handlers.go 5.77 KB
Newer Older
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
1 2 3 4 5 6 7 8
package dht

import (
	"errors"
	"fmt"
	"time"

	peer "github.com/jbenet/go-ipfs/peer"
9
	pb "github.com/jbenet/go-ipfs/routing/dht/pb"
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
10 11
	u "github.com/jbenet/go-ipfs/util"

12
	ds "github.com/jbenet/go-ipfs/Godeps/_workspace/src/github.com/jbenet/go-datastore"
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
13 14
)

Jeromy's avatar
Jeromy committed
15
// The number of closer peers to send on requests.
16 17
var CloserPeerCount = 4

Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
18
// dhthandler specifies the signature of functions that handle DHT messages.
19
type dhtHandler func(peer.Peer, *pb.Message) (*pb.Message, error)
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
20

21
func (dht *IpfsDHT) handlerForMsgType(t pb.Message_MessageType) dhtHandler {
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
22
	switch t {
23
	case pb.Message_GET_VALUE:
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
24
		return dht.handleGetValue
25
	case pb.Message_PUT_VALUE:
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
26
		return dht.handlePutValue
27
	case pb.Message_FIND_NODE:
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
28
		return dht.handleFindPeer
29
	case pb.Message_ADD_PROVIDER:
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
30
		return dht.handleAddProvider
31
	case pb.Message_GET_PROVIDERS:
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
32
		return dht.handleGetProviders
33
	case pb.Message_PING:
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
34 35 36 37 38 39
		return dht.handlePing
	default:
		return nil
	}
}

40
func (dht *IpfsDHT) handleGetValue(p peer.Peer, pmes *pb.Message) (*pb.Message, error) {
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
41
	log.Debugf("%s handleGetValue for key: %s\n", dht.self, pmes.GetKey())
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
42 43

	// setup response
44
	resp := pb.NewMessage(pmes.GetType(), pmes.GetKey(), pmes.GetClusterLevel())
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
45 46 47 48 49 50 51 52

	// first, is the key even a key?
	key := pmes.GetKey()
	if key == "" {
		return nil, errors.New("handleGetValue but no key was provided")
	}

	// let's first check if we have the value locally.
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
53
	log.Debugf("%s handleGetValue looking into ds", dht.self)
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
54
	dskey := u.Key(pmes.GetKey()).DsKey()
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
55
	iVal, err := dht.datastore.Get(dskey)
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
56
	log.Debugf("%s handleGetValue looking into ds GOT %v", dht.self, iVal)
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
57 58

	// if we got an unexpected error, bail.
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
59
	if err != nil && err != ds.ErrNotFound {
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
60 61 62
		return nil, err
	}

63 64 65 66
	// Note: changed the behavior here to return _as much_ info as possible
	// (potentially all of {value, closer peers, provider})

	// if we have the value, send it back
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
67
	if err == nil {
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
68
		log.Debugf("%s handleGetValue success!", dht.self)
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
69 70 71 72 73 74 75 76 77 78 79 80

		byts, ok := iVal.([]byte)
		if !ok {
			return nil, fmt.Errorf("datastore had non byte-slice value for %v", dskey)
		}

		resp.Value = byts
	}

	// if we know any providers for the requested value, return those.
	provs := dht.providers.GetProviders(u.Key(pmes.GetKey()))
	if len(provs) > 0 {
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
81
		log.Debugf("handleGetValue returning %d provider[s]", len(provs))
82
		resp.ProviderPeers = pb.PeersToPBPeers(provs)
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
83 84 85
	}

	// Find closest peer on given cluster to desired key and reply with that info
86
	closer := dht.betterPeersToQuery(pmes, CloserPeerCount)
87
	if closer != nil {
88
		for _, p := range closer {
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
89
			log.Debugf("handleGetValue returning closer peer: '%s'", p)
Jeromy's avatar
Jeromy committed
90
			if len(p.Addresses()) < 1 {
Juan Batiz-Benet's avatar
notes  
Juan Batiz-Benet committed
91
				log.Critical("no addresses on peer being sent!")
Jeromy's avatar
Jeromy committed
92
			}
93
		}
94
		resp.CloserPeers = pb.PeersToPBPeers(closer)
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
95 96 97 98 99 100
	}

	return resp, nil
}

