[#43] cmd/neofs-node: Support hostnames with dns, ipv4 and ipv6 addresses

Signed-off-by: Alex Vanin <alexey@nspcc.ru>
This commit is contained in:
Alex Vanin 2020-09-24 11:27:14 +03:00
parent f930993e3a
commit e158497560
8 changed files with 57 additions and 21 deletions

View file

@ -154,17 +154,7 @@ func initCfg(path string) *cfg {
log, err := logger.NewLogger(viperCfg) log, err := logger.NewLogger(viperCfg)
fatalOnErr(err) fatalOnErr(err)
viperCfg.GetString(cfgListenAddress) netAddr, err := network.AddressFromString(viperCfg.GetString(cfgBootstrapAddress))
endpoint, port, err := net.SplitHostPort(viperCfg.GetString(cfgListenAddress))
fatalOnErr(err)
netAddr, err := network.AddressFromString(strings.Join([]string{
"/ip4",
endpoint,
"tcp",
port,
}, "/"))
fatalOnErr(err) fatalOnErr(err)
return &cfg{ return &cfg{

View file

@ -1,11 +1,24 @@
package network package network
import ( import (
"net"
"strings"
"github.com/multiformats/go-multiaddr" "github.com/multiformats/go-multiaddr"
manet "github.com/multiformats/go-multiaddr-net" manet "github.com/multiformats/go-multiaddr-net"
"github.com/pkg/errors" "github.com/pkg/errors"
) )
/*
HostAddr strings: "localhost:8080", ":8080", "192.168.0.1:8080"
MultiAddr strings: "/dns4/localhost/tcp/8080", "/ip4/192.168.0.1/tcp/8080"
IPAddr strings: "192.168.0.1:8080"
*/
const (
L4Protocol = "tcp"
)
// Address represents the NeoFS node // Address represents the NeoFS node
// network address. // network address.
type Address struct { type Address struct {
@ -18,12 +31,13 @@ type LocalAddressSource interface {
LocalAddress() *Address LocalAddress() *Address
} }
// String returns multiaddr string
func (a Address) String() string { func (a Address) String() string {
return a.ma.String() return a.ma.String()
} }
// NetAddr returns network endpoint address in string format. // IPAddrString returns network endpoint address in string format.
func (a Address) NetAddr() (string, error) { func (a Address) IPAddrString() (string, error) {
ip, err := manet.ToNetAddr(a.ma) ip, err := manet.ToNetAddr(a.ma)
if err != nil { if err != nil {
return "", errors.Wrap(err, "could not get net addr") return "", errors.Wrap(err, "could not get net addr")
@ -32,11 +46,19 @@ func (a Address) NetAddr() (string, error) {
return ip.String(), nil return ip.String(), nil
} }
// AddressFromString restores address from a string representation. // AddressFromString restores address from a multiaddr string representation.
func AddressFromString(s string) (*Address, error) { func AddressFromString(s string) (*Address, error) {
ma, err := multiaddr.NewMultiaddr(s) ma, err := multiaddr.NewMultiaddr(s)
if err != nil { if err != nil {
return nil, err s, err = multiaddrStringFromHostAddr(s)
if err != nil {
return nil, err
}
ma, err = multiaddr.NewMultiaddr(s) // don't want recursion there
if err != nil {
return nil, err
}
} }
return &Address{ return &Address{
@ -44,6 +66,30 @@ func AddressFromString(s string) (*Address, error) {
}, nil }, nil
} }
// multiaddrStringFromHostAddr converts "localhost:8080" to "/dns4/localhost/tcp/8080"
func multiaddrStringFromHostAddr(host string) (string, error) {
endpoint, port, err := net.SplitHostPort(host)
if err != nil {
return "", err
}
var (
prefix = "/dns4"
addr = endpoint
)
if ip := net.ParseIP(endpoint); ip != nil {
addr = ip.String()
if ip.To4() == nil {
prefix = "/ip6"
} else {
prefix = "/ip4"
}
}
return strings.Join([]string{prefix, addr, L4Protocol, port}, "/"), nil
}
// IsLocalAddress returns true if network endpoint from local address // IsLocalAddress returns true if network endpoint from local address
// source is equal to network endpoint of passed address. // source is equal to network endpoint of passed address.
func IsLocalAddress(src LocalAddressSource, addr *Address) bool { func IsLocalAddress(src LocalAddressSource, addr *Address) bool {

View file

@ -24,7 +24,7 @@ func TestAddress_NetAddr(t *testing.T) {
addr, err := AddressFromString(ma.String()) addr, err := AddressFromString(ma.String())
require.NoError(t, err) require.NoError(t, err)
netAddr, err := addr.NetAddr() netAddr, err := addr.IPAddrString()
require.NoError(t, err) require.NoError(t, err)
require.Equal(t, ip+":"+port, netAddr) require.Equal(t, ip+":"+port, netAddr)
} }

View file

@ -22,7 +22,7 @@ func (h *remoteHeader) head(ctx context.Context, prm *Prm, handler func(*object.
return errors.Wrapf(err, "(%T) could not receive private key", h) return errors.Wrapf(err, "(%T) could not receive private key", h)
} }
addr, err := h.node.NetAddr() addr, err := h.node.IPAddrString()
if err != nil { if err != nil {
return err return err
} }

View file

@ -38,7 +38,7 @@ func (t *remoteTarget) Close() (*transformer.AccessIdentifiers, error) {
return nil, errors.Wrapf(err, "(%T) could not receive private key", t) return nil, errors.Wrapf(err, "(%T) could not receive private key", t)
} }
addr, err := t.addr.NetAddr() addr, err := t.addr.IPAddrString()
if err != nil { if err != nil {
return nil, err return nil, err
} }

View file

@ -32,7 +32,7 @@ func (r *remoteRangeWriter) WriteTo(w io.Writer) (int64, error) {
return 0, errors.Wrapf(err, "(%T) could not receive private key", r) return 0, errors.Wrapf(err, "(%T) could not receive private key", r)
} }
addr, err := r.node.NetAddr() addr, err := r.node.IPAddrString()
if err != nil { if err != nil {
return 0, err return 0, err
} }

View file

@ -23,7 +23,7 @@ func (h *remoteHasher) hashRange(ctx context.Context, prm *Prm, handler func([][
return errors.Wrapf(err, "(%T) could not receive private key", h) return errors.Wrapf(err, "(%T) could not receive private key", h)
} }
addr, err := h.node.NetAddr() addr, err := h.node.IPAddrString()
if err != nil { if err != nil {
return err return err
} }

View file

@ -24,7 +24,7 @@ func (s *remoteStream) stream(ctx context.Context, ch chan<- []*object.ID) error
return errors.Wrapf(err, "(%T) could not receive private key", s) return errors.Wrapf(err, "(%T) could not receive private key", s)
} }
addr, err := s.addr.NetAddr() addr, err := s.addr.IPAddrString()
if err != nil { if err != nil {
return err return err
} }