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

import (
	"bytes"
	"fmt"
6
	"io"
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
7
	"path"
8
	"sort"
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
9 10

	cmds "github.com/jbenet/go-ipfs/commands"
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
11
	peer "github.com/jbenet/go-ipfs/p2p/peer"
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
12
	errors "github.com/jbenet/go-ipfs/util/debugerror"
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
13 14 15

	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"
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
16 17 18 19 20 21
)

type stringList struct {
	Strings []string
}

22
var SwarmCmd = &cmds.Command{
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
23 24 25 26 27 28 29 30 31 32 33 34 35
	Helptext: cmds.HelpText{
		Tagline: "swarm inspection tool",
		Synopsis: `
ipfs swarm peers             - List peers with open connections
ipfs swarm connect <address> - Open connection to a given peer
`,
		ShortDescription: `
ipfs swarm is a tool to manipulate the network swarm. The swarm is the
component that opens, listens for, and maintains connections to other
ipfs peers in the internet.
`,
	},
	Subcommands: map[string]*cmds.Command{
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
36 37
		"peers":   swarmPeersCmd,
		"connect": swarmConnectCmd,
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55
	},
}

var swarmPeersCmd = &cmds.Command{
	Helptext: cmds.HelpText{
		Tagline: "List peers with open connections",
		ShortDescription: `
ipfs swarm peers lists the set of peers this node is connected to.
`,
	},
	Run: func(req cmds.Request) (interface{}, error) {

		log.Debug("ipfs swarm peers")
		n, err := req.Context().GetNode()
		if err != nil {
			return nil, err
		}

Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
56
		if n.PeerHost == nil {
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
57 58 59
			return nil, errNotOnline
		}

Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
60
		conns := n.PeerHost.Network().Conns()
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
61 62
		addrs := make([]string, len(conns))
		for i, c := range conns {
63
			pid := c.RemotePeer()
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
64
			addr := c.RemoteMultiaddr()
65
			addrs[i] = fmt.Sprintf("%s/%s", addr, pid.Pretty())
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
66 67
		}

68
		sort.Sort(sort.StringSlice(addrs))
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
69 70 71 72 73
		return &stringList{addrs}, nil
	},
	Marshalers: cmds.MarshalerMap{
		cmds.Text: stringListMarshaler,
	},
74
	Type: stringList{},
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
75 76
}

Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
77 78 79 80 81 82 83 84 85 86 87
var swarmConnectCmd = &cmds.Command{
	Helptext: cmds.HelpText{
		Tagline: "Open connection to a given peer",
		ShortDescription: `
'ipfs swarm connect' opens a connection to a peer address. The address format
is an ipfs multiaddr:

ipfs swarm connect /ip4/104.131.131.82/tcp/4001/QmaCpDMGvV2BGHeYERUEnRQAwe3N8SzbUtfsmvsqQLuvuJ
`,
	},
	Arguments: []cmds.Argument{
88
		cmds.StringArg("address", true, true, "address of peer to connect to").EnableStdin(),
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
89 90 91 92 93 94 95 96 97 98
	},
	Run: func(req cmds.Request) (interface{}, error) {
		ctx := context.TODO()

		log.Debug("ipfs swarm connect")
		n, err := req.Context().GetNode()
		if err != nil {
			return nil, err
		}

99
		addrs := req.Arguments()
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
100

Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
101
		if n.PeerHost == nil {
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
102 103 104 105 106 107 108 109 110 111
			return nil, errNotOnline
		}

		peers, err := peersWithAddresses(n.Peerstore, addrs)
		if err != nil {
			return nil, err
		}

		output := make([]string, len(peers))
		for i, p := range peers {
112
			output[i] = "connect " + p.Pretty()
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
113

Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
114
			err := n.PeerHost.Connect(ctx, peer.PeerInfo{ID: p})
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
115 116 117 118 119 120 121 122 123 124 125 126
			if err != nil {
				output[i] += " failure: " + err.Error()
			} else {
				output[i] += " success"
			}
		}

		return &stringList{output}, nil
	},
	Marshalers: cmds.MarshalerMap{
		cmds.Text: stringListMarshaler,
	},
127
	Type: stringList{},
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
128 129
}

130
func stringListMarshaler(res cmds.Response) (io.Reader, error) {
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
131 132 133 134 135 136 137
	list, ok := res.Output().(*stringList)
	if !ok {
		return nil, errors.New("failed to cast []string")
	}

	var buf bytes.Buffer
	for _, s := range list.Strings {
138 139
		buf.WriteString(s)
		buf.WriteString("\n")
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
140
	}
141
	return &buf, nil
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
142
}
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
143 144 145 146 147 148 149 150 151 152 153 154

// splitAddresses is a function that takes in a slice of string peer addresses
// (multiaddr + peerid) and returns slices of multiaddrs and peerids.
func splitAddresses(addrs []string) (maddrs []ma.Multiaddr, pids []peer.ID, err error) {

	maddrs = make([]ma.Multiaddr, len(addrs))
	pids = make([]peer.ID, len(addrs))
	for i, addr := range addrs {
		a, err := ma.NewMultiaddr(path.Dir(addr))
		if err != nil {
			return nil, nil, cmds.ClientError("invalid peer address: " + err.Error())
		}
155
		id, err := peer.IDB58Decode(path.Base(addr))
156 157 158 159
		if err != nil {
			return nil, nil, err
		}
		pids[i] = id
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
160 161 162 163 164 165 166
		maddrs[i] = a
	}
	return
}

// peersWithAddresses is a function that takes in a slice of string peer addresses
// (multiaddr + peerid) and returns a slice of properly constructed peers
167
func peersWithAddresses(ps peer.Peerstore, addrs []string) ([]peer.ID, error) {
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
168 169 170 171 172
	maddrs, pids, err := splitAddresses(addrs)
	if err != nil {
		return nil, err
	}

173 174
	for i, p := range pids {
		ps.AddAddress(p, maddrs[i])
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
175
	}
176
	return pids, nil
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
177
}