2024-01-14 11:22:14 +00:00
|
|
|
package core
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"net"
|
2024-01-15 16:08:26 +00:00
|
|
|
"sync"
|
|
|
|
"sync/atomic"
|
2024-01-14 11:22:14 +00:00
|
|
|
"time"
|
|
|
|
)
|
|
|
|
|
|
|
|
// Handler is a byte code specific packet handler.
|
|
|
|
type Handler func([]byte, *Client)
|
|
|
|
|
|
|
|
// Server represents a UDP server.
|
|
|
|
type Server struct {
|
|
|
|
socket *net.UDPConn
|
|
|
|
handlers [256]Handler
|
2024-01-15 16:08:26 +00:00
|
|
|
clients sync.Map
|
|
|
|
count atomic.Int64
|
2024-01-14 11:22:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// New creates a new server.
|
|
|
|
func New() *Server {
|
2024-01-15 16:08:26 +00:00
|
|
|
timeout := 3 * time.Second
|
|
|
|
interval := time.Second
|
|
|
|
server := &Server{}
|
|
|
|
|
|
|
|
go func() {
|
|
|
|
ticker := time.NewTicker(interval)
|
|
|
|
defer ticker.Stop()
|
|
|
|
|
|
|
|
for range ticker.C {
|
|
|
|
now := time.Now()
|
|
|
|
|
|
|
|
server.clients.Range(func(key, value interface{}) bool {
|
|
|
|
item := value.(*Client)
|
|
|
|
|
|
|
|
if !item.lastPacket.IsZero() && now.After(item.lastPacket.Add(timeout)) {
|
|
|
|
server.clients.Delete(key)
|
|
|
|
server.count.Add(-1)
|
|
|
|
}
|
|
|
|
|
|
|
|
return true
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
|
|
|
return server
|
2024-01-14 11:22:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// AddHandler adds the handler for the given byte code.
|
|
|
|
func (s *Server) AddHandler(code byte, handler Handler) {
|
|
|
|
s.handlers[code] = handler
|
|
|
|
}
|
|
|
|
|
2024-01-15 16:08:26 +00:00
|
|
|
// Count returns the number of connected clients.
|
|
|
|
func (s *Server) Count() int {
|
|
|
|
return int(s.count.Load())
|
|
|
|
}
|
|
|
|
|
|
|
|
// Send sends the data prefixed with the byte code to the client.
|
|
|
|
func (s *Server) Send(code byte, data []byte, client *Client) {
|
|
|
|
data = append([]byte{code}, data...)
|
2024-01-14 11:22:14 +00:00
|
|
|
_, err := s.socket.WriteToUDP(data, client.address)
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
fmt.Println("Error sending response:", err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Run starts the server on the given port.
|
|
|
|
func (s *Server) Run(port int) {
|
|
|
|
s.socket = listen(port)
|
|
|
|
defer s.socket.Close()
|
|
|
|
|
|
|
|
s.read()
|
|
|
|
}
|
|
|
|
|
|
|
|
// listen creates a socket for the server and starts listening for incoming packets.
|
|
|
|
func listen(port int) *net.UDPConn {
|
|
|
|
addr, err := net.ResolveUDPAddr("udp", fmt.Sprintf(":%d", port))
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
connection, err := net.ListenUDP("udp", addr)
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return connection
|
|
|
|
}
|
|
|
|
|
|
|
|
// read is a blocking call which will read incoming packets and handle them.
|
|
|
|
func (s *Server) read() {
|
|
|
|
buffer := make([]byte, 4096)
|
|
|
|
|
|
|
|
for {
|
|
|
|
n, addr, err := s.socket.ReadFromUDP(buffer)
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
fmt.Println("Error reading from UDP connection:", err)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
if n == 0 {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
go s.handle(buffer[:n], addr)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// handle deals with an incoming packet.
|
|
|
|
func (s *Server) handle(data []byte, addr *net.UDPAddr) {
|
|
|
|
c := s.getClient(addr)
|
|
|
|
c.lastPacket = time.Now()
|
|
|
|
// fmt.Printf("Received %d bytes from %s: %s\n", len(data), c, string(data))
|
|
|
|
|
|
|
|
handler := s.handlers[data[0]]
|
|
|
|
|
|
|
|
if handler == nil {
|
2024-01-15 16:08:26 +00:00
|
|
|
fmt.Printf("No callback registered for packet type %d\n", data[0])
|
2024-01-14 11:22:14 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2024-01-15 16:08:26 +00:00
|
|
|
handler(data[1:], c)
|
2024-01-14 11:22:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// getClient either returns a new or existing client for the requested address.
|
|
|
|
func (s *Server) getClient(addr *net.UDPAddr) *Client {
|
2024-01-15 16:08:26 +00:00
|
|
|
obj, exists := s.clients.Load(addr.String())
|
2024-01-14 11:22:14 +00:00
|
|
|
|
|
|
|
if exists {
|
2024-01-15 16:08:26 +00:00
|
|
|
return obj.(*Client)
|
2024-01-14 11:22:14 +00:00
|
|
|
}
|
|
|
|
|
2024-01-15 16:08:26 +00:00
|
|
|
client := &Client{
|
2024-01-14 11:22:14 +00:00
|
|
|
address: addr,
|
|
|
|
}
|
|
|
|
|
2024-01-15 16:08:26 +00:00
|
|
|
s.clients.Store(addr.String(), client)
|
|
|
|
s.count.Add(1)
|
|
|
|
return client
|
2024-01-14 11:22:14 +00:00
|
|
|
}
|