t/mkrules.t (375 lines of code) (raw):

#!/usr/bin/perl -T use lib '.'; use lib 't'; use SATest; sa_t_init("mkrules"); use Test::More 1.302067; plan tests => 97; use File::Copy; use File::Path; # --------------------------------------------------------------------------- print " script runs, even with nothing to do\n\n"; $workdir =~ s!\\!/!g if $RUNNING_ON_WINDOWS; my $tdir = "$workdir/mkrules_t"; mkpath ([$tdir, "$tdir/rulesrc", "$tdir/rules"]); write_file("$tdir/MANIFEST", [ ]); write_file("$tdir/MANIFEST.SKIP", [ "foo2\n" ]); write_file("$tdir/rules/active.list", [ "" ]); ok (mkrun ("--src $tdir/rulesrc --out $tdir/rules --manifest $tdir/MANIFEST --manifestskip $tdir/MANIFEST.SKIP --active $tdir/rules/active.list", \&patterns_run_cb)); ok ok_all_patterns(); save_tdir(); # --------------------------------------------------------------------------- print " promotion of an active rule\n\n"; %patterns = ( '72_active.cf: WARNING: not listed in manifest file' => manif_found, "body GOOD /foo/" => rule_line_1, "describe GOOD desc_found" => rule_line_2, ); %anti_patterns = ( "describe T_GOOD desc_found" => rule_line_2, ); mkpath ([ "$tdir/rulesrc/sandbox/foo", "$tdir/rules" ]); write_file("$tdir/MANIFEST", [ ]); write_file("$tdir/MANIFEST.SKIP", [ "foo2\n" ]); write_file("$tdir/rules/active.list", [ "GOOD\n" ]); write_file("$tdir/rulesrc/sandbox/foo/20_foo.cf", [ "body GOOD /foo/\n", "describe GOOD desc_found\n" ]); ok (mkrun ("--src $tdir/rulesrc --out $tdir/rules --manifest $tdir/MANIFEST --manifestskip $tdir/MANIFEST.SKIP --active $tdir/rules/active.list 2>&1", \&patterns_run_cb)); checkfile("$tdir/rules/72_active.cf", \&patterns_run_cb); checkfile("$tdir/rules/70_sandbox.cf", \&patterns_run_cb); ok ok_all_patterns(); save_tdir(); # --------------------------------------------------------------------------- print " non-promotion of an inactive rule\n\n"; %patterns = ( '70_sandbox.cf: WARNING: not listed in manifest file' => manif_found, "body T_GOOD /foo/" => rule_line_1, "describe T_GOOD desc_found" => rule_line_2, ); %anti_patterns = ( "describe GOOD desc_found" => rule_line_2, ); mkpath ([ "$tdir/rulesrc/sandbox/foo", "$tdir/rules" ]); write_file("$tdir/MANIFEST", [ ]); write_file("$tdir/MANIFEST.SKIP", [ "foo2\n" ]); write_file("$tdir/rules/active.list", [ "NOT_GOOD\n" ]); write_file("$tdir/rulesrc/sandbox/foo/20_foo.cf", [ "body GOOD /foo/\n", "describe GOOD desc_found\n" ]); ok (mkrun ("--src $tdir/rulesrc --out $tdir/rules --manifest $tdir/MANIFEST --manifestskip $tdir/MANIFEST.SKIP --active $tdir/rules/active.list 2>&1", \&patterns_run_cb)); checkfile("$tdir/rules/72_active.cf", \&patterns_run_cb); checkfile("$tdir/rules/70_sandbox.cf", \&patterns_run_cb); ok ok_all_patterns(); save_tdir(); # --------------------------------------------------------------------------- print " non-promotion of an inactive rule with score set\n\n"; %patterns = ( '70_sandbox.cf: WARNING: not listed in manifest file' => manif_found, "body T_GOOD /foo/" => rule_line_1, "describe T_GOOD desc_found" => rule_line_2, "#score T_GOOD 4.0" => score_good, ); %anti_patterns = ( "describe GOOD desc_found" => rule_line_2, "score GOOD 4.0" => 'score', ); mkpath ([ "$tdir/rulesrc/sandbox/foo", "$tdir/rules" ]); write_file("$tdir/MANIFEST", [ ]); write_file("$tdir/MANIFEST.SKIP", [ "foo2\n" ]); write_file("$tdir/rules/active.list", [ "NOT_GOOD\n" ]); write_file("$tdir/rulesrc/sandbox/foo/20_foo.cf", [ "body GOOD /foo/\n", "score GOOD 4.0\n", "describe GOOD desc_found\n" ]); ok (mkrun ("--src $tdir/rulesrc --out $tdir/rules --manifest $tdir/MANIFEST --manifestskip $tdir/MANIFEST.SKIP --active $tdir/rules/active.list 2>&1", \&patterns_run_cb)); checkfile("$tdir/rules/72_active.cf", \&patterns_run_cb); checkfile("$tdir/rules/70_sandbox.cf", \&patterns_run_cb); ok ok_all_patterns(); save_tdir(); # --------------------------------------------------------------------------- print " non-promotion of a broken rule\n\n"; %patterns = ( '70_sandbox.cf: WARNING: not listed in manifest file' => manif_found, 'LINT FAILED' => lint_failed, ); %anti_patterns = ( "body GOOD" => rule_line_1, "describe GOOD desc_found" => rule_line_2, ); mkpath ([ "$tdir/rulesrc/sandbox/foo", "$tdir/rules" ]); write_file("$tdir/MANIFEST", [ ]); write_file("$tdir/MANIFEST.SKIP", [ "foo2\n" ]); write_file("$tdir/rules/active.list", [ "GOOD\n" ]); write_file("$tdir/rulesrc/sandbox/foo/20_foo.cf", [ "body GOOD /***\n", "describe GOOD desc_found\n" ]); ok (mkrun ("--src $tdir/rulesrc --out $tdir/rules --manifest $tdir/MANIFEST --manifestskip $tdir/MANIFEST.SKIP --active $tdir/rules/active.list 2>&1", \&patterns_run_cb)); checkfile("$tdir/rules/70_sandbox.cf", \&patterns_run_cb); ok (-f "$tdir/rules/72_active.cf"); ok (-s "$tdir/rules/72_active.cf" == 0); ok ok_all_patterns(); save_tdir(); # --------------------------------------------------------------------------- print " promotion of an active meta rule\n\n"; %patterns = ( '70_sandbox.cf: WARNING: not listed in manifest file' => manif_found, '20_foo.cf: 1 active rules, 1 other' => 'foundrule', "body __GOOD /foo/" => rule_line_1, "meta GOOD (__GOOD)" => rule_line_1a, "describe GOOD desc_found" => rule_line_2, ); %anti_patterns = ( "describe T_GOOD desc_found" => rule_line_2, ); mkpath ([ "$tdir/rulesrc/sandbox/foo", "$tdir/rules" ]); write_file("$tdir/MANIFEST", [ ]); write_file("$tdir/MANIFEST.SKIP", [ "foo2\n" ]); write_file("$tdir/rules/active.list", [ "GOOD\n" ]); write_file("$tdir/rulesrc/sandbox/foo/20_foo.cf", [ "body __GOOD /foo/\n", "meta GOOD (__GOOD)\n", "describe GOOD desc_found\n" ]); ok (mkrun ("--src $tdir/rulesrc --out $tdir/rules --manifest $tdir/MANIFEST --manifestskip $tdir/MANIFEST.SKIP --active $tdir/rules/active.list 2>&1", \&patterns_run_cb)); checkfile("$tdir/rules/72_active.cf", \&patterns_run_cb); checkfile("$tdir/rules/70_sandbox.cf", \&patterns_run_cb); ok ok_all_patterns(); save_tdir(); # --------------------------------------------------------------------------- print " inactive meta rule\n\n"; %patterns = ( '70_sandbox.cf: WARNING: not listed in manifest file' => manif_found, '20_foo.cf: 0 active rules, 2 other' => 'foundrule', "body __GOOD /foo/" => rule_line_1, "meta T_GOOD (__GOOD)" => rule_line_1a, "describe T_GOOD desc_found" => rule_line_2, ); %anti_patterns = ( "describe GOOD desc_found" => rule_line_2, ); mkpath ([ "$tdir/rulesrc/sandbox/foo", "$tdir/rules" ]); write_file("$tdir/MANIFEST", [ ]); write_file("$tdir/MANIFEST.SKIP", [ "foo2\n" ]); write_file("$tdir/rules/active.list", [ "NOT_GOOD\n" ]); write_file("$tdir/rulesrc/sandbox/foo/20_foo.cf", [ "body __GOOD /foo/\n", "meta GOOD (__GOOD)\n", "describe GOOD desc_found\n" ]); ok (mkrun ("--src $tdir/rulesrc --out $tdir/rules --manifest $tdir/MANIFEST --manifestskip $tdir/MANIFEST.SKIP --active $tdir/rules/active.list 2>&1", \&patterns_run_cb)); checkfile("$tdir/rules/72_active.cf", \&patterns_run_cb); checkfile("$tdir/rules/70_sandbox.cf", \&patterns_run_cb); ok ok_all_patterns(); save_tdir(); # --------------------------------------------------------------------------- print " active plugin in sandbox\n\n"; %patterns = ( '70_sandbox.cf: WARNING: not listed in manifest file' => manif_found, "loadplugin Good plugin.pm" => loadplugin_found, "body GOOD eval:check_foo()" => rule_line_1, "describe GOOD desc_found" => rule_line_2, "ifplugin Good" => if1, "endif" => endif_found, ); %anti_patterns = ( "describe T_GOOD desc_found" => rule_line_2, ); mkpath ([ "$tdir/rulesrc/sandbox/foo", "$tdir/rules" ]); write_file("$tdir/MANIFEST", [ "rulesrc/sandbox/foo/20_foo.cf\n", "rulesrc/sandbox/foo/plugin.pm\n" ]); write_file("$tdir/MANIFEST.SKIP", [ "foo2\n" ]); write_file("$tdir/rules/active.list", [ "GOOD\n" ]); write_file("$tdir/rulesrc/sandbox/foo/20_foo.cf", [ "loadplugin Good plugin.pm\n", "ifplugin Good\n", "body GOOD eval:check_foo()\n", "describe GOOD desc_found\n", "endif\n", ]); write_file("$tdir/rulesrc/sandbox/foo/plugin.pm", [ 'package Good;', 'use Mail::SpamAssassin::Plugin; our @ISA = qw(Mail::SpamAssassin::Plugin);', 'sub new { my ($class, $m) = @_; $class = ref($class) || $class;', 'my $self = bless $class->SUPER::new($m), $class;', '$self->register_eval_rule("check_foo"); return $self; }', 'sub check_foo { my ($self, $pms) = @_; return 1; }', ]); ok (mkrun ("--src $tdir/rulesrc --out $tdir/rules --manifest $tdir/MANIFEST --manifestskip $tdir/MANIFEST.SKIP --active $tdir/rules/active.list 2>&1", \&patterns_run_cb)); # checkfile("$tdir/rules/72_active.cf", \&patterns_run_cb); checkfile("$tdir/rules/70_sandbox.cf", \&patterns_run_cb); ok (-f "$tdir/rules/plugin.pm"); ok ok_all_patterns(); save_tdir(); # --------------------------------------------------------------------------- print " inactive plugin\n\n"; %patterns = ( '70_sandbox.cf: WARNING: not listed in manifest file' => manif_found, # "WARNING: GOOD: renamed as T_GOOD due to missing T_ prefix" => warning_seen, "loadplugin Good plugin.pm" => loadplugin_found, "body T_GOOD eval:check_foo()" => rule_line_1, "describe T_GOOD desc_found" => rule_line_2, "ifplugin Good" => if1, "endif" => endif_found, ); %anti_patterns = ( "describe GOOD desc_found" => rule_line_2, ); mkpath ([ "$tdir/rulesrc/sandbox/foo", "$tdir/rules" ]); write_file("$tdir/MANIFEST", [ "rulesrc/sandbox/foo/20_foo.cf\n", "rulesrc/sandbox/foo/plugin.pm\n" ]); write_file("$tdir/MANIFEST.SKIP", [ "foo2\n" ]); write_file("$tdir/rules/active.list", [ "NOT_GOOD\n" ]); write_file("$tdir/rulesrc/sandbox/foo/20_foo.cf", [ "loadplugin Good plugin.pm\n", "ifplugin Good\n", "body GOOD eval:check_foo()\n", "describe GOOD desc_found\n", "endif\n", ]); write_file("$tdir/rulesrc/sandbox/foo/plugin.pm", [ 'package Good;', 'use Mail::SpamAssassin::Plugin; our @ISA = qw(Mail::SpamAssassin::Plugin);', 'sub new { my ($class, $m) = @_; $class = ref($class) || $class;', 'my $self = bless $class->SUPER::new($m), $class;', '$self->register_eval_rule("check_foo"); return $self; }', 'sub check_foo { my ($self, $pms) = @_; return 1; }', ]); ok (mkrun ("--src $tdir/rulesrc --out $tdir/rules --manifest $tdir/MANIFEST --manifestskip $tdir/MANIFEST.SKIP --active $tdir/rules/active.list 2>&1", \&patterns_run_cb)); # checkfile("$tdir/rules/72_active.cf", \&patterns_run_cb); checkfile("$tdir/rules/70_sandbox.cf", \&patterns_run_cb); ok (-f "$tdir/rules/plugin.pm"); ok ok_all_patterns(); save_tdir(); # --------------------------------------------------------------------------- print " active plugin, but the .pm file is AWOL\n\n"; %patterns = ( "body GOOD eval:check_foo()" => rule_line_1, "describe GOOD desc_found" => rule_line_2, "ifplugin Good" => if1, "endif" => endif_found, "rulesrc/sandbox/foo/20_foo.cf: WARNING: plugin code file '$workdir/mkrules_t/rulesrc/sandbox/foo/plugin.pm' not found, line ignored: loadplugin Good plugin.pm" => plugin_not_found, ); %anti_patterns = ( "describe T_GOOD desc_found" => rule_line_2, ); rmtree([ $tdir ]); mkpath ([ "$tdir/rulesrc/sandbox/foo", "$tdir/rules" ]); write_file("$tdir/MANIFEST", [ "rulesrc/sandbox/foo/20_foo.cf\n", "rulesrc/sandbox/foo/plugin.pm\n" ]); write_file("$tdir/MANIFEST.SKIP", [ "foo2\n" ]); write_file("$tdir/rules/active.list", [ "GOOD\n" ]); write_file("$tdir/rulesrc/sandbox/foo/20_foo.cf", [ "loadplugin Good plugin.pm\n", "ifplugin Good\n", "body GOOD eval:check_foo()\n", "describe GOOD desc_found\n", "endif\n", ]); ok (mkrun ("--src $tdir/rulesrc --out $tdir/rules --manifest $tdir/MANIFEST --manifestskip $tdir/MANIFEST.SKIP --active $tdir/rules/active.list 2>&1", \&patterns_run_cb)); checkfile("$tdir/rules/72_active.cf", \&patterns_run_cb); # checkfile("$tdir/rules/70_sandbox.cf", \&patterns_run_cb); ok (!-f "$tdir/rules/plugin.pm"); ok ok_all_patterns(); save_tdir(); # --------------------------------------------------------------------------- print " active plugin, but the .pm file is not in MANIFEST\n\n"; %patterns = ( "body GOOD eval:check_foo()" => rule_line_1, "describe GOOD desc_found" => rule_line_2, "ifplugin Good" => if1, "endif" => endif_found, "tryplugin Good plugin.pm" => 'tryplugin', "$workdir/mkrules_t/rulesrc/sandbox/foo/20_foo.cf: WARNING: '$workdir/mkrules_t/rules/plugin.pm' not listed in manifest file, making 'tryplugin': loadplugin Good plugin.pm" => not_found_in_manifest_warning ); %anti_patterns = ( ); rmtree([ $tdir ]); mkpath ([ "$tdir/rulesrc/sandbox/foo", "$tdir/rules" ]); write_file("$tdir/MANIFEST", [ "rulesrc/sandbox/foo/20_foo.cf\n" ]); write_file("$tdir/MANIFEST.SKIP", [ ]); write_file("$tdir/rules/active.list", [ "GOOD\n" ]); write_file("$tdir/rulesrc/sandbox/foo/20_foo.cf", [ "loadplugin Good plugin.pm\n", "ifplugin Good\n", "body GOOD eval:check_foo()\n", "describe GOOD desc_found\n", "endif\n", ]); write_file("$tdir/rulesrc/sandbox/foo/plugin.pm", [ 'package Good;', 'use Mail::SpamAssassin::Plugin; our @ISA = qw(Mail::SpamAssassin::Plugin);', 'sub new { my ($class, $m) = @_; $class = ref($class) || $class;', 'my $self = bless $class->SUPER::new($m), $class;', '$self->register_eval_rule("check_foo"); return $self; }', 'sub check_foo { my ($self, $pms) = @_; return 1; }', ]); ok (mkrun ("--src $tdir/rulesrc --out $tdir/rules --manifest $tdir/MANIFEST --manifestskip $tdir/MANIFEST.SKIP --active $tdir/rules/active.list 2>&1", \&patterns_run_cb)); # checkfile("$tdir/rules/72_active.cf", \&patterns_run_cb); checkfile("$tdir/rules/70_sandbox.cf", \&patterns_run_cb); ok (-f "$tdir/rules/plugin.pm"); ok ok_all_patterns(); save_tdir(); # --------------------------------------------------------------------------- print "meta rule depends on unpromoted subrule in lexically-earlier file\n\n"; # (see mail from Sidney of Oct 16 2006, rules HS_INDEX_PARAM and HS_PHARMA_1) %patterns = ( "header T_GOOD_SUB" => rule_line_1, "header T_BAD_SUB" => rule_line_2, "meta GOOD (T_GOOD_SUB && !T_BAD_SUB)" => meta_found ); %anti_patterns = ( ); rmtree([ $tdir ]); mkpath ([ "$tdir/rulesrc/sandbox/foo", "$tdir/rules" ]); write_file("$tdir/MANIFEST", [ "rules/72_active.cf\n" ]); write_file("$tdir/MANIFEST.SKIP", [ ]); write_file("$tdir/rules/active.list", [ "GOOD\n" ]); write_file("$tdir/rulesrc/sandbox/foo/20_aaa.cf", [ "meta GOOD (GOOD_SUB && !BAD_SUB)\n", ]); write_file("$tdir/rulesrc/sandbox/foo/20_bbb.cf", [ "header GOOD_SUB Foo =~ /good/\n", "header BAD_SUB Foo =~ /bad/\n", ]); ok (mkrun ("--src $tdir/rulesrc --out $tdir/rules --manifest $tdir/MANIFEST --manifestskip $tdir/MANIFEST.SKIP --active $tdir/rules/active.list 2>&1", \&patterns_run_cb)); checkfile("$tdir/rules/72_active.cf", \&patterns_run_cb); ok ok_all_patterns(); save_tdir(); # --------------------------------------------------------------------------- print " nested conditionals\n\n"; %patterns = ( '72_active.cf: WARNING: not listed in manifest file' => manif_found, "body GOOD /foo/" => rule_line_1, "describe GOOD desc_found" => rule_line_2, "ifplugin Mail::SpamAssassin::Plugin::DKIM" => 'ifplugin', "if (version >= 3.002000)" => 'ifversion', ); %anti_patterns = ( "describe T_GOOD desc_found" => rule_line_2, ); mkpath ([ "$tdir/rulesrc/sandbox/foo", "$tdir/rules" ]); write_file("$tdir/MANIFEST", [ ]); write_file("$tdir/MANIFEST.SKIP", [ "foo2\n" ]); write_file("$tdir/rules/active.list", [ "GOOD\n" ]); write_file("$tdir/rulesrc/sandbox/foo/20_foo.cf", [ "ifplugin Mail::SpamAssassin::Plugin::DKIM\n", "if (version >= 3.002000)\n", "body GOOD /foo/\n", "describe GOOD desc_found\n", "endif\n", "endif\n", ]); ok (mkrun ("--src $tdir/rulesrc --out $tdir/rules --manifest $tdir/MANIFEST --manifestskip $tdir/MANIFEST.SKIP --active $tdir/rules/active.list 2>&1", \&patterns_run_cb)); checkfile("$tdir/rules/72_active.cf", \&patterns_run_cb); checkfile("$tdir/rules/70_sandbox.cf", \&patterns_run_cb); ok ok_all_patterns(); save_tdir(); # --------------------------------------------------------------------------- exit; sub write_file { my $file = shift; my $linesref = shift; open (O, ">$file") or die "cannot write to $file"; print O @$linesref; close O or die "cannot save $file"; } sub mkrun { my $args = shift; my $read_sub = shift; my $post_redir = ''; $args =~ s/ 2\>\&1$// and $post_redir = ' 2>&1'; rmtree ("$workdir/outputdir.tmp"); # some tests use this mkdir ("$workdir/outputdir.tmp", 0755); clear_pattern_counters(); my $scrargs = "$perl_path -I../lib ../build/mkrules $args"; print ("\t$scrargs\n"); my $test_number = test_number(); untaint_system ("$scrargs > $workdir/$testname.$test_number $post_redir"); $mk_exitcode = ($?>>8); if ($mk_exitcode != 0) { return undef; } &checkfile ("$workdir/$testname.$test_number", $read_sub) if (defined $read_sub); 1; } sub save_tdir { my $test_number = test_number(); rmtree("$tdir.$test_number"); if (move( "$tdir", "$tdir.$test_number")) { print "\ttest output tree copied to $tdir.$test_number\n"; } }