Skip to content

Commit

Permalink
now with generic value
Browse files Browse the repository at this point in the history
  • Loading branch information
gaissmai committed Jan 10, 2024
1 parent 4f0c06c commit c5b54a2
Show file tree
Hide file tree
Showing 8 changed files with 318 additions and 316 deletions.
31 changes: 15 additions & 16 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -8,40 +8,39 @@

## !!! ATTENTION

API currently not stable!
API is currently not stable!

## Overview

`package cidrtree` is a datastructure for IP routing tables (IPv4/IPv6) with fast lookup (longest prefix match).

The implementation is based on treaps, which have been augmented here for CIDRs. Treaps are randomized, self-balancing binary search trees. Due to the nature of treaps the lookups (readers) and the update (writer) can be easily decoupled. This is the perfect fit for a software router or firewall.

This package is a specialization of the more generic [interval package] of the same author,
but explicit for CIDRs. It has a narrow focus with a specialized API for IP routing tables.
This package is a specialization of the more generic [interval package] of the same author, but explicit for CIDRs. It has a narrow focus with a specialized API for IP routing tables.

[interval package]: https://github.com/gaissmai/interval

## API
```go
import "github.com/gaissmai/cidrtree"

type Table struct { // Has unexported fields. }
type Table[V any] struct { // Has unexported fields. }
Table is an IPv4 and IPv6 routing table. The zero value is ready to use.

func (t Table) Lookup(ip netip.Addr) (lpm netip.Prefix, value any, ok bool)
func (t Table) LookupPrefix(pfx netip.Prefix) (lpm netip.Prefix, value any, ok bool)
func (t Table[V]) Lookup(ip netip.Addr) (lpm netip.Prefix, value V, ok bool)
func (t Table[V]) LookupPrefix(pfx netip.Prefix) (lpm netip.Prefix, value V, ok bool)

func (t *Table) Insert(pfx netip.Prefix, val any)
func (t *Table) Delete(pfx netip.Prefix) bool
func (t *Table) Union(other Table)
func (t *Table[V]) Insert(pfx netip.Prefix, value V)
func (t *Table[V]) Delete(pfx netip.Prefix) bool
func (t *Table[V]) Union(other Table[V])

func (t Table) InsertImmutable(pfx netip.Prefix, val any) *Table
func (t Table) DeleteImmutable(pfx netip.Prefix) (*Table, bool)
func (t Table) UnionImmutable(other Table) *Table
func (t Table) Clone() *Table
func (t Table[V]) InsertImmutable(pfx netip.Prefix, value V) *Table[V]
func (t Table[V]) DeleteImmutable(pfx netip.Prefix) (*Table[V], bool)
func (t Table[V]) UnionImmutable(other Table[V]) *Table[V]
func (t Table[V]) Clone() *Table[V]

func (t Table) String() string
func (t Table) Fprint(w io.Writer) error
func (t Table[V]) String() string
func (t Table[V]) Fprint(w io.Writer) error

func (t Table) Walk(cb func(pfx netip.Prefix, val any) bool)
func (t Table[V]) Walk(cb func(pfx netip.Prefix, value V) bool)
```
10 changes: 5 additions & 5 deletions bench_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -25,7 +25,7 @@ var intMap = map[int]string{

func BenchmarkLookup(b *testing.B) {
for k := 1; k <= 100_000; k *= 10 {
rt := new(cidrtree.Table)
rt := new(cidrtree.Table[any])
cidrs := shuffleFullTable(k)
for _, cidr := range cidrs {
rt.Insert(cidr, nil)
Expand All @@ -45,7 +45,7 @@ func BenchmarkLookup(b *testing.B) {

func BenchmarkLookupPrefix(b *testing.B) {
for k := 1; k <= 100_000; k *= 10 {
rt := new(cidrtree.Table)
rt := new(cidrtree.Table[any])
cidrs := shuffleFullTable(k)
for _, cidr := range cidrs {
rt.Insert(cidr, nil)
Expand All @@ -64,7 +64,7 @@ func BenchmarkLookupPrefix(b *testing.B) {

func BenchmarkClone(b *testing.B) {
for k := 1; k <= 100_000; k *= 10 {
rt := new(cidrtree.Table)
rt := new(cidrtree.Table[any])
for _, cidr := range shuffleFullTable(k) {
rt.Insert(cidr, nil)
}
Expand All @@ -80,7 +80,7 @@ func BenchmarkClone(b *testing.B) {

func BenchmarkInsert(b *testing.B) {
for k := 1; k <= 100_000; k *= 10 {
rt := new(cidrtree.Table)
rt := new(cidrtree.Table[any])
cidrs := shuffleFullTable(k)
for _, cidr := range cidrs {
rt.Insert(cidr, nil)
Expand All @@ -99,7 +99,7 @@ func BenchmarkInsert(b *testing.B) {

func BenchmarkDelete(b *testing.B) {
for k := 1; k <= 100_000; k *= 10 {
rt := new(cidrtree.Table)
rt := new(cidrtree.Table[any])
cidrs := shuffleFullTable(k)
for _, cidr := range cidrs {
rt.Insert(cidr, nil)
Expand Down
8 changes: 4 additions & 4 deletions debug.go
Original file line number Diff line number Diff line change
Expand Up @@ -10,7 +10,7 @@ import (
// fprintBST writes a horizontal tree diagram of the binary search tree (BST) to w.
//
// Note: This is for debugging purposes only.
func (t Table) fprintBST(w io.Writer) error {
func (t Table[V]) fprintBST(w io.Writer) error {
if t.root4 != nil {
if _, err := fmt.Fprint(w, "R "); err != nil {
return err
Expand All @@ -33,7 +33,7 @@ func (t Table) fprintBST(w io.Writer) error {
}

// fprintBST recursive helper.
func (n *node) fprintBST(w io.Writer, pad string) error {
func (n *node[V]) fprintBST(w io.Writer, pad string) error {
// stringify this node
_, err := fmt.Fprintf(w, "%v [prio:%.4g] [subtree maxUpper: %v]\n", n.cidr, float64(n.prio)/math.MaxUint64, n.maxUpper.cidr)
if err != nil {
Expand Down Expand Up @@ -80,7 +80,7 @@ func (n *node) fprintBST(w io.Writer, pad string) error {
// If the skip function is not nil, a true return value defines which nodes must be skipped in the statistics.
//
// Note: This is for debugging and testing purposes only during development.
func (t Table) statistics(skip func(netip.Prefix, any, int) bool) (size int, maxDepth int, average, deviation float64) {
func (t Table[V]) statistics(skip func(netip.Prefix, any, int) bool) (size int, maxDepth int, average, deviation float64) {
// key is depth, value is the sum of nodes with this depth
depths := make(map[int]int)

Expand Down Expand Up @@ -120,7 +120,7 @@ func (t Table) statistics(skip func(netip.Prefix, any, int) bool) (size int, max
}

// walkWithDepth in ascending prefix order.
func (n *node) walkWithDepth(cb func(netip.Prefix, any, int) bool, depth int) bool {
func (n *node[V]) walkWithDepth(cb func(netip.Prefix, any, int) bool, depth int) bool {
if n == nil {
return true
}
Expand Down
123 changes: 63 additions & 60 deletions example_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -8,73 +8,76 @@ import (
"github.com/gaissmai/cidrtree"
)

func addr(s string) netip.Addr {
func mustAddr(s string) netip.Addr {
return netip.MustParseAddr(s)
}

func prfx(s string) netip.Prefix {
func mustPfx(s string) netip.Prefix {
return netip.MustParsePrefix(s)
}

var input = []netip.Prefix{
prfx("fe80::/10"),
prfx("172.16.0.0/12"),
prfx("10.0.0.0/24"),
prfx("::1/128"),
prfx("192.168.0.0/16"),
prfx("10.0.0.0/8"),
prfx("::/0"),
prfx("10.0.1.0/24"),
prfx("169.254.0.0/16"),
prfx("2000::/3"),
prfx("2001:db8::/32"),
prfx("127.0.0.0/8"),
prfx("127.0.0.1/32"),
prfx("192.168.1.0/24"),
var input = []struct {
cidr netip.Prefix
nextHop netip.Addr
}{
{mustPfx("fe80::/10"), mustAddr("::1%lo")},
{mustPfx("172.16.0.0/12"), mustAddr("8.8.8.8")},
{mustPfx("10.0.0.0/24"), mustAddr("8.8.8.8")},
{mustPfx("::1/128"), mustAddr("::1%eth0")},
{mustPfx("192.168.0.0/16"), mustAddr("9.9.9.9")},
{mustPfx("10.0.0.0/8"), mustAddr("9.9.9.9")},
{mustPfx("::/0"), mustAddr("2001:db8::1")},
{mustPfx("10.0.1.0/24"), mustAddr("10.0.0.0")},
{mustPfx("169.254.0.0/16"), mustAddr("10.0.0.0")},
{mustPfx("2000::/3"), mustAddr("2001:db8::1")},
{mustPfx("2001:db8::/32"), mustAddr("2001:db8::1")},
{mustPfx("127.0.0.0/8"), mustAddr("127.0.0.1")},
{mustPfx("127.0.0.1/32"), mustAddr("127.0.0.1")},
{mustPfx("192.168.1.0/24"), mustAddr("127.0.0.1")},
}

func ExampleTable_Lookup() {
rtbl := new(cidrtree.Table)
for _, cidr := range input {
rtbl.Insert(cidr, nil)
rtbl := new(cidrtree.Table[netip.Addr])
for _, item := range input {
rtbl.Insert(item.cidr, item.nextHop)
}
rtbl.Fprint(os.Stdout)

fmt.Println()

ip := addr("42.0.0.0")
ip := mustAddr("42.0.0.0")
lpm, value, ok := rtbl.Lookup(ip)
fmt.Printf("Lookup: %-20v lpm: %-15v value: %v, ok: %v\n", ip, lpm, value, ok)
fmt.Printf("Lookup: %-20v lpm: %-15v value: %11v, ok: %v\n", ip, lpm, value, ok)

ip = addr("10.0.1.17")
ip = mustAddr("10.0.1.17")
lpm, value, ok = rtbl.Lookup(ip)
fmt.Printf("Lookup: %-20v lpm: %-15v value: %v, ok: %v\n", ip, lpm, value, ok)
fmt.Printf("Lookup: %-20v lpm: %-15v value: %11v, ok: %v\n", ip, lpm, value, ok)

ip = addr("2001:7c0:3100:1::111")
ip = mustAddr("2001:7c0:3100:1::111")
lpm, value, ok = rtbl.Lookup(ip)
fmt.Printf("Lookup: %-20v lpm: %-15v value: %v, ok: %v\n", ip, lpm, value, ok)
fmt.Printf("Lookup: %-20v lpm: %-15v value: %11v, ok: %v\n", ip, lpm, value, ok)

// Output:
// ▼
// ├─ 10.0.0.0/8 (<nil>)
// │ ├─ 10.0.0.0/24 (<nil>)
// │ └─ 10.0.1.0/24 (<nil>)
// ├─ 127.0.0.0/8 (<nil>)
// │ └─ 127.0.0.1/32 (<nil>)
// ├─ 169.254.0.0/16 (<nil>)
// ├─ 172.16.0.0/12 (<nil>)
// └─ 192.168.0.0/16 (<nil>)
// └─ 192.168.1.0/24 (<nil>)
// ├─ 10.0.0.0/8 (9.9.9.9)
// │ ├─ 10.0.0.0/24 (8.8.8.8)
// │ └─ 10.0.1.0/24 (10.0.0.0)
// ├─ 127.0.0.0/8 (127.0.0.1)
// │ └─ 127.0.0.1/32 (127.0.0.1)
// ├─ 169.254.0.0/16 (10.0.0.0)
// ├─ 172.16.0.0/12 (8.8.8.8)
// └─ 192.168.0.0/16 (9.9.9.9)
// └─ 192.168.1.0/24 (127.0.0.1)
// ▼
// └─ ::/0 (<nil>)
// ├─ ::1/128 (<nil>)
// ├─ 2000::/3 (<nil>)
// │ └─ 2001:db8::/32 (<nil>)
// └─ fe80::/10 (<nil>)
// └─ ::/0 (2001:db8::1)
// ├─ ::1/128 (::1%eth0)
// ├─ 2000::/3 (2001:db8::1)
// │ └─ 2001:db8::/32 (2001:db8::1)
// └─ fe80::/10 (::1%lo)
//
// Lookup: 42.0.0.0 lpm: invalid Prefix value: <nil>, ok: false
// Lookup: 10.0.1.17 lpm: 10.0.1.0/24 value: <nil>, ok: true
// Lookup: 2001:7c0:3100:1::111 lpm: 2000::/3 value: <nil>, ok: true
// Lookup: 42.0.0.0 lpm: invalid Prefix value: invalid IP, ok: false
// Lookup: 10.0.1.17 lpm: 10.0.1.0/24 value: 10.0.0.0, ok: true
// Lookup: 2001:7c0:3100:1::111 lpm: 2000::/3 value: 2001:db8::1, ok: true
}

func ExampleTable_Walk() {
Expand All @@ -83,25 +86,25 @@ func ExampleTable_Walk() {
return true
}

rtbl := new(cidrtree.Table)
for _, cidr := range input {
rtbl.Insert(cidr, nil)
rtbl := new(cidrtree.Table[any])
for _, item := range input {
rtbl.Insert(item.cidr, item.nextHop)
}
rtbl.Walk(cb)

// Output:
// 10.0.0.0/8 (<nil>)
// 10.0.0.0/24 (<nil>)
// 10.0.1.0/24 (<nil>)
// 127.0.0.0/8 (<nil>)
// 127.0.0.1/32 (<nil>)
// 169.254.0.0/16 (<nil>)
// 172.16.0.0/12 (<nil>)
// 192.168.0.0/16 (<nil>)
// 192.168.1.0/24 (<nil>)
// ::/0 (<nil>)
// ::1/128 (<nil>)
// 2000::/3 (<nil>)
// 2001:db8::/32 (<nil>)
// fe80::/10 (<nil>)
// 10.0.0.0/8 (9.9.9.9)
// 10.0.0.0/24 (8.8.8.8)
// 10.0.1.0/24 (10.0.0.0)
// 127.0.0.0/8 (127.0.0.1)
// 127.0.0.1/32 (127.0.0.1)
// 169.254.0.0/16 (10.0.0.0)
// 172.16.0.0/12 (8.8.8.8)
// 192.168.0.0/16 (9.9.9.9)
// 192.168.1.0/24 (127.0.0.1)
// ::/0 (2001:db8::1)
// ::1/128 (::1%eth0)
// 2000::/3 (2001:db8::1)
// 2001:db8::/32 (2001:db8::1)
// fe80::/10 (::1%lo)
}
24 changes: 12 additions & 12 deletions stringify.go
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,7 @@ import (
)

// String returns a hierarchical tree diagram of the ordered CIDRs as string, just a wrapper for [Tree.Fprint].
func (t Table) String() string {
func (t Table[V]) String() string {
w := new(strings.Builder)
_ = t.Fprint(w)
return w.String()
Expand All @@ -17,7 +17,7 @@ func (t Table) String() string {
//
// The order from top to bottom is in ascending order of the start address
// and the subtree structure is determined by the CIDRs coverage.
func (t Table) Fprint(w io.Writer) error {
func (t Table[V]) Fprint(w io.Writer) error {
if err := t.root4.fprint(w); err != nil {
return err
}
Expand All @@ -27,16 +27,16 @@ func (t Table) Fprint(w io.Writer) error {
return nil
}

func (n *node) fprint(w io.Writer) error {
func (n *node[V]) fprint(w io.Writer) error {
if n == nil {
return nil
}

// pcm = parent-child-mapping
var pcm parentChildsMap
var pcm parentChildsMap[V]

// init map
pcm.pcMap = make(map[*node][]*node)
pcm.pcMap = make(map[*node[V]][]*node[V])

pcm = n.buildParentChildsMap(pcm)

Expand All @@ -50,11 +50,11 @@ func (n *node) fprint(w io.Writer) error {
}

// start recursion with root and empty padding
var root *node
var root *node[V]
return root.walkAndStringify(w, pcm, "")
}

func (n *node) walkAndStringify(w io.Writer, pcm parentChildsMap, pad string) error {
func (n *node[V]) walkAndStringify(w io.Writer, pcm parentChildsMap[V], pad string) error {
// the prefix (pad + glyphe) is already printed on the line on upper level
if n != nil {
if _, err := fmt.Fprintf(w, "%v (%v)\n", n.cidr, n.value); err != nil {
Expand Down Expand Up @@ -92,13 +92,13 @@ func (n *node) walkAndStringify(w io.Writer, pcm parentChildsMap, pad string) er
// parentChildsMap, needed for hierarchical tree printing, this is not BST printing!
//
// CIDR tree, parent->childs relation printed. A parent CIDR covers a child CIDR.
type parentChildsMap struct {
pcMap map[*node][]*node // parent -> []child map
stack []*node // just needed for the algo
type parentChildsMap[T any] struct {
pcMap map[*node[T]][]*node[T] // parent -> []child map
stack []*node[T] // just needed for the algo
}

// buildParentChildsMap, in-order traversal
func (n *node) buildParentChildsMap(pcm parentChildsMap) parentChildsMap {
func (n *node[V]) buildParentChildsMap(pcm parentChildsMap[V]) parentChildsMap[V] {
if n == nil {
return pcm
}
Expand All @@ -114,7 +114,7 @@ func (n *node) buildParentChildsMap(pcm parentChildsMap) parentChildsMap {
}

// pcmForNode, find parent in stack, remove cidrs from stack, put this cidr on stack.
func (n *node) pcmForNode(pcm parentChildsMap) parentChildsMap {
func (n *node[V]) pcmForNode(pcm parentChildsMap[V]) parentChildsMap[V] {
// if this cidr is covered by a prev cidr on stack
for j := len(pcm.stack) - 1; j >= 0; j-- {
that := pcm.stack[j]
Expand Down
Loading

0 comments on commit c5b54a2

Please sign in to comment.