geoip: simplify the get/set data bits

This commit is contained in:
Matt Simerson 2014-11-04 14:43:54 -08:00
parent 1abfd152f2
commit 9862cdc042
2 changed files with 51 additions and 33 deletions

View File

@ -6,8 +6,8 @@ geoip - provide geographic information about mail senders.
=head1 SYNOPSIS =head1 SYNOPSIS
Use MaxMind's GeoIP databases and the Geo::IP perl module to report geographic Use MaxMind's GeoIP databases and the GeoIP2 or Geo::IP perl modules to report
information about incoming connections. geographic information about incoming connections.
=head1 DESCRIPTION =head1 DESCRIPTION
@ -110,9 +110,11 @@ http://smartech.gatech.edu/bitstream/handle/1853/25135/GT-CSE-08-02.pdf
use strict; use strict;
use warnings; use warnings;
use lib 'lib';
use Qpsmtpd::Constants; use Qpsmtpd::Constants;
#use Geo::IP; # eval'ed in register() #use GeoIP2; # eval'ed in register()
#use Geo::IP; # eval loaded if GeoIP2 doesn't
#use Math::Trig; # eval'ed in set_distance_gc #use Math::Trig; # eval'ed in set_distance_gc
sub register { sub register {
@ -122,12 +124,20 @@ sub register {
$self->{_args} = {@args}; $self->{_args} = {@args};
$self->{_args}{db_dir} ||= '/usr/local/share/GeoIP'; $self->{_args}{db_dir} ||= '/usr/local/share/GeoIP';
my $loaded = 0;
eval 'use GeoIP2';
if ($@) {
warn "could not load GeoIP2";
$self->log(LOGERROR, "could not load GeoIP2");
eval 'use Geo::IP'; eval 'use Geo::IP';
if ($@) { if ($@) {
warn "could not load Geo::IP"; warn "could not load Geo::IP";
$self->log(LOGERROR, "could not load Geo::IP"); $self->log(LOGERROR, "could not load Geo::IP");
return; return;
} }
}
# Note that opening the GeoIP DB only in register has caused problems before: # Note that opening the GeoIP DB only in register has caused problems before:
# https://github.com/smtpd/qpsmtpd/commit/29ea9516806e9a8ca6519fcf987dbd684793ebdd#plugins/ident/geoip # https://github.com/smtpd/qpsmtpd/commit/29ea9516806e9a8ca6519fcf987dbd684793ebdd#plugins/ident/geoip
@ -137,10 +147,10 @@ sub register {
$self->init_my_country_code(); $self->init_my_country_code();
$self->register_hook('connect', 'connect_handler'); $self->register_hook('connect', 'geoip_lookup');
} }
sub connect_handler { sub geoip_lookup {
my $self = shift; my $self = shift;
# reopen the DB if Geo::IP failed due to DB update # reopen the DB if Geo::IP failed due to DB update
@ -150,7 +160,6 @@ sub connect_handler {
$self->log(LOGINFO, "skip, no results"); $self->log(LOGINFO, "skip, no results");
return DECLINED; return DECLINED;
}; };
$self->qp->connection->notes('geoip_country', $c_code);
my $c_name = $self->set_country_name(); my $c_name = $self->set_country_name();
my ($city, $continent_code, $distance) = ''; my ($city, $continent_code, $distance) = '';
@ -162,8 +171,9 @@ sub connect_handler {
} }
my @msg_parts; my @msg_parts;
push @msg_parts, $continent_code if ($continent_code && $continent_code ne '--') {
if $continent_code && $continent_code ne '--'; push @msg_parts, $continent_code;
};
push @msg_parts, $c_code if $c_code; push @msg_parts, $c_code if $c_code;
#push @msg_parts, $c_name if $c_name; #push @msg_parts, $c_name if $c_name;
@ -211,9 +221,13 @@ sub init_my_country_code {
sub set_country_code { sub set_country_code {
my $self = shift; my $self = shift;
return $self->get_country_code_gc() if $self->{_geoip_city};
my $remote_ip = $self->qp->connection->remote_ip; my $remote_ip = $self->qp->connection->remote_ip;
my $code = $self->get_country_code();
my $code = $self->{_geoip_city}
? $self->get_country_code_gc($remote_ip)
: $self->get_country_code($remote_ip);
return if ! $code;
$self->qp->connection->notes('geoip_country', $code); $self->qp->connection->notes('geoip_country', $code);
return $code; return $code;
} }
@ -221,7 +235,9 @@ sub set_country_code {
sub get_country_code { sub get_country_code {
my $self = shift; my $self = shift;
my $ip = shift || $self->qp->connection->remote_ip; my $ip = shift || $self->qp->connection->remote_ip;
return $self->get_country_code_gc($ip) if $self->{_geoip_city}; if ($self->{_geoip_city}) {
return $self->get_country_code_gc($ip);
};
return $self->{_geoip}->country_code_by_addr($ip); return $self->{_geoip}->country_code_by_addr($ip);
} }
@ -235,44 +251,45 @@ sub get_country_code_gc {
sub set_country_name { sub set_country_name {
my $self = shift; my $self = shift;
return $self->set_country_name_gc() if $self->{_geoip_city};
my $remote_ip = $self->qp->connection->remote_ip; my $remote_ip = $self->qp->connection->remote_ip;
my $name = $self->{_geoip}->country_name_by_addr($remote_ip) or return;
my $name = $self->{_geoip_city}
? $self->get_country_name_gc($remote_ip)
: $self->{_geoip}->country_name_by_addr($remote_ip);
return if ! $name;
$self->qp->connection->notes('geoip_country_name', $name); $self->qp->connection->notes('geoip_country_name', $name);
return $name; return $name;
} }
sub set_country_name_gc { sub get_country_name_gc {
my $self = shift; my $self = shift;
return if !$self->{_geoip_record}; return if !$self->{_geoip_record};
my $remote_ip = $self->qp->connection->remote_ip; return $self->{_geoip_record}->country_name();
my $name = $self->{_geoip_record}->country_name() or return;
$self->qp->connection->notes('geoip_country_name', $name);
return $name;
} }
sub set_continent { sub set_continent {
my $self = shift; my ($self, $country_code) = @_;
return $self->set_continent_gc() if $self->{_geoip_city}; $country_code or return;
my $c_code = shift or return;
my $continent = $self->{_geoip}->continent_code_by_country_code($c_code) my $continent = $self->{_geoip_city}
or return; ? $self->get_continent_gc()
: $self->{_geoip}->continent_code_by_country_code($country_code);
$continent or return;
$self->qp->connection->notes('geoip_continent', $continent); $self->qp->connection->notes('geoip_continent', $continent);
return $continent; return $continent;
} }
sub set_continent_gc { sub get_continent_gc {
my $self = shift; my $self = shift;
return if !$self->{_geoip_record}; return if !$self->{_geoip_record};
my $continent = $self->{_geoip_record}->continent_code() or return; return $self->{_geoip_record}->continent_code();
$self->qp->connection->notes('geoip_continent', $continent);
return $continent;
} }
sub set_city_gc { sub set_city_gc {
my $self = shift; my $self = shift;
return if !$self->{_geoip_record}; return if !$self->{_geoip_record};
my $remote_ip = $self->qp->connection->remote_ip;
my $city = $self->{_geoip_record}->city() or return; my $city = $self->{_geoip_record}->city() or return;
$self->qp->connection->notes('geoip_city', $city); $self->qp->connection->notes('geoip_city', $city);
return $city; return $city;

View File

@ -3,6 +3,7 @@
use strict; use strict;
use warnings; use warnings;
use lib 'lib';
use Qpsmtpd::Constants; use Qpsmtpd::Constants;
sub register_tests { sub register_tests {
@ -27,7 +28,7 @@ sub test_geoip_lookup {
my $self = shift; my $self = shift;
$self->qp->connection->remote_ip('24.24.24.24'); $self->qp->connection->remote_ip('24.24.24.24');
cmp_ok( $self->connect_handler(), '==', DECLINED, "exit code"); cmp_ok( $self->geoip_lookup(), '==', DECLINED, "exit code");
cmp_ok( $self->connection->notes('geoip_country'), 'eq', 'US', "note"); cmp_ok( $self->connection->notes('geoip_country'), 'eq', 'US', "note");
}; };