go-multiaddr/convert.go

259 lines
5.4 KiB
Go
Raw Normal View History

2014-11-05 00:38:21 -08:00
package manet
2014-10-10 23:56:35 -07:00
import (
"fmt"
"net"
"strings"
2016-05-02 16:30:02 -07:00
"sync"
2014-10-10 23:56:35 -07:00
2016-04-19 12:26:49 -07:00
ma "github.com/jbenet/go-multiaddr"
2015-10-03 15:34:36 -07:00
utp "github.com/jbenet/go-multiaddr-net/utp"
2014-10-10 23:56:35 -07:00
)
var errIncorrectNetAddr = fmt.Errorf("incorrect network addr conversion")
2016-05-02 16:30:02 -07:00
type AddrParser func(a net.Addr) (ma.Multiaddr, error)
type MaddrParser func(ma ma.Multiaddr) (net.Addr, error)
var maddrParsers map[string]MaddrParser
var addrParsers map[string]AddrParser
var addrParsersLock sync.Mutex
func init() {
addrParsers = make(map[string]AddrParser)
maddrParsers = make(map[string]MaddrParser)
registerDefaultAddrParsers()
registerDefaultMaddrParsers()
}
func registerDefaultAddrParsers() {
funcs := map[string]AddrParser{
"tcp": ParseTcpNetAddr,
"udp": ParseUdpNetAddr,
"utp": ParseUtpNetAddr,
"ip": ParseIpNetAddr,
}
for k, v := range funcs {
addrParsers[k] = v
addrParsers[k+"4"] = v
addrParsers[k+"6"] = v
}
addrParsers["ip+net"] = ParseIpPlusNetAddr
}
func RegisterAddressType(netname, maname string, ap AddrParser, mp MaddrParser) {
addrParsersLock.Lock()
defer addrParsersLock.Unlock()
addrParsers[netname] = ap
maddrParsers[maname] = mp
}
func ParseTcpNetAddr(a net.Addr) (ma.Multiaddr, error) {
ac, ok := a.(*net.TCPAddr)
if !ok {
return nil, errIncorrectNetAddr
}
// Get IP Addr
ipm, err := FromIP(ac.IP)
if err != nil {
return nil, errIncorrectNetAddr
}
// Get TCP Addr
tcpm, err := ma.NewMultiaddr(fmt.Sprintf("/tcp/%d", ac.Port))
if err != nil {
return nil, errIncorrectNetAddr
}
// Encapsulate
return ipm.Encapsulate(tcpm), nil
}
func ParseUdpNetAddr(a net.Addr) (ma.Multiaddr, error) {
ac, ok := a.(*net.UDPAddr)
if !ok {
return nil, errIncorrectNetAddr
}
// Get IP Addr
ipm, err := FromIP(ac.IP)
if err != nil {
return nil, errIncorrectNetAddr
}
// Get UDP Addr
udpm, err := ma.NewMultiaddr(fmt.Sprintf("/udp/%d", ac.Port))
if err != nil {
return nil, errIncorrectNetAddr
}
// Encapsulate
return ipm.Encapsulate(udpm), nil
}
func ParseUtpNetAddr(a net.Addr) (ma.Multiaddr, error) {
acc, ok := a.(*utp.Addr)
if !ok {
return nil, errIncorrectNetAddr
}
// Get UDP Addr
ac, ok := acc.Child().(*net.UDPAddr)
if !ok {
return nil, errIncorrectNetAddr
}
// Get IP Addr
ipm, err := FromIP(ac.IP)
if err != nil {
return nil, errIncorrectNetAddr
}
// Get UDP Addr
utpm, err := ma.NewMultiaddr(fmt.Sprintf("/udp/%d/utp", ac.Port))
if err != nil {
return nil, errIncorrectNetAddr
}
// Encapsulate
return ipm.Encapsulate(utpm), nil
}
func ParseIpNetAddr(a net.Addr) (ma.Multiaddr, error) {
ac, ok := a.(*net.IPAddr)
if !ok {
return nil, errIncorrectNetAddr
}
return FromIP(ac.IP)
}
func ParseIpPlusNetAddr(a net.Addr) (ma.Multiaddr, error) {
ac, ok := a.(*net.IPNet)
if !ok {
return nil, errIncorrectNetAddr
}
return FromIP(ac.IP)
}
func getAddrParser(net string) (AddrParser, error) {
addrParsersLock.Lock()
defer addrParsersLock.Unlock()
parser, ok := addrParsers[net]
if !ok {
return nil, fmt.Errorf("unknown network %v", net)
}
return parser, nil
}
2014-10-10 23:56:35 -07:00
// FromNetAddr converts a net.Addr type to a Multiaddr.
func FromNetAddr(a net.Addr) (ma.Multiaddr, error) {
2015-10-08 10:23:05 -07:00
if a == nil {
return nil, fmt.Errorf("nil multiaddr")
}
2016-05-02 16:30:02 -07:00
p, err := getAddrParser(a.Network())
if err != nil {
return nil, err
}
2015-10-08 10:23:05 -07:00
2016-05-02 16:30:02 -07:00
return p(a)
}
2014-11-19 13:24:03 -08:00
2016-05-02 16:30:02 -07:00
func getMaddrParser(name string) (MaddrParser, error) {
addrParsersLock.Lock()
defer addrParsersLock.Unlock()
p, ok := maddrParsers[name]
if !ok {
return nil, fmt.Errorf("network not supported: %s", name)
2014-10-10 23:56:35 -07:00
}
2016-05-02 16:30:02 -07:00
return p, nil
2014-10-10 23:56:35 -07:00
}
// ToNetAddr converts a Multiaddr to a net.Addr
// Must be ThinWaist. acceptable protocol stacks are:
// /ip{4,6}/{tcp, udp}
func ToNetAddr(maddr ma.Multiaddr) (net.Addr, error) {
2016-05-02 16:30:02 -07:00
protos := maddr.Protocols()
final := protos[len(protos)-1]
p, err := getMaddrParser(final.Name)
if err != nil {
return nil, err
}
return p(maddr)
}
func registerDefaultMaddrParsers() {
for _, net := range []string{"tcp", "udp", "utp", "ip", "ip4", "ip6"} {
maddrParsers[net] = parseBasicNetAddr
}
}
func parseBasicNetAddr(maddr ma.Multiaddr) (net.Addr, error) {
2014-10-10 23:56:35 -07:00
network, host, err := DialArgs(maddr)
if err != nil {
return nil, err
}
switch network {
2015-01-12 18:07:14 -08:00
case "tcp", "tcp4", "tcp6":
2014-10-10 23:56:35 -07:00
return net.ResolveTCPAddr(network, host)
2015-01-12 18:07:14 -08:00
case "udp", "udp4", "udp6":
2014-10-10 23:56:35 -07:00
return net.ResolveUDPAddr(network, host)
2015-01-12 18:07:14 -08:00
case "utp", "utp4", "utp6":
return utp.ResolveAddr(network, host)
2015-01-12 18:07:14 -08:00
case "ip", "ip4", "ip6":
2014-10-10 23:56:35 -07:00
return net.ResolveIPAddr(network, host)
}
return nil, fmt.Errorf("network not supported: %s", network)
}
// FromIP converts a net.IP type to a Multiaddr.
func FromIP(ip net.IP) (ma.Multiaddr, error) {
switch {
case ip.To4() != nil:
return ma.NewMultiaddr("/ip4/" + ip.String())
case ip.To16() != nil:
return ma.NewMultiaddr("/ip6/" + ip.String())
default:
return nil, errIncorrectNetAddr
}
}
// DialArgs is a convenience function returning arguments for use in net.Dial
func DialArgs(m ma.Multiaddr) (string, string, error) {
if !IsThinWaist(m) {
return "", "", fmt.Errorf("%s is not a 'thin waist' address", m)
}
str := m.String()
parts := strings.Split(str, "/")[1:]
if len(parts) == 2 { // only IP
return parts[0], parts[1], nil
}
network := parts[2]
2014-11-19 13:24:03 -08:00
if parts[2] == "udp" && len(parts) > 4 && parts[4] == "utp" {
network = parts[4]
}
2014-10-10 23:56:35 -07:00
var host string
switch parts[0] {
case "ip4":
2015-01-12 18:07:14 -08:00
network = network + "4"
2014-10-10 23:56:35 -07:00
host = strings.Join([]string{parts[1], parts[3]}, ":")
case "ip6":
2015-01-12 18:07:14 -08:00
network = network + "6"
2014-10-10 23:56:35 -07:00
host = fmt.Sprintf("[%s]:%s", parts[1], parts[3])
}
return network, host, nil
}