Files
Erupe/server/entrance/entrance_server.go
2024-10-12 22:37:42 +01:00

120 lines
2.7 KiB
Go

package entrance
import (
"encoding/hex"
"fmt"
"io"
"net"
"strings"
"sync"
"erupe-ce/config"
"erupe-ce/network"
"erupe-ce/utils/logger"
"go.uber.org/zap"
)
// EntranceServer is a MHF entrance server.
type EntranceServer struct {
sync.Mutex
logger logger.Logger
listener net.Listener
isShuttingDown bool
}
// NewServer creates a new Server type.
func NewServer() *EntranceServer {
server := &EntranceServer{
logger: logger.Get().Named("entrance"),
}
return server
}
// Start starts the server in a new goroutine.
func (server *EntranceServer) Start() error {
l, err := net.Listen("tcp", fmt.Sprintf(":%d", config.GetConfig().Entrance.Port))
if err != nil {
return err
}
server.listener = l
go server.acceptClients()
return nil
}
// Shutdown exits the server gracefully.
func (server *EntranceServer) Shutdown() {
server.logger.Debug("Shutting down...")
server.Lock()
server.isShuttingDown = true
server.Unlock()
// This will cause the acceptor goroutine to error and exit gracefully.
server.listener.Close()
}
// acceptClients handles accepting new clients in a loop.
func (server *EntranceServer) acceptClients() {
for {
conn, err := server.listener.Accept()
if err != nil {
// Check if we are shutting down and exit gracefully if so.
server.Lock()
shutdown := server.isShuttingDown
server.Unlock()
if shutdown {
break
} else {
continue
}
}
// Start a new goroutine for the connection so that we don't block other incoming connections.
go server.handleEntranceServerConnection(conn)
}
}
func (server *EntranceServer) handleEntranceServerConnection(conn net.Conn) {
defer conn.Close()
// Client initalizes the connection with a one-time buffer of 8 NULL bytes.
nullInit := make([]byte, 8)
n, err := io.ReadFull(conn, nullInit)
if err != nil {
server.logger.Warn("Failed to read 8 NULL init", zap.Error(err))
return
} else if n != len(nullInit) {
server.logger.Warn("io.ReadFull couldn't read the full 8 byte init.")
return
}
// Create a new encrypted connection handler and read a packet from it.
cc := network.NewCryptConn(conn)
pkt, err := cc.ReadPacket()
if err != nil {
server.logger.Warn("Error reading packet", zap.Error(err))
return
}
if config.GetConfig().DebugOptions.LogInboundMessages {
fmt.Printf("[Client] -> [Server]\nData [%d bytes]:\n%s\n", len(pkt), hex.Dump(pkt))
}
local := false
if strings.Split(conn.RemoteAddr().String(), ":")[0] == "127.0.0.1" {
local = true
}
data := makeSv2Resp(server, local)
if len(pkt) > 5 {
data = append(data, makeUsrResp(pkt, server)...)
}
cc.SendPacket(data)
// Close because we only need to send the response once.
// Any further requests from the client will come from a new connection.
}