netgiv/server.go

230 lines
4.7 KiB
Go
Raw Normal View History

package main
import (
"bytes"
"encoding/gob"
"fmt"
"io"
"log"
"net"
"os"
2022-01-14 23:30:14 +10:30
"sync/atomic"
"time"
"github.com/h2non/filetype"
"github.com/tardisx/netgiv/secure"
)
type Server struct {
port int
}
// An NGF is a Netgiv File
type NGF struct {
2022-01-14 23:30:14 +10:30
Id uint32
StorePath string
Filename string // could be empty string if we were not supplied with one
Kind string //
Size uint64 // file size
}
2022-01-14 23:30:14 +10:30
var ngfs []NGF
var globalId uint32
func (s *Server) Run() {
2022-01-13 22:20:35 +10:30
address := fmt.Sprintf(":%d", s.port)
networkAddress, _ := net.ResolveTCPAddr("tcp", address)
listener, err := net.ListenTCP("tcp", networkAddress)
if err != nil {
fmt.Print(err)
os.Exit(2)
}
2022-01-14 23:30:14 +10:30
ngfs = make([]NGF, 0)
for {
conn, err := listener.AcceptTCP()
if err != nil {
fmt.Print(err)
}
go handleConnection(conn)
}
}
func handleConnection(conn *net.TCPConn) {
defer conn.Close()
2022-01-13 22:20:35 +10:30
conn.SetDeadline(time.Now().Add(time.Second * 5))
sharedKey := secure.Handshake(conn)
secureConnection := secure.SecureConnection{Conn: conn, SharedKey: sharedKey, Buffer: &bytes.Buffer{}}
gob.Register(secure.PacketStart{})
gob.Register(secure.PacketSendDataStart{})
dec := gob.NewDecoder(&secureConnection)
2022-01-14 23:30:14 +10:30
enc := gob.NewEncoder(&secureConnection)
2022-01-13 22:20:35 +10:30
// Get the start packet
start := secure.PacketStart{}
2022-01-13 22:20:35 +10:30
err := dec.Decode(&start)
if err == io.EOF {
2022-01-14 23:30:14 +10:30
log.Printf("connection has been closed prematurely")
2022-01-13 22:20:35 +10:30
return
}
if err != nil {
2022-01-14 23:30:14 +10:30
log.Printf("error while expecting PacketStart: %v", err)
2022-01-13 22:20:35 +10:30
return
}
2022-01-13 22:20:35 +10:30
conn.SetDeadline(time.Now().Add(time.Second * 5))
2022-01-13 22:20:35 +10:30
if start.OperationType == secure.OperationTypeSend {
2022-01-14 23:30:14 +10:30
log.Printf("file incoming")
2022-01-13 22:20:35 +10:30
sendStart := secure.PacketSendDataStart{}
2022-01-13 22:20:35 +10:30
err = dec.Decode(&sendStart)
if err != nil {
2022-01-14 23:30:14 +10:30
log.Printf("error - expecting PacketSendDataStart: %v", err)
return
}
2022-01-13 22:20:35 +10:30
file, err := os.CreateTemp("", "netgiv_")
2022-01-14 23:30:14 +10:30
if err != nil {
log.Fatalf("can't open tempfile: %v", err)
}
defer file.Close()
ngf := NGF{
StorePath: file.Name(),
Filename: sendStart.Filename,
Kind: "",
Size: 0,
2022-01-14 23:30:14 +10:30
Id: atomic.AddUint32(&globalId, 1),
}
if err != nil {
2022-01-13 22:20:35 +10:30
log.Printf("got error with temp file: %w", err)
return
}
2022-01-13 22:20:35 +10:30
sendData := secure.PacketSendDataNext{}
determinedKind := false
2022-01-13 22:20:35 +10:30
for {
conn.SetDeadline(time.Now().Add(time.Second * 5))
err = dec.Decode(&sendData)
if err == io.EOF {
break
}
if err != nil {
2022-01-14 23:30:14 +10:30
log.Printf("error while expecting PacketSendDataNext: %s", err)
2022-01-13 22:20:35 +10:30
return
}
// filetype.Match needs a few hundred bytes - I guess there is a chance
// we don't have enough in the very first packet? This might need rework.
if !determinedKind {
kind, _ := filetype.Match(sendData.Data)
ngf.Kind = kind.MIME.Value
determinedKind = true
}
2022-01-13 22:20:35 +10:30
file.Write(sendData.Data)
}
info, err := file.Stat()
if err != nil {
log.Printf("couldn't stat file %s", err)
return
}
ngf.Size = uint64(info.Size())
file.Close()
2022-01-14 23:30:14 +10:30
ngfs = append(ngfs, ngf)
log.Printf("done receiving file")
2022-01-13 22:20:35 +10:30
return
2022-01-15 14:04:13 +10:30
} else if start.OperationType == secure.OperationTypeReceive {
log.Printf("client requesting file receive")
// wait for them to send the request
req := secure.PacketReceiveDataStartRequest{}
err := dec.Decode(&req)
if err != nil {
log.Printf("error expecting PacketReceiveDataStartRequest: %v", err)
return
}
log.Printf("The asked for %v", req)
res := secure.PacketReceiveDataStartResponse{
Status: secure.ReceiveDataStartResponseOK,
Filename: "abcdef",
Kind: "",
TotalSize: 12340,
}
err = enc.Encode(res)
if err != nil {
log.Printf("error sending PacketReceiveDataStartResponse: %v", err)
return
}
log.Printf("%#v", ngfs)
// now just start sending the file in batches
buf := make([]byte, 2048)
filename := ngfs[0].StorePath
log.Printf("opening %s", filename)
f, err := os.Open(filename)
for {
n, err := f.Read(buf)
eof := false
if err != nil && err != io.EOF {
log.Printf("error reading data: %v", err)
break
}
if err == io.EOF {
eof = true
}
chunk := secure.PacketReceiveDataNext{
Size: uint16(n),
Data: buf[:n],
Last: eof,
}
err = enc.Encode(chunk)
if err != nil {
log.Printf("error sending chunk: %v", err)
}
if eof {
break
}
}
log.Printf("sending done")
return
2022-01-14 23:30:14 +10:30
} else if start.OperationType == secure.OperationTypeList {
log.Printf("client requesting file list")
for _, ngf := range ngfs {
p := secure.PacketListData{}
p.FileSize = uint32(ngf.Size)
p.Kind = ngf.Kind
p.Id = ngf.Id
p.Filename = ngf.Filename
enc.Encode(p)
}
log.Printf("done sending list, closing connection")
return
2022-01-13 22:20:35 +10:30
} else {
log.Printf("bad operation")
return
}
2022-01-13 22:20:35 +10:30
}