2025-12-19 17:54:50 +01:00
|
|
|
using System;
|
|
|
|
|
using System.Collections;
|
|
|
|
|
using System.Diagnostics;
|
|
|
|
|
using System.Threading;
|
2026-03-22 18:31:05 +01:00
|
|
|
using ONDServer.Map;
|
2026-03-12 22:33:35 +01:00
|
|
|
using GlobalClassLib;
|
2025-12-19 17:54:50 +01:00
|
|
|
using LiteNetLib;
|
|
|
|
|
using LiteNetLib.Utils;
|
|
|
|
|
using PacketLib;
|
|
|
|
|
|
2026-03-22 18:31:05 +01:00
|
|
|
namespace ONDServer;
|
2025-12-19 17:54:50 +01:00
|
|
|
|
|
|
|
|
public class Server {
|
|
|
|
|
public static ServerPlayer P1;
|
|
|
|
|
public static ServerPlayer P2;
|
2026-01-25 11:16:54 +01:00
|
|
|
public static readonly Dictionary<int, ServerPlayer> Players = new();
|
2026-03-09 20:05:21 +01:00
|
|
|
public static ServerPlayer OtherPlayer(ServerPlayer player) => player.state.pid == P1.state.pid ? P2 : P1;
|
2026-03-21 21:23:33 +01:00
|
|
|
|
|
|
|
|
public static bool AutoStop{ get; set; } = true;
|
2025-12-19 17:54:50 +01:00
|
|
|
|
|
|
|
|
private static EventBasedNetListener listener;
|
|
|
|
|
private static NetManager server;
|
|
|
|
|
|
|
|
|
|
private const int TargetTickRate = 30;
|
|
|
|
|
private const double TargetDeltaTime = 1.0 / TargetTickRate;
|
|
|
|
|
private const long TargetTickTimeMs = 1000 / TargetTickRate;
|
|
|
|
|
|
|
|
|
|
private static NetDataWriter writer;
|
|
|
|
|
private static NetPacketProcessor processor;
|
|
|
|
|
|
|
|
|
|
private static bool isRunning;
|
2026-02-26 16:24:55 +01:00
|
|
|
|
2025-12-19 17:54:50 +01:00
|
|
|
public static void Start(int port) {
|
|
|
|
|
writer = new NetDataWriter();
|
|
|
|
|
processor = new NetPacketProcessor();
|
|
|
|
|
|
|
|
|
|
NestedTypeManager.AutoRegister(processor);
|
|
|
|
|
|
|
|
|
|
processor.SubscribeReusable<JoinPacket, NetPeer>(OnJoinReceived);
|
|
|
|
|
processor.SubscribeReusable<PlayerCommandPacket, NetPeer>(OnCommandReceived);
|
|
|
|
|
|
|
|
|
|
listener = new EventBasedNetListener(); // ← Initialize the listener!
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
server = new NetManager(listener){
|
|
|
|
|
AutoRecycle = true
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
Console.WriteLine($"Starting server on {port}");
|
|
|
|
|
|
|
|
|
|
listener.ConnectionRequestEvent += request => {
|
|
|
|
|
Console.WriteLine($"Connection Request from {request.RemoteEndPoint}");
|
2026-01-25 11:16:54 +01:00
|
|
|
if (Players.Count >= 2){
|
2025-12-19 17:54:50 +01:00
|
|
|
Console.WriteLine($"{request.RemoteEndPoint} denied, server full");
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
request.Accept();
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
listener.NetworkReceiveEvent += (peer, reader, channel, method) => {
|
|
|
|
|
OnNetworkReceive(peer, reader, method);
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
listener.PeerDisconnectedEvent += (peer, info) => {
|
|
|
|
|
OnPeerDisconnected(peer, info);
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
server.Start(port);
|
2026-02-01 14:53:27 +01:00
|
|
|
|
2025-12-19 17:54:50 +01:00
|
|
|
Run();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public static void SendPacket<T>(T packet, NetPeer peer, DeliveryMethod deliveryMethod = DeliveryMethod.ReliableOrdered) where T : class, new() {
|
|
|
|
|
if (peer != null) {
|
|
|
|
|
writer.Reset();
|
|
|
|
|
processor.Write(writer, packet);
|
|
|
|
|
peer.Send(writer, deliveryMethod);
|
|
|
|
|
}
|
|
|
|
|
}
|
2026-01-25 11:16:54 +01:00
|
|
|
public static void SendPacket<T>(T packet, int pid, DeliveryMethod deliveryMethod = DeliveryMethod.ReliableOrdered) where T : class, new() {
|
|
|
|
|
SendPacket(packet, Players[pid].peer, deliveryMethod);
|
2025-12-19 17:54:50 +01:00
|
|
|
}
|
|
|
|
|
public static void SendPacketToAll<T>(T packet, DeliveryMethod deliveryMethod = DeliveryMethod.ReliableOrdered) where T : class, new() {
|
2026-01-25 11:16:54 +01:00
|
|
|
foreach (var player in Players.Values){
|
2025-12-19 17:54:50 +01:00
|
|
|
SendPacket(packet, player.peer, deliveryMethod);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2026-01-25 11:16:54 +01:00
|
|
|
public static void SendUpdate(GameEvent[] gevents, int pid) {
|
2025-12-19 17:54:50 +01:00
|
|
|
// SendPacket(new UpdatePlayerPacket{eventQueue = new EventQueue{Events = events}}, pid);
|
|
|
|
|
SendPacket(new UpdatePlayerPacket{events = gevents}, pid);
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public static void SendUpdateToAll(GameEvent[] gevents) {
|
|
|
|
|
// SendPacketToAll(new UpdatePlayerPacket{eventQueue = new EventQueue{Events = events}});
|
|
|
|
|
SendPacketToAll(new UpdatePlayerPacket{events = gevents});
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
public static void OnJoinReceived(JoinPacket packet, NetPeer peer) {
|
|
|
|
|
Console.WriteLine($"Received join from {packet.username} (pid: {(uint)peer.Id})");
|
|
|
|
|
|
2026-01-25 11:16:54 +01:00
|
|
|
ServerPlayer newPlayer = (Players[peer.Id] = new ServerPlayer {
|
2025-12-19 17:54:50 +01:00
|
|
|
peer = peer,
|
|
|
|
|
state = new PlayerState {
|
2026-01-25 11:16:54 +01:00
|
|
|
pid = peer.Id,
|
2026-02-26 16:24:55 +01:00
|
|
|
camera = GameLogic.START_CAMERA,
|
2026-03-12 22:33:35 +01:00
|
|
|
doorStates = [false, false, false],
|
|
|
|
|
power = Power.MAX_POWER_VALUE,
|
|
|
|
|
poweredOut = false
|
2025-12-19 17:54:50 +01:00
|
|
|
},
|
|
|
|
|
username = packet.username
|
|
|
|
|
});
|
2026-03-08 16:55:49 +01:00
|
|
|
|
|
|
|
|
|
2026-01-25 11:16:54 +01:00
|
|
|
if (Players.Count == 1){
|
2026-03-08 16:55:49 +01:00
|
|
|
newPlayer.state.officeTileId = 10;
|
|
|
|
|
newPlayer.state.neighbouringTiles = [5, 11, 15];
|
|
|
|
|
SendPacket(new JoinAcceptPacket { state = newPlayer.state }, peer);
|
2025-12-19 17:54:50 +01:00
|
|
|
P1 = newPlayer;
|
|
|
|
|
}
|
|
|
|
|
else{
|
2026-03-08 16:55:49 +01:00
|
|
|
newPlayer.state.officeTileId = 14;
|
|
|
|
|
newPlayer.state.neighbouringTiles = [19, 13, 9];
|
|
|
|
|
|
|
|
|
|
SendPacket(new JoinAcceptPacket { state = newPlayer.state }, peer);
|
2025-12-19 17:54:50 +01:00
|
|
|
P2 = newPlayer;
|
2026-03-08 16:55:49 +01:00
|
|
|
|
2026-03-11 22:35:30 +01:00
|
|
|
SendPacket(new OpponentInitPacket{state = newPlayer.state, username = newPlayer.username}, P1.peer);
|
|
|
|
|
SendPacket(new OpponentInitPacket{state = P1.state, username = P1.username}, P2.peer);
|
2026-03-08 16:55:49 +01:00
|
|
|
GameLogic.Init();
|
2025-12-19 17:54:50 +01:00
|
|
|
}
|
2026-03-08 16:55:49 +01:00
|
|
|
|
2025-12-19 17:54:50 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public static void OnNetworkReceive(NetPeer peer, NetPacketReader reader, DeliveryMethod deliveryMethod) {
|
|
|
|
|
processor.ReadAllPackets(reader, peer);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public static void OnPeerDisconnected(NetPeer peer, DisconnectInfo disconnectInfo) {
|
2026-03-21 21:23:33 +01:00
|
|
|
GameLogic.DeclareWinner(OtherPlayer(Players.Values.First(p => Equals(p.peer, peer))));
|
|
|
|
|
|
2025-12-19 17:54:50 +01:00
|
|
|
if (peer.Tag != null) {
|
2026-01-25 11:16:54 +01:00
|
|
|
Players.Remove(peer.Id);
|
2025-12-19 17:54:50 +01:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public static void OnCommandReceived(PlayerCommandPacket packet, NetPeer peer) {
|
2026-01-25 11:16:54 +01:00
|
|
|
CommandProcessor.Evaluate(packet.commands, peer.Id);
|
2025-12-19 17:54:50 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public static void Update() {
|
|
|
|
|
server.PollEvents();
|
|
|
|
|
// Console.WriteLine("update");
|
2026-03-08 16:55:49 +01:00
|
|
|
GameLogic.Update();
|
2025-12-19 17:54:50 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public static void Run(CancellationToken cancellationToken = default)
|
|
|
|
|
{
|
|
|
|
|
isRunning = true;
|
|
|
|
|
var stopwatch = Stopwatch.StartNew();
|
|
|
|
|
long previousTicks = 0;
|
|
|
|
|
|
|
|
|
|
while (isRunning && !cancellationToken.IsCancellationRequested)
|
|
|
|
|
{
|
|
|
|
|
long currentTicks = stopwatch.ElapsedMilliseconds;
|
|
|
|
|
long elapsed = currentTicks - previousTicks;
|
|
|
|
|
|
|
|
|
|
if (elapsed >= TargetTickTimeMs)
|
|
|
|
|
{
|
|
|
|
|
Update();
|
|
|
|
|
|
|
|
|
|
previousTicks = currentTicks;
|
|
|
|
|
if (elapsed > TargetTickTimeMs * 2)
|
|
|
|
|
{
|
|
|
|
|
Console.WriteLine($"Warning: Tick took {elapsed}ms (target: {TargetTickTimeMs}ms)");
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
|
|
|
|
|
int sleepTime = (int)(TargetTickTimeMs - elapsed - 2);
|
|
|
|
|
if (sleepTime > 0)
|
|
|
|
|
{
|
|
|
|
|
Thread.Sleep(sleepTime);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
while (stopwatch.ElapsedMilliseconds - previousTicks < TargetTickTimeMs){
|
|
|
|
|
Thread.SpinWait(1);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public static void Stop()
|
|
|
|
|
{
|
|
|
|
|
isRunning = false;
|
2026-03-21 21:23:33 +01:00
|
|
|
server.Stop();
|
2025-12-19 17:54:50 +01:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|