MaxMind 从 IP 地址获取 Lat/Lng,InvalidDatabaseException 错误

Posted

技术标签:

【中文标题】MaxMind 从 IP 地址获取 Lat/Lng,InvalidDatabaseException 错误【英文标题】:MaxMind Get Lat/Lng from IP address, InvalidDatabaseException Error 【发布时间】:2015-03-03 20:39:14 【问题描述】:

我的最终目标是获取与提供的 IP 地址相关的纬度、经度和国家/地区。我正在使用 java 和 MaxMind GeoIP2 Java API。我的偏好是使用他们的 webapi,因为我需要快速周转时间。

谁能告诉我为什么在调用与 MaxMind api 调用相关的函数时出现此错误(错误后显示的代码)?

我不确定 MaxMind DB 元数据标记是什么。它抱怨我从 MaxMind 网站上抓取的文件。我会假设它是一个有效的 MaxMind DB 文件。

com.maxmind.db.InvalidDatabaseException: Could not find a MaxMind DB metadata marker in this file (GeoLite2-City-Blocks-IPv6.csv). Is this a valid MaxMind DB file?
at com.maxmind.db.Reader.findMetadataStart(Reader.java:231)
at com.maxmind.db.Reader.<init>(Reader.java:82)
at com.maxmind.db.Reader.<init>(Reader.java:74)
at com.maxmind.geoip2.DatabaseReader.<init>(DatabaseReader.java:41)
at com.maxmind.geoip2.DatabaseReader.<init>(DatabaseReader.java:31)
at com.maxmind.geoip2.DatabaseReader$Builder.build(DatabaseReader.java:126)
at com.tutelatechnologies.tapfortap.server.maxmind.MaxMindLookUp.maxMindLookup(MaxMindLookUp.java:20)
at com.tutelatechnologies.tapfortap.server.cidlaclookup.T4TUtilitiesTest.testMaxMindLookUpTest(T4TUtilitiesTest.java:23)
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:57)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
at java.lang.reflect.Method.invoke(Method.java:606)
at org.junit.runners.model.FrameworkMethod$1.runReflectiveCall(FrameworkMethod.java:47)
at org.junit.internal.runners.model.ReflectiveCallable.run(ReflectiveCallable.java:12)
at org.junit.runners.model.FrameworkMethod.invokeExplosively(FrameworkMethod.java:44)
at org.junit.internal.runners.statements.InvokeMethod.evaluate(InvokeMethod.java:17)
at org.junit.runners.ParentRunner.runLeaf(ParentRunner.java:271)
at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:70)
at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:50)
at org.junit.runners.ParentRunner$3.run(ParentRunner.java:238)
at org.junit.runners.ParentRunner$1.schedule(ParentRunner.java:63)
at org.junit.runners.ParentRunner.runChildren(ParentRunner.java:236)
at org.junit.runners.ParentRunner.access$000(ParentRunner.java:53)
at org.junit.runners.ParentRunner$2.evaluate(ParentRunner.java:229)
at org.junit.runners.ParentRunner.run(ParentRunner.java:309)
at org.eclipse.jdt.internal.junit4.runner.JUnit4TestReference.run(JUnit4TestReference.java:50)
at org.eclipse.jdt.internal.junit.runner.TestExecution.run(TestExecution.java:38)
at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.runTests(RemoteTestRunner.java:459)
at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.runTests(RemoteTestRunner.java:675)
at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.run(RemoteTestRunner.java:382)
at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.main(RemoteTestRunner.java:192)

我按照 MaxMind 网站上的说明和示例代码进行了数据使用部分:http://maxmind.github.io/GeoIP2-java/

我选择使用从这里获得的 GeoLite2 csv dbs:http://dev.maxmind.com/geoip/geoip2/geolite2/

我的代码:

public class MaxMindLookUp 

