2018-04-23 16:54:22 +01:00
|
|
|
// Copyright (C) 2018 Storj Labs, Inc.
|
|
|
|
// See LICENSE for copying information.
|
|
|
|
|
2018-04-12 14:50:22 +01:00
|
|
|
package overlay
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
2018-06-22 14:33:57 +01:00
|
|
|
"sync"
|
2018-04-12 14:50:22 +01:00
|
|
|
|
2018-06-05 22:06:37 +01:00
|
|
|
"go.uber.org/zap"
|
2018-06-13 19:22:32 +01:00
|
|
|
"gopkg.in/spacemonkeygo/monkit.v2"
|
2018-06-22 14:33:57 +01:00
|
|
|
"storj.io/storj/pkg/dht"
|
|
|
|
|
2018-04-23 16:54:22 +01:00
|
|
|
proto "storj.io/storj/protos/overlay" // naming proto to avoid confusion with this package
|
2018-06-22 14:33:57 +01:00
|
|
|
"storj.io/storj/storage"
|
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
|
|
|
maxNodes = 40
|
2018-04-12 14:50:22 +01:00
|
|
|
)
|
|
|
|
|
2018-06-19 15:00:15 +01:00
|
|
|
// Server implements our overlay RPC service
|
|
|
|
type Server struct {
|
2018-06-22 14:33:57 +01:00
|
|
|
dht dht.DHT
|
2018-06-13 19:22:32 +01:00
|
|
|
cache *Cache
|
2018-06-05 22:06:37 +01:00
|
|
|
logger *zap.Logger
|
|
|
|
metrics *monkit.Registry
|
|
|
|
}
|
2018-04-12 14:50:22 +01:00
|
|
|
|
|
|
|
// Lookup finds the address of a node in our overlay network
|
2018-06-19 15:00:15 +01:00
|
|
|
func (o *Server) Lookup(ctx context.Context, req *proto.LookupRequest) (*proto.LookupResponse, error) {
|
2018-06-13 19:22:32 +01:00
|
|
|
na, err := o.cache.Get(ctx, req.NodeID)
|
2018-06-19 15:00:15 +01:00
|
|
|
|
2018-06-05 22:06:37 +01:00
|
|
|
if err != nil {
|
|
|
|
o.logger.Error("Error looking up node", zap.Error(err), zap.String("nodeID", req.NodeID))
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return &proto.LookupResponse{
|
2018-06-19 15:00:15 +01:00
|
|
|
Node: &proto.Node{
|
|
|
|
Id: req.GetNodeID(),
|
|
|
|
Address: na,
|
|
|
|
},
|
2018-06-05 22:06:37 +01:00
|
|
|
}, nil
|
2018-04-12 14:50:22 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// FindStorageNodes searches the overlay network for nodes that meet the provided requirements
|
2018-06-19 15:00:15 +01:00
|
|
|
func (o *Server) FindStorageNodes(ctx context.Context, req *proto.FindStorageNodesRequest) (*proto.FindStorageNodesResponse, error) {
|
2018-06-05 22:06:37 +01:00
|
|
|
// NB: call FilterNodeReputation from node_reputation package to find nodes for storage
|
2018-06-29 21:06:25 +01:00
|
|
|
keys, err := o.cache.DB.List(nil, storage.Limit(10))
|
2018-06-05 22:06:37 +01:00
|
|
|
if err != nil {
|
2018-06-22 14:33:57 +01:00
|
|
|
o.logger.Error("Error listing nodes", zap.Error(err))
|
2018-06-05 22:06:37 +01:00
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2018-06-22 14:33:57 +01:00
|
|
|
if len(keys) > maxNodes {
|
|
|
|
// TODO(coyle): determine if this is a set value or they user of the api will specify
|
|
|
|
keys = keys[:maxNodes]
|
|
|
|
}
|
|
|
|
|
|
|
|
nodes := o.getNodes(ctx, keys)
|
|
|
|
|
2018-06-05 22:06:37 +01:00
|
|
|
return &proto.FindStorageNodesResponse{
|
2018-06-19 15:00:15 +01:00
|
|
|
Nodes: nodes,
|
2018-06-05 22:06:37 +01:00
|
|
|
}, nil
|
2018-04-12 14:50:22 +01:00
|
|
|
}
|
2018-06-22 14:33:57 +01:00
|
|
|
|
|
|
|
func (o *Server) getNodes(ctx context.Context, keys storage.Keys) []*proto.Node {
|
|
|
|
wg := &sync.WaitGroup{}
|
|
|
|
ch := make(chan *proto.Node, len(keys))
|
|
|
|
|
|
|
|
wg.Add(len(keys))
|
|
|
|
for _, v := range keys {
|
|
|
|
go func(ch chan *proto.Node, id string) {
|
|
|
|
|
|
|
|
defer wg.Done()
|
|
|
|
na, err := o.cache.Get(ctx, id)
|
|
|
|
if err != nil {
|
|
|
|
o.logger.Error("failed to get key from cache", zap.Error(err))
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
ch <- &proto.Node{Id: id, Address: na}
|
|
|
|
}(ch, v.String())
|
|
|
|
}
|
|
|
|
|
|
|
|
wg.Wait()
|
|
|
|
close(ch)
|
|
|
|
nodes := []*proto.Node{}
|
|
|
|
for node := range ch {
|
|
|
|
nodes = append(nodes, node)
|
|
|
|
}
|
|
|
|
|
|
|
|
return nodes
|
|
|
|
}
|