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

import (
	"fmt"
5
	"time"
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
6

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

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

Steven Allen's avatar
Steven Allen committed
12 13 14
// 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
15 16 17
	errs := make([]error, len(addrs))
	var succeeded int
	for i, a := range addrs {
18
		if err := s.AddListenAddr(a); err != nil {
Jeromy's avatar
Jeromy committed
19
			errs[i] = err
20 21
		} else {
			succeeded++
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
22 23 24
		}
	}

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

Jeromy's avatar
Jeromy committed
31 32 33 34
	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
35 36 37
	return nil
}

Steven Allen's avatar
Steven Allen committed
38 39 40 41 42 43
// 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
44
	}
Jeromy's avatar
Jeromy committed
45

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

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

Jeromy's avatar
Jeromy committed
62 63
	maddr := list.Multiaddr()

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

Steven Allen's avatar
Steven Allen committed
69 70 71 72 73
	go func() {
		defer func() {
			list.Close()
			s.listeners.Lock()
			delete(s.listeners.m, list)
74
			s.listeners.cacheEOL = time.Time{}
Steven Allen's avatar
Steven Allen committed
75 76 77
			s.listeners.Unlock()
			s.refs.Done()
		}()
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
78
		for {
Steven Allen's avatar
Steven Allen committed
79 80
			c, err := list.Accept()
			if err != nil {
Matt Joiner's avatar
Matt Joiner committed
81 82 83
				if s.ctx.Err() == nil {
					log.Errorf("swarm listener accept error: %s", err)
				}
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
84 85
				return
			}
Steven Allen's avatar
Steven Allen committed
86 87 88 89
			log.Debugf("swarm listener accepted connection: %s", c)
			s.refs.Add(1)
			go func() {
				defer s.refs.Done()
90
				_, err := s.addConn(c, network.DirInbound)
Steven Allen's avatar
Steven Allen committed
91 92 93 94 95 96
				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
97
		}
Steven Allen's avatar
Steven Allen committed
98
	}()
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
99 100
	return nil
}