public static final void maxMindLookup(final String ipaddress) 
    File db = new File(
            "src/main/java/com/tutelatechnologies/tapfortap/server/maxmind/GeoLite2-City-Blocks-IPv6.csv");
    try 
        DatabaseReader reader = new DatabaseReader.Builder(db).build();
        InetAddress ip = InetAddress.getByName(ipaddress);

        CityResponse response = reader.city(ip);
        Location loc = response.getLocation();

        Double lat = loc.getLatitude();
        Double lng = loc.getLongitude();
        Integer acc = loc.getAccuracyRadius();

        System.out.println("lat=" + lat + "\nlng=" + lng + "\nacc=" + acc);

     catch (GeoIp2Exception e) 
        // TODO Auto-generated catch block
        e.printStackTrace();

     catch (IOException e) 
        // TODO Auto-generated catch block
        e.printStackTrace();
    


注意:为了运行上面的代码,您需要在 Maven 依赖项部分添加一个插件。

 <dependency>
    <groupId>com.maxmind.geoip2</groupId>
    <artifactId>geoip2</artifactId>
    <version>2.1.0</version>
</dependency>

我没有遇到太多关于 SO 的确切错误。我已经通过调试器,错误发生在这一行:

DatabaseReader reader = new DatabaseReader.Builder(db).build();

这让我相信我从网站上下载的数据库有问题。

此页面底部的异常标题 (http://maxmind.github.io/GeoIP2-java/) 提供了一个死链接“GeoIP2 Precision Web 服务文档”,因此没有任何帮助。

在他们的 Java api 的 Exceptions 部分中似乎没有与此异常相关的任何内容:http://maxmind.github.io/GeoIP2-java/doc/v2.1.0/

更新

根据此处规范中的 DatabaseReader.Builder().build() 描述:http://maxmind.github.io/GeoIP2-java/doc/v2.1.0/ 我认为问题来自读取数据库。我还不确定如何解决这个问题。

更新

我能够使用二进制 db 文件 (GeoLite2-City.mmdb) 让它工作。如果有人知道为什么 CSV db 文件不适合我,我会留下这个问题。

【问题讨论】:

【参考方案1】:

您使用的库用于读取 GeoIP2 的二进制数据库文件。 CSV 文件供人们加载到数据库中(或他们想要使用 CSV 的任何其他方式)。

【讨论】:

谢谢戴夫。我对这一切有点陌生,所以当你说“加载到数据库”时,你的意思是我应该做一些事情,比如将它们转换为 sqlite/sqlback,甚至可以编写一个可以推送值的程序进入 MongoDB? 我将查看规格,看看是否可以在某处找到它。一旦我在网站上验证,我将接受您的回答。 我在 MAxMind 网站上到处搜索了我所建模的示例代码,在他们的 api 中,以及在解释 csv 和二进制数据库的结构的页面上。我没有一次读到您必须在他们的示例中为库使用二进制格式的数据库。我会接受这个答案,因为我不认为它是正确的,它只是一个提供两种格式的网站,然后隐含地暗示只使用二进制文件。谢谢戴夫。 来自dev.maxmind.com/geoip/geoip2/downloadable - "二进制数据库使用 MaxMind DB 文件格式。我们为多种语言提供这种格式的 API,并且我们为语言和平台提供了许多第三方 API不支持我们自己。”然后在dev.maxmind.com/geoip/geoip2/geoip2-csv-databases - “除了我们的 MaxMind DB 二进制格式,我们还提供适合导入 SQL 数据库的 CSV 格式的 GeoIP2 数据库。” 如果有人登陆这里,我遇到了这个问题,因为我错误地下载了 GeoLiteCity.db 而不是 GeoLite2City.mmdb。前者是旧格式,似乎不适用于最新的 Java API。

以上是关于MaxMind 从 IP 地址获取 Lat/Lng,InvalidDatabaseException 错误的主要内容,如果未能解决你的问题,请参考以下文章

根据地址获取 lat,lng

PHP 从 IP 地址获取位置 MaxMind GEOIP

防止 geocomplete 从 lat lng 坐标返回街道地址

如何获取谷歌地图上几个地方的地址,这些地方显示不可用给定的lat,lng

如何使用 Google Maps V3 从 lat/lng 获取城市?

如何从位置(x,y)获取 google maps v3 上的位置(lat/lng)