add more split tools and a bunch of tests

This commit is contained in:
Steven Allen 2018-10-02 14:26:17 -07:00
parent 96897075ba
commit c8d6befb43
3 changed files with 198 additions and 17 deletions

View File

@ -130,20 +130,3 @@ func newComponent(protocol Protocol, bvalue []byte) *Component {
offset: offset,
}
}
// ForEach walks over the multiaddr, component by component.
//
// This function iterates over components *by value* to avoid allocating.
func ForEach(m Multiaddr, cb func(c Component) bool) {
b := m.Bytes()
for len(b) > 0 {
n, c, err := readComponent(b)
if err != nil {
panic(err)
}
if !cb(c) {
return
}
b = b[n:]
}
}

97
util.go
View File

@ -55,3 +55,100 @@ func StringCast(s string) Multiaddr {
}
return m
}
// SplitFirst returns the first component and the rest of the multiaddr.
func SplitFirst(m Multiaddr) (*Component, Multiaddr) {
b := m.Bytes()
if len(b) == 0 {
return nil, nil
}
n, c, err := readComponent(b)
if err != nil {
panic(err)
}
if len(b) == n {
return &c, nil
}
return &c, multiaddr{b[n:]}
}
// SplitLast returns the rest of the multiaddr and the last component.
func SplitLast(m Multiaddr) (Multiaddr, *Component) {
b := m.Bytes()
if len(b) == 0 {
return nil, nil
}
var (
c Component
err error
offset int
)
for {
var n int
n, c, err = readComponent(b[offset:])
if err != nil {
panic(err)
}
if len(b) == n+offset {
// Reached end
if offset == 0 {
// Only one component
return nil, &c
}
return multiaddr{b[:offset]}, &c
}
offset += n
}
}
// SplitFunc splits the multiaddr when the callback first returns true. The
// component on which the callback first returns will be included in the
// *second* multiaddr.
func SplitFunc(m Multiaddr, cb func(Component) bool) (Multiaddr, Multiaddr) {
b := m.Bytes()
if len(b) == 0 {
return nil, nil
}
var (
c Component
err error
offset int
)
for offset < len(b) {
var n int
n, c, err = readComponent(b[offset:])
if err != nil {
panic(err)
}
if cb(c) {
break
}
offset += n
}
switch offset {
case 0:
return nil, m
case len(b):
return m, nil
default:
return multiaddr{b[:offset]}, multiaddr{b[offset:]}
}
}
// ForEach walks over the multiaddr, component by component.
//
// This function iterates over components *by value* to avoid allocating.
func ForEach(m Multiaddr, cb func(c Component) bool) {
b := m.Bytes()
for len(b) > 0 {
n, c, err := readComponent(b)
if err != nil {
panic(err)
}
if !cb(c) {
return
}
b = b[n:]
}
}

101
util_test.go Normal file
View File

@ -0,0 +1,101 @@
package multiaddr
import (
"strings"
"testing"
)
func TestSplitFirstLast(t *testing.T) {
ipStr := "/ip4/0.0.0.0"
tcpStr := "/tcp/123"
quicStr := "/quic"
ipfsStr := "/ipfs/QmPSQnBKM9g7BaUcZCvswUJVscQ1ipjmwxN5PXCjkp9EQ7"
for _, x := range [][]string{
[]string{ipStr, tcpStr, quicStr, ipfsStr},
[]string{ipStr, tcpStr, ipfsStr},
[]string{ipStr, tcpStr},
[]string{ipStr},
[]string{},
} {
addr := StringCast(strings.Join(x, ""))
head, tail := SplitFirst(addr)
rest, last := SplitLast(addr)
if len(x) == 0 {
if head != nil {
t.Error("expected head to be nil")
}
if tail != nil {
t.Error("expected tail to be nil")
}
if rest != nil {
t.Error("expected rest to be nil")
}
if last != nil {
t.Error("expected last to be nil")
}
continue
}
if !head.Equal(StringCast(x[0])) {
t.Errorf("expected %s to be %s", head, x[0])
}
if !last.Equal(StringCast(x[len(x)-1])) {
t.Errorf("expected %s to be %s", head, x[len(x)-1])
}
if len(x) == 1 {
if tail != nil {
t.Error("expected tail to be nil")
}
if rest != nil {
t.Error("expected rest to be nil")
}
continue
}
tailExp := strings.Join(x[1:], "")
if !tail.Equal(StringCast(tailExp)) {
t.Errorf("expected %s to be %s", tail, tailExp)
}
restExp := strings.Join(x[:len(x)-1], "")
if !rest.Equal(StringCast(restExp)) {
t.Errorf("expected %s to be %s", rest, restExp)
}
}
}
func TestSplitFunc(t *testing.T) {
ipStr := "/ip4/0.0.0.0"
tcpStr := "/tcp/123"
quicStr := "/quic"
ipfsStr := "/ipfs/QmPSQnBKM9g7BaUcZCvswUJVscQ1ipjmwxN5PXCjkp9EQ7"
for _, x := range [][]string{
[]string{ipStr, tcpStr, quicStr, ipfsStr},
[]string{ipStr, tcpStr, ipfsStr},
[]string{ipStr, tcpStr},
[]string{ipStr},
} {
addr := StringCast(strings.Join(x, ""))
for i, cs := range x {
target := StringCast(cs)
a, b := SplitFunc(addr, func(c Component) bool {
return c.Equal(target)
})
if i == 0 {
if a != nil {
t.Error("expected nil addr")
}
} else {
if !a.Equal(StringCast(strings.Join(x[:i], ""))) {
t.Error("split failed")
}
if !b.Equal(StringCast(strings.Join(x[i:], ""))) {
t.Error("split failed")
}
}
}
a, b := SplitFunc(addr, func(_ Component) bool { return false })
if !a.Equal(addr) || b != nil {
t.Error("should not have split")
}
}
}