major refactoring :/
All checks were successful
Build and Publish / BuildAndDeploy (push) Successful in 1m56s
All checks were successful
Build and Publish / BuildAndDeploy (push) Successful in 1m56s
This commit is contained in:
94
cart-packet-queue.go
Normal file
94
cart-packet-queue.go
Normal file
@@ -0,0 +1,94 @@
|
|||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"io"
|
||||||
|
"log"
|
||||||
|
"net"
|
||||||
|
"sync"
|
||||||
|
"time"
|
||||||
|
)
|
||||||
|
|
||||||
|
type CartPacketWithData struct {
|
||||||
|
MessageType uint16
|
||||||
|
Id CartId
|
||||||
|
Added time.Time
|
||||||
|
Consumed bool
|
||||||
|
Data []byte
|
||||||
|
}
|
||||||
|
|
||||||
|
type CartPacketQueue struct {
|
||||||
|
mu sync.RWMutex
|
||||||
|
Packets []CartPacketWithData
|
||||||
|
connection net.Conn
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewCartPacketQueue(connection net.Conn) *CartPacketQueue {
|
||||||
|
|
||||||
|
queue := &CartPacketQueue{
|
||||||
|
Packets: make([]CartPacketWithData, 0),
|
||||||
|
connection: connection,
|
||||||
|
}
|
||||||
|
go func() {
|
||||||
|
defer connection.Close()
|
||||||
|
var packet CartPacket
|
||||||
|
for {
|
||||||
|
err := ReadPacket(queue.connection, &packet)
|
||||||
|
ts := time.Now()
|
||||||
|
if err != nil {
|
||||||
|
|
||||||
|
if err == io.EOF {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
log.Printf("Error receiving packet: %v\n", err)
|
||||||
|
//return
|
||||||
|
}
|
||||||
|
|
||||||
|
data, err := GetPacketData(queue.connection, int(packet.DataLength))
|
||||||
|
if err != nil {
|
||||||
|
log.Printf("Error receiving packet data: %v\n", err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
queue.mu.Lock()
|
||||||
|
|
||||||
|
l := make([]CartPacketWithData, 0, len(queue.Packets))
|
||||||
|
|
||||||
|
for _, packet := range queue.Packets {
|
||||||
|
if !packet.Consumed && packet.Added.After(ts.Add(-time.Second)) {
|
||||||
|
l = append(l, packet)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
queue.Packets = append(l, CartPacketWithData{
|
||||||
|
MessageType: packet.MessageType,
|
||||||
|
Id: packet.Id,
|
||||||
|
Added: ts,
|
||||||
|
Data: data,
|
||||||
|
})
|
||||||
|
queue.mu.Unlock()
|
||||||
|
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
return queue
|
||||||
|
}
|
||||||
|
|
||||||
|
func (p *CartPacketQueue) Expect(messageType uint16, id CartId, timeToWait time.Duration) (*CartPacketWithData, error) {
|
||||||
|
start := time.Now().Add(-time.Millisecond)
|
||||||
|
|
||||||
|
for {
|
||||||
|
if time.Since(start) > timeToWait {
|
||||||
|
return nil, fmt.Errorf("timeout waiting for message type %d", messageType)
|
||||||
|
}
|
||||||
|
p.mu.RLock()
|
||||||
|
for _, packet := range p.Packets {
|
||||||
|
if packet.MessageType == messageType && packet.Id == id && packet.Added.After(start) {
|
||||||
|
packet.Consumed = true
|
||||||
|
p.mu.RUnlock()
|
||||||
|
return &packet, nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
p.mu.RUnlock()
|
||||||
|
time.Sleep(time.Millisecond * 2)
|
||||||
|
}
|
||||||
|
}
|
||||||
3
main.go
3
main.go
@@ -173,11 +173,10 @@ func main() {
|
|||||||
// if local
|
// if local
|
||||||
//syncedPool.AddRemote("localhost")
|
//syncedPool.AddRemote("localhost")
|
||||||
|
|
||||||
rpcHandler, err := NewGrainHandler(app.pool, ":1337")
|
_, err = NewGrainHandler(app.pool, ":1337")
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatalf("Error creating handler: %v\n", err)
|
log.Fatalf("Error creating handler: %v\n", err)
|
||||||
}
|
}
|
||||||
go rpcHandler.Serve()
|
|
||||||
|
|
||||||
go func() {
|
go func() {
|
||||||
for range time.Tick(time.Minute) {
|
for range time.Tick(time.Minute) {
|
||||||
|
|||||||
@@ -30,18 +30,23 @@ func NewPacketQueue(connection net.Conn) *PacketQueue {
|
|||||||
}
|
}
|
||||||
go func() {
|
go func() {
|
||||||
defer connection.Close()
|
defer connection.Close()
|
||||||
|
var packet Packet
|
||||||
for {
|
for {
|
||||||
messageType, data, err := ReceivePacket(queue.connection)
|
err := ReadPacket(queue.connection, &packet)
|
||||||
ts := time.Now()
|
ts := time.Now()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Printf("Error receiving packet: %v\n", err)
|
|
||||||
if err == io.EOF {
|
if err == io.EOF {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
log.Printf("Error receiving packet: %v\n", err)
|
||||||
//return
|
//return
|
||||||
}
|
}
|
||||||
|
data, err := GetPacketData(queue.connection, int(packet.DataLength))
|
||||||
|
if err != nil {
|
||||||
|
log.Printf("Error receiving packet data: %v\n", err)
|
||||||
|
return
|
||||||
|
}
|
||||||
queue.mu.Lock()
|
queue.mu.Lock()
|
||||||
|
|
||||||
l := make([]PacketWithData, 0, len(queue.Packets))
|
l := make([]PacketWithData, 0, len(queue.Packets))
|
||||||
@@ -53,7 +58,7 @@ func NewPacketQueue(connection net.Conn) *PacketQueue {
|
|||||||
}
|
}
|
||||||
|
|
||||||
queue.Packets = append(l, PacketWithData{
|
queue.Packets = append(l, PacketWithData{
|
||||||
MessageType: messageType,
|
MessageType: packet.MessageType,
|
||||||
Added: ts,
|
Added: ts,
|
||||||
Data: data,
|
Data: data,
|
||||||
})
|
})
|
||||||
|
|||||||
34
packet.go
34
packet.go
@@ -8,16 +8,18 @@ import (
|
|||||||
)
|
)
|
||||||
|
|
||||||
const (
|
const (
|
||||||
RemoteGetState = uint16(0x01)
|
RemoteGetState = uint16(0x01)
|
||||||
RemoteHandleMessage = uint16(0x02)
|
RemoteHandleMessage = uint16(0x02)
|
||||||
ResponseBody = uint16(0x03)
|
ResponseBody = uint16(0x03)
|
||||||
|
RemoteGetStateReply = uint16(0x04)
|
||||||
|
RemoteHandleMessageReply = uint16(0x05)
|
||||||
)
|
)
|
||||||
|
|
||||||
type CartPacket struct {
|
type CartPacket struct {
|
||||||
Version uint16
|
Version uint16
|
||||||
MessageType uint16
|
MessageType uint16
|
||||||
Id CartId
|
|
||||||
DataLength uint16
|
DataLength uint16
|
||||||
|
Id CartId
|
||||||
}
|
}
|
||||||
|
|
||||||
type Packet struct {
|
type Packet struct {
|
||||||
@@ -37,8 +39,8 @@ func SendCartPacket(conn io.Writer, id CartId, messageType uint16, datafn func(w
|
|||||||
binary.Write(conn, binary.LittleEndian, CartPacket{
|
binary.Write(conn, binary.LittleEndian, CartPacket{
|
||||||
Version: 2,
|
Version: 2,
|
||||||
MessageType: messageType,
|
MessageType: messageType,
|
||||||
Id: id,
|
|
||||||
DataLength: uint16(len(data)),
|
DataLength: uint16(len(data)),
|
||||||
|
Id: id,
|
||||||
})
|
})
|
||||||
_, err = conn.Write(data)
|
_, err = conn.Write(data)
|
||||||
return err
|
return err
|
||||||
@@ -86,19 +88,29 @@ func SendProxyResponse(conn io.Writer, data any) error {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func ReadPacket[V Packet | CartPacket](conn io.Reader, packet *V) error {
|
||||||
|
return binary.Read(conn, binary.LittleEndian, packet)
|
||||||
|
}
|
||||||
|
|
||||||
|
func GetPacketData(conn io.Reader, len int) ([]byte, error) {
|
||||||
|
data := make([]byte, len)
|
||||||
|
l, err := conn.Read(data)
|
||||||
|
if l != len {
|
||||||
|
return nil, fmt.Errorf("expected %d bytes, got %d", len, l)
|
||||||
|
}
|
||||||
|
return data, err
|
||||||
|
}
|
||||||
|
|
||||||
func ReceivePacket(conn io.Reader) (uint16, []byte, error) {
|
func ReceivePacket(conn io.Reader) (uint16, []byte, error) {
|
||||||
var packet Packet
|
var packet Packet
|
||||||
err := binary.Read(conn, binary.LittleEndian, &packet)
|
err := ReadPacket(conn, &packet)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return packet.MessageType, nil, err
|
return packet.MessageType, nil, err
|
||||||
}
|
}
|
||||||
data := make([]byte, packet.DataLength)
|
|
||||||
l, err := conn.Read(data)
|
data, err := GetPacketData(conn, int(packet.DataLength))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return packet.MessageType, nil, err
|
return packet.MessageType, nil, err
|
||||||
}
|
}
|
||||||
if l != int(packet.DataLength) {
|
|
||||||
return packet.MessageType, nil, fmt.Errorf("expected %d bytes, got %d", packet.DataLength, l)
|
|
||||||
}
|
|
||||||
return packet.MessageType, data, nil
|
return packet.MessageType, data, nil
|
||||||
}
|
}
|
||||||
|
|||||||
34
rpc-pool.go
34
rpc-pool.go
@@ -2,11 +2,9 @@ package main
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
||||||
"io"
|
|
||||||
"net"
|
"net"
|
||||||
"strings"
|
"strings"
|
||||||
"sync"
|
"sync"
|
||||||
"time"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
type RemoteGrainPool struct {
|
type RemoteGrainPool struct {
|
||||||
@@ -26,8 +24,7 @@ func ToCartId(id string) CartId {
|
|||||||
}
|
}
|
||||||
|
|
||||||
type RemoteGrain struct {
|
type RemoteGrain struct {
|
||||||
net.Conn
|
*CartClient
|
||||||
*PacketQueue
|
|
||||||
Id CartId
|
Id CartId
|
||||||
Address string
|
Address string
|
||||||
}
|
}
|
||||||
@@ -59,19 +56,18 @@ func (g *RemoteGrain) Connect() error {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (g *RemoteGrain) HandleMessage(message *Message, isReplay bool) ([]byte, error) {
|
func (g *RemoteGrain) HandleMessage(message *Message, isReplay bool) ([]byte, error) {
|
||||||
err := g.Connect()
|
|
||||||
|
data, err := GetData(message.Write)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
err = SendCartPacket(g.connection, g.Id, RemoteHandleMessage, message.Write)
|
reply, err := g.Call(RemoteHandleMessage, g.Id, RemoteHandleMessageReply, data)
|
||||||
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
packet, err := g.Expect(ResponseBody, time.Second)
|
|
||||||
if err != nil {
|
return reply, err
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
return packet.Data, err
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (g *RemoteGrain) GetId() CartId {
|
func (g *RemoteGrain) GetId() CartId {
|
||||||
@@ -79,21 +75,7 @@ func (g *RemoteGrain) GetId() CartId {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (g *RemoteGrain) GetCurrentState() ([]byte, error) {
|
func (g *RemoteGrain) GetCurrentState() ([]byte, error) {
|
||||||
err := g.Connect()
|
return g.Call(RemoteGetState, g.Id, RemoteGetStateReply, nil)
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
err = SendCartPacket(g.connection, g.Id, RemoteGetState, func(w io.Writer) error {
|
|
||||||
return nil
|
|
||||||
})
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
packet, err := g.Expect(ResponseBody, time.Second)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
return packet.Data, nil
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewRemoteGrainPool(addr string) *RemoteGrainPool {
|
func NewRemoteGrainPool(addr string) *RemoteGrainPool {
|
||||||
|
|||||||
133
rpc-server.go
133
rpc-server.go
@@ -1,15 +1,13 @@
|
|||||||
package main
|
package main
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"encoding/binary"
|
"bytes"
|
||||||
"fmt"
|
"fmt"
|
||||||
"io"
|
|
||||||
"net"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
type GrainHandler struct {
|
type GrainHandler struct {
|
||||||
listener net.Listener
|
*CartServer
|
||||||
pool *GrainLocalPool
|
pool *GrainLocalPool
|
||||||
}
|
}
|
||||||
|
|
||||||
func (h *GrainHandler) GetState(id CartId, reply *Grain) error {
|
func (h *GrainHandler) GetState(id CartId, reply *Grain) error {
|
||||||
@@ -22,68 +20,83 @@ func (h *GrainHandler) GetState(id CartId, reply *Grain) error {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func NewGrainHandler(pool *GrainLocalPool, listen string) (*GrainHandler, error) {
|
func NewGrainHandler(pool *GrainLocalPool, listen string) (*GrainHandler, error) {
|
||||||
|
server, err := CartListen(listen)
|
||||||
handler := &GrainHandler{
|
handler := &GrainHandler{
|
||||||
pool: pool,
|
CartServer: server,
|
||||||
|
pool: pool,
|
||||||
}
|
}
|
||||||
l, err := net.Listen("tcp", listen)
|
server.HandleCall(RemoteHandleMessage, handler.RemoteHandleMessageHandler)
|
||||||
handler.listener = l
|
server.HandleCall(RemoteGetState, handler.RemoteGetStateHandler)
|
||||||
return handler, err
|
return handler, err
|
||||||
}
|
}
|
||||||
|
|
||||||
func (h *GrainHandler) Serve() {
|
func (h *GrainHandler) RemoteHandleMessageHandler(id CartId, data []byte) (uint16, []byte, error) {
|
||||||
for {
|
var msg Message
|
||||||
conn, err := h.listener.Accept()
|
err := ReadMessage(bytes.NewReader(data), &msg)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fmt.Println("Error accepting connection:", err)
|
fmt.Println("Error reading message:", err)
|
||||||
continue
|
return RemoteHandleMessageReply, nil, err
|
||||||
}
|
|
||||||
|
|
||||||
go h.handleClient(conn)
|
|
||||||
}
|
}
|
||||||
|
replyData, err := h.pool.Process(id, msg)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println("Error handling message:", err)
|
||||||
|
}
|
||||||
|
if err != nil {
|
||||||
|
return RemoteHandleMessageReply, nil, err
|
||||||
|
}
|
||||||
|
return RemoteHandleMessageReply, replyData, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (h *GrainHandler) handleClient(conn net.Conn) {
|
func (h *GrainHandler) RemoteGetStateHandler(id CartId, data []byte) (uint16, []byte, error) {
|
||||||
var err error
|
data, err := h.pool.Get(id)
|
||||||
|
if err != nil {
|
||||||
defer conn.Close()
|
return RemoteGetStateReply, nil, err
|
||||||
|
|
||||||
var packet CartPacket
|
|
||||||
|
|
||||||
for {
|
|
||||||
err = binary.Read(conn, binary.LittleEndian, &packet)
|
|
||||||
if err != nil {
|
|
||||||
if err == io.EOF {
|
|
||||||
break
|
|
||||||
}
|
|
||||||
fmt.Println("Error in connection:", err)
|
|
||||||
}
|
|
||||||
if packet.Version != 2 {
|
|
||||||
fmt.Printf("Unknown version %d", packet.Version)
|
|
||||||
break
|
|
||||||
}
|
|
||||||
|
|
||||||
switch packet.MessageType {
|
|
||||||
case RemoteHandleMessage:
|
|
||||||
var msg Message
|
|
||||||
err = ReadMessage(conn, &msg)
|
|
||||||
if err != nil {
|
|
||||||
fmt.Println("Error reading message:", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
data, err := h.pool.Process(packet.Id, msg)
|
|
||||||
if err != nil {
|
|
||||||
fmt.Println("Error handling message:", err)
|
|
||||||
}
|
|
||||||
SendRawResponse(conn, data)
|
|
||||||
|
|
||||||
case RemoteGetState:
|
|
||||||
data, err := h.pool.Get(packet.Id)
|
|
||||||
if err != nil {
|
|
||||||
fmt.Println("Error getting grain:", err)
|
|
||||||
}
|
|
||||||
SendRawResponse(conn, data)
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
return RemoteGetStateReply, data, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// func (h *GrainHandler) handleClient(conn net.Conn) {
|
||||||
|
// var err error
|
||||||
|
|
||||||
|
// defer conn.Close()
|
||||||
|
|
||||||
|
// var packet CartPacket
|
||||||
|
|
||||||
|
// for {
|
||||||
|
// err = binary.Read(conn, binary.LittleEndian, &packet)
|
||||||
|
// if err != nil {
|
||||||
|
// if err == io.EOF {
|
||||||
|
// break
|
||||||
|
// }
|
||||||
|
// fmt.Println("Error in connection:", err)
|
||||||
|
// }
|
||||||
|
// if packet.Version != 2 {
|
||||||
|
// fmt.Printf("Unknown version %d", packet.Version)
|
||||||
|
// break
|
||||||
|
// }
|
||||||
|
|
||||||
|
// switch packet.MessageType {
|
||||||
|
// case RemoteHandleMessage:
|
||||||
|
// var msg Message
|
||||||
|
// err = ReadMessage(conn, &msg)
|
||||||
|
// if err != nil {
|
||||||
|
// fmt.Println("Error reading message:", err)
|
||||||
|
// }
|
||||||
|
|
||||||
|
// data, err := h.pool.Process(packet.Id, msg)
|
||||||
|
// if err != nil {
|
||||||
|
// fmt.Println("Error handling message:", err)
|
||||||
|
// }
|
||||||
|
// SendRawResponse(conn, data)
|
||||||
|
|
||||||
|
// case RemoteGetState:
|
||||||
|
// data, err := h.pool.Get(packet.Id)
|
||||||
|
// if err != nil {
|
||||||
|
// fmt.Println("Error getting grain:", err)
|
||||||
|
// }
|
||||||
|
// SendRawResponse(conn, data)
|
||||||
|
// }
|
||||||
|
|
||||||
|
// }
|
||||||
|
|
||||||
|
// }
|
||||||
|
|||||||
104
tcp-cart-client.go
Normal file
104
tcp-cart-client.go
Normal file
@@ -0,0 +1,104 @@
|
|||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"encoding/binary"
|
||||||
|
"io"
|
||||||
|
"net"
|
||||||
|
"time"
|
||||||
|
)
|
||||||
|
|
||||||
|
type CartClient struct {
|
||||||
|
*CartTCPClient
|
||||||
|
}
|
||||||
|
|
||||||
|
func CartDial(address string) (*CartClient, error) {
|
||||||
|
|
||||||
|
mux, err := NewCartTCPClient(address)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
client := &CartClient{
|
||||||
|
CartTCPClient: mux,
|
||||||
|
}
|
||||||
|
return client, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *Client) Close() {
|
||||||
|
c.Conn.Close()
|
||||||
|
}
|
||||||
|
|
||||||
|
type CartTCPClient struct {
|
||||||
|
net.Conn
|
||||||
|
Errors chan error
|
||||||
|
ErrorCount int
|
||||||
|
address string
|
||||||
|
*PacketQueue
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewCartTCPClient(address string) (*CartTCPClient, error) {
|
||||||
|
connection, err := net.Dial("tcp", address)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return &CartTCPClient{
|
||||||
|
Errors: make(chan error),
|
||||||
|
ErrorCount: 0,
|
||||||
|
Conn: connection,
|
||||||
|
address: address,
|
||||||
|
PacketQueue: NewPacketQueue(connection),
|
||||||
|
}, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *CartTCPClient) Connect() error {
|
||||||
|
if m.Conn == nil {
|
||||||
|
connection, err := net.Dial("tcp", m.address)
|
||||||
|
if err != nil {
|
||||||
|
|
||||||
|
m.Errors <- err
|
||||||
|
m.ErrorCount++
|
||||||
|
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
m.ErrorCount = 0
|
||||||
|
m.Conn = connection
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *CartTCPClient) SendPacket(messageType uint16, id CartId, data []byte) error {
|
||||||
|
err := m.Connect()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
err = binary.Write(m.Conn, binary.LittleEndian, CartPacket{
|
||||||
|
Version: 1,
|
||||||
|
MessageType: messageType,
|
||||||
|
DataLength: uint16(len(data)),
|
||||||
|
Id: id,
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
_, err = m.Conn.Write(data)
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *CartTCPClient) SendPacketFn(messageType uint16, id CartId, datafn func(w io.Writer) error) error {
|
||||||
|
data, err := GetData(datafn)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return m.SendPacket(messageType, id, data)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *CartTCPClient) Call(messageType uint16, id CartId, responseType uint16, data []byte) ([]byte, error) {
|
||||||
|
err := m.SendPacket(messageType, id, data)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
packet, err := m.Expect(responseType, time.Second)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return packet.Data, nil
|
||||||
|
}
|
||||||
138
tcp-cart-mux-server.go
Normal file
138
tcp-cart-mux-server.go
Normal file
@@ -0,0 +1,138 @@
|
|||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"encoding/binary"
|
||||||
|
"io"
|
||||||
|
"log"
|
||||||
|
"net"
|
||||||
|
"sync"
|
||||||
|
)
|
||||||
|
|
||||||
|
type CartServer struct {
|
||||||
|
*TCPCartServerMux
|
||||||
|
}
|
||||||
|
|
||||||
|
func CartListen(address string) (*CartServer, error) {
|
||||||
|
listener, err := net.Listen("tcp", address)
|
||||||
|
server := &CartServer{
|
||||||
|
NewCartTCPServerMux(100),
|
||||||
|
}
|
||||||
|
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
go func() {
|
||||||
|
for {
|
||||||
|
conn, err := listener.Accept()
|
||||||
|
if err != nil {
|
||||||
|
log.Printf("Error accepting connection: %v\n", err)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
go server.HandleConnection(conn)
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
return server, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
type TCPCartServerMux struct {
|
||||||
|
mu sync.RWMutex
|
||||||
|
listeners map[uint16]func(CartId, []byte) error
|
||||||
|
functions map[uint16]func(CartId, []byte) (uint16, []byte, error)
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewCartTCPServerMux(maxClients int) *TCPCartServerMux {
|
||||||
|
m := &TCPCartServerMux{
|
||||||
|
mu: sync.RWMutex{},
|
||||||
|
listeners: make(map[uint16]func(CartId, []byte) error),
|
||||||
|
functions: make(map[uint16]func(CartId, []byte) (uint16, []byte, error)),
|
||||||
|
}
|
||||||
|
|
||||||
|
return m
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *TCPCartServerMux) handleListener(messageType uint16, id CartId, data []byte) (bool, error) {
|
||||||
|
m.mu.RLock()
|
||||||
|
handler, ok := m.listeners[messageType]
|
||||||
|
m.mu.RUnlock()
|
||||||
|
if ok {
|
||||||
|
err := handler(id, data)
|
||||||
|
if err != nil {
|
||||||
|
return true, err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *TCPCartServerMux) handleFunction(connection net.Conn, messageType uint16, id CartId, data []byte) (bool, error) {
|
||||||
|
m.mu.RLock()
|
||||||
|
function, ok := m.functions[messageType]
|
||||||
|
m.mu.RUnlock()
|
||||||
|
if ok {
|
||||||
|
responseType, responseData, err := function(id, data)
|
||||||
|
if err != nil {
|
||||||
|
return true, err
|
||||||
|
}
|
||||||
|
err = binary.Write(connection, binary.LittleEndian, CartPacket{
|
||||||
|
Version: 1,
|
||||||
|
MessageType: responseType,
|
||||||
|
DataLength: uint16(len(responseData)),
|
||||||
|
Id: id,
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
|
return true, err
|
||||||
|
}
|
||||||
|
packetsSent.Inc()
|
||||||
|
_, err = connection.Write(responseData)
|
||||||
|
return true, err
|
||||||
|
}
|
||||||
|
return false, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *TCPCartServerMux) HandleConnection(connection net.Conn) error {
|
||||||
|
var packet *CartPacket
|
||||||
|
var err error
|
||||||
|
defer connection.Close()
|
||||||
|
for {
|
||||||
|
err = ReadPacket(connection, packet)
|
||||||
|
if err != nil {
|
||||||
|
if err == io.EOF {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
log.Printf("Error receiving packet: %v\n", err)
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if packet == nil {
|
||||||
|
log.Println("Packet is nil")
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
data, err := GetPacketData(connection, int(packet.DataLength))
|
||||||
|
if err != nil {
|
||||||
|
log.Printf("Error getting packet data: %v\n", err)
|
||||||
|
}
|
||||||
|
status, err := m.handleListener(packet.MessageType, packet.Id, data)
|
||||||
|
if err != nil {
|
||||||
|
log.Printf("Error handling listener: %v\n", err)
|
||||||
|
}
|
||||||
|
if !status {
|
||||||
|
status, err = m.handleFunction(connection, packet.MessageType, packet.Id, data)
|
||||||
|
if err != nil {
|
||||||
|
log.Printf("Error handling function: %v\n", err)
|
||||||
|
}
|
||||||
|
if !status {
|
||||||
|
log.Printf("Unknown message type: %d\n", packet.MessageType)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *TCPCartServerMux) ListenFor(messageType uint16, handler func(CartId, []byte) error) {
|
||||||
|
m.mu.Lock()
|
||||||
|
m.listeners[messageType] = handler
|
||||||
|
m.mu.Unlock()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *TCPCartServerMux) HandleCall(messageType uint16, handler func(CartId, []byte) (uint16, []byte, error)) {
|
||||||
|
m.mu.Lock()
|
||||||
|
m.functions[messageType] = handler
|
||||||
|
m.mu.Unlock()
|
||||||
|
}
|
||||||
@@ -8,26 +8,22 @@ import (
|
|||||||
)
|
)
|
||||||
|
|
||||||
type Client struct {
|
type Client struct {
|
||||||
*TCPClientMux
|
*TCPClient
|
||||||
}
|
}
|
||||||
|
|
||||||
func Dial(address string) (*Client, error) {
|
func Dial(address string) (*Client, error) {
|
||||||
|
|
||||||
mux, err := NewTCPClientMux(address)
|
mux, err := NewTCPClient(address)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
client := &Client{
|
client := &Client{
|
||||||
TCPClientMux: mux,
|
TCPClient: mux,
|
||||||
}
|
}
|
||||||
return client, nil
|
return client, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *Client) Close() {
|
type TCPClient struct {
|
||||||
c.Conn.Close()
|
|
||||||
}
|
|
||||||
|
|
||||||
type TCPClientMux struct {
|
|
||||||
net.Conn
|
net.Conn
|
||||||
Errors chan error
|
Errors chan error
|
||||||
ErrorCount int
|
ErrorCount int
|
||||||
@@ -35,12 +31,12 @@ type TCPClientMux struct {
|
|||||||
*PacketQueue
|
*PacketQueue
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewTCPClientMux(address string) (*TCPClientMux, error) {
|
func NewTCPClient(address string) (*TCPClient, error) {
|
||||||
connection, err := net.Dial("tcp", address)
|
connection, err := net.Dial("tcp", address)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
return &TCPClientMux{
|
return &TCPClient{
|
||||||
Errors: make(chan error),
|
Errors: make(chan error),
|
||||||
ErrorCount: 0,
|
ErrorCount: 0,
|
||||||
Conn: connection,
|
Conn: connection,
|
||||||
@@ -49,7 +45,7 @@ func NewTCPClientMux(address string) (*TCPClientMux, error) {
|
|||||||
}, nil
|
}, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (m *TCPClientMux) Connect() error {
|
func (m *TCPClient) Connect() error {
|
||||||
if m.Conn == nil {
|
if m.Conn == nil {
|
||||||
connection, err := net.Dial("tcp", m.address)
|
connection, err := net.Dial("tcp", m.address)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@@ -65,11 +61,11 @@ func (m *TCPClientMux) Connect() error {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (m *TCPClientMux) Close() {
|
func (m *TCPClient) Close() {
|
||||||
m.Conn.Close()
|
m.Conn.Close()
|
||||||
}
|
}
|
||||||
|
|
||||||
func (m *TCPClientMux) SendPacket(messageType uint16, data []byte) error {
|
func (m *TCPClient) SendPacket(messageType uint16, data []byte) error {
|
||||||
err := m.Connect()
|
err := m.Connect()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
@@ -86,7 +82,7 @@ func (m *TCPClientMux) SendPacket(messageType uint16, data []byte) error {
|
|||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
func (m *TCPClientMux) SendPacketFn(messageType uint16, datafn func(w io.Writer) error) error {
|
func (m *TCPClient) SendPacketFn(messageType uint16, datafn func(w io.Writer) error) error {
|
||||||
data, err := GetData(datafn)
|
data, err := GetData(datafn)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
@@ -94,7 +90,7 @@ func (m *TCPClientMux) SendPacketFn(messageType uint16, datafn func(w io.Writer)
|
|||||||
return m.SendPacket(messageType, data)
|
return m.SendPacket(messageType, data)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (m *TCPClientMux) Call(messageType uint16, responseType uint16, data []byte) ([]byte, error) {
|
func (m *TCPClient) Call(messageType uint16, responseType uint16, data []byte) ([]byte, error) {
|
||||||
err := m.SendPacket(messageType, data)
|
err := m.SendPacket(messageType, data)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
@@ -35,18 +35,16 @@ func Listen(address string) (*Server, error) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
type TCPServerMux struct {
|
type TCPServerMux struct {
|
||||||
mu sync.RWMutex
|
mu sync.RWMutex
|
||||||
listeners map[uint16]func(data []byte) error
|
listeners map[uint16]func(data []byte) error
|
||||||
functions map[uint16]func(data []byte) (uint16, []byte, error)
|
functions map[uint16]func(data []byte) (uint16, []byte, error)
|
||||||
connections []net.Conn
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewTCPServerMux(maxClients int) *TCPServerMux {
|
func NewTCPServerMux(maxClients int) *TCPServerMux {
|
||||||
m := &TCPServerMux{
|
m := &TCPServerMux{
|
||||||
connections: make([]net.Conn, 0, maxClients),
|
mu: sync.RWMutex{},
|
||||||
mu: sync.RWMutex{},
|
listeners: make(map[uint16]func(data []byte) error),
|
||||||
listeners: make(map[uint16]func(data []byte) error),
|
functions: make(map[uint16]func(data []byte) (uint16, []byte, error)),
|
||||||
functions: make(map[uint16]func(data []byte) (uint16, []byte, error)),
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return m
|
return m
|
||||||
@@ -90,9 +88,7 @@ func (m *TCPServerMux) handleFunction(connection net.Conn, messageType uint16, d
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (m *TCPServerMux) HandleConnection(connection net.Conn) error {
|
func (m *TCPServerMux) HandleConnection(connection net.Conn) error {
|
||||||
m.mu.Lock()
|
|
||||||
m.connections = append(m.connections, connection)
|
|
||||||
m.mu.Unlock()
|
|
||||||
defer connection.Close()
|
defer connection.Close()
|
||||||
for {
|
for {
|
||||||
messageType, data, err := ReceivePacket(connection)
|
messageType, data, err := ReceivePacket(connection)
|
||||||
|
|||||||
Reference in New Issue
Block a user