| # <@LICENSE> |
| # Licensed to the Apache Software Foundation (ASF) under one or more |
| # contributor license agreements. See the NOTICE file distributed with |
| # this work for additional information regarding copyright ownership. |
| # The ASF licenses this file to you under the Apache License, Version 2.0 |
| # (the "License"); you may not use this file except in compliance with |
| # the License. You may obtain a copy of the License at: |
| # |
| # http://www.apache.org/licenses/LICENSE-2.0 |
| # |
| # Unless required by applicable law or agreed to in writing, software |
| # distributed under the License is distributed on an "AS IS" BASIS, |
| # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| # See the License for the specific language governing permissions and |
| # limitations under the License. |
| # </@LICENSE> |
| |
| =head1 NAME |
| |
| Mail::SpamAssassin::Plugin::WLBLEval - SpamAssassin plugin for blocklist and welcomelist evaluation |
| |
| =head1 SYNOPSIS |
| |
| loadplugin Mail::SpamAssassin::Plugin::WLBLEval |
| |
| # In your rules: |
| header FROM_IN_BLOCKLIST eval:check_from_in_blocklist() |
| header TO_IN_WELCOMELIST eval:check_to_in_welcomelist() |
| |
| =head1 DESCRIPTION |
| |
| This plugin provides functions for blocklists and welcomelists for sender and recipient addresses, as well as URI host lists. |
| It supports matching by address, domain, and relay information. |
| |
| =head1 METHODS |
| |
| =over 4 |
| |
| =item check_from_in_blocklist() |
| |
| Checks if the sender address is in the blocklist. |
| |
| =item check_to_in_blocklist() |
| |
| Checks if the recipient address is in the blocklist. |
| |
| =item check_to_in_welcomelist() |
| |
| Checks if the recipient address is in the welcomelist. |
| |
| =item check_to_in_more_spam() |
| |
| Checks if the recipient address is in the more_spam_to list. |
| |
| =item check_to_in_all_spam() |
| |
| Checks if the recipient address is in the all_spam_to list. |
| |
| =item check_from_in_list($list) |
| |
| Checks if the sender address is in the specified list. |
| |
| =item check_replyto_in_list($list) |
| |
| Checks if the Reply-To address is in the specified list. |
| |
| =item check_wb_list($params) |
| |
| Checks if an address is in a welcomelist or blocklist, based on parameters. |
| |
| =item check_to_in_list($list) |
| |
| Checks if the recipient address is in the specified list. |
| |
| =item check_from_in_welcomelist() |
| |
| Checks if the sender address is in the welcomelist. |
| |
| =item check_forged_in_welcomelist() |
| |
| Checks if the sender address is forged with respect to the welcomelist. |
| |
| =item check_from_in_default_welcomelist() |
| |
| Checks if the sender address is in the default welcomelist. |
| |
| =item check_forged_in_default_welcomelist() |
| |
| Checks if the sender address is forged with respect to the default welcomelist. |
| |
| =item check_mailfrom_matches_rcvd() |
| |
| Checks if the envelope sender domain matches the relay domain. |
| |
| =item check_uri_host_in_blocklist() |
| |
| Checks if any URI host is in the blocklist. |
| |
| =item check_uri_host_in_welcomelist() |
| |
| Checks if any URI host is in the welcomelist. |
| |
| =item check_uri_host_listed($subname) |
| |
| Checks if any URI host is listed in the specified list. |
| |
| =back |
| |
| =cut |
| |
| package Mail::SpamAssassin::Plugin::WLBLEval; |
| |
| use strict; |
| use warnings; |
| # use bytes; |
| use re 'taint'; |
| |
| use NetAddr::IP 4.000; |
| |
| use Mail::SpamAssassin::Plugin; |
| use Mail::SpamAssassin::Logger; |
| |
| our @ISA = qw(Mail::SpamAssassin::Plugin); |
| |
| # constructor: register the eval rule |
| sub new { |
| my $class = shift; |
| my $mailsaobject = shift; |
| |
| # some boilerplate... |
| $class = ref($class) || $class; |
| my $self = $class->SUPER::new($mailsaobject); |
| bless ($self, $class); |
| |
| # the important bit! |
| $self->register_eval_rule("check_from_in_blocklist", $Mail::SpamAssassin::Conf::TYPE_HEAD_EVALS); |
| $self->register_eval_rule("check_from_in_blacklist", $Mail::SpamAssassin::Conf::TYPE_HEAD_EVALS); #Stub - Remove in SA 4.1 |
| $self->register_eval_rule("check_to_in_blocklist", $Mail::SpamAssassin::Conf::TYPE_HEAD_EVALS); |
| $self->register_eval_rule("check_to_in_blacklist", $Mail::SpamAssassin::Conf::TYPE_HEAD_EVALS); #Stub - Remove in SA 4.1 |
| $self->register_eval_rule("check_to_in_welcomelist", $Mail::SpamAssassin::Conf::TYPE_HEAD_EVALS); |
| $self->register_eval_rule("check_to_in_whitelist", $Mail::SpamAssassin::Conf::TYPE_HEAD_EVALS); #Stub - Remove in SA 4.1 |
| $self->register_eval_rule("check_to_in_more_spam", $Mail::SpamAssassin::Conf::TYPE_HEAD_EVALS); |
| $self->register_eval_rule("check_to_in_all_spam", $Mail::SpamAssassin::Conf::TYPE_HEAD_EVALS); |
| $self->register_eval_rule("check_from_in_list", $Mail::SpamAssassin::Conf::TYPE_HEAD_EVALS); |
| $self->register_eval_rule("check_replyto_in_list", $Mail::SpamAssassin::Conf::TYPE_HEAD_EVALS); |
| $self->register_eval_rule("check_to_in_list", $Mail::SpamAssassin::Conf::TYPE_HEAD_EVALS); |
| $self->register_eval_rule("check_from_in_welcomelist", $Mail::SpamAssassin::Conf::TYPE_HEAD_EVALS); |
| $self->register_eval_rule("check_from_in_whitelist", $Mail::SpamAssassin::Conf::TYPE_HEAD_EVALS); #Stub - Remove in SA 4.1 |
| $self->register_eval_rule("check_forged_in_welcomelist", $Mail::SpamAssassin::Conf::TYPE_HEAD_EVALS); |
| $self->register_eval_rule("check_forged_in_whitelist", $Mail::SpamAssassin::Conf::TYPE_HEAD_EVALS); #Stub - Remove in SA 4.1 |
| $self->register_eval_rule("check_from_in_default_welcomelist", $Mail::SpamAssassin::Conf::TYPE_HEAD_EVALS); |
| $self->register_eval_rule("check_from_in_default_whitelist", $Mail::SpamAssassin::Conf::TYPE_HEAD_EVALS); #Stub - Remove in SA 4.1 |
| $self->register_eval_rule("check_forged_in_default_welcomelist", $Mail::SpamAssassin::Conf::TYPE_HEAD_EVALS); |
| $self->register_eval_rule("check_forged_in_default_whitelist", $Mail::SpamAssassin::Conf::TYPE_HEAD_EVALS); #Stub - Remove in SA 4.1 |
| $self->register_eval_rule("check_mailfrom_matches_rcvd", $Mail::SpamAssassin::Conf::TYPE_HEAD_EVALS); |
| $self->register_eval_rule("check_uri_host_listed", $Mail::SpamAssassin::Conf::TYPE_HEAD_EVALS); |
| # same as: eval:check_uri_host_listed('BLOCK') : |
| $self->register_eval_rule("check_uri_host_in_blocklist"); # type does not matter |
| $self->register_eval_rule("check_uri_host_in_blacklist"); # type does not matter #Stub - Remove in SA 4.1 |
| # same as: eval:check_uri_host_listed('WELCOME') : |
| $self->register_eval_rule("check_uri_host_in_welcomelist"); # type does not matter |
| $self->register_eval_rule("check_uri_host_in_whitelist"); # type does not matter #Stub - Remove in SA 4.1 |
| |
| return $self; |
| } |
| |
| sub check_from_in_blocklist { |
| my ($self, $pms) = @_; |
| foreach ($pms->all_from_addrs()) { |
| if ($self->_check_welcomelist ($self->{main}->{conf}->{blocklist_from}, $_)) { |
| return 1; |
| } |
| } |
| return 0; |
| } |
| *check_from_in_blacklist = \&check_from_in_blocklist; # removed in 4.1 |
| |
| sub check_to_in_blocklist { |
| my ($self, $pms) = @_; |
| foreach ($pms->all_to_addrs()) { |
| if ($self->_check_welcomelist ($self->{main}->{conf}->{blocklist_to}, $_)) { |
| return 1; |
| } |
| } |
| return 0; |
| } |
| *check_to_in_blacklist = \&check_to_in_blocklist; # removed in 4.1 |
| |
| sub check_to_in_welcomelist { |
| my ($self, $pms) = @_; |
| foreach ($pms->all_to_addrs()) { |
| if ($self->_check_welcomelist ($self->{main}->{conf}->{welcomelist_to}, $_)) { |
| return 1; |
| } |
| } |
| return 0; |
| } |
| *check_to_in_whitelist = \&check_to_in_welcomelist; # removed in 4.1 |
| |
| sub check_to_in_more_spam { |
| my ($self, $pms) = @_; |
| foreach ($pms->all_to_addrs()) { |
| if ($self->_check_welcomelist ($self->{main}->{conf}->{more_spam_to}, $_)) { |
| return 1; |
| } |
| } |
| return 0; |
| } |
| |
| sub check_to_in_all_spam { |
| my ($self, $pms) = @_; |
| foreach ($pms->all_to_addrs()) { |
| if ($self->_check_welcomelist ($self->{main}->{conf}->{all_spam_to}, $_)) { |
| return 1; |
| } |
| } |
| return 0; |
| } |
| |
| sub check_from_in_list { |
| my ($self, $pms, $list) = @_; |
| my $list_ref = $pms->{conf}->{$list}; |
| unless (defined $list_ref) { |
| warn "eval: could not find list $list"; |
| return 0; |
| } |
| |
| foreach my $addr ($pms->all_from_addrs()) { |
| if ($self->_check_welcomelist ($list_ref, $addr)) { |
| return 1; |
| } |
| } |
| |
| return 0; |
| } |
| |
| sub check_replyto_in_list { |
| my ($self, $pms, $list) = @_; |
| my $list_ref = $pms->{conf}->{$list}; |
| unless (defined $list_ref) { |
| warn "eval: could not find list $list"; |
| return 0; |
| } |
| |
| my $replyto = $pms->get("Reply-To:addr"); |
| return 0 if $replyto eq ''; |
| |
| if ($self->_check_welcomelist ($list_ref, $replyto)) { |
| return 1; |
| } |
| |
| return 0; |
| } |
| |
| # TODO: this should be moved to a utility module off PerMsgStatus, |
| # rather than a plugin API; it's used in Bayes.pm as a utility |
| sub check_wb_list { |
| my ($self, $params) = @_; |
| |
| return unless (defined $params->{permsgstatus}); |
| return unless (defined $params->{type}); |
| return unless (defined $params->{list}); |
| |
| if (lc $params->{type} eq "to") { |
| return $self->check_to_in_list($params->{permsgstatus}, $params->{list}); |
| } |
| elsif (lc $params->{type} eq "from") { |
| return $self->check_from_in_list($params->{permsgstatus}, $params->{list}); |
| } |
| |
| return; |
| } |
| |
| sub check_to_in_list { |
| my ($self,$pms,$list) = @_; |
| my $list_ref = $pms->{conf}->{$list}; |
| unless (defined $list_ref) { |
| warn "eval: could not find list $list"; |
| return 0; |
| } |
| |
| foreach my $addr ($pms->all_to_addrs()) { |
| if ($self->_check_welcomelist ($list_ref, $addr)) { |
| return 1; |
| } |
| } |
| |
| return 0; |
| } |
| |
| ########################################################################### |
| # |
| |
| sub check_from_in_welcomelist { |
| my ($self, $pms) = @_; |
| $self->_check_from_in_welcomelist($pms) unless exists $pms->{from_in_welcomelist}; |
| return ($pms->{from_in_welcomelist} > 0); |
| } |
| *check_from_in_whitelist = \&check_from_in_welcomelist; # removed in 4.1 |
| |
| sub check_forged_in_welcomelist { |
| my ($self, $pms) = @_; |
| $self->_check_from_in_welcomelist($pms) unless exists $pms->{from_in_welcomelist}; |
| $self->_check_from_in_default_welcomelist($pms) unless exists $pms->{from_in_default_welcomelist}; |
| return ($pms->{from_in_welcomelist} < 0) && ($pms->{from_in_default_welcomelist} == 0); |
| } |
| *check_forged_in_whitelist = \&check_forged_in_welcomelist; # removed in 4.1 |
| |
| sub check_from_in_default_welcomelist { |
| my ($self, $pms) = @_; |
| $self->_check_from_in_default_welcomelist($pms) unless exists $pms->{from_in_default_welcomelist}; |
| return ($pms->{from_in_default_welcomelist} > 0); |
| } |
| *check_from_in_default_whitelist = \&check_from_in_default_welcomelist; # removed in 4.1 |
| |
| sub check_forged_in_default_welcomelist { |
| my ($self, $pms) = @_; |
| $self->_check_from_in_default_welcomelist($pms) unless exists $pms->{from_in_default_welcomelist}; |
| $self->_check_from_in_welcomelist($pms) unless exists $pms->{from_in_welcomelist}; |
| return ($pms->{from_in_default_welcomelist} < 0) && ($pms->{from_in_welcomelist} == 0); |
| } |
| *check_forged_in_default_whitelist = \&check_forged_in_default_welcomelist; # removed in 4.1 |
| |
| ########################################################################### |
| |
| sub _check_from_in_welcomelist { |
| my ($self, $pms) = @_; |
| my $found_match = 0; |
| foreach ($pms->all_from_addrs()) { |
| if ($self->_check_welcomelist ($self->{main}->{conf}->{welcomelist_from}, $_)) { |
| $pms->{from_in_welcomelist} = 1; |
| return; |
| } |
| my $wh = $self->_check_welcomelist_rcvd ($pms, $self->{main}->{conf}->{welcomelist_from_rcvd}, $_); |
| if ($wh == 1) { |
| $pms->{from_in_welcomelist} = 1; |
| return; |
| } |
| elsif ($wh == -1) { |
| $found_match = -1; |
| } |
| } |
| |
| $pms->{from_in_welcomelist} = $found_match; |
| return; |
| } |
| |
| ########################################################################### |
| |
| sub _check_from_in_default_welcomelist { |
| my ($self, $pms) = @_; |
| my $found_match = 0; |
| foreach ($pms->all_from_addrs()) { |
| my $wh = $self->_check_welcomelist_rcvd ($pms, $self->{main}->{conf}->{def_welcomelist_from_rcvd}, $_); |
| if ($wh == 1) { |
| $pms->{from_in_default_welcomelist} = 1; |
| return; |
| } |
| elsif ($wh == -1) { |
| $found_match = -1; |
| } |
| } |
| |
| $pms->{from_in_default_welcomelist} = $found_match; |
| return; |
| } |
| |
| ########################################################################### |
| |
| # check if domain name of an envelope sender address matches a domain name |
| # of the first untrusted relay (if any), or any trusted relay otherwise |
| sub check_mailfrom_matches_rcvd { |
| my ($self, $pms) = @_; |
| my $sender = $pms->get("EnvelopeFrom:addr"); |
| return 0 if $sender eq ''; |
| return $self->_check_addr_matches_rcvd($pms,$sender); |
| } |
| |
| # check if domain name of a supplied e-mail address matches a domain name |
| # of the first untrusted relay (if any), or any trusted relay otherwise |
| sub _check_addr_matches_rcvd { |
| my ($self, $pms, $addr) = @_; |
| |
| local $1; |
| return 0 if $addr !~ / \@ ( [^\@]+ \. [^\@]+ ) \z/x; |
| my $addr_domain = lc $1; |
| |
| my @relays; |
| if ($pms->{num_relays_untrusted} > 0) { |
| # check against the first untrusted, if present |
| @relays = $pms->{relays_untrusted}->[0]; |
| } elsif ($pms->{num_relays_trusted} > 0) { |
| # otherwise try all trusted ones, but only do so |
| # if there are no untrusted relays to avoid forgery |
| push(@relays, @{$pms->{relays_trusted}}); |
| } |
| return 0 if !@relays; |
| |
| my($adrh,$adrd) = |
| $self->{main}->{registryboundaries}->split_domain($addr_domain); |
| my $match = 0; |
| my $any_tried = 0; |
| foreach my $rly (@relays) { |
| my $relay_rdns = $rly->{lc_rdns}; |
| next if !defined $relay_rdns || $relay_rdns eq ''; |
| my($rlyh,$rlyd) = |
| $self->{main}->{registryboundaries}->split_domain($relay_rdns); |
| $any_tried = 1; |
| if ($adrd eq $rlyd) { |
| dbg("rules: $addr MATCHES relay $relay_rdns ($adrd)"); |
| $match = 1; last; |
| } |
| } |
| if ($any_tried && !$match) { |
| dbg("rules: %s does NOT match relay(s) %s", |
| $addr, join(', ', map { $_->{lc_rdns} } @relays)); |
| } |
| return $match; |
| } |
| |
| ########################################################################### |
| |
| # look up $addr and trusted relays in a welcomelist with rcvd |
| # note if it appears to be a forgery and $addr is not in any-relay list |
| sub _check_welcomelist_rcvd { |
| my ($self, $pms, $list, $addr) = @_; |
| |
| # we can only match this if we have at least 1 trusted or untrusted header |
| return 0 unless ($pms->{num_relays_untrusted}+$pms->{num_relays_trusted} > 0); |
| |
| my @relays; |
| # try the untrusted one first |
| if ($pms->{num_relays_untrusted} > 0) { |
| @relays = $pms->{relays_untrusted}->[0]; |
| } |
| # then try the trusted ones; the user could have welcomelisted a trusted |
| # relay, totally permitted |
| # but do not do this if any untrusted relays, to avoid forgery -- bug 4425 |
| if ($pms->{num_relays_trusted} > 0 && !$pms->{num_relays_untrusted} ) { |
| push (@relays, @{$pms->{relays_trusted}}); |
| } |
| |
| $addr = lc $addr; |
| my $found_forged = 0; |
| foreach my $welcome_addr (keys %{$list}) { |
| my $regexp = $list->{$welcome_addr}{re}; |
| foreach my $domain (@{$list->{$welcome_addr}{domain}}) { |
| # $domain is a second param in welcomelist_from_rcvd: a domain name or an IP address |
| |
| if ($addr =~ $regexp) { |
| # From or sender address matching the first param in welcomelist_from_rcvd |
| my $match; |
| foreach my $lastunt (@relays) { |
| local($1,$2); |
| if ($domain =~ m{^ \[ (.*) \] ( / \d{1,3} )? \z}sx) { |
| # matching by IP address |
| my($wl_ip, $rly_ip) = ($1, $lastunt->{ip}); |
| $wl_ip .= $2 if defined $2; # allow prefix len even after bracket |
| |
| if (!defined $rly_ip || $rly_ip eq '') { |
| # relay's IP address not provided or unparseable |
| |
| } elsif ($wl_ip =~ /^\d+\.\d+\.\d+\.\d+\z/s) { |
| # an IPv4 welcomelist entry can only be matched by an IPv4 relay |
| if ($wl_ip eq $rly_ip) { $match = 1; last } # exact match |
| |
| } elsif ($wl_ip =~ /^[\d\.]+\z/s) { # an IPv4 classful subnet? |
| $wl_ip =~ s/\.*\z/./; # enforce trailing dot |
| if ($rly_ip =~ /^\Q$wl_ip\E/) { $match = 1; last } # subnet |
| |
| } else { # either an wl entry is an IPv6 addr, or has a prefix len |
| my $rly_ip_obj = NetAddr::IP->new($rly_ip); # TCP-info field |
| if (!defined $rly_ip_obj) { |
| dbg("rules: bad IP address in relay: %s, sender: %s", |
| $rly_ip, $addr); |
| } else { |
| my $wl_ip_obj = NetAddr::IP->new($wl_ip); # welcomelist 2nd param |
| if (!defined $wl_ip_obj) { |
| info("rules: bad IP address in welcomelist: %s", $wl_ip); |
| } elsif ($wl_ip_obj->contains($rly_ip_obj)) { |
| # note: an IPv4-compatible IPv6 address can match an IPv4 addr |
| dbg("rules: relay addr %s matches welcomelist %s, sender: %s", |
| $rly_ip, $wl_ip_obj, $addr); |
| $match = 1; last; |
| } else { |
| dbg("rules: relay addr %s does not match wl %s, sender %s", |
| $rly_ip, $wl_ip_obj, $addr); |
| } |
| } |
| } |
| |
| } else { # match by an rdns name |
| my $rdns = $lastunt->{lc_rdns}; |
| if ($rdns =~ /(?:^|\.)\Q${domain}\E$/i) { $match=1; last } |
| } |
| } |
| if ($match) { |
| dbg("rules: address %s matches (def_)welcomelist_from_rcvd %s %s", |
| $addr, $list->{$welcome_addr}{re}, $domain); |
| return 1; |
| } |
| # found address match but no relay match. note as possible forgery |
| $found_forged = -1; |
| } |
| } |
| } |
| if ($found_forged) { # might be forgery. check if in list of exempted |
| my $wlist = $pms->{conf}->{welcomelist_allows_relays}; |
| foreach my $regexp (values %{$wlist}) { |
| if ($addr =~ $regexp) { |
| $found_forged = 0; |
| last; |
| } |
| } |
| } |
| return $found_forged; |
| } |
| |
| ########################################################################### |
| |
| sub _check_welcomelist { |
| my ($self, $list, $addr) = @_; |
| $addr = lc $addr; |
| if (defined ($list->{$addr})) { return 1; } |
| foreach my $regexp (values %{$list}) { |
| if ($addr =~ $regexp) { |
| dbg("rules: address $addr matches welcomelist or blocklist regexp: $regexp"); |
| return 1; |
| } |
| } |
| |
| return 0; |
| } |
| |
| ########################################################################### |
| |
| sub check_uri_host_in_blocklist { |
| my ($self, $pms) = @_; |
| $self->check_uri_host_listed($pms, 'BLOCK'); |
| } |
| *check_uri_host_in_blacklist = \&check_uri_host_in_blocklist; # removed in 4.1 |
| |
| sub check_uri_host_in_welcomelist { |
| my ($self, $pms) = @_; |
| $self->check_uri_host_listed($pms, 'WELCOME'); |
| } |
| *check_uri_host_in_whitelist = \&check_uri_host_in_welcomelist; # removed in 4.1 |
| |
| sub check_uri_host_listed { |
| my ($self, $pms, $subname) = @_; |
| my $host_enlisted_ref = $self->_check_uri_host_listed($pms); |
| if ($host_enlisted_ref) { |
| my $matched_host = $host_enlisted_ref->{$subname}; |
| if ($matched_host) { |
| dbg("rules: uri host enlisted (%s): %s", $subname, $matched_host); |
| $pms->test_log("URI: $matched_host"); |
| return 1; |
| } |
| } |
| return 0; |
| } |
| |
| sub _check_uri_host_listed { |
| my ($self, $pms) = @_; |
| |
| if ($pms->{'uri_host_enlisted'}) { |
| return $pms->{'uri_host_enlisted'}; # just provide a cached result |
| } |
| |
| my $uri_lists_href = $pms->{conf}->{uri_host_lists}; |
| if (!$uri_lists_href || !%$uri_lists_href) { |
| $pms->{'uri_host_enlisted'} = {}; # no URI host lists |
| return $pms->{'uri_host_enlisted'}; |
| } |
| |
| my %host_enlisted; |
| my @uri_listnames = sort keys %$uri_lists_href; |
| if (would_log("dbg","rules")) { |
| foreach my $nm (@uri_listnames) { |
| dbg("rules: check_uri_host_listed: (%s) %s", |
| $nm, join(', ', map { $uri_lists_href->{$nm}{$_} ? $_ : '!'.$_ } |
| sort keys %{$uri_lists_href->{$nm}})); |
| } |
| } |
| # obtain a complete list of html-parsed domains |
| my $uris = $pms->get_uri_detail_list(); |
| my %seen; |
| while (my($uri,$info) = each %$uris) { |
| next if $uri =~ /^mailto:/i; # we may want to skip mailto: uris (?) |
| while (my($host,$domain) = each( %{$info->{hosts}} )) { # typically one |
| next if $seen{$host}; |
| $seen{$host} = 1; |
| local($1,$2); |
| my @query_keys; |
| if ($host =~ /^\[(.*)\]\z/) { # looks like an address literal |
| @query_keys = ( $1 ); |
| } elsif ($host =~ /^\d+\.\d+\.\d+\.\d+\z/) { # IPv4 address |
| @query_keys = ( $host ); |
| } elsif ($host ne '') { |
| my($h) = $host; |
| for (;;) { |
| shift @query_keys if @query_keys > 10; # sanity limit, keep tail |
| push(@query_keys, $h); # sub.example.com, example.com, com |
| last if $h !~ s{^([^.]*)\.(.*)\z}{$2}s; |
| } |
| } |
| foreach my $nm (@uri_listnames) { |
| my $match; |
| my $verdict; |
| my $hash_nm_ref = $uri_lists_href->{$nm}; |
| foreach my $q (@query_keys) { |
| $verdict = $hash_nm_ref->{$q}; |
| if (defined $verdict) { |
| $match = $q eq $host ? $host : "$host ($q)"; |
| $match = '!' if !$verdict; |
| last; |
| } |
| } |
| if (defined $verdict) { |
| $host_enlisted{$nm} = $match if $verdict; |
| dbg("rules: check_uri_host_listed %s, (%s): %s, search: %s", |
| $uri, $nm, $match, join(', ',@query_keys)); |
| } |
| } |
| } |
| } |
| $pms->{'uri_host_enlisted'} = \%host_enlisted; |
| return $pms->{'uri_host_enlisted'}; |
| } |
| |
| 1; |