package reflector import ( "encoding/json" "net" "strconv" "github.com/lbryio/errors.go" log "github.com/sirupsen/logrus" ) type Client struct { conn net.Conn connected bool } func (c *Client) Connect(address string) error { var err error c.conn, err = net.Dial("tcp", address) if err != nil { return err } c.connected = true return c.doHandshake(protocolVersion1) } func (c *Client) Close() error { c.connected = false return c.conn.Close() } func (c *Client) SendBlob(blob []byte) error { if !c.connected { return errors.Err("not connected") } if len(blob) != maxBlobSize { return errors.Err("blob must be exactly " + strconv.Itoa(maxBlobSize) + " bytes") } blobHash := getBlobHash(blob) sendRequest, err := json.Marshal(sendBlobRequest{ BlobSize: len(blob), BlobHash: blobHash, }) if err != nil { return err } _, err = c.conn.Write(sendRequest) if err != nil { return err } dec := json.NewDecoder(c.conn) var sendResp sendBlobResponse err = dec.Decode(&sendResp) if err != nil { return err } if !sendResp.SendBlob { return ErrBlobExists } log.Println("Sending blob " + blobHash[:8]) _, err = c.conn.Write(blob) if err != nil { return err } var transferResp blobTransferResponse err = dec.Decode(&transferResp) if err != nil { return err } if !transferResp.ReceivedBlob { return errors.Err("server did not received blob") } return nil } func (c *Client) doHandshake(version int) error { if !c.connected { return errors.Err("not connected") } handshake, err := json.Marshal(handshakeRequestResponse{Version: version}) if err != nil { return err } _, err = c.conn.Write(handshake) if err != nil { return err } var resp handshakeRequestResponse dec := json.NewDecoder(c.conn) err = dec.Decode(&resp) if err != nil { return err } else if resp.Version != version { return errors.Err("handshake version mismatch") } return nil }