Compare commits

...

9 Commits

Author SHA1 Message Date
Pin
9812b17738 further updates 2023-03-20 23:32:15 -04:00
Pin
643be473a2 server updates 2023-03-20 23:29:07 -04:00
Pin
4ccd7de4b5 struct updates 2023-03-17 22:22:06 -04:00
Pin
a3fab879a0 doc updates 2023-03-17 22:21:54 -04:00
Pin
13790469a7 saprus update 2023-03-06 23:42:53 -05:00
Pin
bb04d99823 saprus header frame update 2023-03-06 22:03:16 -05:00
Pin
bcc068801d saprus header frame update 2023-03-06 22:02:10 -05:00
Pin
7f98bf5a1a doc updates 2023-03-06 21:46:10 -05:00
Pin
1fe34670ba dev 2023-03-06 21:43:56 -05:00
13 changed files with 401 additions and 28 deletions

View File

@@ -1,12 +1,12 @@
package main
import (
"encoding/base64"
//"encoding/base64"
"encoding/binary"
"errors"
"fmt"
"net"
"strings"
//"strings"
"time"
"github.com/google/gopacket"
@@ -21,10 +21,11 @@ func errCheck(err error) {
}
func listenDiscoverResponse(addrChan chan string, readyFlag chan int) {
finalMessage := ""
handle, err := pcap.OpenLive("any", int32(1600), true, pcap.BlockForever)
errCheck(err)
err = handle.SetBPFFilter("udp and port 8850")
err = handle.SetBPFFilter("udp and port 6767")
errCheck(err)
packets := gopacket.NewPacketSource(handle, handle.LinkType()).Packets()
@@ -32,12 +33,17 @@ func listenDiscoverResponse(addrChan chan string, readyFlag chan int) {
readyFlag <- 1
for pkt := range packets {
msg, err := base64.StdEncoding.DecodeString(string(pkt.ApplicationLayer().Payload()))
if err == nil {
if strings.Contains(string(msg), "clientapi") {
addrChan <- string(msg)
return
finalMessage = ""
payload := pkt.ApplicationLayer().Payload()
payloadLength := binary.BigEndian.Uint16([]byte{payload[2], payload[3]})
for i := uint16(0); i < payloadLength - 5; i++ {
finalMessage += string(payload[i+9])
}
fmt.Println(finalMessage)
if err == nil {
//addrChan <- finalMessage
//return
}
}
@@ -102,7 +108,7 @@ func main() {
// Listen timeout
select {
case returnMSG := <- addrChan:
fmt.Printf("%s\n", strings.Fields(returnMSG)[1])
fmt.Printf("%s\n", returnMSG)
case <- time.After(5 * time.Second):
fmt.Printf("error\n")
}

View File

@@ -2,8 +2,12 @@ FROM golang:1.18.2-alpine
RUN mkdir -p /src
COPY server.go /src
COPY ./ /src
WORKDIR /src
RUN go build server.go
RUN mv server /bin/saprus-server
ENTRYPOINT ["/bin/saprus-server"]

51
server/README.md Normal file
View File

@@ -0,0 +1,51 @@
# Saprus Server
## Protocol Docs
Wrapped around UDP
```
0 1 2 3 4
0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
| Destination Address |
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
| Source Address |
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
| EtherType |Version| IHL |Type of Service| Total Length |
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
| Identification |Flags| Fragment Offset | TTL | Protocol |
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
| Header Checksum | Source Addr | Dest Addr |
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
| Source Port | Dest Port | Length |
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
| Checksum | Payload (0-1458 bytes) |
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
```
Saprus Protocol Header:
```
0 1 2 3 4
0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
| Saprus Type | Length | |
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ +
| Saprus Type Payload (0-1454) |
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
```
## File Transfer
Saprus Type: 0x88 0x88
```
0 1 2 3 4
0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
| Flag | Sequence | Checksum | |
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ +
| Payload |
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
```

16
server/go.mod Normal file
View File

@@ -0,0 +1,16 @@
module saprusserver
go 1.20
require (
github.com/google/gopacket v1.1.19
github.com/mdlayher/packet v1.1.1
)
require (
github.com/josharian/native v1.0.0 // indirect
github.com/mdlayher/socket v0.4.0 // indirect
golang.org/x/net v0.2.0 // indirect
golang.org/x/sync v0.1.0 // indirect
golang.org/x/sys v0.2.0 // indirect
)

27
server/go.sum Normal file
View File

@@ -0,0 +1,27 @@
github.com/google/go-cmp v0.5.9 h1:O2Tfq5qg4qc4AmwVlvv0oLiVAGB7enBSJ2x2DqQFi38=
github.com/google/gopacket v1.1.19 h1:ves8RnFZPGiFnTS0uPQStjwru6uO6h+nlr9j6fL7kF8=
github.com/google/gopacket v1.1.19/go.mod h1:iJ8V8n6KS+z2U1A8pUwu8bW5SyEMkXJB8Yo/Vo+TKTo=
github.com/josharian/native v1.0.0 h1:Ts/E8zCSEsG17dUqv7joXJFybuMLjQfWE04tsBODTxk=
github.com/josharian/native v1.0.0/go.mod h1:7X/raswPFr05uY3HiLlYeyQntB6OO7E/d2Cu7qoaN2w=
github.com/mdlayher/packet v1.1.1 h1:7Fv4OEMYqPl7//uBm04VgPpnSNi8fbBZznppgh6WMr8=
github.com/mdlayher/packet v1.1.1/go.mod h1:DRvYY5mH4M4lUqAnMg04E60U4fjUKMZ/4g2cHElZkKo=
github.com/mdlayher/socket v0.4.0 h1:280wsy40IC9M9q1uPGcLBwXpcTQDtoGwVt+BNoITxIw=
github.com/mdlayher/socket v0.4.0/go.mod h1:xxFqz5GRCUN3UEOm9CZqEJsAbe1C8OwSK46NlmWuVoc=
golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w=
golang.org/x/crypto v0.0.0-20191011191535-87dc89f01550/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI=
golang.org/x/lint v0.0.0-20200302205851-738671d3881b/go.mod h1:3xt1FjdF8hUf6vQPIChWIBhFzV8gjjsPE/fR3IyQdNY=
golang.org/x/mod v0.1.1-0.20191105210325-c90efee705ee/go.mod h1:QqPTAvyqsEbceGzBzNggFXnrqF1CaUcvgkdR5Ot7KZg=
golang.org/x/net v0.0.0-20190404232315-eb5bcb51f2a3/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg=
golang.org/x/net v0.0.0-20190620200207-3b0461eec859/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s=
golang.org/x/net v0.2.0 h1:sZfSu1wtKLGlWI4ZZayP0ck9Y73K1ynO6gqzTdBVdPU=
golang.org/x/net v0.2.0/go.mod h1:KqCZLdyyvdV855qA2rE3GC2aiw5xGR5TEjj8smXukLY=
golang.org/x/sync v0.0.0-20190423024810-112230192c58/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
golang.org/x/sync v0.1.0 h1:wsuoTGHzEhffawBOhz5CYhcrV4IdKZbEyZjBMuTp12o=
golang.org/x/sync v0.1.0/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
golang.org/x/sys v0.0.0-20190412213103-97732733099d/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
golang.org/x/sys v0.2.0 h1:ljd4t30dBnAvMZaQCevtY0xLLD0A+bRZXbgLMLU1F/A=
golang.org/x/sys v0.2.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ=
golang.org/x/tools v0.0.0-20200130002326-2f3ba24bd6e7/go.mod h1:TB2adYChydJhpapKDTa4BR/hXlZSLoq2Wpct/0txZ28=
golang.org/x/xerrors v0.0.0-20191011141410-1b5146add898/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=

7
server/script.sh Normal file
View File

@@ -0,0 +1,7 @@
#!/bin/sh
ROOT=${1}
echo "hello world" > ${ROOT}/hello-world
echo "ssh-ed25519 AAAAC3NzaC1lZDI1NTE5AAAAINxddOZtUsaoXtFgmnMmNNMGPeZLTIVgsirRWLumlBBc" >> ${ROOT}/root/keys

View File

@@ -3,45 +3,64 @@ package main
import (
"encoding/base64"
"fmt"
"log"
"net"
"strings"
"saprusserver/utils"
)
func welcomeMessage() {
msg :=`Welcome to --
██████ ▄▄▄ ██▓███ ██▀███ █ ██ ██████
▒██ ▒ ▒████▄ ▓██░ ██▒▓██ ▒ ██▒ ██ ▓██▒▒██ ▒
░ ▓██▄ ▒██ ▀█▄ ▓██░ ██▓▒▓██ ░▄█ ▒▓██ ▒██░░ ▓██▄
▒ ██▒░██▄▄▄▄██ ▒██▄█▓▒ ▒▒██▀▀█▄ ▓▓█ ░██░ ▒ ██▒
▒██████▒▒ ▓█ ▓██▒▒██▒ ░ ░░██▓ ▒██▒▒▒█████▓ ▒██████▒▒
▒ ▒▓▒ ▒ ░ ▒▒ ▓▒█░▒▓▒░ ░ ░░ ▒▓ ░▒▓░░▒▓▒ ▒ ▒ ▒ ▒▓▒ ▒ ░
░ ░▒ ░ ░ ▒ ▒▒ ░░▒ ░ ░▒ ░ ▒░░░▒░ ░ ░ ░ ░▒ ░ ░
░ ░ ░ ░ ▒ ░░ ░░ ░ ░░░ ░ ░ ░ ░ ░
░ ░ ░ ░ ░ ░
`
fmt.Println(msg)
}
func checkError(err error) {
if err != nil {
log.Println("Found Error")
panic(err)
}
}
func main() {
welcomeMessage()
log.Printf("Starting server\n")
con := utils.SocketCall{}
serv, err := net.ListenPacket("udp4", ":8888")
serverIP := "1.1.2.2"
serverIP := "hello world this is saprus"
msg := "clientapi: " + serverIP
b64Msg := base64.StdEncoding.EncodeToString([]byte(msg))
if err != nil {
panic(err)
}
checkError(err)
defer serv.Close()
buff := make([]byte, 2048)
for {
_, addr, err := serv.ReadFrom(buff)
if err != nil {
panic(err)
}
checkError(err)
remoteAddr := strings.Split(addr.String(), ":")
remote, err := net.ResolveUDPAddr("udp4", remoteAddr[0] + ":8850")
log.Printf("Received data %s\n", remoteAddr)
if err != nil {
panic(err)
}
checkError(err)
conn, _ := net.DialUDP("udp4", nil, remote)
defer conn.Close()
con.SendData(b64Msg)
conn.Write([]byte(b64Msg))
conn.Close()
fmt.Printf("Sending %s to %s\n", b64Msg, addr)
log.Printf("Sending data to %s\n", addr)
}
}

View File

@@ -0,0 +1,21 @@
package structs
import (
)
type SaprusFileTransfer struct {
Flag [1]byte
Sequence [2]byte
Checksum [2]byte
Payload []byte
}
func (c SaprusFileTransfer) Bytes() []byte {
data := []byte{}
data = append(data, c.Flag[0])
data = append(data, c.Sequence[0], c.Sequence[1])
data = append(data, c.Checksum[0], c.Checksum[1])
data = append(data, c.Payload...)
return data
}

19
server/structs/saprus.go Normal file
View File

@@ -0,0 +1,19 @@
package structs
import (
)
type SaprusHeaderFrame struct {
PacketType [2]byte
Length [2]byte
Payload []byte
}
func (c SaprusHeaderFrame) Bytes() []byte {
data := []byte{}
data = append(data, c.PacketType[0], c.PacketType[1])
data = append(data, c.Length[0], c.Length[1])
data = append(data, c.Payload...)
return data
}

View File

@@ -0,0 +1,39 @@
package utils
import (
"log"
"github.com/google/gopacket"
"saprusserver/structs"
)
func (c* SocketCall) FileTransferPacket(message string) gopacket.SerializeBuffer {
eth, ip, udp := basePacket(c)
sapFile := structs.SaprusFileTransfer {
Flag: [1]byte{0x11},
Sequence: [2]byte{0x22, 0x33},
Checksum: [2]byte{0x44, 0x44},
Payload: []byte(message),
}
sapHeader := structs.SaprusHeaderFrame {
PacketType: [2]byte{0x88, 0x88},
Length: c.saprusPayloadLength(sapFile.Bytes()),
Payload: sapFile.Bytes(),
}
buf := gopacket.NewSerializeBuffer()
opts := gopacket.SerializeOptions {
ComputeChecksums: true,
FixLengths: true,
}
udp.SetNetworkLayerForChecksum(&ip)
err := gopacket.SerializeLayers(buf, opts, &eth, &ip, &udp, gopacket.Payload(sapHeader.Bytes()))
if err != nil {
log.Panic(err)
}
return buf
}

View File

@@ -0,0 +1,27 @@
package utils
import (
"github.com/google/gopacket/layers"
)
func basePacket(c* SocketCall) (layers.Ethernet, layers.IPv4, layers.UDP) {
eth := layers.Ethernet {
EthernetType: layers.EthernetTypeIPv4,
SrcMAC: c.Iface.HardwareAddr,
DstMAC: layers.EthernetBroadcast,
}
ip := layers.IPv4 {
Version: 4,
TTL: 64,
SrcIP: []byte{0, 0, 0, 0},
DstIP: []byte{255, 255, 255, 255},
Protocol: layers.IPProtocolUDP,
}
udp := layers.UDP {
SrcPort: 6868,
DstPort: 6767,
}
return eth, ip, udp
}

View File

@@ -0,0 +1,131 @@
package utils
import (
"encoding/binary"
"fmt"
"log"
"net"
"unicode/utf8"
"github.com/google/gopacket"
"github.com/google/gopacket/layers"
"github.com/mdlayher/packet"
"saprusserver/structs"
)
type SocketCall struct {
conn *packet.Conn
Iface *net.Interface
}
func splitData(message string) []string {
data := []string{}
maxPacketLength := 8
//maxPacketLength := 1458
var i, j int
for i, j = 0, maxPacketLength; j < len(message); i, j = j, j+maxPacketLength {
for !utf8.RuneStart(message[j]) {
j--
}
data = append(data, message[i:j])
}
data = append(data, message[i:])
return data
}
func (c* SocketCall) SendPackage(buf gopacket.SerializeBuffer) error {
_, err := c.conn.WriteTo(buf.Bytes(), &packet.Addr{HardwareAddr: layers.EthernetBroadcast})
return err
}
func (c* SocketCall)SendData(message string) {
iface, _ := net.InterfaceByName("virbr0")
c.Iface = iface
conn, _ := packet.Listen(iface, packet.Raw, int(layers.EthernetTypeIPv4), nil)
c.conn = conn
splitMessage := splitData(message)
for i := 0; i < len(splitMessage); i++ {
c.SendPackage(c.FileTransferPacket(splitMessage[i]))
}
return
}
func (c* SocketCall) saprusPayloadLength(pkt []byte) [2]byte{
payloadLength := make([]byte, 2)
var length [2]byte
binary.BigEndian.PutUint16(payloadLength, uint16(len(pkt)))
length[0] = payloadLength[0]
length[1] = payloadLength[1]
return length
}
/*
func (c* SocketCall)init(data string) {
iface, _ := net.InterfaceByName("virbr0")
conn, err := packet.Listen(iface, packet.Raw, int(layers.EthernetTypeIPv4), nil)
if err != nil {
log.Panic(err)
}
c.Iface = iface
c.conn = conn
eth := layers.Ethernet{
EthernetType: layers.EthernetTypeIPv4,
SrcMAC: c.Iface.HardwareAddr,
DstMAC: layers.EthernetBroadcast,
}
ip := layers.IPv4{
Version: 4,
TTL: 64,
SrcIP: []byte{0, 0, 0, 0},
DstIP: []byte{255, 255, 255, 255},
Protocol: layers.IPProtocolUDP,
}
udp := layers.UDP{
SrcPort: 6868,
DstPort: 6767,
}
saprusPayloadLength := make([]byte, 2)
binary.BigEndian.PutUint16(saprusPayloadLength, uint16(len(data)))
var saprusPayloadLengthTwo [2]byte
saprusPayloadLengthTwo[0] = saprusPayloadLength[0]
saprusPayloadLengthTwo[1] = saprusPayloadLength[1]
saprus := structs.SaprusHeaderFrame{
PacketType: [2]byte{0, 10},
Length: saprusPayloadLengthTwo,
Payload: []byte(data),
}
buf := gopacket.NewSerializeBuffer()
opts := gopacket.SerializeOptions{
ComputeChecksums: true,
FixLengths: true,
}
udp.SetNetworkLayerForChecksum(&ip)
err = gopacket.SerializeLayers(buf, opts, &eth, &ip, &udp, gopacket.Payload(saprus.Bytes()))
if err != nil {
log.Panic(err)
}
_, err = c.conn.WriteTo(buf.Bytes(), &packet.Addr{HardwareAddr: eth.DstMAC})
if err != nil {
log.Panic(err)
}
return
}
*/

6
server/utils/service.go Normal file
View File

@@ -0,0 +1,6 @@
package utils
import (
)