2015-07-29 16:21:03 -05:00
|
|
|
import os
|
2019-08-10 11:55:22 +02:00
|
|
|
import pathlib
|
2017-01-19 12:16:04 -05:00
|
|
|
from unittest import mock, skipUnless
|
2015-07-29 16:21:03 -05:00
|
|
|
|
|
|
|
from django.conf import settings
|
|
|
|
from django.contrib.gis.geoip2 import HAS_GEOIP2
|
2017-05-04 11:03:23 -04:00
|
|
|
from django.contrib.gis.geos import GEOSGeometry
|
2018-03-03 23:43:26 +02:00
|
|
|
from django.test import SimpleTestCase
|
2023-11-30 09:51:42 +00:00
|
|
|
from django.utils.deprecation import RemovedInDjango60Warning
|
2015-07-29 16:21:03 -05:00
|
|
|
|
|
|
|
if HAS_GEOIP2:
|
|
|
|
from django.contrib.gis.geoip2 import GeoIP2, GeoIP2Exception
|
|
|
|
|
|
|
|
|
|
|
|
# Note: Requires both the GeoIP country and city datasets.
|
|
|
|
# The GEOIP_DATA path should be the only setting set (the directory
|
|
|
|
# should contain links or the actual database files 'GeoLite2-City.mmdb' and
|
|
|
|
# 'GeoLite2-City.mmdb'.
|
2016-04-07 22:04:45 -04:00
|
|
|
@skipUnless(
|
|
|
|
HAS_GEOIP2 and getattr(settings, "GEOIP_PATH", None),
|
|
|
|
"GeoIP is required along with the GEOIP_PATH setting.",
|
|
|
|
)
|
2018-03-03 23:43:26 +02:00
|
|
|
class GeoIPTest(SimpleTestCase):
|
2021-01-29 11:00:12 +01:00
|
|
|
addr = "129.237.192.1"
|
|
|
|
fqdn = "ku.edu"
|
2015-07-29 16:21:03 -05:00
|
|
|
|
|
|
|
def test01_init(self):
|
|
|
|
"GeoIP initialization."
|
|
|
|
g1 = GeoIP2() # Everything inferred from GeoIP path
|
|
|
|
path = settings.GEOIP_PATH
|
|
|
|
g2 = GeoIP2(path, 0) # Passing in data path explicitly.
|
|
|
|
g3 = GeoIP2.open(path, 0) # MaxMind Python API syntax.
|
2019-08-10 11:55:22 +02:00
|
|
|
# path accepts str and pathlib.Path.
|
|
|
|
if isinstance(path, str):
|
|
|
|
g4 = GeoIP2(pathlib.Path(path))
|
|
|
|
else:
|
|
|
|
g4 = GeoIP2(str(path))
|
2015-07-29 16:21:03 -05:00
|
|
|
|
2019-08-10 11:55:22 +02:00
|
|
|
for g in (g1, g2, g3, g4):
|
2015-07-29 16:21:03 -05:00
|
|
|
self.assertTrue(g._country)
|
|
|
|
self.assertTrue(g._city)
|
|
|
|
|
|
|
|
# Only passing in the location of one database.
|
|
|
|
city = os.path.join(path, "GeoLite2-City.mmdb")
|
|
|
|
cntry = os.path.join(path, "GeoLite2-Country.mmdb")
|
|
|
|
g4 = GeoIP2(city, country="")
|
|
|
|
self.assertIsNone(g4._country)
|
|
|
|
g5 = GeoIP2(cntry, city="")
|
|
|
|
self.assertIsNone(g5._city)
|
|
|
|
|
|
|
|
# Improper parameters.
|
|
|
|
bad_params = (23, "foo", 15.23)
|
|
|
|
for bad in bad_params:
|
2016-01-17 14:56:39 +03:30
|
|
|
with self.assertRaises(GeoIP2Exception):
|
|
|
|
GeoIP2(cache=bad)
|
2016-12-29 16:27:49 +01:00
|
|
|
if isinstance(bad, str):
|
2015-07-29 16:21:03 -05:00
|
|
|
e = GeoIP2Exception
|
|
|
|
else:
|
|
|
|
e = TypeError
|
2016-01-17 14:56:39 +03:30
|
|
|
with self.assertRaises(e):
|
|
|
|
GeoIP2(bad, 0)
|
2015-07-29 16:21:03 -05:00
|
|
|
|
2018-03-03 23:43:26 +02:00
|
|
|
def test_no_database_file(self):
|
|
|
|
invalid_path = os.path.join(os.path.dirname(__file__), "data")
|
|
|
|
msg = "Could not load a database from %s." % invalid_path
|
|
|
|
with self.assertRaisesMessage(GeoIP2Exception, msg):
|
|
|
|
GeoIP2(invalid_path)
|
|
|
|
|
2015-07-29 16:21:03 -05:00
|
|
|
def test02_bad_query(self):
|
|
|
|
"GeoIP query parameter checking."
|
|
|
|
cntry_g = GeoIP2(city="<foo>")
|
|
|
|
# No city database available, these calls should fail.
|
2016-01-17 14:56:39 +03:30
|
|
|
with self.assertRaises(GeoIP2Exception):
|
|
|
|
cntry_g.city("tmc.edu")
|
2015-07-29 16:21:03 -05:00
|
|
|
|
|
|
|
# Non-string query should raise TypeError
|
2016-01-17 14:56:39 +03:30
|
|
|
with self.assertRaises(TypeError):
|
|
|
|
cntry_g.country_code(17)
|
|
|
|
with self.assertRaises(TypeError):
|
|
|
|
cntry_g.country_name(GeoIP2)
|
2015-07-29 16:21:03 -05:00
|
|
|
|
2015-09-20 22:23:28 +03:00
|
|
|
@mock.patch("socket.gethostbyname")
|
|
|
|
def test03_country(self, gethostbyname):
|
2015-07-29 16:21:03 -05:00
|
|
|
"GeoIP country querying methods."
|
2015-09-20 22:23:28 +03:00
|
|
|
gethostbyname.return_value = "128.249.1.1"
|
2015-07-29 16:21:03 -05:00
|
|
|
g = GeoIP2(city="<foo>")
|
|
|
|
|
|
|
|
for query in (self.fqdn, self.addr):
|
|
|
|
self.assertEqual(
|
|
|
|
"US",
|
|
|
|
g.country_code(query),
|
|
|
|
"Failed for func country_code and query %s" % query,
|
|
|
|
)
|
|
|
|
self.assertEqual(
|
|
|
|
"United States",
|
|
|
|
g.country_name(query),
|
|
|
|
"Failed for func country_name and query %s" % query,
|
|
|
|
)
|
|
|
|
self.assertEqual(
|
|
|
|
{"country_code": "US", "country_name": "United States"},
|
|
|
|
g.country(query),
|
|
|
|
)
|
|
|
|
|
2015-09-20 22:23:28 +03:00
|
|
|
@mock.patch("socket.gethostbyname")
|
|
|
|
def test04_city(self, gethostbyname):
|
2015-07-29 16:21:03 -05:00
|
|
|
"GeoIP city querying methods."
|
2021-01-29 11:00:12 +01:00
|
|
|
gethostbyname.return_value = "129.237.192.1"
|
2015-07-29 16:21:03 -05:00
|
|
|
g = GeoIP2(country="<foo>")
|
|
|
|
|
|
|
|
for query in (self.fqdn, self.addr):
|
|
|
|
# Country queries should still work.
|
|
|
|
self.assertEqual(
|
|
|
|
"US",
|
|
|
|
g.country_code(query),
|
|
|
|
"Failed for func country_code and query %s" % query,
|
|
|
|
)
|
|
|
|
self.assertEqual(
|
|
|
|
"United States",
|
|
|
|
g.country_name(query),
|
|
|
|
"Failed for func country_name and query %s" % query,
|
|
|
|
)
|
|
|
|
self.assertEqual(
|
|
|
|
{"country_code": "US", "country_name": "United States"},
|
|
|
|
g.country(query),
|
|
|
|
)
|
|
|
|
|
|
|
|
# City information dictionary.
|
|
|
|
d = g.city(query)
|
2017-10-25 19:30:42 +11:00
|
|
|
self.assertEqual("NA", d["continent_code"])
|
|
|
|
self.assertEqual("North America", d["continent_name"])
|
2015-07-29 16:21:03 -05:00
|
|
|
self.assertEqual("US", d["country_code"])
|
2021-01-29 11:00:12 +01:00
|
|
|
self.assertEqual("Lawrence", d["city"])
|
|
|
|
self.assertEqual("KS", d["region"])
|
2017-05-23 03:07:14 +02:00
|
|
|
self.assertEqual("America/Chicago", d["time_zone"])
|
2019-05-07 21:31:44 +10:00
|
|
|
self.assertFalse(d["is_in_european_union"])
|
2015-07-29 16:21:03 -05:00
|
|
|
geom = g.geos(query)
|
|
|
|
self.assertIsInstance(geom, GEOSGeometry)
|
|
|
|
|
2018-10-24 01:46:49 +01:00
|
|
|
for e1, e2 in (
|
|
|
|
geom.tuple,
|
|
|
|
g.lon_lat(query),
|
|
|
|
g.lat_lon(query),
|
|
|
|
):
|
|
|
|
self.assertIsInstance(e1, float)
|
|
|
|
self.assertIsInstance(e2, float)
|
2015-07-29 16:21:03 -05:00
|
|
|
|
|
|
|
def test06_ipv6_query(self):
|
|
|
|
"GeoIP can lookup IPv6 addresses."
|
|
|
|
g = GeoIP2()
|
|
|
|
d = g.city("2002:81ed:c9a5::81ed:c9a5") # IPv6 address for www.nhm.ku.edu
|
|
|
|
self.assertEqual("US", d["country_code"])
|
|
|
|
self.assertEqual("Lawrence", d["city"])
|
|
|
|
self.assertEqual("KS", d["region"])
|
2015-07-30 14:16:46 +01:00
|
|
|
|
|
|
|
def test_repr(self):
|
|
|
|
path = settings.GEOIP_PATH
|
|
|
|
g = GeoIP2(path=path)
|
|
|
|
meta = g._reader.metadata()
|
|
|
|
version = "%s.%s" % (
|
|
|
|
meta.binary_format_major_version,
|
|
|
|
meta.binary_format_minor_version,
|
|
|
|
)
|
|
|
|
country_path = g._country_file
|
|
|
|
city_path = g._city_file
|
|
|
|
expected = (
|
|
|
|
'<GeoIP2 [v%(version)s] _country_file="%(country)s", _city_file="%(city)s">'
|
|
|
|
% {
|
|
|
|
"version": version,
|
|
|
|
"country": country_path,
|
|
|
|
"city": city_path,
|
|
|
|
}
|
2022-02-03 20:24:19 +01:00
|
|
|
)
|
2015-07-30 14:16:46 +01:00
|
|
|
self.assertEqual(repr(g), expected)
|
2017-04-22 16:44:24 +02:00
|
|
|
|
|
|
|
@mock.patch("socket.gethostbyname", return_value="expected")
|
|
|
|
def test_check_query(self, gethostbyname):
|
|
|
|
g = GeoIP2()
|
|
|
|
self.assertEqual(g._check_query("127.0.0.1"), "127.0.0.1")
|
|
|
|
self.assertEqual(
|
|
|
|
g._check_query("2002:81ed:c9a5::81ed:c9a5"), "2002:81ed:c9a5::81ed:c9a5"
|
|
|
|
)
|
|
|
|
self.assertEqual(g._check_query("invalid-ip-address"), "expected")
|
2023-11-30 09:51:42 +00:00
|
|
|
|
|
|
|
def test_coords_deprecation_warning(self):
|
|
|
|
g = GeoIP2()
|
|
|
|
msg = "GeoIP2.coords() is deprecated. Use GeoIP2.lon_lat() instead."
|
|
|
|
with self.assertWarnsMessage(RemovedInDjango60Warning, msg):
|
|
|
|
e1, e2 = g.coords(self.fqdn)
|
|
|
|
self.assertIsInstance(e1, float)
|
|
|
|
self.assertIsInstance(e2, float)
|