using System; using System.Text; using System.Net; using System.Linq; using System.Collections.Generic; using System.Threading; using System.Reflection; using System.IO; using System.Net.Http; using Newtonsoft.Json; using Lidgren.Network; namespace CoopServer { internal class IpInfo { public string ip { get; set; } public string country { get; set; } } internal class Server { private static readonly string _compatibleVersion = "V1_4"; private static long _currentTick = 0; public static readonly Settings MainSettings = Util.Read("Settings.xml"); private readonly Blocklist _mainBlocklist = Util.Read("Blocklist.xml"); private readonly Allowlist _mainAllowlist = Util.Read("Allowlist.xml"); public static NetServer MainNetServer; public static Resource RunningResource = null; public static Dictionary> Commands; public static readonly List Clients = new(); public Server() { Logging.Info("================"); Logging.Info($"Server version: {Assembly.GetCallingAssembly().GetName().Version}"); Logging.Info($"Compatible RAGECOOP versions: {_compatibleVersion.Replace('_', '.')}.x"); Logging.Info("================"); // 623c92c287cc392406e7aaaac1c0f3b0 = RAGECOOP NetPeerConfiguration config = new("623c92c287cc392406e7aaaac1c0f3b0") { MaximumConnections = MainSettings.MaxPlayers, Port = MainSettings.Port, EnableUPnP = MainSettings.UPnP }; config.EnableMessageType(NetIncomingMessageType.ConnectionApproval); config.EnableMessageType(NetIncomingMessageType.ConnectionLatencyUpdated); MainNetServer = new NetServer(config); MainNetServer.Start(); Logging.Info(string.Format("Server listening on {0}:{1}", config.LocalAddress.ToString(), config.Port)); if (MainSettings.UPnP) { Logging.Info(string.Format("Attempting to forward port {0}", MainSettings.Port)); if (MainNetServer.UPnP.ForwardPort(MainSettings.Port, "RAGECOOP server")) { Logging.Info(string.Format("Server available on {0}:{1}", MainNetServer.UPnP.GetExternalIP().ToString(), config.Port)); } else { Logging.Error("Port forwarding failed! Your router may not support UPnP."); Logging.Warning("If you and your friends can join this server, please ignore this error or set UPnP in Settings.xml to false!"); } } if (MainSettings.AnnounceSelf) { Logging.Info("Announcing to master server..."); #region -- MASTERSERVER -- new Thread(async () => { try { // TLS only ServicePointManager.Expect100Continue = true; ServicePointManager.SecurityProtocol = SecurityProtocolType.Tls13 | SecurityProtocolType.Tls12; ServicePointManager.ServerCertificateValidationCallback = delegate { return true; }; HttpClient httpClient = new(); IpInfo info; try { string data = await httpClient.GetStringAsync("https://ipinfo.io/json"); info = JsonConvert.DeserializeObject(data); } catch { info = new() { ip = MainNetServer.Configuration.LocalAddress.ToString(), country = "?" }; } while (!Program.ReadyToStop) { string msg = "{ " + "\"address\": \"" + info.ip + "\", " + "\"port\": \"" + MainSettings.Port + "\", " + "\"name\": \"" + MainSettings.Name + "\", " + "\"version\": \"" + _compatibleVersion.Replace("_", ".") + "\", " + "\"players\": \"" + MainNetServer.ConnectionsCount + "\", " + "\"maxPlayers\": \"" + MainSettings.MaxPlayers + "\", " + "\"allowlist\": \"" + _mainAllowlist.Username.Any() + "\", " + "\"mods\": \"" + MainSettings.ModsAllowed + "\", " + "\"npcs\": \"" + MainSettings.NpcsAllowed + "\", " + "\"country\": \"" + info.country + "\"" + " }"; HttpResponseMessage response = null; try { response = await httpClient.PostAsync(MainSettings.MasterServer, new StringContent(msg, Encoding.UTF8, "application/json")); if (response == null) { Logging.Error("MasterServer: Something went wrong!"); break; } } catch (Exception ex) { Logging.Error($"MasterServer: {ex.Message}"); break; } if (response.StatusCode != System.Net.HttpStatusCode.OK) { Logging.Error($"MasterServer: [{(int)response.StatusCode}]"); // Wait 5 seconds before trying again Thread.Sleep(5000); continue; } // Sleep for 12.5s Thread.Sleep(12500); } } catch (HttpRequestException ex) { Logging.Error($"MasterServer: {ex.InnerException.Message}"); } catch (Exception ex) { Logging.Error($"MasterServer: {ex.Message}"); } }).Start(); #endregion } if (!string.IsNullOrEmpty(MainSettings.Resource)) { Commands = new(); try { string resourcepath = AppDomain.CurrentDomain.BaseDirectory + "resources" + Path.DirectorySeparatorChar + MainSettings.Resource + ".dll"; Logging.Info($"Loading resource \"{MainSettings.Resource}.dll\"..."); Assembly asm = Assembly.LoadFrom(resourcepath); Type[] types = asm.GetExportedTypes(); IEnumerable validTypes = types.Where(t => !t.IsInterface && !t.IsAbstract).Where(t => typeof(ServerScript).IsAssignableFrom(t)); Type[] enumerable = validTypes as Type[] ?? validTypes.ToArray(); if (!enumerable.Any()) { Logging.Error("ERROR: No classes that inherit from ServerScript have been found in the assembly. Starting freeroam."); } else { if (Activator.CreateInstance(enumerable.ToArray()[0]) is ServerScript script) { RunningResource = new(script); } else { Logging.Warning("Could not create resource: it is null."); } } } catch (Exception e) { Logging.Error(e.InnerException.Message); } } Logging.Info("Searching for client-side files..."); DownloadManager.CheckForDirectoryAndFiles(); Listen(); } private void Listen() { Logging.Info("Listening for clients"); Logging.Info("Please use CTRL + C if you want to stop the server!"); while (!Program.ReadyToStop) { if (RunningResource != null) { RunningResource.InvokeTick(++_currentTick); } // Only new clients that did not receive files on connection will receive the current files in "clientside" if (DownloadManager.AnyFileExists) { lock (Clients) { Clients.ForEach(client => { if (!client.FilesSent) { DownloadManager.InsertClient(client.NetHandle); client.FilesSent = true; } }); } DownloadManager.Tick(); } NetIncomingMessage message; while ((message = MainNetServer.ReadMessage()) != null) { switch (message.MessageType) { case NetIncomingMessageType.ConnectionApproval: Logging.Info($"New incoming connection from: [{message.SenderConnection.RemoteEndPoint}]"); if (message.ReadByte() != (byte)PacketTypes.Handshake) { Logging.Info($"IP [{message.SenderConnection.RemoteEndPoint.Address}] was blocked, reason: Wrong packet!"); message.SenderConnection.Deny("Wrong packet!"); } else { try { int len = message.ReadInt32(); byte[] data = message.ReadBytes(len); Packets.Handshake packet = new(); packet.NetIncomingMessageToPacket(data); GetHandshake(message.SenderConnection, packet); } catch (Exception e) { Logging.Info($"IP [{message.SenderConnection.RemoteEndPoint.Address}] was blocked, reason: {e.Message}"); message.SenderConnection.Deny(e.Message); } } break; case NetIncomingMessageType.StatusChanged: NetConnectionStatus status = (NetConnectionStatus)message.ReadByte(); if (status == NetConnectionStatus.Disconnected) { long nethandle = message.SenderConnection.RemoteUniqueIdentifier; DownloadManager.RemoveClient(nethandle); SendPlayerDisconnectPacket(nethandle); } else if (status == NetConnectionStatus.Connected) { SendPlayerConnectPacket(message.SenderConnection); } break; case NetIncomingMessageType.Data: // Get packet type byte type = message.ReadByte(); switch (type) { case (byte)PacketTypes.FullSyncPlayer: { try { int len = message.ReadInt32(); byte[] data = message.ReadBytes(len); Packets.FullSyncPlayer packet = new(); packet.NetIncomingMessageToPacket(data); FullSyncPlayer(packet); } catch (Exception e) { DisconnectAndLog(message.SenderConnection, type, e); } } break; case (byte)PacketTypes.FullSyncPlayerVeh: { try { int len = message.ReadInt32(); byte[] data = message.ReadBytes(len); Packets.FullSyncPlayerVeh packet = new(); packet.NetIncomingMessageToPacket(data); FullSyncPlayerVeh(packet); } catch (Exception e) { DisconnectAndLog(message.SenderConnection, type, e); } } break; case (byte)PacketTypes.LightSyncPlayer: { try { int len = message.ReadInt32(); byte[] data = message.ReadBytes(len); Packets.LightSyncPlayer packet = new(); packet.NetIncomingMessageToPacket(data); LightSyncPlayer(packet); } catch (Exception e) { DisconnectAndLog(message.SenderConnection, type, e); } } break; case (byte)PacketTypes.LightSyncPlayerVeh: { try { int len = message.ReadInt32(); byte[] data = message.ReadBytes(len); Packets.LightSyncPlayerVeh packet = new(); packet.NetIncomingMessageToPacket(data); LightSyncPlayerVeh(packet); } catch (Exception e) { DisconnectAndLog(message.SenderConnection, type, e); } } break; case (byte)PacketTypes.ChatMessage: { try { int len = message.ReadInt32(); byte[] data = message.ReadBytes(len); Packets.ChatMessage packet = new(); packet.NetIncomingMessageToPacket(data); SendChatMessage(packet); } catch (Exception e) { DisconnectAndLog(message.SenderConnection, type, e); } } break; case (byte)PacketTypes.FullSyncNpc: { if (MainSettings.NpcsAllowed) { try { int len = message.ReadInt32(); byte[] data = message.ReadBytes(len); Packets.FullSyncNpc packet = new(); packet.NetIncomingMessageToPacket(data); FullSyncNpc(message.SenderConnection, packet); } catch (Exception e) { DisconnectAndLog(message.SenderConnection, type, e); } } else { message.SenderConnection.Disconnect("Npcs are not allowed!"); } } break; case (byte)PacketTypes.FullSyncNpcVeh: { if (MainSettings.NpcsAllowed) { try { int len = message.ReadInt32(); byte[] data = message.ReadBytes(len); Packets.FullSyncNpcVeh packet = new(); packet.NetIncomingMessageToPacket(data); FullSyncNpcVeh(message.SenderConnection, packet); } catch (Exception e) { DisconnectAndLog(message.SenderConnection, type, e); } } else { message.SenderConnection.Disconnect("Npcs are not allowed!"); } } break; case (byte)PacketTypes.NativeResponse: { try { int len = message.ReadInt32(); byte[] data = message.ReadBytes(len); Packets.NativeResponse packet = new(); packet.NetIncomingMessageToPacket(data); Client client = Clients.Find(x => x.NetHandle == message.SenderConnection.RemoteUniqueIdentifier); if (client != null) { if (client.Callbacks.ContainsKey(packet.ID)) { client.Callbacks[packet.ID].Invoke(packet.Args[0]); client.Callbacks.Remove(packet.ID); } } } catch (Exception e) { DisconnectAndLog(message.SenderConnection, type, e); } } break; case (byte)PacketTypes.Mod: { if (MainSettings.ModsAllowed) { try { int len = message.ReadInt32(); byte[] data = message.ReadBytes(len); Packets.Mod packet = new Packets.Mod(); packet.NetIncomingMessageToPacket(data); bool resourceResult = false; if (RunningResource != null) { if (RunningResource.InvokeModPacketReceived(packet.NetHandle, packet.Target, packet.Name, packet.CustomPacketID, packet.Bytes)) { resourceResult = true; } } if (!resourceResult && packet.Target != -1) { NetOutgoingMessage outgoingMessage = MainNetServer.CreateMessage(); packet.PacketToNetOutGoingMessage(outgoingMessage); if (packet.Target != 0) { NetConnection target = MainNetServer.Connections.Find(x => x.RemoteUniqueIdentifier == packet.Target); if (target == null) { Logging.Error($"[ModPacket] target \"{packet.Target}\" not found!"); } else { // Send back to target MainNetServer.SendMessage(outgoingMessage, target, NetDeliveryMethod.ReliableOrdered, (byte)ConnectionChannel.Mod); } } else { // Send back to all players MainNetServer.SendMessage(outgoingMessage, MainNetServer.Connections, NetDeliveryMethod.ReliableOrdered, (byte)ConnectionChannel.Mod); } } } catch (Exception e) { DisconnectAndLog(message.SenderConnection, type, e); } } else { message.SenderConnection.Disconnect("Mods are not allowed!"); } } break; case (byte)PacketTypes.FileTransferComplete: { try { if (DownloadManager.AnyFileExists) { int len = message.ReadInt32(); byte[] data = message.ReadBytes(len); Packets.FileTransferComplete packet = new(); packet.NetIncomingMessageToPacket(data); Client client = Clients.Find(x => x.NetHandle == message.SenderConnection.RemoteUniqueIdentifier); if (client != null && !client.FilesReceived) { DownloadManager.TryToRemoveClient(client.NetHandle, packet.ID); } } } catch (Exception e) { DisconnectAndLog(message.SenderConnection, type, e); } } break; default: Logging.Error("Unhandled Data / Packet type"); break; } break; case NetIncomingMessageType.ConnectionLatencyUpdated: { Client client = Clients.Find(x => x.NetHandle == message.SenderConnection.RemoteUniqueIdentifier); if (client != null) { client.Latency = message.ReadFloat(); } } break; case NetIncomingMessageType.ErrorMessage: Logging.Error(message.ReadString()); break; case NetIncomingMessageType.WarningMessage: Logging.Warning(message.ReadString()); break; case NetIncomingMessageType.DebugMessage: case NetIncomingMessageType.VerboseDebugMessage: Logging.Debug(message.ReadString()); break; default: Logging.Error(string.Format("Unhandled type: {0} {1} bytes {2} | {3}", message.MessageType, message.LengthBytes, message.DeliveryMethod, message.SequenceChannel)); break; } MainNetServer.Recycle(message); } // 16 milliseconds to sleep to reduce CPU usage Thread.Sleep(1000 / 60); } Logging.Warning("Server is shutting down!"); if (RunningResource != null) { // Waiting for resource... while (!RunningResource.ReadyToStop) { // 16 milliseconds to sleep to reduce CPU usage Thread.Sleep(1000 / 60); } } if (MainNetServer.Connections.Count > 0) { MainNetServer.Connections.ForEach(x => x.Disconnect("Server is shutting down!")); // We have to wait some time for all Disconnect() messages to be sent successfully // Sleep for 1 second Thread.Sleep(1000); } } private void DisconnectAndLog(NetConnection senderConnection, byte type, Exception e) { Logging.Error($"Error receiving a packet of type {type}"); Logging.Error(e.Message); Logging.Error(e.StackTrace); senderConnection.Disconnect(e.Message); } #region -- PLAYER -- // Before we approve the connection, we must shake hands private void GetHandshake(NetConnection local, Packets.Handshake packet) { Logging.Debug("New handshake from: [Name: " + packet.Username + " | Address: " + local.RemoteEndPoint.Address.ToString() + "]"); if (!packet.ModVersion.StartsWith(_compatibleVersion)) { local.Deny($"RAGECOOP version {_compatibleVersion.Replace('_', '.')}.x required!"); return; } if (string.IsNullOrWhiteSpace(packet.Username)) { local.Deny("Username is empty or contains spaces!"); return; } if (packet.Username.Any(p => !char.IsLetterOrDigit(p) && !(p == '_') && !(p=='-'))) { local.Deny("Username contains special chars!"); return; } if (_mainAllowlist.Username.Any() && !_mainAllowlist.Username.Contains(packet.Username.ToLower())) { local.Deny("This Username is not on the allow list!"); return; } if (_mainBlocklist.Username.Contains(packet.Username.ToLower())) { local.Deny("This Username has been blocked by this server!"); return; } if (_mainBlocklist.IP.Contains(local.RemoteEndPoint.ToString().Split(":")[0])) { local.Deny("This IP was blocked by this server!"); return; } if (Clients.Any(x => x.Player.Username.ToLower() == packet.Username.ToLower())) { local.Deny("Username is already taken!"); return; } long localNetHandle = local.RemoteUniqueIdentifier; Client tmpClient; // Add the player to Players lock (Clients) { Clients.Add( tmpClient = new Client() { NetHandle = localNetHandle, Player = new() { Username = packet.Username } } ); } NetOutgoingMessage outgoingMessage = MainNetServer.CreateMessage(); // Create a new handshake packet new Packets.Handshake() { NetHandle = localNetHandle, Username = string.Empty, ModVersion = string.Empty, NPCsAllowed = MainSettings.NpcsAllowed }.PacketToNetOutGoingMessage(outgoingMessage); // Accept the connection and send back a new handshake packet with the connection ID local.Approve(outgoingMessage); if (RunningResource != null) { RunningResource.InvokePlayerHandshake(tmpClient); } } // The connection has been approved, now we need to send all other players to the new player and the new player to all players private static void SendPlayerConnectPacket(NetConnection local) { Client localClient = Clients.Find(x => x.NetHandle == local.RemoteUniqueIdentifier); if (localClient == null) { local.Disconnect("No data found!"); return; } List clients; if ((clients = Util.FilterAllLocal(local)).Count > 0) { // Send all players to local clients.ForEach(targetPlayer => { long targetNetHandle = targetPlayer.RemoteUniqueIdentifier; Client targetClient = Clients.Find(x => x.NetHandle == targetNetHandle); if (targetClient != null) { NetOutgoingMessage outgoingMessage = MainNetServer.CreateMessage(); new Packets.PlayerConnect() { NetHandle = targetNetHandle, Username = targetClient.Player.Username }.PacketToNetOutGoingMessage(outgoingMessage); MainNetServer.SendMessage(outgoingMessage, local, NetDeliveryMethod.ReliableOrdered, 0); } }); // Send local to all players NetOutgoingMessage outgoingMessage = MainNetServer.CreateMessage(); new Packets.PlayerConnect() { NetHandle = local.RemoteUniqueIdentifier, Username = localClient.Player.Username }.PacketToNetOutGoingMessage(outgoingMessage); MainNetServer.SendMessage(outgoingMessage, clients, NetDeliveryMethod.ReliableOrdered, 0); } if (RunningResource != null) { RunningResource.InvokePlayerConnected(localClient); } else { Logging.Info($"Player {localClient.Player.Username} connected!"); } if (!string.IsNullOrEmpty(MainSettings.WelcomeMessage)) { SendChatMessage(new Packets.ChatMessage() { Username = "Server", Message = MainSettings.WelcomeMessage }, new List() { local }); } } // Send all players a message that someone has left the server private static void SendPlayerDisconnectPacket(long nethandle) { List clients = MainNetServer.Connections.FindAll(x => x.RemoteUniqueIdentifier != nethandle); if (clients.Count > 0) { NetOutgoingMessage outgoingMessage = MainNetServer.CreateMessage(); new Packets.PlayerDisconnect() { NetHandle = nethandle }.PacketToNetOutGoingMessage(outgoingMessage); MainNetServer.SendMessage(outgoingMessage, clients, NetDeliveryMethod.ReliableOrdered, 0); } Client localClient = Clients.FirstOrDefault(x => x.NetHandle == nethandle); if (localClient == null) { return; } Clients.Remove(localClient); if (RunningResource != null) { RunningResource.InvokePlayerDisconnected(localClient); } else { Logging.Info($"Player {localClient.Player.Username} disconnected!"); } } private static void FullSyncPlayer(Packets.FullSyncPlayer packet) { Client client = Util.GetClientByNetHandle(packet.NetHandle); if (client == null) { return; } // Save the new data client.Player.PedHandle = packet.PedHandle; client.Player.IsInVehicle = false; client.Player.Position = packet.Position; client.Player.Health = packet.Health; // Override the latency packet.Latency = client.Latency; MainNetServer.Connections.FindAll(x => x.RemoteUniqueIdentifier != packet.NetHandle).ForEach(x => { NetOutgoingMessage outgoingMessage = MainNetServer.CreateMessage(); if (Clients.First(y => y.NetHandle == x.RemoteUniqueIdentifier).Player.IsInRangeOf(packet.Position, 550f)) { packet.PacketToNetOutGoingMessage(outgoingMessage); MainNetServer.SendMessage(outgoingMessage, x, NetDeliveryMethod.UnreliableSequenced, (byte)ConnectionChannel.PlayerFull); } else { new Packets.SuperLightSync() { NetHandle = packet.NetHandle, Position = packet.Position, Latency = packet.Latency }.PacketToNetOutGoingMessage(outgoingMessage); MainNetServer.SendMessage(outgoingMessage, x, NetDeliveryMethod.UnreliableSequenced, (byte)ConnectionChannel.PlayerSuperLight); } }); if (RunningResource != null) { RunningResource.InvokePlayerUpdate(client); } } private static void FullSyncPlayerVeh(Packets.FullSyncPlayerVeh packet) { Client client = Util.GetClientByNetHandle(packet.NetHandle); if (client == null) { return; } // Save the new data client.Player.PedHandle = packet.PedHandle; client.Player.VehicleHandle = packet.VehicleHandle; client.Player.IsInVehicle = true; client.Player.Position = packet.Position; client.Player.Health = packet.Health; // Override the latency packet.Latency = client.Latency; MainNetServer.Connections.FindAll(x => x.RemoteUniqueIdentifier != packet.NetHandle).ForEach(x => { NetOutgoingMessage outgoingMessage = MainNetServer.CreateMessage(); if (Clients.First(y => y.NetHandle == x.RemoteUniqueIdentifier).Player.IsInRangeOf(packet.Position, 550f)) { packet.PacketToNetOutGoingMessage(outgoingMessage); MainNetServer.SendMessage(outgoingMessage, x, NetDeliveryMethod.UnreliableSequenced, (byte)ConnectionChannel.PlayerFull); } else { new Packets.SuperLightSync() { NetHandle = packet.NetHandle, Position = packet.Position, Latency = packet.Latency }.PacketToNetOutGoingMessage(outgoingMessage); MainNetServer.SendMessage(outgoingMessage, x, NetDeliveryMethod.UnreliableSequenced, (byte)ConnectionChannel.PlayerSuperLight); } }); if (RunningResource != null) { RunningResource.InvokePlayerUpdate(client); } } private static void LightSyncPlayer(Packets.LightSyncPlayer packet) { Client client = Util.GetClientByNetHandle(packet.NetHandle); if (client == null) { return; } // Save the new data client.Player.Position = packet.Position; client.Player.Health = packet.Health; // Override the latency packet.Latency = client.Latency; MainNetServer.Connections.FindAll(x => x.RemoteUniqueIdentifier != packet.NetHandle).ForEach(x => { NetOutgoingMessage outgoingMessage = MainNetServer.CreateMessage(); if (Clients.First(y => y.NetHandle == x.RemoteUniqueIdentifier).Player.IsInRangeOf(packet.Position, 550f)) { packet.PacketToNetOutGoingMessage(outgoingMessage); MainNetServer.SendMessage(outgoingMessage, x, NetDeliveryMethod.UnreliableSequenced, (byte)ConnectionChannel.PlayerLight); } else { new Packets.SuperLightSync() { NetHandle = packet.NetHandle, Position = packet.Position, Latency = packet.Latency }.PacketToNetOutGoingMessage(outgoingMessage); MainNetServer.SendMessage(outgoingMessage, x, NetDeliveryMethod.UnreliableSequenced, (byte)ConnectionChannel.PlayerSuperLight); } }); if (RunningResource != null) { RunningResource.InvokePlayerUpdate(client); } } private static void LightSyncPlayerVeh(Packets.LightSyncPlayerVeh packet) { Client client = Util.GetClientByNetHandle(packet.NetHandle); if (client == null) { return; } // Save the new data client.Player.Position = packet.Position; client.Player.Health = packet.Health; // Override the latency packet.Latency = client.Latency; MainNetServer.Connections.FindAll(x => x.RemoteUniqueIdentifier != packet.NetHandle).ForEach(x => { NetOutgoingMessage outgoingMessage = MainNetServer.CreateMessage(); if (Clients.First(y => y.NetHandle == x.RemoteUniqueIdentifier).Player.IsInRangeOf(packet.Position, 550f)) { packet.PacketToNetOutGoingMessage(outgoingMessage); MainNetServer.SendMessage(outgoingMessage, x, NetDeliveryMethod.UnreliableSequenced, (byte)ConnectionChannel.PlayerLight); } else { new Packets.SuperLightSync() { NetHandle = packet.NetHandle, Position = packet.Position, Latency = packet.Latency }.PacketToNetOutGoingMessage(outgoingMessage); MainNetServer.SendMessage(outgoingMessage, x, NetDeliveryMethod.UnreliableSequenced, (byte)ConnectionChannel.PlayerSuperLight); } }); if (RunningResource != null) { RunningResource.InvokePlayerUpdate(client); } } // Send a message to targets or all players private static void SendChatMessage(Packets.ChatMessage packet, List targets = null) { if (RunningResource != null) { if (packet.Message.StartsWith('/')) { string[] cmdArgs = packet.Message.Split(" "); string cmdName = cmdArgs[0].Remove(0, 1); if (Commands.Any(x => x.Key.Name == cmdName)) { string[] argsWithoutCmd = cmdArgs.Skip(1).ToArray(); CommandContext ctx = new() { Client = Clients.Find(x => x.Player.Username == packet.Username), Args = argsWithoutCmd }; KeyValuePair> command = Commands.First(x => x.Key.Name == cmdName); if (command.Key.Usage != null && command.Key.ArgsLength != argsWithoutCmd.Length) { NetConnection userConnection = Util.GetConnectionByUsername(packet.Username); if (userConnection == default) { return; } SendChatMessage("Server", command.Key.Usage, userConnection); return; } command.Value.Invoke(ctx); } else { NetConnection userConnection = Util.GetConnectionByUsername(packet.Username); if (userConnection == default) { return; } SendChatMessage("Server", "Command not found!", userConnection); } return; } if (RunningResource.InvokeChatMessage(packet.Username, packet.Message)) { return; } } packet.Message = packet.Message.Replace("~", ""); SendChatMessage(packet.Username, packet.Message, targets); Logging.Info(packet.Username + ": " + packet.Message); } internal static void SendChatMessage(string username, string message, List targets = null) { NetOutgoingMessage outgoingMessage = MainNetServer.CreateMessage(); new Packets.ChatMessage() { Username = username, Message = message }.PacketToNetOutGoingMessage(outgoingMessage); MainNetServer.SendMessage(outgoingMessage, targets ?? MainNetServer.Connections, NetDeliveryMethod.ReliableOrdered, (byte)ConnectionChannel.Chat); } internal static void SendChatMessage(string username, string message, NetConnection target) { SendChatMessage(username, message, new List() { target }); } #endregion #region -- NPC -- private static void FullSyncNpc(NetConnection local, Packets.FullSyncNpc packet) { List clients; if ((clients = Util.GetAllInRange(packet.Position, 550f, local)).Count == 0) { return; } NetOutgoingMessage outgoingMessage = MainNetServer.CreateMessage(); packet.PacketToNetOutGoingMessage(outgoingMessage); MainNetServer.SendMessage(outgoingMessage, clients, NetDeliveryMethod.UnreliableSequenced, (byte)ConnectionChannel.NPCFull); } private static void FullSyncNpcVeh(NetConnection local, Packets.FullSyncNpcVeh packet) { List clients; if ((clients = Util.GetAllInRange(packet.Position, 550f, local)).Count == 0) { return; } NetOutgoingMessage outgoingMessage = MainNetServer.CreateMessage(); packet.PacketToNetOutGoingMessage(outgoingMessage); MainNetServer.SendMessage(outgoingMessage, clients, NetDeliveryMethod.UnreliableSequenced, (byte)ConnectionChannel.NPCFull); } #endregion public static void RegisterCommand(string name, string usage, short argsLength, Action callback) { Command command = new(name) { Usage = usage, ArgsLength = argsLength }; if (Commands.ContainsKey(command)) { throw new Exception("Command \"" + command.Name + "\" was already been registered!"); } Commands.Add(command, callback); } public static void RegisterCommand(string name, Action callback) { Command command = new(name); if (Commands.ContainsKey(command)) { throw new Exception("Command \"" + command.Name + "\" was already been registered!"); } Commands.Add(command, callback); } public static void RegisterCommands() { IEnumerable commands = typeof(T).GetMethods().Where(method => method.GetCustomAttributes(typeof(Command), false).Any()); foreach (MethodInfo method in commands) { Command attribute = method.GetCustomAttribute(true); RegisterCommand(attribute.Name, attribute.Usage, attribute.ArgsLength, (Action)Delegate.CreateDelegate(typeof(Action), method)); } } } }