package nat import ( "errors" "fmt" "strconv" "strings" "sync" "time" ma "github.com/jbenet/go-ipfs/Godeps/_workspace/src/github.com/jbenet/go-multiaddr" manet "github.com/jbenet/go-ipfs/Godeps/_workspace/src/github.com/jbenet/go-multiaddr-net" nat "github.com/jbenet/go-ipfs/Godeps/_workspace/src/github.com/fd/go-nat" goprocess "github.com/jbenet/go-ipfs/Godeps/_workspace/src/github.com/jbenet/goprocess" periodic "github.com/jbenet/go-ipfs/Godeps/_workspace/src/github.com/jbenet/goprocess/periodic" eventlog "github.com/jbenet/go-ipfs/thirdparty/eventlog" notifier "github.com/jbenet/go-ipfs/thirdparty/notifier" ) var ( // ErrNoMapping signals no mapping exists for an address ErrNoMapping = errors.New("mapping not established") ) var log = eventlog.Logger("nat") // MappingDuration is a default port mapping duration. // Port mappings are renewed every (MappingDuration / 3) const MappingDuration = time.Second * 60 // DiscoverGateway looks for a NAT device in the network and // returns an object that can manage port mappings. func DiscoverGateway() *NAT { nat, err := nat.DiscoverGateway() if err != nil { log.Debug("DiscoverGateway error:", err) return nil } addr, err := nat.GetDeviceAddress() if err != nil { log.Debug("DiscoverGateway address error:", err) } else { log.Debug("DiscoverGateway address:", addr) } return newNAT(nat) } // NAT is an object that manages address port mappings in // NATs (Network Address Translators). It is a long-running // service that will periodically renew port mappings, // and keep an up-to-date list of all the external addresses. type NAT struct { nat nat.NAT proc goprocess.Process // manages nat mappings lifecycle mappingmu sync.RWMutex // guards mappings mappings []*mapping Notifier } func newNAT(realNAT nat.NAT) *NAT { return &NAT{ nat: realNAT, proc: goprocess.WithParent(goprocess.Background()), } } // Close shuts down all port mappings. NAT can no longer be used. func (nat *NAT) Close() error { return nat.proc.Close() } // Notifier is an object that assists NAT in notifying listeners. // It is implemented using github.com/jbenet/go-ipfs/thirdparty/notifier type Notifier struct { n notifier.Notifier } func (n *Notifier) notifyAll(notify func(n Notifiee)) { n.n.NotifyAll(func(n notifier.Notifiee) { notify(n.(Notifiee)) }) } // Notify signs up notifiee to listen to NAT events. func (n *Notifier) Notify(notifiee Notifiee) { n.n.Notify(n) } // StopNotify stops signaling events to notifiee. func (n *Notifier) StopNotify(notifiee Notifiee) { n.n.StopNotify(notifiee) } // Notifiee is an interface objects must implement to listen to NAT events. type Notifiee interface { // Called every time a successful mapping happens // Warning: the port mapping may have changed. If that is the // case, both MappingSuccess and MappingChanged are called. MappingSuccess(nat *NAT, m Mapping) // Called when mapping a port succeeds, but the mapping is // with a different port than an earlier success. MappingChanged(nat *NAT, m Mapping, oldport int, newport int) // Called when a port mapping fails. NAT will continue attempting after // the next period. To stop trying, use: mapping.Close(). After this failure, // mapping.ExternalPort() will be zero, and nat.ExternalAddrs() will not // return the address for this mapping. With luck, the next attempt will // succeed, without the client needing to do anything. MappingFailed(nat *NAT, m Mapping, oldport int, err error) } // Mapping represents a port mapping in a NAT. type Mapping interface { // NAT returns the NAT object this Mapping belongs to. NAT() *NAT // Protocol returns the protocol of this port mapping. This is either // "tcp" or "udp" as no other protocols are likely to be NAT-supported. Protocol() string // InternalPort returns the internal device port. Mapping will continue to // try to map InternalPort() to an external facing port. InternalPort() int // ExternalPort returns the external facing port. If the mapping is not // established, port will be 0 ExternalPort() int // InternalAddr returns the internal address. InternalAddr() ma.Multiaddr // ExternalAddr returns the external facing address. If the mapping is not // established, addr will be nil, and and ErrNoMapping will be returned. ExternalAddr() (addr ma.Multiaddr, err error) } // keeps republishing type mapping struct { sync.Mutex // guards all fields nat *NAT proto string intport int extport int intaddr ma.Multiaddr proc goprocess.Process } func (m *mapping) NAT() *NAT { m.Lock() defer m.Unlock() return m.nat } func (m *mapping) Protocol() string { m.Lock() defer m.Unlock() return m.proto } func (m *mapping) InternalPort() int { m.Lock() defer m.Unlock() return m.intport } func (m *mapping) ExternalPort() int { m.Lock() defer m.Unlock() return m.extport } func (m *mapping) setExternalPort(p int) { m.Lock() defer m.Unlock() m.extport = p } func (m *mapping) InternalAddr() ma.Multiaddr { m.Lock() defer m.Unlock() return m.intaddr } func (m *mapping) ExternalAddr() (ma.Multiaddr, error) { if m.ExternalPort() == 0 { // dont even try right now. return nil, ErrNoMapping } ip, err := m.nat.nat.GetExternalAddress() if err != nil { return nil, err } ipmaddr, err := manet.FromIP(ip) if err != nil { return nil, fmt.Errorf("error parsing ip") } // call m.ExternalPort again, as mapping may have changed under our feet. (tocttou) extport := m.ExternalPort() if extport == 0 { return nil, ErrNoMapping } tcp, err := ma.NewMultiaddr(fmt.Sprintf("/%s/%d", m.Protocol(), extport)) if err != nil { return nil, err } maddr2 := ipmaddr.Encapsulate(tcp) return maddr2, nil } func (m *mapping) Close() error { return m.proc.Close() } // Mappings returns a slice of all NAT mappings func (nat *NAT) Mappings() []Mapping { nat.mappingmu.Lock() maps2 := make([]Mapping, len(nat.mappings)) for i, m := range nat.mappings { maps2[i] = m } nat.mappingmu.Unlock() return maps2 } func (nat *NAT) addMapping(m *mapping) { // make mapping automatically close when nat is closed. nat.proc.AddChild(m.proc) nat.mappingmu.Lock() nat.mappings = append(nat.mappings, m) nat.mappingmu.Unlock() } // NewMapping attemps to construct a mapping on protocol and internal port // It will also periodically renew the mapping until the returned Mapping // -- or its parent NAT -- is Closed. // // May not succeed, and mappings may change over time; // NAT devices may not respect our port requests, and even lie. // Clients should not store the mapped results, but rather always // poll our object for the latest mappings. func (nat *NAT) NewMapping(maddr ma.Multiaddr) (Mapping, error) { if nat == nil { return nil, fmt.Errorf("no nat available") } network, addr, err := manet.DialArgs(maddr) if err != nil { return nil, fmt.Errorf("DialArgs failed on addr:", maddr.String()) } switch network { case "tcp", "tcp4", "tcp6": network = "tcp" case "udp", "udp4", "udp6": network = "udp" default: return nil, fmt.Errorf("transport not supported by NAT: %s", network) } intports := strings.Split(addr, ":")[1] intport, err := strconv.Atoi(intports) if err != nil { return nil, err } m := &mapping{ nat: nat, proto: network, intport: intport, intaddr: maddr, } m.proc = periodic.Every(MappingDuration/3, func(worker goprocess.Process) { nat.establishMapping(m) }) nat.addMapping(m) // do it once synchronously, so first mapping is done right away, and before exiting, // allowing users -- in the optimistic case -- to use results right after. nat.establishMapping(m) return m, nil } func (nat *NAT) establishMapping(m *mapping) { oldport := m.ExternalPort() log.Debugf("Attempting port map: %s/%d", m.Protocol(), m.InternalPort()) newport, err := nat.nat.AddPortMapping(m.Protocol(), m.InternalPort(), "http", MappingDuration) failure := func() { m.setExternalPort(0) // clear mapping // TODO: log.Event log.Infof("failed to establish port mapping: %s", err) nat.Notifier.notifyAll(func(n Notifiee) { n.MappingFailed(nat, m, oldport, err) }) // we do not close if the mapping failed, // because it may work again next time. } if err != nil || newport == 0 { failure() return } m.setExternalPort(newport) ext, err := m.ExternalAddr() if err != nil { log.Debugf("NAT Mapping addr error: %s %s", m.InternalAddr(), err) failure() return } log.Debugf("NAT Mapping: %s --> %s", m.InternalAddr(), ext) if oldport != 0 && newport != oldport { log.Infof("failed to renew same port mapping: ch %d -> %d", oldport, newport) nat.Notifier.notifyAll(func(n Notifiee) { n.MappingChanged(nat, m, oldport, newport) }) } nat.Notifier.notifyAll(func(n Notifiee) { n.MappingSuccess(nat, m) }) } // PortMapAddrs attempts to open (and continue to keep open) // port mappings for given addrs. This function blocks until // all addresses have been tried. This allows clients to // retrieve results immediately after: // // nat.PortMapAddrs(addrs) // mapped := nat.ExternalAddrs() // // Some may not succeed, and mappings may change over time; // NAT devices may not respect our port requests, and even lie. // Clients should not store the mapped results, but rather always // poll our object for the latest mappings. func (nat *NAT) PortMapAddrs(addrs []ma.Multiaddr) { // spin off addr mappings independently. var wg sync.WaitGroup for _, addr := range addrs { // do all of them concurrently wg.Add(1) go func() { defer wg.Done() nat.NewMapping(addr) }() } wg.Wait() } // MappedAddrs returns address mappings NAT believes have been // successfully established. Unsuccessful mappings are nil. This is: // // map[internalAddr]externalAddr // // This set of mappings _may not_ be correct, as NAT devices are finicky. // Consider this with _best effort_ semantics. func (nat *NAT) MappedAddrs() map[ma.Multiaddr]ma.Multiaddr { mappings := nat.Mappings() addrmap := make(map[ma.Multiaddr]ma.Multiaddr, len(mappings)) for _, m := range mappings { i := m.InternalAddr() e, err := m.ExternalAddr() if err != nil { addrmap[i] = nil } else { addrmap[i] = e } } return addrmap } // ExternalAddrs returns a list of addresses that NAT believes have // been successfully established. Unsuccessful mappings are omitted, // so nat.ExternalAddrs() may return less addresses than nat.InternalAddrs(). // To see which addresses are mapped, use nat.MappedAddrs(). // // This set of mappings _may not_ be correct, as NAT devices are finicky. // Consider this with _best effort_ semantics. func (nat *NAT) ExternalAddrs() []ma.Multiaddr { mappings := nat.Mappings() addrs := make([]ma.Multiaddr, 0, len(mappings)) for _, m := range mappings { a, err := m.ExternalAddr() if err != nil { continue // this mapping not currently successful. } addrs = append(addrs, a) } return addrs }