mirror of
https://github.com/logos-messaging/go-multiaddr.git
synced 2026-01-02 21:13:12 +00:00
112 lines
2.4 KiB
Go
112 lines
2.4 KiB
Go
package multiaddr
|
|
|
|
import (
|
|
"fmt"
|
|
"strings"
|
|
)
|
|
|
|
// Multiaddr is the data structure representing a multiaddr
|
|
type Multiaddr struct {
|
|
Bytes []byte
|
|
}
|
|
|
|
// NewMultiaddr parses and validates an input string, returning a *Multiaddr
|
|
func NewMultiaddr(s string) (*Multiaddr, error) {
|
|
b, err := stringToBytes(s)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return &Multiaddr{Bytes: b}, nil
|
|
}
|
|
|
|
// String returns the string representation of a Multiaddr
|
|
func (m *Multiaddr) String() (string, error) {
|
|
return bytesToString(m.Bytes)
|
|
}
|
|
|
|
// Protocols returns the list of protocols this Multiaddr has.
|
|
func (m *Multiaddr) Protocols() (ret []*Protocol, err error) {
|
|
|
|
// panic handler, in case we try accessing bytes incorrectly.
|
|
defer func() {
|
|
if e := recover(); e != nil {
|
|
ret = nil
|
|
err = e.(error)
|
|
}
|
|
}()
|
|
|
|
ps := []*Protocol{}
|
|
b := m.Bytes[:]
|
|
for len(b) > 0 {
|
|
p := ProtocolWithCode(int(b[0]))
|
|
if p == nil {
|
|
return nil, fmt.Errorf("no protocol with code %d", b[0])
|
|
}
|
|
ps = append(ps, p)
|
|
b = b[1+(p.Size/8):]
|
|
}
|
|
return ps, nil
|
|
}
|
|
|
|
// Encapsulate wraps a given Multiaddr, returning the resulting joined Multiaddr
|
|
func (m *Multiaddr) Encapsulate(o *Multiaddr) *Multiaddr {
|
|
b := make([]byte, len(m.Bytes)+len(o.Bytes))
|
|
b = append(m.Bytes, o.Bytes...)
|
|
return &Multiaddr{Bytes: b}
|
|
}
|
|
|
|
// Decapsulate unwraps Multiaddr up until the given Multiaddr is found.
|
|
func (m *Multiaddr) Decapsulate(o *Multiaddr) (*Multiaddr, error) {
|
|
s1, err := m.String()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
s2, err := o.String()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
i := strings.LastIndex(s1, s2)
|
|
if i < 0 {
|
|
return nil, fmt.Errorf("%s not contained in %s", s2, s1)
|
|
}
|
|
return NewMultiaddr(s1[:i])
|
|
}
|
|
|
|
// DialArgs is a convenience function returning arguments for use in net.Dial
|
|
func (m *Multiaddr) DialArgs() (string, string, error) {
|
|
if !m.IsThinWaist() {
|
|
return "", "", fmt.Errorf("%s is not a 'thin waist' address", m)
|
|
}
|
|
|
|
str, err := m.String()
|
|
if err != nil {
|
|
return "", "", err
|
|
}
|
|
|
|
parts := strings.Split(str, "/")[1:]
|
|
network := parts[2]
|
|
host := strings.Join([]string{parts[1], parts[3]}, ":")
|
|
return network, host, nil
|
|
}
|
|
|
|
// IsThinWaist returns whether this multiaddr includes "Thin Waist" Protocols.
|
|
// This means: /{IP4, IP6}/{TCP, UDP}
|
|
func (m *Multiaddr) IsThinWaist() bool {
|
|
p, err := m.Protocols()
|
|
if err != nil {
|
|
return false
|
|
}
|
|
|
|
if p[0].Code != P_IP4 && p[0].Code != P_IP6 {
|
|
return false
|
|
}
|
|
|
|
if p[1].Code != P_TCP && p[1].Code != P_UDP {
|
|
return false
|
|
}
|
|
|
|
return true
|
|
}
|