package conn import ( "sync" context "github.com/jbenet/go-ipfs/Godeps/_workspace/src/code.google.com/p/go.net/context" ma "github.com/jbenet/go-ipfs/Godeps/_workspace/src/github.com/jbenet/go-multiaddr" peer "github.com/jbenet/go-ipfs/peer" u "github.com/jbenet/go-ipfs/util" ) // Duplex is a simple duplex channel type Duplex struct { In chan []byte Out chan []byte } // MultiConn represents a single connection to another Peer (IPFS Node). type MultiConn struct { // connections, mapped by a string, which uniquely identifies the connection. // this string is: /addr1/peer1/addr2/peer2 (peers ordered lexicographically) conns map[string]Conn local *peer.Peer remote *peer.Peer // fan-in/fan-out duplex Duplex // for adding/removing connections concurrently sync.RWMutex ContextCloser } // NewMultiConn constructs a new connection func NewMultiConn(ctx context.Context, local, remote *peer.Peer, conns []Conn) (Conn, error) { c := &MultiConn{ local: local, remote: remote, conns: map[string]Conn{}, duplex: Duplex{ In: make(chan []byte, 10), Out: make(chan []byte, 10), }, } // must happen before Adds / fanOut c.ContextCloser = NewContextCloser(ctx, c.close) log.Info("adding %d...", len(conns)) if conns != nil && len(conns) > 0 { c.Add(conns...) } go c.fanOut() log.Info("newMultiConn: %v to %v", local, remote) return c, nil } // Add adds given Conn instances to multiconn. func (c *MultiConn) Add(conns ...Conn) { c.Lock() defer c.Unlock() for _, c2 := range conns { log.Info("MultiConn: adding %s", c2) if c.LocalPeer() != c2.LocalPeer() || c.RemotePeer() != c2.RemotePeer() { log.Error("%s", c2) panic("connection addresses mismatch") } c.conns[c2.ID()] = c2 go c.fanInSingle(c2) log.Info("MultiConn: added %s", c2) } } // Remove removes given Conn instances from multiconn. func (c *MultiConn) Remove(conns ...Conn) { // first remove them to avoid sending any more messages through it. { c.Lock() for _, c1 := range conns { c2, found := c.conns[c1.ID()] if !found { panic("Conn not in MultiConn") } if c1 != c2 { panic("different Conn objects for same id.") } delete(c.conns, c2.ID()) } c.Unlock() } // close all in parallel, but wait for all to be done closing. CloseConns(conns) } // CloseConns closes multiple connections in parallel, and waits for all // to finish closing. func CloseConns(conns []Conn) { var wg sync.WaitGroup for _, child := range conns { select { case <-child.Closed(): // if already closed, continue continue default: } wg.Add(1) go func(child Conn) { child.Close() wg.Done() }(child) } wg.Wait() } // fanOut is the multiplexor out -- it sends outgoing messages over the // underlying single connections. func (c *MultiConn) fanOut() { c.Children().Add(1) defer c.Children().Done() for { select { case <-c.Closing(): return // send data out through our "best connection" case m, more := <-c.duplex.Out: if !more { return } sc := c.BestConn() if sc == nil { // maybe this should be a logged error, not a panic. panic("sending out multiconn without any live connection") } sc.Out() <- m } } } // fanInSingle is a multiplexor in -- it receives incoming messages over the // underlying single connections. func (c *MultiConn) fanInSingle(child Conn) { c.Children().Add(1) child.Children().Add(1) // yep, on the child too. // cleanup all data associated with this child Connection. defer func() { // in case it still is in the map, remove it. c.Lock() delete(c.conns, child.ID()) c.Unlock() c.Children().Done() child.Children().Done() }() for { select { case <-c.Closing(): // multiconn closing return case <-child.Closing(): // child closing return case m, more := <-child.In(): // receiving data if !more { return // closed } c.duplex.In <- m } } } // close is the internal close function, called by ContextCloser.Close func (c *MultiConn) close() error { log.Debug("%s closing Conn with %s", c.local, c.remote) // get connections c.RLock() conns := make([]Conn, 0, len(c.conns)) for _, c := range c.conns { conns = append(conns, c) } c.RUnlock() // close underlying connections CloseConns(conns) return nil } // BestConn is the best connection in this MultiConn func (c *MultiConn) BestConn() Conn { c.RLock() defer c.RUnlock() var id1 string var c1 Conn for id2, c2 := range c.conns { if id1 == "" || id1 < id2 { id1 = id2 c1 = c2 } } return c1 } // ID is an identifier unique to this connection. // In MultiConn, this is all the children IDs XORed together. func (c *MultiConn) ID() string { c.RLock() defer c.RUnlock() ids := []byte(nil) for i := range c.conns { if ids == nil { ids = []byte(i) } else { ids = u.XOR(ids, []byte(i)) } } return string(ids) } func (c *MultiConn) String() string { return String(c, "MultiConn") } // LocalMultiaddr is the Multiaddr on this side func (c *MultiConn) LocalMultiaddr() ma.Multiaddr { return c.BestConn().LocalMultiaddr() } // RemoteMultiaddr is the Multiaddr on the remote side func (c *MultiConn) RemoteMultiaddr() ma.Multiaddr { return c.BestConn().RemoteMultiaddr() } // LocalPeer is the Peer on this side func (c *MultiConn) LocalPeer() *peer.Peer { return c.local } // RemotePeer is the Peer on the remote side func (c *MultiConn) RemotePeer() *peer.Peer { return c.remote } // In returns a readable message channel func (c *MultiConn) In() <-chan []byte { return c.duplex.In } // Out returns a writable message channel func (c *MultiConn) Out() chan<- []byte { return c.duplex.Out }