2

Maxmind 有两个免费的 GeoLite2 cvs 文件,我们可以使用它们来查找与 IP 地址范围相关的位置。我想在 SQL Server 中创建一个存储过程,该过程使用导入的表([GeoLite2_City_Blocks] 和 [GeoLite2_City_Locations]),采用传入 IP 地址的 varchar(45) 字符串并返回与该 IP 地址关联的位置。我原以为网络上会存在相关的 T-SQL 代码,但我找不到。

我检查了Querying GeoLite2 Country CSV in SQL,但我根本不明白 Prefix_Length (在 Blocks 表中称为 Network_mask_Length )是做什么的。如果它们与 CIDR 前缀相同(请参阅:http ://en.wikipedia.org/wiki/IPv6_subnetting_reference ),我会更加困惑,因为 Network_mask_Length 可能是 118、119 或任何其他未显示的数字那个维基百科页面。

我在 SQL Server 中看到了一个谈论 IPv6 的网页,该网页建议用零填充进行比较。我想不能简单地查询免费的 GeoLite2,而无需在表中添加至少一列以获取范围的最终 IP 地址?

我想我可以从 IPv4 地址前面去掉“::ffff:”,如果这是正确的说法的话。但是我仍然不知道如何将 network_mask_length (前缀)字段翻译成我理解的内容。然后对 IPv6 地址进行零填充,但我仍然不知道那个前缀是关于什么的。

我的问题是:

  1. 是否有任何地方可用的示例代码可以让我走上正轨?或者,
  2. 我需要做什么来创建一个接受 IPv4 或 IPv6 地址并返回相关位置或 geoname_id 的 proc?
4

2 回答 2

7

我使用这个查询(实际上是它的 MySQL)来导入 Block-File,将 IP 地址转换为 from-to ip 范围来做一个简单的BETWEEN查询。

我在表中添加了两个字段:start_ipend_ip. start_ip用 的数值填充network_start_ipend_ip用 计算的 ip 范围的末端填充network_mask_length。正如@oschwald 所说,我从前缀长度中减去了 96 并删除了 ::FFFF:。

LOAD DATA LOCAL INFILE 'GeoLite2-City-Blocks.csv' 
INTO TABLE geoip_blocks 
COLUMNS TERMINATED BY ',' 
OPTIONALLY ENCLOSED BY '"' 
LINES STARTING BY '::ffff:' -- import only ipv4 and cut off the ipv6 prefix
IGNORE 2 LINES 
(network_start_ip, network_mask_length, geoname_id, 
  registered_country_geoname_id, represented_country_geoname_id, postal_code, 
  latitude, longitude, is_anonymous_proxy, is_satellite_provider)
SET start_ip = inet_aton(network_start_ip),  -- set start ip as given
  end_ip = inet_aton(network_start_ip) + 
   (pow(2, (32 - (network_mask_length -96))) - 1) - 1, -- calc end_ip using mask
  network_mask_length = network_mask_length - 96 -- subtract 96 as we use ipv4

使用查询数据

SELECT geoname_id
FROM geoip_blocks
WHERE
INET_ATON('123.123.123.123') BETWEEN start_ip AND end_ip
LIMIT 1
于 2014-05-03T15:38:48.137 回答
1

前缀长度是网络地址中的有效位数。对于任何 IPv6 地址,这是一个介于 0-128 之间的数字,因为 IPv6 地址的长度为 128 位。例如,前缀为 128 的网络将只包含一个 IP 地址。前缀为 127 的网络将包含两个地址。126 会有 4 个,依此类推。

正如您所注意到的,格式为 ::FFFF:10.11.12.13 的 IPv6 地址对应于删除了 '::FFFF:' 的 IPv4 地址。'::FFFF:' 对应于地址的前 96 位。如果 IPv6 地址的前缀长度为 126,则 IPv4 地址的前缀长度为 30(即,最后两位有效)。

谷歌搜索应该显示代码片段以将 CIDR 表示法转换为范围。如果您只关心 IPv4,则最简单的方法可能是预先过滤数据,删除 ::FFFF:,从前缀长度中减去 96,并丢弃未映射到 IPv4 的地址范围。

于 2014-02-27T14:51:57.160 回答