package downloader import ( "context" "encoding/binary" "fmt" "log" "time" "google.golang.org/grpc" pb "github.com/harmony-one/harmony/api/services/syncing/downloader/proto" ) // Client is the client model for downloader package. type Client struct { dlClient pb.DownloaderClient opts []grpc.DialOption conn *grpc.ClientConn } // ClientSetup setups a Client given ip and port. func ClientSetup(ip, port string) *Client { client := Client{} client.opts = append(client.opts, grpc.WithInsecure()) var err error client.conn, err = grpc.Dial(fmt.Sprintf("%s:%s", ip, port), client.opts...) if err != nil { log.Printf("client.go:ClientSetup fail to dial: %v", err) return nil } client.dlClient = pb.NewDownloaderClient(client.conn) return &client } // Close closes the Client. func (client *Client) Close() { err := client.conn.Close() if err != nil { log.Printf("unable to close connection %v with error %v ", client.conn, err) } } // GetBlockHashes gets block hashes from all the peers by calling grpc request. func (client *Client) GetBlockHashes(startHash []byte) *pb.DownloaderResponse { ctx, cancel := context.WithTimeout(context.Background(), 10*time.Second) defer cancel() request := &pb.DownloaderRequest{Type: pb.DownloaderRequest_HEADER, BlockHash: startHash} response, err := client.dlClient.Query(ctx, request) if err != nil { log.Printf("[SYNC] GetBlockHashes query failed with error %v", err) } return response } // GetBlocks gets blocks in serialization byte array by calling a grpc request. func (client *Client) GetBlocks(hashes [][]byte) *pb.DownloaderResponse { ctx, cancel := context.WithTimeout(context.Background(), 10*time.Second) defer cancel() request := &pb.DownloaderRequest{Type: pb.DownloaderRequest_BLOCK} request.Hashes = make([][]byte, len(hashes)) for i := range hashes { request.Hashes[i] = make([]byte, len(hashes[i])) copy(request.Hashes[i], hashes[i]) } response, err := client.dlClient.Query(ctx, request) if err != nil { log.Printf("[SYNC] downloader/client.go:GetBlocks query failed with error %v", err) } return response } // Register will register node's ip/port information to peers receive newly created blocks in future // hash is the bytes of "ip:port" string representation func (client *Client) Register(hash []byte) *pb.DownloaderResponse { ctx, cancel := context.WithTimeout(context.Background(), 3*time.Second) defer cancel() request := &pb.DownloaderRequest{Type: pb.DownloaderRequest_REGISTER} request.PeerHash = make([]byte, len(hash)) copy(request.PeerHash, hash) response, err := client.dlClient.Query(ctx, request) if err != nil { log.Printf("[SYNC] client.go:Register failed with code %v", err) } return response } // PushNewBlock will send the lastest verified blow to registered nodes func (client *Client) PushNewBlock(peerID uint32, blockHash []byte, timeout bool) *pb.DownloaderResponse { ctx, cancel := context.WithTimeout(context.Background(), 10*time.Second) defer cancel() peerHash := make([]byte, 4) binary.BigEndian.PutUint32(peerHash, peerID) request := &pb.DownloaderRequest{Type: pb.DownloaderRequest_NEWBLOCK} request.BlockHash = make([]byte, len(blockHash)) copy(request.BlockHash, blockHash) request.PeerHash = make([]byte, len(peerHash)) copy(request.PeerHash, peerHash) if timeout { request.Type = pb.DownloaderRequest_REGISTERTIMEOUT } response, err := client.dlClient.Query(ctx, request) if err != nil { log.Printf("[SYNC] unable to send new block to unsync node with error: %v", err) } return response }