cidranger

Fast IP to CIDR lookup in Golang

  • 所有者: yl2chen/cidranger
  • 平台:
  • 许可证: MIT License
  • 分类:
  • 主题:
  • 喜欢:
    0
      比较:

Github星跟踪图

cidranger

Fast IP to CIDR block(s) lookup using trie in Golang, inspired by IPv4 route lookup linux. Possible use cases include detecting if a IP address is from published cloud provider CIDR blocks (e.g. 52.95.110.1 is contained in published AWS Route53 CIDR 52.95.110.0/24), IP routing rules, etc.

GoDoc Reference
Build Status
Coverage Status
Go Report Card

This is visualization of a trie storing CIDR blocks 128.0.0.0/2 192.0.0.0/2 200.0.0.0/5 without path compression, the 0/1 number on the path indicates the bit value of the IP address at specified bit position, hence the path from root node to a child node represents a CIDR block that contains all IP ranges of its children, and children's children.

Visualization of trie storing same CIDR blocks with path compression, improving both lookup speed and memory footprint.

Getting Started

Configure imports.

import (
  "net"

  "github.com/yl2chen/cidranger"
)

Create a new ranger implemented using Path-Compressed prefix trie.

ranger := NewPCTrieRanger()

Inserts CIDR blocks.

_, network1, _ := net.ParseCIDR("192.168.1.0/24")
_, network2, _ := net.ParseCIDR("128.168.1.0/24")
ranger.Insert(NewBasicRangerEntry(*network1))
ranger.Insert(NewBasicRangerEntry(*network2))

To attach any additional value(s) to the entry, simply create custom struct
storing the desired value(s) that implements the RangerEntry interface:

type RangerEntry interface {
	Network() net.IPNet
}

The prefix trie can be visualized as:

0.0.0.0/0 (target_pos:31:has_entry:false), 1--> 128.0.0.0/1 (target_pos:30:has_entry:false), 0--> 128.168.1.0/24 (target_pos:7:has_entry:true), 1--> 192.168.1.0/24 (target_pos:7:has_entry:true)

To test if given IP is contained in constructed ranger,

contains, err = ranger.Contains(net.ParseIP("128.168.1.0")) // returns true, nil
contains, err = ranger.Contains(net.ParseIP("192.168.2.0")) // returns false, nil

To get all the networks given is contained in,

containingNetworks, err = ranger.ContainingNetworks(net.ParseIP("128.168.1.0"))

To get all networks in ranger,

entries, err := ranger.CoveredNetworks(*AllIPv4) // for IPv4
entries, err := ranger.CoveredNetworks(*AllIPv6) // for IPv6

Benchmark

Compare hit/miss case for IPv4/IPv6 using PC trie vs brute force implementation, Ranger is initialized with published AWS ip ranges (889 IPv4 CIDR blocks and 360 IPv6)

// Ipv4 lookup hit scenario
BenchmarkPCTrieHitIPv4UsingAWSRanges-4         	 5000000	       353   ns/op
BenchmarkBruteRangerHitIPv4UsingAWSRanges-4    	  100000	     13719   ns/op

// Ipv6 lookup hit scenario, counter-intuitively faster then IPv4 due to less IPv6 CIDR
// blocks in the AWS dataset, hence the constructed trie has less path splits and depth.
BenchmarkPCTrieHitIPv6UsingAWSRanges-4         	10000000	       143   ns/op
BenchmarkBruteRangerHitIPv6UsingAWSRanges-4    	  300000	      5178   ns/op

// Ipv4 lookup miss scenario
BenchmarkPCTrieMissIPv4UsingAWSRanges-4        	20000000	        96.5 ns/op
BenchmarkBruteRangerMissIPv4UsingAWSRanges-4   	   50000	     24781   ns/op

// Ipv6 lookup miss scenario
BenchmarkPCTrieHMissIPv6UsingAWSRanges-4       	10000000	       115   ns/op
BenchmarkBruteRangerMissIPv6UsingAWSRanges-4   	  100000	     10824   ns/op

Example of IPv6 trie:

::/0 (target_pos:127:has_entry:false), 0--> 2400::/14 (target_pos:113:has_entry:false), 0--> 2400:6400::/22 (target_pos:105:has_entry:false), 0--> 2400:6500::/32 (target_pos:95:has_entry:false), 0--> 2400:6500::/39 (target_pos:88:has_entry:false), 0--> 2400:6500:0:7000::/53 (target_pos:74:has_entry:false), 0--> 2400:6500:0:7000::/54 (target_pos:73:has_entry:false), 0--> 2400:6500:0:7000::/55 (target_pos:72:has_entry:false), 0--> 2400:6500:0:7000::/56 (target_pos:71:has_entry:true), 1--> 2400:6500:0:7100::/56 (target_pos:71:has_entry:true), 1--> 2400:6500:0:7200::/56 (target_pos:71:has_entry:true), 1--> 2400:6500:0:7400::/55 (target_pos:72:has_entry:false), 0--> 2400:6500:0:7400::/56 (target_pos:71:has_entry:true), 1--> 2400:6500:0:7500::/56 (target_pos:71:has_entry:true), 1--> 2400:6500:100:7000::/54 (target_pos:73:has_entry:false), 0--> 2400:6500:100:7100::/56 (target_pos:71:has_entry:true), 1--> 2400:6500:100:7200::/56 (target_pos:71:has_entry:true), 1--> 2400:6500:ff00::/64 (target_pos:63:has_entry:true), 1--> 2400:6700:ff00::/64 (target_pos:63:has_entry:true), 1--> 2403:b300:ff00::/64 (target_pos:63:has_entry:true)

主要指标

概览
名称与所有者yl2chen/cidranger
主编程语言Go
编程语言Go (语言数: 1)
平台
许可证MIT License
所有者活动
创建于2017-08-21 05:50:14
推送于2023-06-05 20:01:20
最后一次提交2021-09-27 19:18:09
发布数3
最新版本名称v1.0.2 (发布于 )
第一版名称v1.0.0 (发布于 )
用户参与
星数0.9k
关注者数21
派生数108
提交数65
已启用问题?
问题数20
打开的问题数4
拉请求数18
打开的拉请求数5
关闭的拉请求数7
项目设置
已启用Wiki?
已存档?
是复刻?
已锁定?
是镜像?
是私有?