swarm_listen.go 1.98 KB
Newer Older
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
1 2 3 4 5
package swarm

import (
	"fmt"

6 7
	"github.com/libp2p/go-libp2p-core/network"

Jeromy's avatar
Jeromy committed
8
	ma "github.com/multiformats/go-multiaddr"
Jeromy's avatar
Jeromy committed
9
)
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
10

Steven Allen's avatar
Steven Allen committed
11 12 13
// Listen sets up listeners for all of the given addresses.
// It returns as long as we successfully listen on at least *one* address.
func (s *Swarm) Listen(addrs ...ma.Multiaddr) error {
Jeromy's avatar
Jeromy committed
14 15 16
	errs := make([]error, len(addrs))
	var succeeded int
	for i, a := range addrs {
17
		if err := s.AddListenAddr(a); err != nil {
Jeromy's avatar
Jeromy committed
18
			errs[i] = err
19 20
		} else {
			succeeded++
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
21 22 23
		}
	}

Jeromy's avatar
Jeromy committed
24 25
	for i, e := range errs {
		if e != nil {
Jeromy's avatar
Jeromy committed
26
			log.Warningf("listen on %s failed: %s", addrs[i], errs[i])
Jeromy's avatar
Jeromy committed
27
		}
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
28
	}
29

Jeromy's avatar
Jeromy committed
30 31 32 33
	if succeeded == 0 && len(addrs) > 0 {
		return fmt.Errorf("failed to listen on any addresses: %s", errs)
	}

Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
34 35 36
	return nil
}

Steven Allen's avatar
Steven Allen committed
37 38 39 40 41 42
// AddListenAddr tells the swarm to listen on a single address. Unlike Listen,
// this method does not attempt to filter out bad addresses.
func (s *Swarm) AddListenAddr(a ma.Multiaddr) error {
	tpt := s.TransportForListening(a)
	if tpt == nil {
		return ErrNoTransport
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
43
	}
Jeromy's avatar
Jeromy committed
44

Steven Allen's avatar
Steven Allen committed
45
	list, err := tpt.Listen(a)
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
46 47 48 49
	if err != nil {
		return err
	}

Steven Allen's avatar
Steven Allen committed
50 51 52 53 54
	s.listeners.Lock()
	if s.listeners.m == nil {
		s.listeners.Unlock()
		list.Close()
		return ErrSwarmClosed
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
55
	}
Steven Allen's avatar
Steven Allen committed
56 57 58
	s.refs.Add(1)
	s.listeners.m[list] = struct{}{}
	s.listeners.Unlock()
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
59

Jeromy's avatar
Jeromy committed
60 61
	maddr := list.Multiaddr()

Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
62
	// signal to our notifiees on successful conn.
63
	s.notifyAll(func(n network.Notifiee) {
Steven Allen's avatar
Steven Allen committed
64
		n.Listen(s, maddr)
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
65 66
	})

Steven Allen's avatar
Steven Allen committed
67 68 69 70 71 72 73 74
	go func() {
		defer func() {
			list.Close()
			s.listeners.Lock()
			delete(s.listeners.m, list)
			s.listeners.Unlock()
			s.refs.Done()
		}()
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
75
		for {
Steven Allen's avatar
Steven Allen committed
76 77
			c, err := list.Accept()
			if err != nil {
Matt Joiner's avatar
Matt Joiner committed
78 79 80
				if s.ctx.Err() == nil {
					log.Errorf("swarm listener accept error: %s", err)
				}
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
81 82
				return
			}
Steven Allen's avatar
Steven Allen committed
83 84 85 86
			log.Debugf("swarm listener accepted connection: %s", c)
			s.refs.Add(1)
			go func() {
				defer s.refs.Done()
87
				_, err := s.addConn(c, network.DirInbound)
Steven Allen's avatar
Steven Allen committed
88 89 90 91 92 93
				if err != nil {
					// Probably just means that the swarm has been closed.
					log.Warningf("add conn failed: ", err)
					return
				}
			}()
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
94
		}
Steven Allen's avatar
Steven Allen committed
95
	}()
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
96 97
	return nil
}