mirror of
https://github.com/ipfs/kubo.git
synced 2025-10-24 07:03:53 +08:00
2
Godeps/Godeps.json
generated
2
Godeps/Godeps.json
generated
@ -196,7 +196,7 @@
|
|||||||
},
|
},
|
||||||
{
|
{
|
||||||
"ImportPath": "github.com/jbenet/go-peerstream",
|
"ImportPath": "github.com/jbenet/go-peerstream",
|
||||||
"Rev": "cfdc29a19c1a209d548670f5c33c5cda2e040143"
|
"Rev": "f90119e97e8be7b2bdd5e598067b0dc44df63381"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"ImportPath": "github.com/jbenet/go-random",
|
"ImportPath": "github.com/jbenet/go-random",
|
||||||
|
11
Godeps/_workspace/src/github.com/jbenet/go-peerstream/conn.go
generated
vendored
11
Godeps/_workspace/src/github.com/jbenet/go-peerstream/conn.go
generated
vendored
@ -44,6 +44,9 @@ type Conn struct {
|
|||||||
|
|
||||||
streams map[*Stream]struct{}
|
streams map[*Stream]struct{}
|
||||||
streamLock sync.RWMutex
|
streamLock sync.RWMutex
|
||||||
|
|
||||||
|
closed bool
|
||||||
|
closeLock sync.Mutex
|
||||||
}
|
}
|
||||||
|
|
||||||
func newConn(nconn net.Conn, tconn smux.Conn, s *Swarm) *Conn {
|
func newConn(nconn net.Conn, tconn smux.Conn, s *Swarm) *Conn {
|
||||||
@ -114,6 +117,14 @@ func (c *Conn) Streams() []*Stream {
|
|||||||
|
|
||||||
// Close closes this connection
|
// Close closes this connection
|
||||||
func (c *Conn) Close() error {
|
func (c *Conn) Close() error {
|
||||||
|
c.closeLock.Lock()
|
||||||
|
defer c.closeLock.Unlock()
|
||||||
|
|
||||||
|
if c.closed {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
c.closed = true
|
||||||
|
|
||||||
// close streams
|
// close streams
|
||||||
streams := c.Streams()
|
streams := c.Streams()
|
||||||
for _, s := range streams {
|
for _, s := range streams {
|
||||||
|
@ -56,6 +56,8 @@ type msgQueue struct {
|
|||||||
out bsmsg.BitSwapMessage
|
out bsmsg.BitSwapMessage
|
||||||
network bsnet.BitSwapNetwork
|
network bsnet.BitSwapNetwork
|
||||||
|
|
||||||
|
refcnt int
|
||||||
|
|
||||||
work chan struct{}
|
work chan struct{}
|
||||||
done chan struct{}
|
done chan struct{}
|
||||||
}
|
}
|
||||||
@ -101,13 +103,13 @@ func (pm *WantManager) SendBlock(ctx context.Context, env *engine.Envelope) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (pm *WantManager) startPeerHandler(p peer.ID) *msgQueue {
|
func (pm *WantManager) startPeerHandler(p peer.ID) *msgQueue {
|
||||||
_, ok := pm.peers[p]
|
mq, ok := pm.peers[p]
|
||||||
if ok {
|
if ok {
|
||||||
// TODO: log an error?
|
mq.refcnt++
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
mq := pm.newMsgQueue(p)
|
mq = pm.newMsgQueue(p)
|
||||||
|
|
||||||
// new peer, we will want to give them our full wantlist
|
// new peer, we will want to give them our full wantlist
|
||||||
fullwantlist := bsmsg.New(true)
|
fullwantlist := bsmsg.New(true)
|
||||||
@ -129,6 +131,11 @@ func (pm *WantManager) stopPeerHandler(p peer.ID) {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pq.refcnt--
|
||||||
|
if pq.refcnt > 0 {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
close(pq.done)
|
close(pq.done)
|
||||||
delete(pm.peers, p)
|
delete(pm.peers, p)
|
||||||
}
|
}
|
||||||
@ -247,6 +254,7 @@ func (wm *WantManager) newMsgQueue(p peer.ID) *msgQueue {
|
|||||||
mq.work = make(chan struct{}, 1)
|
mq.work = make(chan struct{}, 1)
|
||||||
mq.network = wm.network
|
mq.network = wm.network
|
||||||
mq.p = p
|
mq.p = p
|
||||||
|
mq.refcnt = 1
|
||||||
|
|
||||||
return mq
|
return mq
|
||||||
}
|
}
|
||||||
|
@ -73,13 +73,19 @@ type initCfg struct {
|
|||||||
Bootstrap string
|
Bootstrap string
|
||||||
PortStart int
|
PortStart int
|
||||||
Mdns bool
|
Mdns bool
|
||||||
|
Utp bool
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *initCfg) swarmAddrForPeer(i int) string {
|
func (c *initCfg) swarmAddrForPeer(i int) string {
|
||||||
if c.PortStart == 0 {
|
str := "/ip4/0.0.0.0/tcp/%d"
|
||||||
return "/ip4/0.0.0.0/tcp/0"
|
if c.Utp {
|
||||||
|
str = "/ip4/0.0.0.0/udp/%d/utp"
|
||||||
}
|
}
|
||||||
return fmt.Sprintf("/ip4/0.0.0.0/tcp/%d", c.PortStart+i)
|
|
||||||
|
if c.PortStart == 0 {
|
||||||
|
return fmt.Sprintf(str, 0)
|
||||||
|
}
|
||||||
|
return fmt.Sprintf(str, c.PortStart+i)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *initCfg) apiAddrForPeer(i int) string {
|
func (c *initCfg) apiAddrForPeer(i int) string {
|
||||||
@ -250,6 +256,20 @@ func IpfsKillAll() error {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func envForDaemon(n int) []string {
|
||||||
|
envs := os.Environ()
|
||||||
|
npath := "IPFS_PATH=" + IpfsDirN(n)
|
||||||
|
for i, e := range envs {
|
||||||
|
p := strings.Split(e, "=")
|
||||||
|
if p[0] == "IPFS_PATH" {
|
||||||
|
envs[i] = npath
|
||||||
|
return envs
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return append(envs, npath)
|
||||||
|
}
|
||||||
|
|
||||||
func IpfsStart(waitall bool) error {
|
func IpfsStart(waitall bool) error {
|
||||||
var addrs []string
|
var addrs []string
|
||||||
n := GetNumNodes()
|
n := GetNumNodes()
|
||||||
@ -257,7 +277,7 @@ func IpfsStart(waitall bool) error {
|
|||||||
dir := IpfsDirN(i)
|
dir := IpfsDirN(i)
|
||||||
cmd := exec.Command("ipfs", "daemon")
|
cmd := exec.Command("ipfs", "daemon")
|
||||||
cmd.Dir = dir
|
cmd.Dir = dir
|
||||||
cmd.Env = append(os.Environ(), "IPFS_PATH="+dir)
|
cmd.Env = envForDaemon(i)
|
||||||
|
|
||||||
cmd.SysProcAttr = &syscall.SysProcAttr{Setsid: true}
|
cmd.SysProcAttr = &syscall.SysProcAttr{Setsid: true}
|
||||||
|
|
||||||
@ -441,6 +461,11 @@ func IpfsShell(n int) error {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func ConnectNodes(from, to int) error {
|
func ConnectNodes(from, to int) error {
|
||||||
|
if from == to {
|
||||||
|
// skip connecting to self..
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
fmt.Printf("connecting %d -> %d\n", from, to)
|
||||||
cmd := exec.Command("ipfs", "id", "-f", "<addrs>")
|
cmd := exec.Command("ipfs", "id", "-f", "<addrs>")
|
||||||
cmd.Env = []string{"IPFS_PATH=" + IpfsDirN(to)}
|
cmd.Env = []string{"IPFS_PATH=" + IpfsDirN(to)}
|
||||||
out, err := cmd.Output()
|
out, err := cmd.Output()
|
||||||
@ -449,7 +474,6 @@ func ConnectNodes(from, to int) error {
|
|||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
addr := strings.Split(string(out), "\n")[0]
|
addr := strings.Split(string(out), "\n")[0]
|
||||||
fmt.Println("ADDR: ", addr)
|
|
||||||
|
|
||||||
connectcmd := exec.Command("ipfs", "swarm", "connect", addr)
|
connectcmd := exec.Command("ipfs", "swarm", "connect", addr)
|
||||||
connectcmd.Env = []string{"IPFS_PATH=" + IpfsDirN(from)}
|
connectcmd.Env = []string{"IPFS_PATH=" + IpfsDirN(from)}
|
||||||
@ -461,6 +485,55 @@ func ConnectNodes(from, to int) error {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func parseRange(s string) ([]int, error) {
|
||||||
|
if strings.HasPrefix(s, "[") && strings.HasSuffix(s, "]") {
|
||||||
|
ranges := strings.Split(s[1:len(s)-1], ",")
|
||||||
|
var out []int
|
||||||
|
for _, r := range ranges {
|
||||||
|
rng, err := expandDashRange(r)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
out = append(out, rng...)
|
||||||
|
}
|
||||||
|
return out, nil
|
||||||
|
} else {
|
||||||
|
i, err := strconv.Atoi(s)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
return []int{i}, nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func expandDashRange(s string) ([]int, error) {
|
||||||
|
parts := strings.Split(s, "-")
|
||||||
|
if len(parts) == 0 {
|
||||||
|
i, err := strconv.Atoi(s)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return []int{i}, nil
|
||||||
|
}
|
||||||
|
low, err := strconv.Atoi(parts[0])
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
hi, err := strconv.Atoi(parts[1])
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
var out []int
|
||||||
|
for i := low; i <= hi; i++ {
|
||||||
|
out = append(out, i)
|
||||||
|
}
|
||||||
|
return out, nil
|
||||||
|
}
|
||||||
|
|
||||||
func GetAttr(attr string, node int) (string, error) {
|
func GetAttr(attr string, node int) (string, error) {
|
||||||
switch attr {
|
switch attr {
|
||||||
case "id":
|
case "id":
|
||||||
@ -518,9 +591,10 @@ func main() {
|
|||||||
cfg := new(initCfg)
|
cfg := new(initCfg)
|
||||||
kingpin.Flag("n", "number of ipfs nodes to initialize").Short('n').IntVar(&cfg.Count)
|
kingpin.Flag("n", "number of ipfs nodes to initialize").Short('n').IntVar(&cfg.Count)
|
||||||
kingpin.Flag("port", "port to start allocations from").Default("4002").Short('p').IntVar(&cfg.PortStart)
|
kingpin.Flag("port", "port to start allocations from").Default("4002").Short('p').IntVar(&cfg.PortStart)
|
||||||
kingpin.Flag("f", "force initialization (overwrite existing configs)").BoolVar(&cfg.Force)
|
kingpin.Flag("force", "force initialization (overwrite existing configs)").Short('f').BoolVar(&cfg.Force)
|
||||||
kingpin.Flag("mdns", "turn on mdns for nodes").BoolVar(&cfg.Mdns)
|
kingpin.Flag("mdns", "turn on mdns for nodes").BoolVar(&cfg.Mdns)
|
||||||
kingpin.Flag("bootstrap", "select bootstrapping style for cluster").Default("star").StringVar(&cfg.Bootstrap)
|
kingpin.Flag("bootstrap", "select bootstrapping style for cluster").Default("star").StringVar(&cfg.Bootstrap)
|
||||||
|
kingpin.Flag("utp", "use utp for addresses").BoolVar(&cfg.Utp)
|
||||||
|
|
||||||
wait := kingpin.Flag("wait", "wait for nodes to come fully online before exiting").Bool()
|
wait := kingpin.Flag("wait", "wait for nodes to come fully online before exiting").Bool()
|
||||||
|
|
||||||
@ -576,22 +650,26 @@ func main() {
|
|||||||
os.Exit(1)
|
os.Exit(1)
|
||||||
}
|
}
|
||||||
|
|
||||||
from, err := strconv.Atoi(args[1])
|
from, err := parseRange(args[1])
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fmt.Printf("failed to parse: %s\n", err)
|
fmt.Printf("failed to parse: %s\n", err)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
to, err := strconv.Atoi(args[2])
|
to, err := parseRange(args[2])
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fmt.Printf("failed to parse: %s\n", err)
|
fmt.Printf("failed to parse: %s\n", err)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
err = ConnectNodes(from, to)
|
for _, f := range from {
|
||||||
if err != nil {
|
for _, t := range to {
|
||||||
fmt.Printf("failed to connect: %s\n", err)
|
err = ConnectNodes(f, t)
|
||||||
return
|
if err != nil {
|
||||||
|
fmt.Printf("failed to connect: %s\n", err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
case "get":
|
case "get":
|
||||||
|
81
test/sharness/t0130-multinode.sh
Executable file
81
test/sharness/t0130-multinode.sh
Executable file
@ -0,0 +1,81 @@
|
|||||||
|
#!/bin/sh
|
||||||
|
#
|
||||||
|
# Copyright (c) 2015 Jeromy Johnson
|
||||||
|
# MIT Licensed; see the LICENSE file in this repository.
|
||||||
|
#
|
||||||
|
|
||||||
|
test_description="Test multiple ipfs nodes"
|
||||||
|
|
||||||
|
. lib/test-lib.sh
|
||||||
|
|
||||||
|
export IPTB_ROOT="`pwd`/.iptb"
|
||||||
|
|
||||||
|
ipfsi() {
|
||||||
|
dir="$1"
|
||||||
|
shift
|
||||||
|
IPFS_PATH="$IPTB_ROOT/$dir" ipfs $@
|
||||||
|
}
|
||||||
|
|
||||||
|
check_has_connection() {
|
||||||
|
node=$1
|
||||||
|
ipfsi $node swarm peers | grep ipfs > /dev/null
|
||||||
|
}
|
||||||
|
|
||||||
|
startup_cluster() {
|
||||||
|
test_expect_success "start up nodes" '
|
||||||
|
iptb start
|
||||||
|
'
|
||||||
|
|
||||||
|
test_expect_success "connect nodes to eachother" '
|
||||||
|
iptb connect [1-4] 0
|
||||||
|
'
|
||||||
|
|
||||||
|
test_expect_success "nodes are connected" '
|
||||||
|
check_has_connection 0 &&
|
||||||
|
check_has_connection 1 &&
|
||||||
|
check_has_connection 2 &&
|
||||||
|
check_has_connection 3 &&
|
||||||
|
check_has_connection 4
|
||||||
|
'
|
||||||
|
}
|
||||||
|
|
||||||
|
check_file_fetch() {
|
||||||
|
node=$1
|
||||||
|
fhash=$2
|
||||||
|
fname=$3
|
||||||
|
|
||||||
|
test_expect_success "can fetch file" '
|
||||||
|
ipfsi $node cat $fhash > fetch_out
|
||||||
|
'
|
||||||
|
|
||||||
|
test_expect_success "file looks good" '
|
||||||
|
test_cmp $fname fetch_out
|
||||||
|
'
|
||||||
|
}
|
||||||
|
|
||||||
|
run_basic_test() {
|
||||||
|
startup_cluster
|
||||||
|
|
||||||
|
test_expect_success "add a file on node1" '
|
||||||
|
random 1000000 > filea &&
|
||||||
|
FILEA_HASH=$(ipfsi 1 add -q filea)
|
||||||
|
'
|
||||||
|
|
||||||
|
check_file_fetch 4 $FILEA_HASH filea
|
||||||
|
check_file_fetch 3 $FILEA_HASH filea
|
||||||
|
check_file_fetch 2 $FILEA_HASH filea
|
||||||
|
check_file_fetch 1 $FILEA_HASH filea
|
||||||
|
check_file_fetch 0 $FILEA_HASH filea
|
||||||
|
|
||||||
|
test_expect_success "shut down nodes" '
|
||||||
|
iptb stop
|
||||||
|
'
|
||||||
|
}
|
||||||
|
|
||||||
|
test_expect_success "set up tcp testbed" '
|
||||||
|
iptb init -n 5 -p 0 -f --bootstrap=none
|
||||||
|
'
|
||||||
|
|
||||||
|
run_basic_test
|
||||||
|
|
||||||
|
test_done
|
Reference in New Issue
Block a user