mirror of
https://github.com/WiiLink24/wfc-server.git
synced 2026-04-24 23:47:50 -05:00
172 lines
3.7 KiB
Go
172 lines
3.7 KiB
Go
package gpcm
|
|
|
|
import (
|
|
"bufio"
|
|
"context"
|
|
"errors"
|
|
"fmt"
|
|
"io"
|
|
"log"
|
|
"net"
|
|
"os"
|
|
"time"
|
|
"wwfc/common"
|
|
"wwfc/logging"
|
|
|
|
"github.com/jackc/pgx/v4/pgxpool"
|
|
"github.com/logrusorgru/aurora/v3"
|
|
)
|
|
|
|
var (
|
|
ctx = context.Background()
|
|
pool *pgxpool.Pool
|
|
userId int64
|
|
)
|
|
|
|
func checkError(err error) {
|
|
if err != nil {
|
|
log.Fatalf("GPCM server has encountered a fatal error! Reason: %v\n", err)
|
|
}
|
|
}
|
|
|
|
func StartServer() {
|
|
// Get config
|
|
config := common.GetConfig()
|
|
|
|
// Start SQL
|
|
dbString := fmt.Sprintf("postgres://%s:%s@%s/%s", config.Username, config.Password, config.DatabaseAddress, config.DatabaseName)
|
|
dbConf, err := pgxpool.ParseConfig(dbString)
|
|
checkError(err)
|
|
pool, err = pgxpool.ConnectConfig(ctx, dbConf)
|
|
checkError(err)
|
|
|
|
l, err := net.Listen("tcp", "127.0.0.1:29900")
|
|
if err != nil {
|
|
fmt.Println("Error listening:", err.Error())
|
|
os.Exit(1)
|
|
}
|
|
// Close the listener when the application closes.
|
|
defer l.Close()
|
|
fmt.Println("Listening on " + "127.0.0.1:29900")
|
|
for {
|
|
// Listen for an incoming connection.
|
|
conn, err := l.Accept()
|
|
if err != nil {
|
|
fmt.Println("Error accepting: ", err.Error())
|
|
os.Exit(1)
|
|
}
|
|
|
|
// Handle connections in a new goroutine.
|
|
go handleRequest(conn)
|
|
}
|
|
}
|
|
|
|
// Handles incoming requests.
|
|
func handleRequest(conn net.Conn) {
|
|
defer conn.Close()
|
|
|
|
// Set session ID and challenge
|
|
challenge := common.RandomString(10)
|
|
|
|
err := conn.(*net.TCPConn).SetKeepAlive(true)
|
|
if err != nil {
|
|
logging.Notice("GPCM", "Unable to set keepalive:", err.Error())
|
|
}
|
|
|
|
err = conn.(*net.TCPConn).SetKeepAlivePeriod(time.Hour * 1000)
|
|
if err != nil {
|
|
logging.Notice("GPCM", "Unable to set keepalive:", err.Error())
|
|
}
|
|
|
|
conn.Write([]byte(fmt.Sprintf(`\lc\1\challenge\%s\id\1\final\`, challenge)))
|
|
|
|
logging.Notice("GPCM", "Connection established from", conn.RemoteAddr().String())
|
|
|
|
loggedIn := false
|
|
|
|
// Here we go into the listening loop
|
|
for {
|
|
buffer := make([]byte, 1024)
|
|
_, err := bufio.NewReader(conn).Read(buffer)
|
|
if err != nil {
|
|
if errors.Is(err, io.EOF) {
|
|
// Client closed connection, terminate.
|
|
logging.Notice("GPCM", "Client closed connection")
|
|
return
|
|
}
|
|
}
|
|
|
|
commands, err := common.ParseGameSpyMessage(string(buffer))
|
|
if err != nil {
|
|
logging.Notice("GPCM", "Error parsing message:", err.Error())
|
|
logging.Notice("GPCM", "Raw data:", string(buffer))
|
|
return
|
|
}
|
|
|
|
for _, command := range commands {
|
|
logging.Notice("GPCM", "Command:", aurora.Yellow(command.Command).String())
|
|
|
|
if loggedIn == false {
|
|
if command.Command != "login" {
|
|
logging.Notice("GPCM", "Attempt to run command before login!!!")
|
|
return
|
|
}
|
|
|
|
payload := login(pool, ctx, command, challenge)
|
|
if userId != 0 {
|
|
loggedIn = true
|
|
}
|
|
|
|
conn.Write([]byte(payload))
|
|
}
|
|
}
|
|
|
|
// Make sure commands that update the profile run before getprofile
|
|
for _, command := range commands {
|
|
switch command.Command {
|
|
case "login":
|
|
// User should already be authenticated
|
|
break
|
|
|
|
case "logout":
|
|
// Bye
|
|
return
|
|
|
|
case "updatepro":
|
|
updateProfile(pool, ctx, command)
|
|
break
|
|
|
|
case "status":
|
|
logging.Notice("GPCM", "statstring:", aurora.Cyan(command.OtherValues["statstring"]).String())
|
|
if command.OtherValues["locstring"] == "" {
|
|
logging.Notice("GPCM", "locstring: (empty)")
|
|
} else {
|
|
logging.Notice("GPCM", "locstring:", aurora.Cyan(command.OtherValues["locstring"]).String())
|
|
}
|
|
break
|
|
|
|
case "addbuddy":
|
|
addFriend(pool, ctx, command)
|
|
break
|
|
|
|
case "delbuddy":
|
|
removeFriend(pool, ctx, command)
|
|
break
|
|
}
|
|
}
|
|
|
|
for _, command := range commands {
|
|
switch command.Command {
|
|
case "ka":
|
|
conn.Write([]byte(`\ka\\final\`))
|
|
break
|
|
|
|
case "getprofile":
|
|
payload := getProfile(pool, ctx, command)
|
|
conn.Write([]byte(payload))
|
|
break
|
|
}
|
|
}
|
|
}
|
|
}
|