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 (
|
2018-05-16 19:47:59 +01:00
|
|
|
"context"
|
|
|
|
"flag"
|
|
|
|
"fmt"
|
|
|
|
"net"
|
|
|
|
|
|
|
|
"go.uber.org/zap"
|
2018-04-12 14:50:22 +01:00
|
|
|
"google.golang.org/grpc"
|
2018-05-16 19:47:59 +01:00
|
|
|
monkit "gopkg.in/spacemonkeygo/monkit.v2"
|
|
|
|
|
|
|
|
"storj.io/storj/pkg/kademlia"
|
|
|
|
proto "storj.io/storj/protos/overlay"
|
|
|
|
"storj.io/storj/storage/redis"
|
|
|
|
)
|
2018-04-23 16:54:22 +01:00
|
|
|
|
2018-05-16 19:47:59 +01:00
|
|
|
var (
|
|
|
|
redisAddress string
|
|
|
|
redisPassword string
|
|
|
|
db int
|
2018-04-12 14:50:22 +01:00
|
|
|
)
|
|
|
|
|
2018-05-16 19:47:59 +01:00
|
|
|
func init() {
|
|
|
|
flag.StringVar(&redisAddress, "cache", "", "The <IP:PORT> string to use for connection to a redis cache")
|
|
|
|
flag.StringVar(&redisPassword, "password", "", "The password used for authentication to a secured redis instance")
|
|
|
|
flag.IntVar(&db, "db", 0, "The network cache database")
|
|
|
|
}
|
|
|
|
|
2018-04-23 16:54:22 +01:00
|
|
|
// NewServer creates a new Overlay Service Server
|
|
|
|
func NewServer() *grpc.Server {
|
|
|
|
|
2018-04-12 14:50:22 +01:00
|
|
|
grpcServer := grpc.NewServer()
|
2018-05-16 19:47:59 +01:00
|
|
|
proto.RegisterOverlayServer(grpcServer, &Overlay{})
|
2018-04-12 14:50:22 +01:00
|
|
|
|
2018-04-23 16:54:22 +01:00
|
|
|
return grpcServer
|
2018-04-12 14:50:22 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// NewClient connects to grpc server at the provided address with the provided options
|
|
|
|
// returns a new instance of an overlay Client
|
2018-05-16 19:47:59 +01:00
|
|
|
func NewClient(serverAddr *string, opts ...grpc.DialOption) (proto.OverlayClient, error) {
|
2018-04-12 14:50:22 +01:00
|
|
|
conn, err := grpc.Dial(*serverAddr, opts...)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2018-05-16 19:47:59 +01:00
|
|
|
return proto.NewOverlayClient(conn), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Service contains all methods needed to implement the process.Service interface
|
|
|
|
type Service struct {
|
|
|
|
logger *zap.Logger
|
|
|
|
metrics *monkit.Registry
|
|
|
|
}
|
|
|
|
|
|
|
|
// Process is the main function that executes the service
|
|
|
|
func (s *Service) Process(ctx context.Context) error {
|
|
|
|
// bootstrap network
|
|
|
|
kad := kademlia.Kademlia{}
|
|
|
|
|
|
|
|
kad.Bootstrap(ctx)
|
|
|
|
// bootstrap cache
|
2018-06-05 12:48:19 +01:00
|
|
|
cache, err := redis.NewOverlayClient(redisAddress, redisPassword, db, &kad)
|
2018-05-16 19:47:59 +01:00
|
|
|
if err != nil {
|
|
|
|
s.logger.Error("Failed to create a new overlay client", zap.Error(err))
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if err := cache.Bootstrap(ctx); err != nil {
|
|
|
|
s.logger.Error("Failed to boostrap cache", zap.Error(err))
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// send off cache refreshes concurrently
|
|
|
|
go cache.Refresh(ctx)
|
|
|
|
|
|
|
|
lis, err := net.Listen("tcp", fmt.Sprintf(":%d", 0))
|
|
|
|
if err != nil {
|
|
|
|
s.logger.Error("Failed to initialize TCP connection", zap.Error(err))
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
grpcServer := grpc.NewServer()
|
|
|
|
proto.RegisterOverlayServer(grpcServer, &Overlay{})
|
|
|
|
|
|
|
|
defer grpcServer.GracefulStop()
|
|
|
|
return grpcServer.Serve(lis)
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
// SetLogger adds the initialized logger to the Service
|
|
|
|
func (s *Service) SetLogger(l *zap.Logger) error {
|
|
|
|
s.logger = l
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// SetMetricHandler adds the initialized metric handler to the Service
|
|
|
|
func (s *Service) SetMetricHandler(m *monkit.Registry) error {
|
|
|
|
s.metrics = m
|
|
|
|
return nil
|
2018-04-12 14:50:22 +01:00
|
|
|
}
|
2018-05-30 15:03:44 +01:00
|
|
|
|
|
|
|
// InstanceID implements Service.InstanceID
|
|
|
|
func (s *Service) InstanceID() string { return "" }
|