use strict;
use Apache;
use Apache::Constants qw(REMOTE_HOST);
use vars qw($VERSION $gip);
my $GEOIP_DBFILE;
@Apache::Geo::IP::ISA = qw(Apache);
$VERSION = '1.215';
use constant GEOIP_STANDARD => 0;
use constant GEOIP_MEMORY_CACHE => 1;
use constant GEOIP_CHECK_CACHE => 2;
sub new {
my ($class, $r) = @_;
init($r) unless ($gip);
return bless { r => $r }, $class;
}
sub init {
my $r = shift;
my $file = $r->dir_config('GeoIPDBFile') || $GEOIP_DBFILE;
if ($file) {
unless (-e $file) {
$r->warn("Cannot find GeoIP database file '$file'");
die;
}
}
else {
$r->warn("Must specify GeoIP database file");
die;
}
my $flag = $r->dir_config('GeoIPFlag') || '';
if ($flag) {
unless ($flag =~ /^(STANDARD|MEMORY_CACHE|CHECK_CACHE)$/i) {
$r->warn("GeoIP flag '$flag' not understood");
die;
}
}
$flag = uc($flag);
FLAG: {
($flag && $flag eq 'MEMORY_CACHE') && do {
$flag = GEOIP_MEMORY_CACHE;
last FLAG;
};
($flag && $flag eq 'CHECK_CACHE') && do {
$flag = GEOIP_CHECK_CACHE;
last FLAG;
};
$flag = GEOIP_STANDARD;
}
unless ($gip = Apache::GeoIP->open($file, $flag)) {
$r->warn("Couldn't make GeoIP object");
die;
}
}
sub country_code_by_addr {
my $self = shift;
my $ip = shift || $self->connection->remote_ip;
return $gip->_country_code_by_addr($ip);
}
sub country_code_by_name {
my $self = shift;
my $host = shift || $self->get_remote_host(REMOTE_HOST);
return $gip->_country_code_by_name($host);
}
sub country_code3_by_addr {
my $self = shift;
my $ip = shift || $self->connection->remote_ip;
return $gip->_country_code3_by_addr($ip);
}
sub country_code3_by_name {
my $self = shift;
my $host = shift || $self->get_remote_host(REMOTE_HOST);
return $gip->_country_code3_by_name($host);
}
sub country_name_by_addr {
my $self = shift;
my $ip = shift || $self->connection->remote_ip;
return $gip->_country_name_by_addr($ip);
}
sub country_name_by_name {
my $self = shift;
my $host = shift || $self->get_remote_host(REMOTE_HOST);
return $gip->_country_name_by_name($host);
}
1;
__END__
=head1 NAME
Apache::Geo::IP - Look up country by IP Address
=head1 SYNOPSIS
# in httpd.conf
# PerlModule Apache::HelloIP
#<Location /ip>
# SetHandler perl-script
# PerlHandler Apache::HelloIP
# PerlSetVar GeoIPDBFile "/usr/local/share/GeoIP/GeoIP.dat"
# PerlSetVar GeoIPFlag Standard
#</Location>
# file Apache::HelloIP
use Apache::Geo::IP;
use strict;
use Apache::Constants qw(OK);
sub handler {
my $r = Apache::Geo::IP->new(shift);
$r->content_type('text/plain');
my $country = uc($r->country_code_by_addr());
$r->print($country);
return OK;
}
1;
=head1 DESCRIPTION
This module constitutes a mod_perl (version 1) interface to the
I<Geo::IP> module, which looks up in a database a country of origin of
an IP address. This database simply contains
IP blocks as keys, and countries as values. This database should be more
complete and accurate than reverse DNS lookups.
This module can be used to automatically select the geographically
closest mirror, to analyze your web server logs
to determine the countries of your visiters, for credit card fraud
detection, and for software export controls.
If you are on Win32 and have installed this package using the
ActivePerl ppm utility, the database F<GeoIP.dat> is expected
to be found under the F<C:\Program Files\GeoIP\> directory.
To find a country for an IP address, this module a finds the Network
that contains the IP address, then returns the country the Network is
assigned to.
=head1 CONFIGURATION
This module subclasses I<Apache>, and can be used as follows
in an Apache module.
# file Apache::HelloIP
use Apache::Geo::IP;
use strict;
sub handler {
my $r = Apache::Geo::IP->new(shift);
# continue along
}
The directives in F<httpd.conf> are as follows:
PerlModule Apache::HelloIP
<Location /ip>
PerlSetVar GeoIPDBFile "/usr/local/share/GeoIP/GeoIP.dat"
PerlSetVar GeoIPFlag Standard
# other directives
</Location>
The C<PerlSetVar> directives available are
=over 4
=item PerlSetVar GeoIPDBFile "/path/to/GeoIP.dat"
This specifies the location of the F<GeoIP.dat> file.
If not given, it defaults to the location specified
upon installing the module.
=item PerlSetVar GeoIPFlag Standard
This can be set to I<STANDARD>, or for faster performance
but at a cost of using more memory, I<MEMORY_CACHE>.
When using memory
cache you can force a reload if the file is updated by
using I<CHECK_CACHE>.
If not specified, I<STANDARD> is used.
=back
=head1 METHODS
The available methods are as follows.
=over 4
=item $code = $r->country_code_by_addr( [$ipaddr] );
Returns the ISO 3166 country code for an IP address.
If I<$ipaddr> is not given, the value obtained by
C<$r-E<gt>connection-E<gt>remote_ip> is used.
=item $code = $r->country_code_by_name( [$ipname] );
Returns the ISO 3166 country code for a hostname.
If I<$ipname> is not given, the value obtained by
C<$r-E<gt>get_remote_host(REMOTE_HOST)> is used.
=item $code = $r->country_code3_by_addr( [$ipaddr] );
Returns the 3 letter country code for an IP address.
If I<$ipaddr> is not given, the value obtained by
C<$r-E<gt>connection-E<gt>remote_ip> is used.
=item $code = $r->country_code3_by_name( [$ipname] );
Returns the 3 letter country code for a hostname.
If I<$ipname> is not given, the value obtained by
C<$r-E<gt>get_remote_host(REMOTE_HOST)> is used.
=item $name = $r->country_name_by_addr( [$ipaddr] );
Returns the full country name for an IP address.
If I<$ipaddr> is not given, the value obtained by
C<$r-E<gt>connection-E<gt>remote_ip> is used.
=item $name = $r->country_name_by_name( [$ipname] );
Returns the full country name for a hostname.
If I<$ipname> is not given, the value obtained by
C<$r-E<gt>get_remote_host(REMOTE_HOST)> is used.
=back
=head1 VERSION
1.11
=head1 SEE ALSO
L<Geo::IP> and L<Apache>.
=head1 AUTHOR
The look-up code for associating a country with an IP address
is based on the GeoIP library and the Geo::IP Perl module, and is
Copyright (c) 2002, T.J. Mather, tjmather@tjmather.com, New York, NY,
USA. See http://www.maxmind.com/ for details. The mod_perl interface is
Copyright (c) 2002, Randy Kobes <randy@theoryx5.uwinnipeg.ca>.
All rights reserved. This package is free software; you can redistribute it
and/or modify it under the same terms as Perl itself.
=cut