mirror of
https://github.com/Sanae6/SmoOnlineServer.git
synced 2024-11-22 03:05:16 +00:00
664 lines
No EOL
26 KiB
C#
664 lines
No EOL
26 KiB
C#
using System.Collections.Concurrent;
|
|
using System.Net;
|
|
using System.Numerics;
|
|
using System.Text;
|
|
using System.Text.Json;
|
|
using Server;
|
|
using Shared;
|
|
using Shared.Packet.Packets;
|
|
using Timer = System.Timers.Timer;
|
|
|
|
Server.Server server = new Server.Server();
|
|
HashSet<int> shineBag = new HashSet<int>();
|
|
CancellationTokenSource cts = new CancellationTokenSource();
|
|
bool restartRequested = false;
|
|
Task listenTask = server.Listen(cts.Token);
|
|
Logger consoleLogger = new Logger("Console");
|
|
DiscordBot bot = new DiscordBot();
|
|
await bot.Run();
|
|
|
|
async Task PersistShines()
|
|
{
|
|
if (!Settings.Instance.PersistShines.Enabled)
|
|
{
|
|
return;
|
|
}
|
|
|
|
try
|
|
{
|
|
string shineJson = JsonSerializer.Serialize(shineBag);
|
|
await File.WriteAllTextAsync(Settings.Instance.PersistShines.Filename, shineJson);
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
consoleLogger.Error(ex);
|
|
}
|
|
}
|
|
|
|
async Task LoadShines()
|
|
{
|
|
if (!Settings.Instance.PersistShines.Enabled)
|
|
{
|
|
return;
|
|
}
|
|
|
|
try
|
|
{
|
|
string shineJson = await File.ReadAllTextAsync(Settings.Instance.PersistShines.Filename);
|
|
var loadedShines = JsonSerializer.Deserialize<HashSet<int>>(shineJson);
|
|
|
|
if (loadedShines is not null) shineBag = loadedShines;
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
consoleLogger.Error(ex);
|
|
}
|
|
}
|
|
|
|
// Load shines table from file
|
|
await LoadShines();
|
|
|
|
server.ClientJoined += (c, _) => {
|
|
if (Settings.Instance.BanList.Enabled
|
|
&& (Settings.Instance.BanList.Players.Contains(c.Id)
|
|
|| Settings.Instance.BanList.IpAddresses.Contains(
|
|
((IPEndPoint) c.Socket!.RemoteEndPoint!).Address.ToString())))
|
|
throw new Exception($"Banned player attempted join: {c.Name}");
|
|
c.Metadata["shineSync"] = new ConcurrentBag<int>();
|
|
c.Metadata["loadedSave"] = false;
|
|
c.Metadata["scenario"] = (byte?) 0;
|
|
c.Metadata["2d"] = false;
|
|
c.Metadata["speedrun"] = false;
|
|
foreach (Client client in server.ClientsConnected) {
|
|
try {
|
|
c.Send((GamePacket) client.Metadata["lastGamePacket"]!, client).Wait();
|
|
} catch {
|
|
// lol who gives a fuck
|
|
}
|
|
}
|
|
};
|
|
|
|
async Task ClientSyncShineBag(Client client) {
|
|
if (!Settings.Instance.Shines.Enabled) return;
|
|
try {
|
|
if ((bool?) client.Metadata["speedrun"] ?? false) return;
|
|
ConcurrentBag<int> clientBag = (ConcurrentBag<int>) (client.Metadata["shineSync"] ??= new ConcurrentBag<int>());
|
|
foreach (int shine in shineBag.Except(clientBag).ToArray()) {
|
|
clientBag.Add(shine);
|
|
await client.Send(new ShinePacket {
|
|
ShineId = shine
|
|
});
|
|
}
|
|
} catch {
|
|
// errors that can happen when sending will crash the server :)
|
|
}
|
|
}
|
|
|
|
async void SyncShineBag() {
|
|
try {
|
|
await PersistShines();
|
|
await Parallel.ForEachAsync(server.Clients.ToArray(), async (client, _) => await ClientSyncShineBag(client));
|
|
} catch {
|
|
// errors that can happen shines change will crash the server :)
|
|
}
|
|
}
|
|
|
|
Timer timer = new Timer(120000);
|
|
timer.AutoReset = true;
|
|
timer.Enabled = true;
|
|
timer.Elapsed += (_, _) => { SyncShineBag(); };
|
|
timer.Start();
|
|
|
|
float MarioSize(bool is2d) => is2d ? 180 : 160;
|
|
|
|
server.PacketHandler = (c, p) => {
|
|
switch (p) {
|
|
case GamePacket gamePacket: {
|
|
c.Logger.Info($"Got game packet {gamePacket.Stage}->{gamePacket.ScenarioNum}");
|
|
c.Metadata["scenario"] = gamePacket.ScenarioNum;
|
|
c.Metadata["2d"] = gamePacket.Is2d;
|
|
c.Metadata["lastGamePacket"] = gamePacket;
|
|
switch (gamePacket.Stage) {
|
|
case "CapWorldHomeStage" when gamePacket.ScenarioNum == 0:
|
|
c.Metadata["speedrun"] = true;
|
|
((ConcurrentBag<int>) (c.Metadata["shineSync"] ??= new ConcurrentBag<int>())).Clear();
|
|
shineBag.Clear();
|
|
Task.Run(async () => {
|
|
await PersistShines();
|
|
});
|
|
c.Logger.Info("Entered Cap on new save, preventing moon sync until Cascade");
|
|
break;
|
|
case "WaterfallWorldHomeStage":
|
|
bool wasSpeedrun = (bool) c.Metadata["speedrun"]!;
|
|
c.Metadata["speedrun"] = false;
|
|
if (wasSpeedrun)
|
|
Task.Run(async () => {
|
|
c.Logger.Info("Entered Cascade with moon sync disabled, enabling moon sync");
|
|
await Task.Delay(15000);
|
|
await ClientSyncShineBag(c);
|
|
});
|
|
break;
|
|
}
|
|
|
|
if (Settings.Instance.Scenario.MergeEnabled) {
|
|
server.BroadcastReplace(gamePacket, c, (from, to, gp) => {
|
|
gp.ScenarioNum = (byte?) to.Metadata["scenario"] ?? 200;
|
|
#pragma warning disable CS4014
|
|
to.Send(gp, from)
|
|
.ContinueWith(x => { if (x.Exception != null) { consoleLogger.Error(x.Exception.ToString()); } });
|
|
#pragma warning restore CS4014
|
|
});
|
|
return false;
|
|
}
|
|
|
|
break;
|
|
}
|
|
case TagPacket tagPacket: {
|
|
if ((tagPacket.UpdateType & TagPacket.TagUpdate.State) != 0) c.Metadata["seeking"] = tagPacket.IsIt;
|
|
if ((tagPacket.UpdateType & TagPacket.TagUpdate.Time) != 0)
|
|
c.Metadata["time"] = new Time(tagPacket.Minutes, tagPacket.Seconds, DateTime.Now);
|
|
break;
|
|
}
|
|
case CostumePacket:
|
|
#pragma warning disable CS4014
|
|
ClientSyncShineBag(c); //no point logging since entire def has try/catch
|
|
#pragma warning restore CS4014
|
|
c.Metadata["loadedSave"] = true;
|
|
break;
|
|
case ShinePacket shinePacket: {
|
|
if (c.Metadata["loadedSave"] is false) break;
|
|
ConcurrentBag<int> playerBag = (ConcurrentBag<int>)c.Metadata["shineSync"]!;
|
|
shineBag.Add(shinePacket.ShineId);
|
|
if (playerBag.Contains(shinePacket.ShineId)) break;
|
|
c.Logger.Info($"Got moon {shinePacket.ShineId}");
|
|
playerBag.Add(shinePacket.ShineId);
|
|
SyncShineBag();
|
|
break;
|
|
}
|
|
case PlayerPacket playerPacket when Settings.Instance.Flip.Enabled
|
|
&& Settings.Instance.Flip.Pov is FlipOptions.Both or FlipOptions.Others
|
|
&& Settings.Instance.Flip.Players.Contains(c.Id): {
|
|
playerPacket.Position += Vector3.UnitY * MarioSize((bool) c.Metadata["2d"]!);
|
|
playerPacket.Rotation *= Quaternion.CreateFromRotationMatrix(Matrix4x4.CreateRotationX(MathF.PI))
|
|
* Quaternion.CreateFromRotationMatrix(Matrix4x4.CreateRotationY(MathF.PI));
|
|
#pragma warning disable CS4014
|
|
server.Broadcast(playerPacket, c)
|
|
.ContinueWith(x => { if (x.Exception != null) { consoleLogger.Error(x.Exception.ToString()); } });
|
|
#pragma warning restore CS4014
|
|
return false;
|
|
}
|
|
case PlayerPacket playerPacket when Settings.Instance.Flip.Enabled
|
|
&& Settings.Instance.Flip.Pov is FlipOptions.Both or FlipOptions.Self
|
|
&& !Settings.Instance.Flip.Players.Contains(c.Id): {
|
|
server.BroadcastReplace(playerPacket, c, (from, to, sp) => {
|
|
if (Settings.Instance.Flip.Players.Contains(to.Id)) {
|
|
sp.Position += Vector3.UnitY * MarioSize((bool) c.Metadata["2d"]!);
|
|
sp.Rotation *= Quaternion.CreateFromRotationMatrix(Matrix4x4.CreateRotationX(MathF.PI))
|
|
* Quaternion.CreateFromRotationMatrix(Matrix4x4.CreateRotationY(MathF.PI));
|
|
}
|
|
#pragma warning disable CS4014
|
|
to.Send(sp, from)
|
|
.ContinueWith(x => { if (x.Exception != null) { consoleLogger.Error(x.Exception.ToString()); } });
|
|
#pragma warning restore CS4014
|
|
});
|
|
return false;
|
|
}
|
|
}
|
|
|
|
return true;
|
|
};
|
|
|
|
(HashSet<string> failToFind, HashSet<Client> toActUpon, List<(string arg, IEnumerable<string> amb)> ambig) MultiUserCommandHelper(string[] args) {
|
|
HashSet<string> failToFind = new();
|
|
HashSet<Client> toActUpon;
|
|
List<(string arg, IEnumerable<string> amb)> ambig = new();
|
|
if (args[0] == "*")
|
|
toActUpon = new(server.Clients.Where(c => c.Connected));
|
|
else {
|
|
toActUpon = args[0] == "!*" ? new(server.Clients.Where(c => c.Connected)) : new();
|
|
for (int i = (args[0] == "!*" ? 1 : 0); i < args.Length; i++) {
|
|
string arg = args[i];
|
|
IEnumerable<Client> search = server.Clients.Where(c => c.Connected &&
|
|
(c.Name.ToLower().StartsWith(arg.ToLower()) || (Guid.TryParse(arg, out Guid res) && res == c.Id)));
|
|
if (!search.Any())
|
|
failToFind.Add(arg); //none found
|
|
else if (search.Count() > 1) {
|
|
Client? exact = search.FirstOrDefault(x => x.Name == arg);
|
|
if (!ReferenceEquals(exact, null)) {
|
|
//even though multiple matches, since exact match, it isn't ambiguous
|
|
if (args[0] == "!*")
|
|
toActUpon.Remove(exact);
|
|
else
|
|
toActUpon.Add(exact);
|
|
}
|
|
else {
|
|
if (!ambig.Any(x => x.arg == arg))
|
|
ambig.Add((arg, search.Select(x => x.Name))); //more than one match
|
|
foreach (var rem in search.ToList()) //need copy because can't remove from list while iterating over it
|
|
toActUpon.Remove(rem);
|
|
}
|
|
}
|
|
else {
|
|
//only one match, so autocomplete
|
|
if (args[0] == "!*")
|
|
toActUpon.Remove(search.First());
|
|
else
|
|
toActUpon.Add(search.First());
|
|
}
|
|
}
|
|
}
|
|
return (failToFind, toActUpon, ambig);
|
|
}
|
|
|
|
CommandHandler.RegisterCommand("rejoin", args => {
|
|
if (args.Length == 0) {
|
|
return "Usage: rejoin <* | !* (usernames to not rejoin...) | (usernames to rejoin...)>";
|
|
}
|
|
|
|
var res = MultiUserCommandHelper(args);
|
|
|
|
StringBuilder sb = new StringBuilder();
|
|
sb.Append(res.toActUpon.Count > 0 ? "Rejoined: " + string.Join(", ", res.toActUpon.Select(x => $"\"{x.Name}\"")) : "");
|
|
sb.Append(res.failToFind.Count > 0 ? "\nFailed to find matches for: " + string.Join(", ", res.failToFind.Select(x => $"\"{x.ToLower()}\"")) : "");
|
|
if (res.ambig.Count > 0) {
|
|
res.ambig.ForEach(x => {
|
|
sb.Append($"\nAmbiguous for \"{x.arg}\": {string.Join(", ", x.amb.Select(x => $"\"{x}\""))}");
|
|
});
|
|
}
|
|
|
|
foreach (Client user in res.toActUpon) {
|
|
user.Dispose();
|
|
}
|
|
|
|
return sb.ToString();
|
|
});
|
|
|
|
CommandHandler.RegisterCommand("crash", args => {
|
|
if (args.Length == 0) {
|
|
return "Usage: crash <* | !* (usernames to not crash...) | (usernames to crash...)>";
|
|
}
|
|
|
|
var res = MultiUserCommandHelper(args);
|
|
|
|
StringBuilder sb = new StringBuilder();
|
|
sb.Append(res.toActUpon.Count > 0 ? "Crashed: " + string.Join(", ", res.toActUpon.Select(x => $"\"{x.Name}\"")) : "");
|
|
sb.Append(res.failToFind.Count > 0 ? "\nFailed to find matches for: " + string.Join(", ", res.failToFind.Select(x => $"\"{x.ToLower()}\"")) : "");
|
|
if (res.ambig.Count > 0) {
|
|
res.ambig.ForEach(x => {
|
|
sb.Append($"\nAmbiguous for \"{x.arg}\": {string.Join(", ", x.amb.Select(x => $"\"{x}\""))}");
|
|
});
|
|
}
|
|
|
|
foreach (Client user in res.toActUpon) {
|
|
Task.Run(async () => {
|
|
await user.Send(new ChangeStagePacket {
|
|
Id = "$among$us/SubArea",
|
|
Stage = "$agogusStage",
|
|
Scenario = 21,
|
|
SubScenarioType = 69 // invalid id
|
|
});
|
|
user.Dispose();
|
|
});
|
|
}
|
|
|
|
return sb.ToString();
|
|
});
|
|
|
|
CommandHandler.RegisterCommand("ban", args => {
|
|
if (args.Length == 0) {
|
|
return "Usage: ban <* | !* (usernames to not ban...) | (usernames to ban...)>";
|
|
}
|
|
|
|
var res = MultiUserCommandHelper(args);
|
|
|
|
StringBuilder sb = new StringBuilder();
|
|
sb.Append(res.toActUpon.Count > 0 ? "Banned: " + string.Join(", ", res.toActUpon.Select(x => $"\"{x.Name}\"")) : "");
|
|
sb.Append(res.failToFind.Count > 0 ? "\nFailed to find matches for: " + string.Join(", ", res.failToFind.Select(x => $"\"{x.ToLower()}\"")) : "");
|
|
if (res.ambig.Count > 0) {
|
|
res.ambig.ForEach(x => {
|
|
sb.Append($"\nAmbiguous for \"{x.arg}\": {string.Join(", ", x.amb.Select(x => $"\"{x}\""))}");
|
|
});
|
|
}
|
|
|
|
foreach (Client user in res.toActUpon) {
|
|
Task.Run(async () => {
|
|
await user.Send(new ChangeStagePacket {
|
|
Id = "$agogus/banned4lyfe",
|
|
Stage = "$ejected",
|
|
Scenario = 69,
|
|
SubScenarioType = 21 // invalid id
|
|
});
|
|
IPEndPoint? endpoint = (IPEndPoint?) user.Socket?.RemoteEndPoint;
|
|
Settings.Instance.BanList.Players.Add(user.Id);
|
|
if (endpoint != null) Settings.Instance.BanList.IpAddresses.Add(endpoint.ToString());
|
|
user.Dispose();
|
|
});
|
|
}
|
|
|
|
Settings.SaveSettings();
|
|
return sb.ToString();
|
|
});
|
|
|
|
CommandHandler.RegisterCommand("send", args => {
|
|
const string optionUsage = "Usage: send <stage> <id> <scenario[-1..127]> <player/*>";
|
|
if (args.Length < 4)
|
|
return optionUsage;
|
|
|
|
string stage = args[0];
|
|
string id = args[1];
|
|
|
|
if (Constants.MapNames.TryGetValue(stage.ToLower(), out string? mapName)) {
|
|
stage = mapName;
|
|
}
|
|
|
|
if (!stage.Contains("Stage") && !stage.Contains("Zone")) {
|
|
return "Invalid Stage Name! ```cap -> Cap Kingdom\ncascade -> Cascade Kingdom\nsand -> Sand Kingdom\nlake -> Lake Kingdom\nwooded -> Wooded Kingdom\ncloud -> Cloud Kingdom\nlost -> Lost Kingdom\nmetro -> Metro Kingdom\nsea -> Sea Kingdom\nsnow -> Snow Kingdom\nlunch -> Luncheon Kingdom\nruined -> Ruined Kingdom\nbowser -> Bowser's Kingdom\nmoon -> Moon Kingdom\nmush -> Mushroom Kingdom\ndark -> Dark Side\ndarker -> Darker Side```";
|
|
}
|
|
|
|
if (!sbyte.TryParse(args[2], out sbyte scenario) || scenario < -1)
|
|
return $"Invalid scenario number {args[2]} (range: [-1 to 127])";
|
|
Client[] players = args[3] == "*"
|
|
? server.Clients.Where(c => c.Connected).ToArray()
|
|
: server.Clients.Where(c =>
|
|
c.Connected
|
|
&& args[3..].Any(x => c.Name.StartsWith(x) || (Guid.TryParse(x, out Guid result) && result == c.Id)))
|
|
.ToArray();
|
|
Parallel.ForEachAsync(players, async (c, _) => {
|
|
await c.Send(new ChangeStagePacket {
|
|
Stage = stage,
|
|
Id = id,
|
|
Scenario = scenario,
|
|
SubScenarioType = 0
|
|
});
|
|
}).Wait();
|
|
return $"Sent players to {stage}:{scenario}";
|
|
});
|
|
|
|
CommandHandler.RegisterCommand("sendall", args => {
|
|
const string optionUsage = "Usage: sendall <stage>";
|
|
if (args.Length < 1)
|
|
return optionUsage;
|
|
|
|
string stage = args[0];
|
|
|
|
if (Constants.MapNames.TryGetValue(stage.ToLower(), out string? mapName)) {
|
|
stage = mapName;
|
|
}
|
|
|
|
if (!stage.Contains("Stage") && !stage.Contains("Zone")) {
|
|
return "Invalid Stage Name! ```cap -> Cap Kingdom\ncascade -> Cascade Kingdom\nsand -> Sand Kingdom\nlake -> Lake Kingdom\nwooded -> Wooded Kingdom\ncloud -> Cloud Kingdom\nlost -> Lost Kingdom\nmetro -> Metro Kingdom\nsea -> Sea Kingdom\nsnow -> Snow Kingdom\nlunch -> Luncheon Kingdom\nruined -> Ruined Kingdom\nbowser -> Bowser's Kingdom\nmoon -> Moon Kingdom\nmush -> Mushroom Kingdom\ndark -> Dark Side\ndarker -> Darker Side```";
|
|
}
|
|
|
|
Client[] players = server.Clients.Where(c => c.Connected).ToArray();
|
|
|
|
Parallel.ForEachAsync(players, async (c, _) => {
|
|
await c.Send(new ChangeStagePacket {
|
|
Stage = stage,
|
|
Id = "",
|
|
Scenario = -1,
|
|
SubScenarioType = 0
|
|
});
|
|
}).Wait();
|
|
|
|
return $"Sent players to {stage}:{-1}";
|
|
});
|
|
|
|
CommandHandler.RegisterCommand("scenario", args => {
|
|
const string optionUsage = "Valid options: merge [true/false]";
|
|
if (args.Length < 1)
|
|
return optionUsage;
|
|
switch (args[0]) {
|
|
case "merge" when args.Length == 2: {
|
|
if (bool.TryParse(args[1], out bool result)) {
|
|
Settings.Instance.Scenario.MergeEnabled = result;
|
|
Settings.SaveSettings();
|
|
return result ? "Enabled scenario merge" : "Disabled scenario merge";
|
|
}
|
|
|
|
return optionUsage;
|
|
}
|
|
case "merge" when args.Length == 1: {
|
|
return $"Scenario merging is {Settings.Instance.Scenario.MergeEnabled}";
|
|
}
|
|
default:
|
|
return optionUsage;
|
|
}
|
|
});
|
|
|
|
CommandHandler.RegisterCommand("tag", args => {
|
|
const string optionUsage =
|
|
"Valid options:\n\ttime <user/*> <minutes[0-65535]> <seconds[0-59]>\n\tseeking <user/*> <true/false>\n\tstart <time> <seekers>";
|
|
if (args.Length < 3)
|
|
return optionUsage;
|
|
switch (args[0]) {
|
|
case "time" when args.Length == 4: {
|
|
if (args[1] != "*" && server.Clients.All(x => x.Name != args[1])) return $"Cannot find user {args[1]}";
|
|
Client? client = server.Clients.FirstOrDefault(x => x.Name == args[1]);
|
|
if (!ushort.TryParse(args[2], out ushort minutes))
|
|
return $"Invalid time for minutes {args[2]} (range: 0-65535)";
|
|
if (!byte.TryParse(args[3], out byte seconds) || seconds >= 60)
|
|
return $"Invalid time for seconds {args[3]} (range: 0-59)";
|
|
TagPacket tagPacket = new TagPacket {
|
|
UpdateType = TagPacket.TagUpdate.Time,
|
|
Minutes = minutes,
|
|
Seconds = seconds
|
|
};
|
|
if (args[1] == "*")
|
|
server.Broadcast(tagPacket);
|
|
else
|
|
client?.Send(tagPacket);
|
|
return $"Set time for {(args[1] == "*" ? "everyone" : args[1])} to {minutes}:{seconds}";
|
|
}
|
|
case "seeking" when args.Length == 3: {
|
|
if (args[1] != "*" && server.Clients.All(x => x.Name != args[1])) return $"Cannot find user {args[1]}";
|
|
Client? client = server.Clients.FirstOrDefault(x => x.Name == args[1]);
|
|
if (!bool.TryParse(args[2], out bool seeking)) return $"Usage: tag seeking {args[1]} <true/false>";
|
|
TagPacket tagPacket = new TagPacket {
|
|
UpdateType = TagPacket.TagUpdate.State,
|
|
IsIt = seeking
|
|
};
|
|
if (args[1] == "*")
|
|
server.Broadcast(tagPacket);
|
|
else
|
|
client?.Send(tagPacket);
|
|
return $"Set {(args[1] == "*" ? "everyone" : args[1])} to {(seeking ? "seeker" : "hider")}";
|
|
}
|
|
case "start" when args.Length > 2: {
|
|
if (!byte.TryParse(args[1], out byte time)) return $"Invalid countdown seconds {args[1]} (range: 0-255)";
|
|
string[] seekerNames = args[2..];
|
|
Client[] seekers = server.Clients.Where(c => seekerNames.Contains(c.Name)).ToArray();
|
|
if (seekers.Length != seekerNames.Length)
|
|
return
|
|
$"Couldn't find seeker{(seekerNames.Length > 1 ? "s" : "")}: {string.Join(", ", seekerNames.Where(name => server.Clients.All(c => c.Name != name)))}";
|
|
Task.Run(async () => {
|
|
int realTime = 1000 * time;
|
|
await Task.Delay(realTime);
|
|
await Task.WhenAll(
|
|
Parallel.ForEachAsync(seekers, async (seeker, _) =>
|
|
await server.Broadcast(new TagPacket {
|
|
UpdateType = TagPacket.TagUpdate.State,
|
|
IsIt = true
|
|
}, seeker)),
|
|
Parallel.ForEachAsync(server.Clients.Except(seekers), async (hider, _) =>
|
|
await server.Broadcast(new TagPacket {
|
|
UpdateType = TagPacket.TagUpdate.State,
|
|
IsIt = false
|
|
}, hider)
|
|
)
|
|
);
|
|
consoleLogger.Info($"Started game with seekers {string.Join(", ", seekerNames)}");
|
|
});
|
|
return $"Starting game in {time} seconds with seekers {string.Join(", ", seekerNames)}";
|
|
}
|
|
default:
|
|
return optionUsage;
|
|
}
|
|
});
|
|
|
|
CommandHandler.RegisterCommand("maxplayers", args => {
|
|
const string optionUsage = "Valid usage: maxplayers <playercount>";
|
|
if (args.Length != 1) return optionUsage;
|
|
if (!ushort.TryParse(args[0], out ushort maxPlayers)) return optionUsage;
|
|
Settings.Instance.Server.MaxPlayers = maxPlayers;
|
|
Settings.SaveSettings();
|
|
foreach (Client client in server.Clients)
|
|
client.Dispose(); // reconnect all players
|
|
return $"Saved and set max players to {maxPlayers}";
|
|
});
|
|
|
|
CommandHandler.RegisterCommand("list",
|
|
_ => $"List: {string.Join("\n\t", server.Clients.Where(x => x.Connected).Select(x => $"{x.Name} ({x.Id})"))}");
|
|
|
|
CommandHandler.RegisterCommand("flip", args => {
|
|
const string optionUsage =
|
|
"Valid options: \n\tlist\n\tadd <user id>\n\tremove <user id>\n\tset <true/false>\n\tpov <both/self/others>";
|
|
if (args.Length < 1)
|
|
return optionUsage;
|
|
switch (args[0]) {
|
|
case "list" when args.Length == 1:
|
|
return "User ids: " + string.Join(", ", Settings.Instance.Flip.Players.ToList());
|
|
case "add" when args.Length == 2: {
|
|
if (Guid.TryParse(args[1], out Guid result)) {
|
|
Settings.Instance.Flip.Players.Add(result);
|
|
Settings.SaveSettings();
|
|
return $"Added {result} to flipped players";
|
|
}
|
|
|
|
return $"Invalid user id {args[1]}";
|
|
}
|
|
case "remove" when args.Length == 2: {
|
|
if (Guid.TryParse(args[1], out Guid result)) {
|
|
string output = Settings.Instance.Flip.Players.Remove(result)
|
|
? $"Removed {result} to flipped players"
|
|
: $"User {result} wasn't in the flipped players list";
|
|
Settings.SaveSettings();
|
|
return output;
|
|
}
|
|
|
|
return $"Invalid user id {args[1]}";
|
|
}
|
|
case "set" when args.Length == 2: {
|
|
if (bool.TryParse(args[1], out bool result)) {
|
|
Settings.Instance.Flip.Enabled = result;
|
|
Settings.SaveSettings();
|
|
return result ? "Enabled player flipping" : "Disabled player flipping";
|
|
}
|
|
|
|
return optionUsage;
|
|
}
|
|
case "pov" when args.Length == 2: {
|
|
if (Enum.TryParse(args[1], true, out FlipOptions result)) {
|
|
Settings.Instance.Flip.Pov = result;
|
|
Settings.SaveSettings();
|
|
return $"Point of view set to {result}";
|
|
}
|
|
|
|
return optionUsage;
|
|
}
|
|
default:
|
|
return optionUsage;
|
|
}
|
|
});
|
|
|
|
CommandHandler.RegisterCommand("shine", args => {
|
|
const string optionUsage = "Valid options: list, clear, sync, send, set";
|
|
if (args.Length < 1)
|
|
return optionUsage;
|
|
switch (args[0]) {
|
|
case "list" when args.Length == 1:
|
|
return $"Shines: {string.Join(", ", shineBag)}";
|
|
case "clear" when args.Length == 1:
|
|
shineBag.Clear();
|
|
foreach (ConcurrentBag<int> playerBag in server.Clients.Select(serverClient =>
|
|
(ConcurrentBag<int>)serverClient.Metadata["shineSync"]!)) playerBag?.Clear();
|
|
|
|
return "Cleared shine bags";
|
|
case "sync" when args.Length == 1:
|
|
SyncShineBag();
|
|
return "Synced shine bag automatically";
|
|
case "send" when args.Length >= 3:
|
|
if (int.TryParse(args[1], out int id)) {
|
|
Client[] players = args[2] == "*"
|
|
? server.Clients.Where(c => c.Connected).ToArray()
|
|
: server.Clients.Where(c => c.Connected && args[3..].Contains(c.Name)).ToArray();
|
|
Parallel.ForEachAsync(players, async (c, _) => {
|
|
await c.Send(new ShinePacket {
|
|
ShineId = id
|
|
});
|
|
}).Wait();
|
|
return $"Sent Shine Num {id}";
|
|
}
|
|
|
|
return optionUsage;
|
|
case "set" when args.Length == 2: {
|
|
if (bool.TryParse(args[1], out bool result)) {
|
|
Settings.Instance.Shines.Enabled = result;
|
|
Settings.SaveSettings();
|
|
return result ? "Enabled shine sync" : "Disabled shine sync";
|
|
}
|
|
|
|
return optionUsage;
|
|
}
|
|
default:
|
|
return optionUsage;
|
|
}
|
|
});
|
|
|
|
CommandHandler.RegisterCommand("loadsettings", _ => {
|
|
Settings.LoadSettings();
|
|
return "Loaded settings.json";
|
|
});
|
|
|
|
CommandHandler.RegisterCommand("restartserver", args =>
|
|
{
|
|
if (args.Length != 0)
|
|
{
|
|
return "Usage: restartserver (no arguments)";
|
|
}
|
|
else
|
|
{
|
|
consoleLogger.Info("Received restartserver command");
|
|
restartRequested = true;
|
|
cts.Cancel();
|
|
return "Restarting...";
|
|
}
|
|
});
|
|
|
|
Console.CancelKeyPress += (_, e) => {
|
|
e.Cancel = true;
|
|
consoleLogger.Info("Received Ctrl+C");
|
|
cts.Cancel();
|
|
};
|
|
|
|
CommandHandler.RegisterCommandAliases(_ => {
|
|
cts.Cancel();
|
|
return "Shutting down";
|
|
}, "exit", "quit", "q");
|
|
|
|
#pragma warning disable CS4014
|
|
Task.Run(() => {
|
|
consoleLogger.Info("Run help command for valid commands.");
|
|
while (true) {
|
|
string? text = Console.ReadLine();
|
|
if (text != null) {
|
|
foreach (string returnString in CommandHandler.GetResult(text).ReturnStrings) {
|
|
consoleLogger.Info(returnString);
|
|
}
|
|
}
|
|
}
|
|
}).ContinueWith(x => { if (x.Exception != null) { consoleLogger.Error(x.Exception.ToString()); } });
|
|
#pragma warning restore CS4014
|
|
|
|
await listenTask;
|
|
if (restartRequested) //need to do this here because this needs to happen after the listener closes, and there isn't an
|
|
//easy way to sync in the restartserver command without it exiting Main()
|
|
{
|
|
string? path = System.Reflection.Assembly.GetEntryAssembly()?.GetName().Name;
|
|
const string unableToStartMsg = "Unable to ascertain the executable location, you'll need to re-run the server manually.";
|
|
if (path != null) //path is probably just "Server", but in the context of the assembly, that's all you need to restart it.
|
|
{
|
|
Console.WriteLine($"Server Running on (pid): {System.Diagnostics.Process.Start(path)?.Id.ToString() ?? unableToStartMsg}");
|
|
}
|
|
else
|
|
consoleLogger.Info(unableToStartMsg);
|
|
} |