#!perl -w =head1 NAME dkim: validate DomainKeys and (DKIM) Domain Keys Indentified Messages =head1 SYNOPSIS Validate the DKIM and Domainkeys signatures of a message, enforce DKIM sending policies, and DKIM sign outgoing messages. =head1 CONFIGURATION =head2 reject [ 0 | 1 | naughty ] dkim reject 0 Reject is a boolean that toggles message rejection on or off, or naughty, which offloads a deferred rejection to the B plugin. Default: 1 =head2 reject_type dkim reject_type [ temp | perm ] Default: perm =head1 HOW TO SIGN =head2 generate DKIM key(s) mkdir -p ~smtpd/config/dkim/example.org cd ~smtpd/config/dkim/example.org echo 'mar2013' > selector openssl genrsa -out private 2048 chmod 400 private openssl rsa -in private -out public -pubout chown -R smtpd:smtpd ~smtpd/config/dkim/example.org After running the commands, you'll have a directory with three files: example.org example.org/selector example.org/private example.org/public =head3 selector The selector can be any value that is a valid DNS label. =head3 key length The minimum recommended key length for short duration keys (ones that will be replaced within a few months) is 1024. If you are unlikely to rotate your keys frequently, go with 2048, at the expense of more CPU. =head2 publish public key in DNS mar2013._domainkey TXT "v=DKIM1;p=[public key stripped of whitespace];" hash: h=[ sha1 | sha256 ] test; t=[ s | s:y ] granularity: g=[ ] notes: n=[ ] services: s=[email] keytypes: [ rsa ] Prepare the DNS record with these commands: cd ~smtpd/config/dkim/example.org cat selector | tr -d "\n" > dns echo -n '._domainkey TXT "v=DKIM1;p=' >> dns grep -v -e '^-' public | tr -d "\n" >> dns echo '"' >> dns The contents of I are ready to be copy/pasted into a BIND zone file, or better yet, NicTool, and published to most any DNS server. =head2 testing After confirming that the DKIM public key can be fetched with DNS, send test messages via QP to a Gmail box and check the Authentication-Results header. There are also DKIM relays (check-auth@verifier.port25.com, checkmyauth@auth.returnpath.net) that provide more debugging information in a nice email report. =head2 Sign for others Following the directions above will configure QP to DKIM sign messages from authenticated senders from example.org. Suppose you host client.com and would like to DKIM sign their messages too? Do that as follows: cd ~smtpd/config/dkim/example.org ln -s example.org client.com QP will follow the symlink target and sign client.com emails with the example.org DKIM key. =head1 SEE ALSO http://www.dkim.org/ http://tools.ietf.org/html/rfc6376 - DKIM Signatures http://tools.ietf.org/html/rfc5863 - DKIM Development, Deployment, & Operations http://tools.ietf.org/html/rfc5617 - DKIM ADSP (Author Domain Signing Practices) http://tools.ietf.org/html/rfc5585 - DKIM Service Overview http://tools.ietf.org/html/rfc5016 - DKIM Signing Practices Protocol http://tools.ietf.org/html/rfc4871 - DKIM Signatures http://tools.ietf.org/html/rfc4870 - DomainKeys http://dkimcore.org/tools/ http://www.protodave.com/tools/dkim-key-checker/ =head1 AUTHORS 2012 - Matt Simerson - initial plugin =head1 ACKNOWLEDGEMENTS David Summers - http://www.nntp.perl.org/group/perl.qpsmtpd/2010/08/msg9417.html Matthew Harrell - http://alecto.bittwiddlers.com/files/qpsmtpd/dkimcheck I first attempted to fix the dkimcheck plugin, but soon scrapped that effort and wrote this one. Why? =over 4 The nine 'if' brackets with 19 conditionals, and my inability to easily determine which of the 15 possible permutations (5 signature validation results x 3 possible policy results) were covered. The use of $dkim->fetch_author_policy, which is deprecated by Mail::DKIM. The paradim of a single policy, when DKIM supports 0 or many. Although I may yet implement the 'local' policy idea, so long as I'm confident it will never result in a false positive. The OBF programming style, which is nigh impossible to test. =back =cut use strict; use warnings; use Qpsmtpd::Constants; # use Mail::DKIM::Verifier; # eval'ed in register() use Socket qw(:DEFAULT :crlf); sub init { my ($self, $qp) = (shift, shift); $self->{_args} = { @_ }; $self->{_args}{reject} = 1 if ! defined $self->{_args}{reject}; $self->{_args}{reject_type} ||= 'perm'; } sub register { my $self = shift; foreach my $mod ( qw/ Mail::DKIM::Verifier Mail::DKIM::Signer Mail::DKIM::TextWrap / ) { eval "use $mod"; if ( $@ ) { warn "error, plugin disabled, could not load $mod\n"; $self->log(LOGERROR, "skip, plugin disabled, is Mail::DKIM installed?"); return; }; }; $self->register_hook('data_post', 'data_post_handler'); }; sub data_post_handler { my ($self, $transaction) = @_; if ( $self->qp->connection->relay_client() ) { # this is one of our authenticated users sending a message. return $self->sign_it( $transaction ); }; return DECLINED if $self->is_immune(); return $self->validate_it( $transaction ); }; sub validate_it { my ($self, $transaction) = @_; # Incoming message, perform DKIM validation my $dkim = Mail::DKIM::Verifier->new() or do { $self->log(LOGERROR, "error, could not instantiate a new Mail::DKIM::Verifier"); return DECLINED; }; $self->send_message_to_dkim( $dkim, $transaction ); my $result = $dkim->result; my $mess = $self->get_details( $dkim ); foreach my $r ( qw/ pass fail invalid temperror none / ) { my $handler = 'handle_sig_' . $r; if ( $result eq $r && $self->can( $handler ) ) { #$self->log(LOGINFO, "dispatching $result to $handler"); return $self->$handler( $dkim, $mess ); }; }; $self->log( LOGERROR, "unknown result: $result, $mess" ); return DECLINED; } sub sign_it { my ($self, $transaction) = @_; my ($domain, $keydir) = $self->get_keydir($transaction) or return DECLINED; my $selector = $self->get_selector($keydir); my $dkim = Mail::DKIM::Signer->new( Algorithm => "rsa-sha256", Method => "relaxed", Domain => $domain, Selector => $selector, KeyFile => "$keydir/private", ); $self->send_message_to_dkim( $dkim, $transaction ); my $signature = $dkim->signature; # what is the signature result? $self->qp->transaction->header->add( 'DKIM-Signature', $signature->as_string, 0 ); $self->log(LOGINFO, "pass, signed message, ", $signature->as_string ); return DECLINED; }; sub get_details { my ($self, $dkim ) = @_; my @data; my $string; push @data, "domain: " . $dkim->signature->domain if $dkim->signature; push @data, "selector: " . $dkim->signature->selector if $dkim->signature; push @data, "result: " . $dkim->result_detail if $dkim->result_detail; foreach my $policy ( $dkim->policies ) { next if ! $policy; push @data, "policy: " . $policy->as_string; push @data, "name: " . $policy->name; push @data, "policy_location: " . $policy->location if $policy->location; my $policy_result; $policy_result = $policy->apply($dkim); $policy_result or next; push @data, "policy_result: " . $policy_result if $policy_result; }; return join(', ', @data); }; sub handle_sig_fail { my ( $self, $dkim, $mess ) = @_; $self->adjust_karma( -1 ); return $self->get_reject( "DKIM signature invalid: " . $dkim->result_detail, $mess ); }; sub handle_sig_temperror { my ( $self, $dkim, $mess ) = @_; $self->log(LOGINFO, "error, $mess" ); return ( DENYSOFT, "Please try again later - $dkim->result_detail" ); }; sub handle_sig_invalid { my ( $self, $dkim, $mess ) = @_; my ( $prs, $policies) = $self->get_policy_results( $dkim ); foreach my $policy ( @$policies ) { if ( $policy->signall && ! $policy->is_implied_default_policy ) { $self->log(LOGINFO, $mess ); return $self->get_reject( "invalid DKIM signature with sign-all policy", "invalid signature, sign-all policy" ); } }; $self->adjust_karma( -1 ); $self->log(LOGINFO, $mess ); if ( $prs->{accept} ) { $self->add_header( $mess ); $self->log( LOGERROR, "error, invalid signature but accept policy!?" ); return DECLINED; } elsif ( $prs->{neutral} ) { $self->add_header( $mess ); $self->log( LOGERROR, "error, invalid signature but neutral policy?!" ); return DECLINED; } elsif ( $prs->{reject} ) { return $self->get_reject( "invalid DKIM signature: " . $dkim->result_detail, "fail, invalid signature, reject policy" ); } # this should never happen $self->log( LOGINFO, "error, invalid signature, unhandled" ); $self->add_header( $mess ); return DECLINED; }; sub handle_sig_pass { my ( $self, $dkim, $mess ) = @_; my ($prs) = $self->get_policy_results( $dkim ); if ( $prs->{accept} ) { $self->add_header( $mess ); $self->log(LOGINFO, "pass, valid signature, accept policy"); $self->adjust_karma( 1 ); return DECLINED; } elsif ( $prs->{neutral} ) { $self->add_header( $mess ); $self->log(LOGINFO, "pass, valid signature, neutral policy"); $self->log(LOGINFO, $mess ); return DECLINED; } elsif ( $prs->{reject} ) { $self->log(LOGINFO, $mess ); $self->adjust_karma( -1 ); return $self->get_reject( "DKIM signature valid but fails policy, $mess", "fail, valid sig, reject policy" ); }; # this should never happen $self->add_header( $mess ); $self->log(LOGERROR, "pass, valid sig, no policy results" ); $self->log(LOGINFO, $mess ); return DECLINED; }; sub handle_sig_none { my ( $self, $dkim, $mess ) = @_; my ( $prs, $policies) = $self->get_policy_results( $dkim ); foreach my $policy ( @$policies ) { if ( $policy->signall && ! $policy->is_implied_default_policy ) { $self->log(LOGINFO, $mess ); return $self->get_reject( "no DKIM signature with sign-all policy", "no signature, sign-all policy" ); } }; if ( $prs->{accept} ) { $self->log( LOGINFO, "pass, no signature, accept policy" ); return DECLINED; } elsif ( $prs->{neutral} ) { $self->log( LOGINFO, "pass, no signature, neutral policy" ); return DECLINED; } elsif ( $prs->{reject} ) { $self->log(LOGINFO, $mess ); $self->get_reject( "no DKIM signature, policy says reject: " . $dkim->result_detail, "no signature, reject policy" ); }; # should never happen $self->log( LOGINFO, "error, no signature, no policy" ); $self->log(LOGINFO, $mess ); return DECLINED; }; sub get_keydir { my ($self, $transaction) = @_; my $domain = $transaction->sender->host; my $dir = "config/dkim/$domain"; if ( -l $dir ) { $dir = readlink($dir); $dir = "config/dkim/$dir" if $dir !~ /\//; # no /, relative path ($domain) = (split /\//, $dir)[-1]; }; if ( ! -d $dir ) { $self->log(LOGINFO, "skip, DKIM not configured for $domain"); return; }; if ( ! -r $dir ) { $self->log(LOGINFO, "error, unable to read key from $dir"); return; }; if ( ! -r "$dir/private" ) { $self->log(LOGINFO, "error, unable to read dkim key from $dir/private"); return; }; return ($domain, $dir); }; sub send_message_to_dkim { my ($self, $dkim, $transaction) = @_; foreach ( split ( /\n/s, $transaction->header->as_string ) ) { $_ =~ s/\r?$//s; eval { $dkim->PRINT ( $_ . CRLF ); }; $self->log(LOGERROR, $@ ) if $@; } $transaction->body_resetpos; while (my $line = $transaction->body_getline) { chomp $line; $line =~ s/\015$//; eval { $dkim->PRINT($line . CRLF ); }; $self->log(LOGERROR, $@ ) if $@; }; $dkim->CLOSE; }; sub get_policies { my ($self, $dkim) = @_; my @policies; eval { @policies = $dkim->policies }; $self->log(LOGERROR, $@ ) if $@; return @policies; }; sub get_policy_results { my ( $self, $dkim ) = @_; my %prs; my @policies = $self->get_policies( $dkim ); foreach my $policy ( @policies ) { my $policy_result; eval { $policy_result = $policy->apply($dkim); }; # accept, reject, neutral if ( $@ ) { $self->log(LOGERROR, $@ ); }; $prs{$policy_result}++ if $policy_result; }; return \%prs, \@policies; }; sub get_selector { my ($self, $keydir) = @_; open my $SFH, '<', "$keydir/selector" or do { $self->log(LOGINFO, "error, unable to read selector from $keydir/selector"); return DECLINED; }; my $selector = <$SFH>; chomp $selector; close $SFH; $self->log(LOGINFO, "info, selector: $selector"); return $selector; }; sub add_header { my $self = shift; my $header = shift or return; # consider adding Authentication-Results header, (RFC 5451) $self->qp->transaction->header->add( 'X-DKIM-Authentication', $header, 0 ); }