swarm.go 4.19 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"
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
8 9

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

	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
15 16 17 18 19 20
)

type stringList struct {
	Strings []string
}

21
var SwarmCmd = &cmds.Command{
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
22 23 24 25 26 27 28 29 30 31 32 33 34
	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
35 36
		"peers":   swarmPeersCmd,
		"connect": swarmConnectCmd,
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54
	},
}

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
55
		if n.PeerHost == nil {
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
56 57 58
			return nil, errNotOnline
		}

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

		return &stringList{addrs}, nil
	},
	Marshalers: cmds.MarshalerMap{
		cmds.Text: stringListMarshaler,
	},
	Type: &stringList{},
}

Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96
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{
		cmds.StringArg("address", true, true, "address of peer to connect to"),
	},
	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
		}

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

Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
99
		if n.PeerHost == nil {
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
100 101 102 103 104 105 106 107 108 109
			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 {
110
			output[i] = "connect " + p.Pretty()
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
111

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

		return &stringList{output}, nil
	},
	Marshalers: cmds.MarshalerMap{
		cmds.Text: stringListMarshaler,
	},
	Type: &stringList{},
}

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

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

// 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())
		}
153
		id, err := peer.IDB58Decode(path.Base(addr))
154 155 156 157
		if err != nil {
			return nil, nil, err
		}
		pids[i] = id
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
158 159 160 161 162 163 164
		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
165
func peersWithAddresses(ps peer.Peerstore, addrs []string) ([]peer.ID, error) {
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
166 167 168 169 170
	maddrs, pids, err := splitAddresses(addrs)
	if err != nil {
		return nil, err
	}

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