224 lines
4.5 KiB
Go
224 lines
4.5 KiB
Go
package reflector
|
|
|
|
import (
|
|
"bufio"
|
|
"encoding/json"
|
|
"io"
|
|
"net"
|
|
"strconv"
|
|
|
|
"github.com/lbryio/reflector.go/store"
|
|
|
|
"github.com/lbryio/errors.go"
|
|
|
|
log "github.com/sirupsen/logrus"
|
|
)
|
|
|
|
type Server struct {
|
|
store store.BlobStore
|
|
}
|
|
|
|
func NewServer(store store.BlobStore) *Server {
|
|
return &Server{
|
|
store: store,
|
|
}
|
|
}
|
|
|
|
func (s *Server) ListenAndServe(address string) error {
|
|
log.Println("Listening on " + address)
|
|
l, err := net.Listen("tcp", address)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
defer l.Close()
|
|
|
|
for {
|
|
conn, err := l.Accept()
|
|
if err != nil {
|
|
log.Error(err)
|
|
} else {
|
|
go s.handleConn(conn)
|
|
}
|
|
}
|
|
}
|
|
|
|
func (s *Server) handleConn(conn net.Conn) {
|
|
// TODO: connection should time out eventually
|
|
defer conn.Close()
|
|
|
|
err := s.doHandshake(conn)
|
|
if err != nil {
|
|
if err == io.EOF {
|
|
return
|
|
}
|
|
s.doError(conn, err)
|
|
return
|
|
}
|
|
|
|
for {
|
|
err = s.receiveBlob(conn)
|
|
if err != nil {
|
|
if err != io.EOF {
|
|
s.doError(conn, err)
|
|
}
|
|
return
|
|
}
|
|
}
|
|
}
|
|
|
|
func (s *Server) doError(conn net.Conn, err error) error {
|
|
log.Errorln(err)
|
|
if e2, ok := err.(*json.SyntaxError); ok {
|
|
log.Printf("syntax error at byte offset %d", e2.Offset)
|
|
}
|
|
resp, err := json.Marshal(errorResponse{Error: err.Error()})
|
|
if err != nil {
|
|
return err
|
|
}
|
|
_, err = conn.Write(resp)
|
|
return err
|
|
}
|
|
|
|
func (s *Server) receiveBlob(conn net.Conn) error {
|
|
blobSize, blobHash, isSdBlob, err := s.readBlobRequest(conn)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
blobExists := false
|
|
if !isSdBlob {
|
|
// we have to say sd blobs are missing because if we say we have it, they wont try to send any content blobs
|
|
has, err := s.store.Has(blobHash)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
blobExists = has
|
|
}
|
|
|
|
err = s.sendBlobResponse(conn, blobExists, isSdBlob)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if blobExists {
|
|
return nil
|
|
}
|
|
|
|
blob := make([]byte, blobSize)
|
|
_, err = io.ReadFull(bufio.NewReader(conn), blob)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
receivedBlobHash := getBlobHash(blob)
|
|
if blobHash != receivedBlobHash {
|
|
return errors.Err("hash of received blob data does not match hash from send request")
|
|
// this can also happen if the blob size is wrong, because the server will read the wrong number of bytes from the stream
|
|
}
|
|
|
|
log.Println("Got blob " + blobHash[:8])
|
|
|
|
if isSdBlob {
|
|
err = s.store.PutSD(blobHash, blob)
|
|
} else {
|
|
err = s.store.Put(blobHash, blob)
|
|
}
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
return s.sendTransferResponse(conn, true, isSdBlob)
|
|
}
|
|
|
|
func (s *Server) doHandshake(conn net.Conn) error {
|
|
var handshake handshakeRequestResponse
|
|
err := json.NewDecoder(conn).Decode(&handshake)
|
|
if err != nil {
|
|
return err
|
|
} else if handshake.Version != protocolVersion1 && handshake.Version != protocolVersion2 {
|
|
return errors.Err("protocol version not supported")
|
|
}
|
|
|
|
resp, err := json.Marshal(handshakeRequestResponse{Version: handshake.Version})
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
_, err = conn.Write(resp)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (s *Server) readBlobRequest(conn net.Conn) (int, string, bool, error) {
|
|
var sendRequest sendBlobRequest
|
|
err := json.NewDecoder(conn).Decode(&sendRequest)
|
|
if err != nil {
|
|
return 0, "", false, err
|
|
}
|
|
|
|
if sendRequest.SdBlobHash != "" && sendRequest.BlobHash != "" {
|
|
return 0, "", false, errors.Err("invalid request")
|
|
}
|
|
|
|
var blobHash string
|
|
var blobSize int
|
|
isSdBlob := sendRequest.SdBlobHash != ""
|
|
|
|
if blobSize > maxBlobSize {
|
|
return 0, "", isSdBlob, errors.Err("blob cannot be more than " + strconv.Itoa(maxBlobSize) + " bytes")
|
|
}
|
|
|
|
if isSdBlob {
|
|
blobSize = sendRequest.SdBlobSize
|
|
blobHash = sendRequest.SdBlobHash
|
|
} else {
|
|
blobSize = sendRequest.BlobSize
|
|
blobHash = sendRequest.BlobHash
|
|
}
|
|
|
|
return blobSize, blobHash, isSdBlob, nil
|
|
}
|
|
|
|
func (s *Server) sendBlobResponse(conn net.Conn, blobExists, isSdBlob bool) error {
|
|
var response []byte
|
|
var err error
|
|
|
|
if isSdBlob {
|
|
response, err = json.Marshal(sendSdBlobResponse{SendSdBlob: !blobExists})
|
|
} else {
|
|
response, err = json.Marshal(sendBlobResponse{SendBlob: !blobExists})
|
|
}
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
_, err = conn.Write(response)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (s *Server) sendTransferResponse(conn net.Conn, receivedBlob, isSdBlob bool) error {
|
|
var response []byte
|
|
var err error
|
|
|
|
if isSdBlob {
|
|
response, err = json.Marshal(sdBlobTransferResponse{ReceivedSdBlob: receivedBlob})
|
|
|
|
} else {
|
|
response, err = json.Marshal(blobTransferResponse{ReceivedBlob: receivedBlob})
|
|
}
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
_, err = conn.Write(response)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return nil
|
|
}
|