// Store a value in this peer local storage
101
func (dht *IpfsDHT) handlePutValue(p peer.Peer, pmes *pb.Message) (*pb.Message, error) {
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
102 103
	dht.dslock.Lock()
	defer dht.dslock.Unlock()
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
104
	dskey := u.Key(pmes.GetKey()).DsKey()
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
105
	err := dht.datastore.Put(dskey, pmes.GetValue())
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
106
	log.Debugf("%s handlePutValue %v %v\n", dht.self, dskey, pmes.GetValue())
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
107
	return pmes, err
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
108 109
}

110
func (dht *IpfsDHT) handlePing(p peer.Peer, pmes *pb.Message) (*pb.Message, error) {
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
111
	log.Debugf("%s Responding to ping from %s!\n", dht.self, p)
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
112
	return pmes, nil
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
113 114
}

115 116
func (dht *IpfsDHT) handleFindPeer(p peer.Peer, pmes *pb.Message) (*pb.Message, error) {
	resp := pb.NewMessage(pmes.GetType(), "", pmes.GetClusterLevel())
117
	var closest []peer.Peer
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
118 119

	// if looking for self... special case where we send it on CloserPeers.
120 121
	if peer.ID(pmes.GetKey()).Equal(dht.self.ID()) {
		closest = []peer.Peer{dht.self}
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
122
	} else {
123
		closest = dht.betterPeersToQuery(pmes, CloserPeerCount)
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
124 125 126
	}

	if closest == nil {
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
127
		log.Errorf("handleFindPeer: could not find anything.")
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
128 129 130
		return resp, nil
	}

131
	var withAddresses []peer.Peer
132
	for _, p := range closest {
133
		if len(p.Addresses()) > 0 {
134 135
			withAddresses = append(withAddresses, p)
		}
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
136 137
	}

138
	for _, p := range withAddresses {
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
139
		log.Debugf("handleFindPeer: sending back '%s'", p)
140
	}
141
	resp.CloserPeers = pb.PeersToPBPeers(withAddresses)
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
142 143 144
	return resp, nil
}

145 146
func (dht *IpfsDHT) handleGetProviders(p peer.Peer, pmes *pb.Message) (*pb.Message, error) {
	resp := pb.NewMessage(pmes.GetType(), pmes.GetKey(), pmes.GetClusterLevel())
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
147 148

	// check if we have this value, to add ourselves as provider.
Jeromy's avatar
Jeromy committed
149
	log.Debugf("handling GetProviders: '%s'", u.Key(pmes.GetKey()))
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
150 151
	dsk := u.Key(pmes.GetKey()).DsKey()
	has, err := dht.datastore.Has(dsk)
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
152
	if err != nil && err != ds.ErrNotFound {
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
153
		log.Errorf("unexpected datastore error: %v\n", err)
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
154 155 156 157 158 159 160 161 162 163 164
		has = false
	}

	// setup providers
	providers := dht.providers.GetProviders(u.Key(pmes.GetKey()))
	if has {
		providers = append(providers, dht.self)
	}

	// if we've got providers, send thos those.
	if providers != nil && len(providers) > 0 {
165
		resp.ProviderPeers = pb.PeersToPBPeers(providers)
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
166 167 168
	}

	// Also send closer peers.
169
	closer := dht.betterPeersToQuery(pmes, CloserPeerCount)
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
170
	if closer != nil {
171
		resp.CloserPeers = pb.PeersToPBPeers(closer)
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
172 173 174 175 176 177 178
	}

	return resp, nil
}

type providerInfo struct {
	Creation time.Time
179
	Value    peer.Peer
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
180 181
}

182
func (dht *IpfsDHT) handleAddProvider(p peer.Peer, pmes *pb.Message) (*pb.Message, error) {
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
183
	key := u.Key(pmes.GetKey())
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
184

Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
185
	log.Debugf("%s adding %s as a provider for '%s'\n", dht.self, p, peer.ID(key))
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
186

187
	// add provider should use the address given in the message
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
188 189
	for _, pb := range pmes.GetProviderPeers() {
		pid := peer.ID(pb.GetId())
190
		if pid.Equal(p.ID()) {
191 192 193

			addr, err := pb.Address()
			if err != nil {
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
194
				log.Errorf("provider %s error with address %s", p, *pb.Addr)
195 196
				continue
			}
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
197

Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
198
			log.Infof("received provider %s %s for %s", p, addr, key)
199
			p.AddAddress(addr)
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
200 201 202
			dht.providers.AddProvider(key, p)

		} else {
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
203
			log.Errorf("handleAddProvider received provider %s from %s", pid, p)
204 205 206
		}
	}

Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
207
	return pmes, nil // send back same msg as confirmation.
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
208
}