From 521aa4919f818bc7502ba02db15c56cfd07f98e7 Mon Sep 17 00:00:00 2001 From: Matt Simerson Date: Mon, 21 May 2012 17:30:11 -0400 Subject: [PATCH 1/5] basicheaders, add reject option, loglevel added reject option document the existence of the loglevel option factored date validity tests into their own sub added tests improved POD --- Changes | 2 + UPGRADING | 9 +++ docs/config.pod | 1 - plugins/check_basicheaders | 97 +++++++++++++++++++++++-------- t/plugin_tests/check_basicheaders | 64 +++++++++++++++++--- 5 files changed, 139 insertions(+), 34 deletions(-) diff --git a/Changes b/Changes index ac9d2cf..ead962e 100644 --- a/Changes +++ b/Changes @@ -1,6 +1,8 @@ Next Version + check_basicheaders. see UPGRADING (Matt Simerson) + sender_permitted_from. see UPGRADING (Matt Simerson) dspam plugin added (Matt Simerson) diff --git a/UPGRADING b/UPGRADING index 5b15721..207b5ac 100644 --- a/UPGRADING +++ b/UPGRADING @@ -3,6 +3,15 @@ When upgrading from: v 0.84 or below +CHECK_BASICHEADERS: + + Deprecated 'days' option in favor of past/future. See 'perldoc plugins/check_basicheaders'. + + +CHECK_RELAY, CHECK_NORELAY, RELAY_ONLY + + All 3 plugins are deprecated and replaced with a new 'relay' plugin. The new plugin reads the same config files (see 'perldoc plugins/relay') as the previous plugins. To get the equivalent functionality of enabling 'relay_only', use the 'only' argument to the relay plugin as documented in the RELAY ONLY section of plugins/relay. + GREYLISTING plugin: 'mode' config argument is deprecated. Use reject and reject_type instead. diff --git a/docs/config.pod b/docs/config.pod index 158aee4..4103eb5 100644 --- a/docs/config.pod +++ b/docs/config.pod @@ -119,7 +119,6 @@ only be used for some extremly rude clients: if list is too big it will slow down accepting new connections. =item relayclients - =item morerelayclients Plugin: F diff --git a/plugins/check_basicheaders b/plugins/check_basicheaders index 114867a..70f765f 100644 --- a/plugins/check_basicheaders +++ b/plugins/check_basicheaders @@ -2,43 +2,69 @@ =head1 NAME -check_basicheaders - Make sure both From and Date headers are present, and -do optional range checking on the Date header. +check_basicheaders =head1 DESCRIPTION -Rejects messages that do not have a From or Date header or are completely -empty. +Checks for missing or empty values in the From or Date headers. -Can also reject messages where the date in the Date header is more than -some number of the days in the past or future. +Optionally test if the Date header is too many days in the past or future. If +I or I are not defined, they are not tested. =head1 CONFIGURATION -The following optional parameters exist: +The following optional settings exist: + +=head2 future + +The number of days in the future beyond which messages are invalid. + + check_basicheaders [ future 1 ] + +=head2 past + +The number of days in the past beyond which a message is invalid. The Date header is added by the MUA, so there are many valid reasons a message may have an older date in the header. It could have been delayed by the client, the sending server, connectivity problems, recipient server problem, recipient server configuration, etc. The I setting should take those factors into consideration. + +I would be surprised if a valid message ever had a date header older than a week. + + check_basicheaders [ past 5 ] =head2 days -The number of days in the future or past beyond which to reject messages. When -unset, messages are not rejected based on the date. +Deprecated. Use I and I instead. + +The number of days in the future or past beyond which messages are invalid. check_basicheaders [ days 3 ] +=head2 reject + +Determine if the connection is denied. Use the I option when first enabling the plugin, and then watch your logs to see what would have been rejected. When you are no longer concerned that valid messages will be rejected, enable with I. + + check_basicheaders [ reject 0 | 1 ] + +Default policy is to reject. + =head2 reject_type Whether to issue a permanent or temporary rejection. The default is permanent. check_basicheaders reject_type [ temp | perm ] -Switching to a temporary rejection is most useful when testing the plugin. It -allows an administrator to watch for a test period and make sure no valid mail -is getting rejected. +Using a temporary rejection is a cautious way to enable rejections. It allows an administrator to watch for a trial period and assure no valid messages are rejected. If a deferral of valid mail is noticed, I can be set to permit the deferred message to be delivered. + +Default policy is a permanent rejection. + +=head2 loglevel + +Adjust the quantity of logging for this plugin. See docs/logging.pod =head1 AUTHOR 2004 - Written by Jim Winstead Jr. 2012 - added logging, named arguments, reject_type, tests - Matt Simerson + - deprecate days for I & I. Improved POD =head1 LICENSE @@ -46,13 +72,18 @@ Released to the public domain, 26 March 2004. =cut +use strict; +use warnings; + +use Qpsmtpd::Constants; + use Date::Parse qw(str2time); sub register { my ($self, $qp, @args) = @_; if ( @args == 1 ) { - $self->log(LOGWARN, "deprecated arguments. Update your arguments to this plugin"); + $self->log(LOGWARN, "deprecated arguments. Update your config."); $self->{_args}{days} = $args[0]; } elsif ( @args % 2 ) { @@ -61,16 +92,27 @@ sub register { else { $self->{_args} = { @args }; }; +# provide backwards comptibility with the old 'days' argument + if ( $self->{_args}{days} ) { + $self->log(LOGWARN, "deprecated argument 'days', update your config."); + if ( ! defined $self->{_args}{future} ) { + $self->{_args}{future} = $self->{_args}{days}; + }; + if ( ! defined $self->{_args}{past} ) { + $self->{_args}{past} = $self->{_args}{days}; + }; + }; } sub hook_data_post { my ($self, $transaction) = @_; my $deny = $self->{_args}{reject_type} eq 'temp' ? DENYSOFT : DENY; + $deny = DECLINED if defined $self->{_args}{reject} && ! $self->{_args}{reject}; if ( $transaction->data_size == 0 ) { $self->log(LOGINFO, "fail: no data"); - return ($deny, "You have to send some data first"); + return ($deny, "You must send some data first"); }; my $header = $transaction->header or do { @@ -88,27 +130,34 @@ sub hook_data_post { return ($deny, "We require a valid Date header"); }; - my $days = $self->{_args}{days}; - if ( ! defined $days ) { - $self->log(LOGINFO, "pass: no days arg"); - return (DECLINED); + my $err_msg = $self->invalid_date_range($date); + if ( $err_msg ) { + return ($deny, $err_msg ); }; + return (DECLINED); +}; + +sub invalid_date_range { + my ($self, $date) = @_; + my $ts = str2time($date) or do { $self->log(LOGINFO, "skip: date not parseable ($date)"); - return (DECLINED); + return; }; - if ( $ts < time - ($days*24*3600) ) { + my $past = $self->{_args}{past}; + if ( $past && $ts < time - ($past*24*3600) ) { $self->log(LOGINFO, "fail: date too old ($date)"); - return ($deny, "The Date in the header is too far in the past") + return "The Date header is too far in the past"; }; - if ( $ts > time + ($days*24*3600) ) { + my $future = $self->{_args}{future}; + if ( $future && $ts > time + ($future*24*3600) ) { $self->log(LOGINFO, "fail: date in future ($date)"); - return ($deny, "The Date in the header is too far in the future") + return "The Date header is too far in the future"; }; $self->log(LOGINFO, "pass"); - return (DECLINED); + return; } diff --git a/t/plugin_tests/check_basicheaders b/t/plugin_tests/check_basicheaders index 921030e..2ac5748 100644 --- a/t/plugin_tests/check_basicheaders +++ b/t/plugin_tests/check_basicheaders @@ -7,51 +7,97 @@ use POSIX qw(strftime); use Qpsmtpd::Address; use Qpsmtpd::Constants; +my $test_email = 'matt@example.com'; sub register_tests { my $self = shift; $self->register_test("test_hook_data_post", 7); + $self->register_test('test_invalid_date_range', 7); } -sub test_hook_data_post { +sub setup_test_headers { my $self = shift; - my $reject = $self->{_args}{reject_type}; - my $deny = $reject =~ /^temp|soft$/i ? DENYSOFT : DENY; - my $transaction = $self->qp->transaction; - my $test_email = 'matt@example.com'; my $address = Qpsmtpd::Address->new( "<$test_email>" ); my $header = Mail::Header->new(Modify => 0, MailFrom => "COERCE"); my $now = strftime "%a %b %e %H:%M:%S %Y", localtime time; - my $future = strftime "%a %b %e %H:%M:%S %Y", localtime time + 518400; #6d - my $past = strftime "%a %b %e %H:%M:%S %Y", localtime time - 518400; #6d - $self->{_args}{days} = 5; $transaction->sender($address); $transaction->header($header); $transaction->header->add('From', "<$test_email>"); $transaction->header->add('Date', $now ); $transaction->body_write( "test message body " ); +}; + +sub test_invalid_date_range { + my $self = shift; + + my $now = strftime "%a %b %e %H:%M:%S %Y", localtime time; + ok( ! $self->invalid_date_range($now), "valid +"); + + $self->{_args}{future} = 2; + + my $future_6 = strftime "%a %b %e %H:%M:%S %Y", localtime time + 518400; #6d + my $r = $self->invalid_date_range( $future_6 ); + ok( $r, "too new -" ); + + my $future_3 = strftime "%a %b %e %H:%M:%S %Y", localtime time + 259200; #3d + $r = $self->invalid_date_range( $future_3 ); + ok( $r, "too new -" ); + + my $future_1 = strftime "%a %b %e %H:%M:%S %Y", localtime time + 86400; #1d + $r = $self->invalid_date_range( $future_1 ); + ok( ! $r, "a little new, +" ); + + + $self->{_args}{past} = 2; + + my $past_6 = strftime "%a %b %e %H:%M:%S %Y", localtime time - 518400; #6d + $r = $self->invalid_date_range( $past_6 ); + ok( $r, "too old -" ); + + my $past_3 = strftime "%a %b %e %H:%M:%S %Y", localtime time - 259200; #3d + $r = $self->invalid_date_range( $past_3 ); + ok( $r, "too old -" ); + + my $past_1 = strftime "%a %b %e %H:%M:%S %Y", localtime time - 86400; #1d + $r = $self->invalid_date_range( $past_1 ); + ok( ! $r, "a little old +" ); +}; + +sub test_hook_data_post { + my $self = shift; + + my $reject = $self->{_args}{reject_type}; + my $deny = $reject =~ /^temp|soft$/i ? DENYSOFT : DENY; + + $self->setup_test_headers(); + my $transaction = $self->qp->transaction; my ($code, $mess) = $self->hook_data_post( $transaction ); - cmp_ok( DECLINED, '==', $code, "okay" ); + cmp_ok( DECLINED, '==', $code, "okay +" ); $transaction->header->delete('Date'); ($code, $mess) = $self->hook_data_post( $transaction ); cmp_ok( $deny, '==', $code, "missing date ( $mess )" ); + my $now = strftime "%a %b %e %H:%M:%S %Y", localtime time; $transaction->header->add('Date', $now ); $transaction->header->delete('From'); ($code, $mess) = $self->hook_data_post( $transaction ); cmp_ok( $deny, '==', $code, "missing from ( $mess )" ); $transaction->header->add('From', "<$test_email>"); + $self->{_args}{future} = 5; + my $future = strftime "%a %b %e %H:%M:%S %Y", localtime time + 518400; #6d $transaction->header->replace('Date', $future ); ($code, $mess) = $self->hook_data_post( $transaction ); cmp_ok( $deny, '==', $code, "too new ( $mess )" ); + $self->{_args}{past} = 5; + my $past = strftime "%a %b %e %H:%M:%S %Y", localtime time - 518400; #6d $transaction->header->replace('Date', $past ); ($code, $mess) = $self->hook_data_post( $transaction ); cmp_ok( $deny, '==', $code, "too old ( $mess )" ); From 80b94eb47ac92535736b87be6d0ad4e1f924020a Mon Sep 17 00:00:00 2001 From: Matt Simerson Date: Tue, 22 May 2012 18:14:10 -0400 Subject: [PATCH 2/5] removed newline --- plugins/check_basicheaders | 1 + 1 file changed, 1 insertion(+) diff --git a/plugins/check_basicheaders b/plugins/check_basicheaders index 70f765f..8f0e1c5 100644 --- a/plugins/check_basicheaders +++ b/plugins/check_basicheaders @@ -129,6 +129,7 @@ sub hook_data_post { $self->log(LOGINFO, "fail: no date"); return ($deny, "We require a valid Date header"); }; + chomp $date; my $err_msg = $self->invalid_date_range($date); if ( $err_msg ) { From 09935b0bf6e6a51c245acd6efdaffca8726ea993 Mon Sep 17 00:00:00 2001 From: Matt Simerson Date: Wed, 23 May 2012 17:12:26 -0400 Subject: [PATCH 3/5] basicheaders: added whitelist support because alerts.etrade.com doesn't set a Date header in alerts --- plugins/check_basicheaders | 25 +++++++++++++++++++++++++ 1 file changed, 25 insertions(+) diff --git a/plugins/check_basicheaders b/plugins/check_basicheaders index 8f0e1c5..889fac0 100644 --- a/plugins/check_basicheaders +++ b/plugins/check_basicheaders @@ -11,6 +11,8 @@ Checks for missing or empty values in the From or Date headers. Optionally test if the Date header is too many days in the past or future. If I or I are not defined, they are not tested. +If the remote IP is whitelisted, header validation is skipped. + =head1 CONFIGURATION The following optional settings exist: @@ -120,6 +122,8 @@ sub hook_data_post { return ($deny, "missing header"); }; + return DECLINED if $self->is_immune(); + if ( ! $header->get('From') ) { $self->log(LOGINFO, "fail: no from"); return ($deny, "We require a valid From header") @@ -162,3 +166,24 @@ sub invalid_date_range { $self->log(LOGINFO, "pass"); return; } + +sub is_immune { + my $self = shift; + + if ( $self->qp->connection->relay_client() ) { + $self->log(LOGINFO, "skip: relay client"); + return 1; + }; + + if ( $self->qp->connection->notes('whitelisthost') ) { + $self->log(LOGINFO, "skip: whitelisted host"); + return 1; + }; + + if ( $self->qp->transaction->notes('whitelistsender') ) { + $self->log(LOGINFO, "skip: whitelisted sender"); + return 1; + }; + + return; +}; From 23f06fde7ac83e3ed30b3ba7c0333cf691c82c91 Mon Sep 17 00:00:00 2001 From: Matt Simerson Date: Wed, 23 May 2012 17:56:06 -0400 Subject: [PATCH 4/5] basicheaders: removed deprecated argument warning --- UPGRADING | 5 ----- plugins/check_basicheaders | 12 +----------- 2 files changed, 1 insertion(+), 16 deletions(-) diff --git a/UPGRADING b/UPGRADING index 207b5ac..7a3b478 100644 --- a/UPGRADING +++ b/UPGRADING @@ -3,11 +3,6 @@ When upgrading from: v 0.84 or below -CHECK_BASICHEADERS: - - Deprecated 'days' option in favor of past/future. See 'perldoc plugins/check_basicheaders'. - - CHECK_RELAY, CHECK_NORELAY, RELAY_ONLY All 3 plugins are deprecated and replaced with a new 'relay' plugin. The new plugin reads the same config files (see 'perldoc plugins/relay') as the previous plugins. To get the equivalent functionality of enabling 'relay_only', use the 'only' argument to the relay plugin as documented in the RELAY ONLY section of plugins/relay. diff --git a/plugins/check_basicheaders b/plugins/check_basicheaders index 889fac0..684b9a4 100644 --- a/plugins/check_basicheaders +++ b/plugins/check_basicheaders @@ -31,14 +31,6 @@ I would be surprised if a valid message ever had a date header older than a week check_basicheaders [ past 5 ] -=head2 days - -Deprecated. Use I and I instead. - -The number of days in the future or past beyond which messages are invalid. - - check_basicheaders [ days 3 ] - =head2 reject Determine if the connection is denied. Use the I option when first enabling the plugin, and then watch your logs to see what would have been rejected. When you are no longer concerned that valid messages will be rejected, enable with I. @@ -85,7 +77,6 @@ sub register { my ($self, $qp, @args) = @_; if ( @args == 1 ) { - $self->log(LOGWARN, "deprecated arguments. Update your config."); $self->{_args}{days} = $args[0]; } elsif ( @args % 2 ) { @@ -94,9 +85,8 @@ sub register { else { $self->{_args} = { @args }; }; -# provide backwards comptibility with the old 'days' argument +# provide backwards comptibility with the previous unnamed 'days' argument if ( $self->{_args}{days} ) { - $self->log(LOGWARN, "deprecated argument 'days', update your config."); if ( ! defined $self->{_args}{future} ) { $self->{_args}{future} = $self->{_args}{days}; }; From 162f2c13e7b51c912737a41622ec25017cdc9bf4 Mon Sep 17 00:00:00 2001 From: Matt Simerson Date: Wed, 23 May 2012 18:07:15 -0400 Subject: [PATCH 5/5] basicheaders: updated Changes with brief summary --- Changes | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Changes b/Changes index ead962e..547bac5 100644 --- a/Changes +++ b/Changes @@ -1,7 +1,7 @@ Next Version - check_basicheaders. see UPGRADING (Matt Simerson) + check_basicheaders. New arguments available: past, future, reject, reject_type sender_permitted_from. see UPGRADING (Matt Simerson)