From f30dcf5ef41d54c588d7b42c48be8d941abba72e Mon Sep 17 00:00:00 2001 From: Michael Date: Sun, 8 Jan 2017 09:57:10 -0500 Subject: Initial upload --- ShiftOS.Server/App.config | 6 + ShiftOS.Server/Program.cs | 964 ++++++++++++++++++++++++ ShiftOS.Server/Properties/AssemblyInfo.cs | 36 + ShiftOS.Server/Properties/Resources.Designer.cs | 97 +++ ShiftOS.Server/Properties/Resources.resx | 127 ++++ ShiftOS.Server/Resources/Home.txt | 31 + ShiftOS.Server/Resources/NotFound.txt | 9 + ShiftOS.Server/ShiftOS.Server.csproj | 108 +++ ShiftOS.Server/WebAdmin.cs | 91 +++ ShiftOS.Server/packages.config | 8 + 10 files changed, 1477 insertions(+) create mode 100644 ShiftOS.Server/App.config create mode 100644 ShiftOS.Server/Program.cs create mode 100644 ShiftOS.Server/Properties/AssemblyInfo.cs create mode 100644 ShiftOS.Server/Properties/Resources.Designer.cs create mode 100644 ShiftOS.Server/Properties/Resources.resx create mode 100644 ShiftOS.Server/Resources/Home.txt create mode 100644 ShiftOS.Server/Resources/NotFound.txt create mode 100644 ShiftOS.Server/ShiftOS.Server.csproj create mode 100644 ShiftOS.Server/WebAdmin.cs create mode 100644 ShiftOS.Server/packages.config (limited to 'ShiftOS.Server') diff --git a/ShiftOS.Server/App.config b/ShiftOS.Server/App.config new file mode 100644 index 0000000..88fa402 --- /dev/null +++ b/ShiftOS.Server/App.config @@ -0,0 +1,6 @@ + + + + + + \ No newline at end of file diff --git a/ShiftOS.Server/Program.cs b/ShiftOS.Server/Program.cs new file mode 100644 index 0000000..768ed08 --- /dev/null +++ b/ShiftOS.Server/Program.cs @@ -0,0 +1,964 @@ +using System; +using System.Collections.Generic; +using System.Linq; +using System.Text; +using System.Threading.Tasks; +using ShiftOS.Objects; +using NetSockets; +using System.IO; +using Newtonsoft.Json; +using System.Net; +using System.Net.Sockets; +using Nancy.Hosting.Self; +using Nancy; +using Nancy.Authentication.Basic; +using Nancy.Security; +using Nancy.TinyIoc; +using Nancy.Bootstrapper; + +namespace ShiftOS.Server +{ + public interface IUserMapper + { + /// + /// Get the real username from an identifier + /// + /// User identifier + /// The current NancyFx context + /// Matching populated IUserIdentity object, or empty + IUserIdentity GetUserFromIdentifier(Guid identifier, NancyContext context); + } + + public class MUDUserValidator : IUserValidator + { + public IUserIdentity Validate(string username, string password) + { + if(username == Program.AdminUsername && password == Program.AdminPassword) + { + return null; + } + else + { + return null; + } + } + } + + public class MUDUserIdentity : IUserIdentity + { + public IEnumerable Claims + { + get + { + return null; + } + } + + public string UserName + { + get + { + return uname; + } + } + + public string uname = ""; + + public MUDUserIdentity(string username) + { + uname = username; + } + } + + public class AuthenticationBootstrapper : DefaultNancyBootstrapper + { + protected override void ApplicationStartup(TinyIoCContainer container, IPipelines pipelines) + { + base.ApplicationStartup(container, pipelines); + + pipelines.EnableBasicAuthentication(new BasicAuthenticationConfiguration( + container.Resolve(), + "MUD", UserPromptBehaviour.NonAjax)); + } + } + + public class Program + { + public static string AdminUsername = "admin"; + public static string AdminPassword = "admin"; + + + + public static NetObjectServer server; + + public delegate void StringEventHandler(string str); + + public static event StringEventHandler ServerStarted; + + public static void SaveChats() + { + List saved = new List(); + foreach(var chat in chats) + { + saved.Add(new Channel + { + ID = chat.ID, + Name = chat.Name, + MaxUsers = chat.MaxUsers, + Topic = chat.Topic, + Users = new List() + }); + } + File.WriteAllText("chats.json", JsonConvert.SerializeObject(saved)); + } + + public static void LoadChats() + { + chats = Newtonsoft.Json.JsonConvert.DeserializeObject>(File.ReadAllText("chats.json")); + } + + public static void Main(string[] args) + { + if (!Directory.Exists("saves")) + { + Directory.CreateDirectory("saves"); + } + + if(!File.Exists("chats.json")) + { + SaveChats(); + } + else + { + LoadChats(); + } + + if(!Directory.Exists("scripts")) + { + Console.WriteLine("Creating scripts directory..."); + Directory.CreateDirectory("scripts"); + Console.WriteLine("NOTE: This MUD is not just gonna generate scripts for you. You're going to need to write them. YOU are DevX."); + } + + Console.WriteLine("Starting server..."); + server = new NetObjectServer(); + + server.OnStarted += (o, a) => + { + Console.WriteLine($"Server started on address {server.Address}, port {server.Port}."); + ServerStarted?.Invoke(server.Address.MapToIPv4().ToString()); + }; + + server.OnStopped += (o, a) => + { + Console.WriteLine("WARNING! Server stopped."); + }; + + server.OnError += (o, a) => + { + Console.WriteLine("ERROR: " + a.Exception.Message); + }; + + server.OnClientAccepted += (o, a) => + { + Console.WriteLine("Client connected."); + server.DispatchTo(a.Guid, new NetObject("welcome", new ServerMessage { Name = "Welcome", Contents = a.Guid.ToString(), GUID = "Server" })); + }; + + server.OnReceived += (o, a) => + { + var obj = a.Data.Object; + + var msg = obj as ServerMessage; + + if(msg != null) + { + Interpret(msg); + } + }; + + IPAddress defaultAddress = null; + + var host = Dns.GetHostEntry(Dns.GetHostName()); + foreach (var ip in host.AddressList) + { + if (ip.AddressFamily == AddressFamily.InterNetwork) + { + defaultAddress = ip; + } + } + + try + { + server.Start(defaultAddress, 13370); + } + catch + { + Console.WriteLine("So we tried to bind the server to your IP address automatically, but your operating system or the .NET environment you are in (possibly Mono on Linux) is preventing us from doing so. We'll try to bind to the loopback IP address (127.0.0.1) and if that doesn't work, the multi-user domain software may not be compatible with this OS or .NET environment."); + server.Stop(); + server.Start(IPAddress.Loopback, 13370); + + } + + var hConf = new HostConfiguration(); + hConf.UrlReservations.CreateAutomatically = true; + + var nancy = new NancyHost(hConf, new Uri("http://localhost:13371/")); + + server.OnStopped += (o, a) => + { + nancy.Stop(); + }; + + nancy.Start(); + } + + public static bool UserInChat(Channel chan, Save user) + { + foreach(var usr in chan.Users) + { + if(usr.Username == user.Username) + { + return true; + } + } + return false; + } + + + + public static void Interpret(ServerMessage msg) + { + Dictionary args = null; + + try + { + Console.WriteLine($@"Message received from {msg.GUID}: {msg.Name} + +Contents: +{msg.Contents}"); + + if (!string.IsNullOrWhiteSpace(msg.Contents)) + { + try + { + //It's gotta be JSON. + if (msg.Contents.StartsWith("{")) + { + args = JsonConvert.DeserializeObject>(msg.Contents); + } + } + catch + { + //Damnit, we were wrong. + args = null; + } + } + + switch (msg.Name) + { + case "mud_login": + if (args["username"] != null && args["password"] != null) + { + foreach(var savefile in Directory.GetFiles("saves")) + { + try + { + var save = JsonConvert.DeserializeObject(File.ReadAllText(savefile)); + + if(save.Username == args["username"].ToString() && save.Password == args["password"].ToString()) + { + server.DispatchTo(new Guid(msg.GUID), new NetObject("mud_savefile", new ServerMessage + { + Name = "mud_savefile", + GUID = "server", + Contents = File.ReadAllText(savefile) + })); + return; + } + } + catch { } + } + server.DispatchTo(new Guid(msg.GUID), new NetObject("auth_failed", new ServerMessage + { + Name = "mud_login_denied", + GUID = "server" + })); + } + else + { + server.DispatchTo(new Guid(msg.GUID), new NetObject("auth_failed", new ServerMessage + { + Name = "mud_login_denied", + GUID = "server" + })); + } + break; + case "legion_create": + List legions = new List(); + if (File.Exists("legions.json")) + legions = JsonConvert.DeserializeObject>(File.ReadAllText("legions.json")); + + var l = JsonConvert.DeserializeObject(msg.Contents); + + legions.Add(l); + + File.WriteAllText("legions.json", JsonConvert.SerializeObject(legions, Formatting.Indented)); + break; + case "legion_get_all": + List allLegions = new List(); + + if (File.Exists("legions.json")) + allLegions = JsonConvert.DeserializeObject>(File.ReadAllText("legions.json")); + + server.DispatchTo(new Guid(msg.GUID), new NetObject("alllegions", new ServerMessage + { + Name = "legion_all", + GUID = "server", + Contents = JsonConvert.SerializeObject(allLegions) + })); + break; + case "legion_get_users": + var lgn = JsonConvert.DeserializeObject(msg.Contents); + + List userIDs = new List(); + + foreach (var savfile in Directory.GetFiles("saves")) + { + try + { + var savefilecontents = JsonConvert.DeserializeObject(File.ReadAllText(savfile)); + if (savefilecontents.CurrentLegions.Contains(lgn.ShortName)) + { + userIDs.Add($"{savefilecontents.Username}@{savefilecontents.SystemName}"); + } + } + catch { } + } + + server.DispatchTo(new Guid(msg.GUID), new NetObject("userlist", new ServerMessage + { + Name = "legion_users_found", + GUID = "server", + Contents = JsonConvert.SerializeObject(userIDs) + })); + break; + case "user_get_legion": + var userSave = JsonConvert.DeserializeObject(msg.Contents); + + if (File.Exists("legions.json")) + { + var legionList = JsonConvert.DeserializeObject>(File.ReadAllText("legions.json")); + foreach (var legion in legionList) + { + if (userSave.CurrentLegions.Contains(legion.ShortName)) + { + server.DispatchTo(new Guid(msg.GUID), new NetObject("reply", new ServerMessage + { + Name = "user_legion", + GUID = "server", + Contents = JsonConvert.SerializeObject(legion) + })); + return; + } + } + } + + server.DispatchTo(new Guid(msg.GUID), new NetObject("fuck", new ServerMessage + { + Name = "user_not_found_in_legion", + GUID = "server" + })); + + break; + case "mud_save": + var sav = JsonConvert.DeserializeObject(msg.Contents); + File.WriteAllText("saves/" + sav.Username + ".save", JsonConvert.SerializeObject(sav, Formatting.Indented)); + + server.DispatchTo(new Guid(msg.GUID), new NetObject("auth_failed", new ServerMessage + { + Name = "mud_saved", + GUID = "server" + })); + + break; + case "mud_checkuserexists": + if (args["username"] != null && args["password"] != null) + { + foreach (var savefile in Directory.GetFiles("saves")) + { + try + { + var save = JsonConvert.DeserializeObject(File.ReadAllText(savefile)); + + if (save.Username == args["username"].ToString() && save.Password == args["password"].ToString()) + { + server.DispatchTo(new Guid(msg.GUID), new NetObject("mud_savefile", new ServerMessage + { + Name = "mud_found", + GUID = "server", + })); + return; + } + } + catch { } + } + server.DispatchTo(new Guid(msg.GUID), new NetObject("auth_failed", new ServerMessage + { + Name = "mud_notfound", + GUID = "server" + })); + } + else + { + server.DispatchTo(new Guid(msg.GUID), new NetObject("auth_failed", new ServerMessage + { + Name = "mud_notfound", + GUID = "server" + })); + } + break; + break; + case "pong_gethighscores": + if (File.Exists("pong_highscores.json")) + { + server.DispatchTo(new Guid(msg.GUID), new NetObject("pongstuff", new ServerMessage + { + Name = "pong_highscores", + GUID = "server", + Contents = File.ReadAllText("pong_highscores.json") + })); + } + break; + case "get_memos_for_user": + if(args["username"] != null) + { + string usrname = args["username"].ToString(); + + List mmos = new List(); + + if (File.Exists("memos.json")) + { + foreach(var mmo in JsonConvert.DeserializeObject(File.ReadAllText("memos.json"))) + { + if(mmo.UserTo == usrname) + { + mmos.Add(mmo); + } + } + } + + server.DispatchTo(new Guid(msg.GUID), new NetObject("mud_memos", new ServerMessage + { + Name = "mud_usermemos", + GUID = "server", + Contents = JsonConvert.SerializeObject(mmos) + })); + } + break; + case "mud_post_memo": + MUDMemo memo = JsonConvert.DeserializeObject(msg.Contents); + List memos = new List(); + + if (File.Exists("memos.json")) + memos = JsonConvert.DeserializeObject>(File.ReadAllText("memos.json")); + + memos.Add(memo); + File.WriteAllText("memos.txt", JsonConvert.SerializeObject(memos)); + + + break; + case "pong_sethighscore": + var hs = new List(); + if (File.Exists("pong_highscores.json")) + hs = JsonConvert.DeserializeObject>(File.ReadAllText("ponghighscores.json")); + + var newHS = JsonConvert.DeserializeObject(msg.Contents); + for (int i = 0; i <= hs.Count; i++) + { + try + { + if (hs[i].UserName == newHS.UserName) + { + if (newHS.HighestLevel > hs[i].HighestLevel) + hs[i].HighestLevel = newHS.HighestLevel; + if (newHS.HighestCodepoints > hs[i].HighestCodepoints) + hs[i].HighestCodepoints = newHS.HighestCodepoints; + File.WriteAllText("pong_highscores.json", JsonConvert.SerializeObject(hs)); + return; + + } + } + catch + { + + } + } + hs.Add(newHS); + File.WriteAllText("pong_highscores.json", JsonConvert.SerializeObject(hs)); + return; + case "getvirusdb": + if (!File.Exists("virus.db")) + File.WriteAllText("virus.db", "{}"); + + server.DispatchTo(new Guid(msg.GUID), new NetObject("vdb", new ServerMessage + { + Name = "virusdb", + GUID = "server", + Contents = File.ReadAllText("virus.db") + })); + break; + case "getvirus": + Dictionary virusDB = new Dictionary(); + + if (File.Exists("virus.db")) + virusDB = JsonConvert.DeserializeObject>(File.ReadAllText("virus.db")); + + foreach (var kv in virusDB) + { + if (kv.Key == msg.Contents) + { + server.DispatchTo(new Guid(msg.GUID), new NetObject("response", new ServerMessage + { + Name = "mud_virus", + GUID = "server", + Contents = kv.Value, + })); + return; + } + } + + + break; + case "mud_scanvirus": + Dictionary _virusDB = new Dictionary(); + + bool addIfNotFound = true; + + if (msg.Contents.Contains("||scanonly")) + addIfNotFound = false; + + msg.Contents = msg.Contents.Replace("||scanonly", ""); + + if(File.Exists("virus.db")) + _virusDB = JsonConvert.DeserializeObject>(File.ReadAllText("virus.db")); + + foreach(var kv in _virusDB) + { + if(kv.Value == msg.Contents) + { + server.DispatchTo(new Guid(msg.GUID), new NetObject("response", new ServerMessage + { + Name = "mud_virus_signature", + GUID = "server", + Contents = kv.Key, + })); + return; + } + } + + if (addIfNotFound == true) + { + string newguid = Guid.NewGuid().ToString(); + _virusDB.Add(newguid, msg.Contents); + File.WriteAllText("virus.db", JsonConvert.SerializeObject(_virusDB, Formatting.Indented)); + server.DispatchTo(new Guid(msg.GUID), new NetObject("response", new ServerMessage + { + Name = "mud_virus_signature", + GUID = "server", + Contents = newguid, + })); + } + else + { + server.DispatchTo(new Guid(msg.GUID), new NetObject("response", new ServerMessage + { + Name = "mud_virus_signature", + GUID = "server", + Contents = "unknown", + })); + } + return; + + case "chat_join": + if (args.ContainsKey("id")) + { + var cuser = new Save(); + if (args.ContainsKey("user")) + { + cuser = JsonConvert.DeserializeObject(JsonConvert.SerializeObject(args["user"])); + + + } + int index = -1; + string chat_id = args["id"] as string; + foreach(var chat in chats) + { + if(chat.ID == chat_id) + { + if(chat.Users.Count < chat.MaxUsers || chat.MaxUsers == 0) + { + //user can join chat. + if(cuser != null) + { + index = chats.IndexOf(chat); + } + } + } + } + if(index > -1) + { + chats[index].Users.Add(cuser); + server.DispatchAll(new NetObject("broadcast", new ServerMessage + { + Name = "cbroadcast", + GUID = "server", + Contents = $"{chat_id}: {cuser.Username} {{CHAT_HAS_JOINED}}" + })); + } + else + { + server.DispatchTo(new Guid(msg.GUID), new NetObject("broadcast", new ServerMessage + { + Name = "cbroadcast", + GUID = "server", + Contents = $"{chat_id}: {{CHAT_NOT_FOUND_OR_TOO_MANY_MEMBERS}}" + })); + } + } + break; + case "chat": + if (args.ContainsKey("id")) + { + var cuser = new Save(); + if (args.ContainsKey("user")) + { + cuser = JsonConvert.DeserializeObject(JsonConvert.SerializeObject(args["user"])); + + + } + string message = ""; + if (args.ContainsKey("msg")) + message = args["msg"] as string; + + int index = -1; + string chat_id = args["id"] as string; + foreach (var chat in chats) + { + if (chat.ID == chat_id) + { + if (cuser != null && !string.IsNullOrWhiteSpace(message) && UserInChat(chat, cuser)) + { + index = chats.IndexOf(chat); + } + } + } + if (index > -1) + { + server.DispatchAll(new NetObject("broadcast", new ServerMessage + { + Name = "cbroadcast", + GUID = "server", + Contents = $"{chat_id}/{cuser.Username}: {message}" + })); + } + else + { + server.DispatchTo(new Guid(msg.GUID), new NetObject("broadcast", new ServerMessage + { + Name = "cbroadcast", + GUID = "server", + Contents = $"{chats[index].ID}: {{CHAT_NOT_FOUND_OR_NOT_IN_CHAT}}" + })); + } + } + + break; + case "chat_leave": + if (args.ContainsKey("id")) + { + var cuser = new Save(); + if (args.ContainsKey("user")) + { + cuser = JsonConvert.DeserializeObject(JsonConvert.SerializeObject(args["user"])); + + + } + int index = -1; + string chat_id = args["id"] as string; + foreach (var chat in chats) + { + if (chat.ID == chat_id) + { + if (cuser != null && UserInChat(chat, cuser)) + { + index = chats.IndexOf(chat); + } + } + } + if (index > -1) + { + server.DispatchAll(new NetObject("broadcast", new ServerMessage + { + Name = "cbroadcast", + GUID = "server", + Contents = $"{chats[index].ID}: {cuser.Username} {{HAS_LEFT_CHAT}}" + })); + } + else + { + server.DispatchTo(new Guid(msg.GUID), new NetObject("broadcast", new ServerMessage + { + Name = "cbroadcast", + GUID = "server", + Contents = $"{chat_id}: {{CHAT_NOT_FOUND_OR_NOT_IN_CHAT}}" + })); + } + } + break; + case "chat_create": + string id = ""; + string topic = ""; + string name = ""; + int max_users = 0; + + if (args.ContainsKey("id")) + id = args["id"] as string; + if (args.ContainsKey("topic")) + name = args["topic"] as string; + if (args.ContainsKey("name")) + topic = args["name"] as string; + if (args.ContainsKey("max_users")) + max_users = Convert.ToInt32(args["max_users"].ToString()); + + bool id_taken = false; + + foreach(var chat in chats) + { + if (chat.ID == id) + id_taken = true; + } + + if (id_taken == false) + { + chats.Add(new Channel + { + ID = id, + Name = name, + Topic = topic, + MaxUsers = max_users, + Users = new List() + }); + SaveChats(); + server.DispatchTo(new Guid(msg.GUID), new NetObject("broadcast", new ServerMessage + { + Name = "cbroadcast", + GUID = "server", + Contents = $"{id}: {{SUCCESSFULLY_CREATED_CHAT}}" + })); + } + else + { + server.DispatchTo(new Guid(msg.GUID), new NetObject("broadcast", new ServerMessage + { + Name = "cbroadcast", + GUID = "server", + Contents = $"{id}: {{ID_TAKEN}}" + })); + } + + break; + case "broadcast": + string text = msg.Contents; + if (!string.IsNullOrWhiteSpace(text)) + { + server.DispatchTo(new Guid(msg.GUID), new NetObject("runme", new ServerMessage + { + Name = "broadcast", + GUID = "Server", + Contents = text + + })); + } + break; + case "lua_up": + string lua = msg.Contents; + string firstLine = lua.Split(new[] { Environment.NewLine }, StringSplitOptions.None)[0]; + firstLine = firstLine.Remove(0, 3); //delete the comment + string[] a = firstLine.Split('.'); + if(!Directory.Exists("scripts/" + a[0])) + { + Directory.CreateDirectory($"scripts/{a[0]}"); + } + File.WriteAllText($"scripts/{a[0]}/{a[1]}.lua", lua); + break; + case "mudhack_init": + if (MUDHackPasswords.ContainsKey(msg.GUID)) + MUDHackPasswords.Remove(msg.GUID); + + MUDHackPasswords.Add(msg.GUID, GenerateRandomPassword()); + + + server.DispatchTo(new Guid(msg.GUID), new NetObject("mudhack_init", new ServerMessage + { + Name = "mudhack_init", + GUID = "SERVER", + Contents = MUDHackPasswords[msg.GUID], + })); + + break; + case "mudhack_verify": + if (!MUDHackPasswords.ContainsKey(msg.GUID)) + { + + server.DispatchTo(new Guid(msg.GUID), new NetObject("mudhack_init", new ServerMessage + { + Name = "server_error", + GUID = "SERVER", + Contents = "{SRV_HACK_NOT_INITIATED}", + })); + return; + } + + string pass = ""; + if (args.ContainsKey("pass")) + pass = args["pass"] as string; + + if(pass == MUDHackPasswords[msg.GUID]) + { + server.DispatchTo(new Guid(msg.GUID), new NetObject("mudhack_init", new ServerMessage + { + Name = "mudhack_granted", + GUID = "SERVER", + })); + + } + else + { + + server.DispatchTo(new Guid(msg.GUID), new NetObject("mudhack_init", new ServerMessage + { + Name = "mudhack_denied", + GUID = "SERVER", + })); + } + break; + case "mudhack_killpass": + if (MUDHackPasswords.ContainsKey(msg.GUID)) + MUDHackPasswords.Remove(msg.GUID); + break; + case "mudhack_getallusers": + List users = new List(); + + foreach (var chat in chats) + { + foreach(var usr in chat.Users) + { + var ousr = new OnlineUser(); + ousr.Username = usr.Username; + ousr.OnlineChat = chat.ID; + users.Add(ousr); + } + } + + server.DispatchTo(new Guid(msg.GUID), new NetObject("mudhack_users", new ServerMessage + { + Name = "mudhack_users", + GUID = "SERVER", + Contents = JsonConvert.SerializeObject(users), + })); + break; + case "getguid_reply": + msg.GUID = "server"; + //The message's GUID was manipulated by the client to send to another client. + //So we can just bounce back the message to the other client. + server.DispatchTo(new Guid(msg.GUID), new NetObject("bounce", msg)); + break; + case "getguid_send": + string username = msg.Contents; + string guid = msg.GUID; + server.DispatchAll(new NetObject("are_you_this_guy", new ServerMessage + { + Name = "getguid_fromserver", + GUID = guid, + Contents = username, + })); + break; + case "script": + string user = ""; + string script = ""; + string sArgs = ""; + + if (!args.ContainsKey("user")) + throw new Exception("No 'user' arg specified in message to server"); + + if (!args.ContainsKey("script")) + throw new Exception("No 'script' arg specified in message to server"); + + if (!args.ContainsKey("args")) + throw new Exception("No 'args' arg specified in message to server"); + + user = args["user"] as string; + script = args["script"] as string; + sArgs = args["args"] as string; + + if(File.Exists($"scripts/{user}/{script}.lua")) + { + var script_arguments = JsonConvert.DeserializeObject>(sArgs); + server.DispatchTo(new Guid(msg.GUID), new NetObject("runme", new ServerMessage { + Name="run", + GUID="Server", + Contents = $@"{{ + script:""{File.ReadAllText($"scripts/{user}/{script}.lua").Replace("\"", "\\\"")}"", + args:""{sArgs}"" +}}" + })); + } + else + { + throw new Exception($"{user}.{script}: Script not found."); + } + break; + default: + throw new Exception($"Server couldn't decipher this message:\n\n{JsonConvert.SerializeObject(msg)}"); + } + } + catch(Exception ex) + { + Console.WriteLine("An error occurred with that one."); + Console.WriteLine(ex); + + server.DispatchTo(new Guid(msg.GUID), new NetObject("error", new ServerMessage { Name = "Error", GUID = "Server", Contents = JsonConvert.SerializeObject(ex) })); + } + } + + public static string GenerateRandomPassword() + { + return Guid.NewGuid().ToString(); + } + + public static Dictionary MUDHackPasswords = new Dictionary(); + + public static void Stop() + { + try + { + if (server.IsOnline) + { + try + { + server.Stop(); + } + catch + { + + } + } + server = null; + } + catch { } + } + + public static List chats = new List(); + } +} diff --git a/ShiftOS.Server/Properties/AssemblyInfo.cs b/ShiftOS.Server/Properties/AssemblyInfo.cs new file mode 100644 index 0000000..631aa86 --- /dev/null +++ b/ShiftOS.Server/Properties/AssemblyInfo.cs @@ -0,0 +1,36 @@ +using System.Reflection; +using System.Runtime.CompilerServices; +using System.Runtime.InteropServices; + +// General Information about an assembly is controlled through the following +// set of attributes. Change these attribute values to modify the information +// associated with an assembly. +[assembly: AssemblyTitle("ShiftOS.Server")] +[assembly: AssemblyDescription("")] +[assembly: AssemblyConfiguration("")] +[assembly: AssemblyCompany("")] +[assembly: AssemblyProduct("ShiftOS.Server")] +[assembly: AssemblyCopyright("Copyright © 2016")] +[assembly: AssemblyTrademark("")] +[assembly: AssemblyCulture("")] + +// Setting ComVisible to false makes the types in this assembly not visible +// to COM components. If you need to access a type in this assembly from +// COM, set the ComVisible attribute to true on that type. +[assembly: ComVisible(false)] + +// The following GUID is for the ID of the typelib if this project is exposed to COM +[assembly: Guid("226c63b4-e60d-4949-b4e7-7a2ddbb96776")] + +// Version information for an assembly consists of the following four values: +// +// Major Version +// Minor Version +// Build Number +// Revision +// +// You can specify all the values or you can default the Build and Revision Numbers +// by using the '*' as shown below: +// [assembly: AssemblyVersion("1.0.*")] +[assembly: AssemblyVersion("1.0.0.0")] +[assembly: AssemblyFileVersion("1.0.0.0")] diff --git a/ShiftOS.Server/Properties/Resources.Designer.cs b/ShiftOS.Server/Properties/Resources.Designer.cs new file mode 100644 index 0000000..4ceddac --- /dev/null +++ b/ShiftOS.Server/Properties/Resources.Designer.cs @@ -0,0 +1,97 @@ +//------------------------------------------------------------------------------ +// +// This code was generated by a tool. +// Runtime Version:4.0.30319.42000 +// +// Changes to this file may cause incorrect behavior and will be lost if +// the code is regenerated. +// +//------------------------------------------------------------------------------ + +namespace ShiftOS.Server.Properties { + using System; + + + /// + /// A strongly-typed resource class, for looking up localized strings, etc. + /// + // This class was auto-generated by the StronglyTypedResourceBuilder + // class via a tool like ResGen or Visual Studio. + // To add or remove a member, edit your .ResX file then rerun ResGen + // with the /str option, or rebuild your VS project. + [global::System.CodeDom.Compiler.GeneratedCodeAttribute("System.Resources.Tools.StronglyTypedResourceBuilder", "4.0.0.0")] + [global::System.Diagnostics.DebuggerNonUserCodeAttribute()] + [global::System.Runtime.CompilerServices.CompilerGeneratedAttribute()] + internal class Resources { + + private static global::System.Resources.ResourceManager resourceMan; + + private static global::System.Globalization.CultureInfo resourceCulture; + + [global::System.Diagnostics.CodeAnalysis.SuppressMessageAttribute("Microsoft.Performance", "CA1811:AvoidUncalledPrivateCode")] + internal Resources() { + } + + /// + /// Returns the cached ResourceManager instance used by this class. + /// + [global::System.ComponentModel.EditorBrowsableAttribute(global::System.ComponentModel.EditorBrowsableState.Advanced)] + internal static global::System.Resources.ResourceManager ResourceManager { + get { + if (object.ReferenceEquals(resourceMan, null)) { + global::System.Resources.ResourceManager temp = new global::System.Resources.ResourceManager("ShiftOS.Server.Properties.Resources", typeof(Resources).Assembly); + resourceMan = temp; + } + return resourceMan; + } + } + + /// + /// Overrides the current thread's CurrentUICulture property for all + /// resource lookups using this strongly typed resource class. + /// + [global::System.ComponentModel.EditorBrowsableAttribute(global::System.ComponentModel.EditorBrowsableState.Advanced)] + internal static global::System.Globalization.CultureInfo Culture { + get { + return resourceCulture; + } + set { + resourceCulture = value; + } + } + + /// + /// Looks up a localized string similar to <html> + /// <head> + /// <title>ShiftOS Multi-User Domain &bull; Admin Panel</title> + /// </head> + /// <body> + /// <h1>Welcome.</h1> + /// <p>There's nothing here, but hey, it works!</p> + /// </body> + ///</html>. + /// + internal static string Home { + get { + return ResourceManager.GetString("Home", resourceCulture); + } + } + + /// + /// Looks up a localized string similar to <html> + /// <head> + /// <title>Multi-User Domain &bull; Administration Panel - Page not found.</title> + /// </head> + /// <body> + /// <h1>This page wasn't found.</h1> + /// <p>We couldn't find this page...</p> + /// </body> + ///</html>. + /// + internal static string NotFound { + get { + return ResourceManager.GetString("NotFound", resourceCulture); + } + } + } +} diff --git a/ShiftOS.Server/Properties/Resources.resx b/ShiftOS.Server/Properties/Resources.resx new file mode 100644 index 0000000..cd10c06 --- /dev/null +++ b/ShiftOS.Server/Properties/Resources.resx @@ -0,0 +1,127 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + text/microsoft-resx + + + 2.0 + + + System.Resources.ResXResourceReader, System.Windows.Forms, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089 + + + System.Resources.ResXResourceWriter, System.Windows.Forms, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089 + + + + ..\Resources\Home.txt;System.String, mscorlib, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089;Windows-1252 + + + ..\Resources\NotFound.txt;System.String, mscorlib, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089;Windows-1252 + + \ No newline at end of file diff --git a/ShiftOS.Server/Resources/Home.txt b/ShiftOS.Server/Resources/Home.txt new file mode 100644 index 0000000..41c8995 --- /dev/null +++ b/ShiftOS.Server/Resources/Home.txt @@ -0,0 +1,31 @@ + + + ShiftOS Multi-User Domain • Admin Panel + + + + + + + + + +
+ + {BODY} + +
+

MUD server on {IP_ADDR}:{PORT}

+
+
+ + \ No newline at end of file diff --git a/ShiftOS.Server/Resources/NotFound.txt b/ShiftOS.Server/Resources/NotFound.txt new file mode 100644 index 0000000..cd65829 --- /dev/null +++ b/ShiftOS.Server/Resources/NotFound.txt @@ -0,0 +1,9 @@ + + + Multi-User Domain • Administration Panel - Page not found. + + +

This page wasn't found.

+

We couldn't find this page...

+ + \ No newline at end of file diff --git a/ShiftOS.Server/ShiftOS.Server.csproj b/ShiftOS.Server/ShiftOS.Server.csproj new file mode 100644 index 0000000..8db678e --- /dev/null +++ b/ShiftOS.Server/ShiftOS.Server.csproj @@ -0,0 +1,108 @@ + + + + + Debug + AnyCPU + {226C63B4-E60D-4949-B4E7-7A2DDBB96776} + Exe + Properties + ShiftOS.Server + ShiftOS.Server + v4.5.2 + 512 + true + + + AnyCPU + true + full + false + bin\Debug\ + DEBUG;TRACE + prompt + 4 + + + AnyCPU + pdbonly + true + bin\Release\ + TRACE + prompt + 4 + + + + ..\packages\DynamicLua.1.1.2.0\lib\net40-Client\DynamicLua.dll + True + + + ..\packages\Nancy.1.4.1\lib\net40\Nancy.dll + True + + + ..\packages\Nancy.Authentication.Basic.1.4.1\lib\net40\Nancy.Authentication.Basic.dll + True + + + ..\packages\Nancy.Hosting.Self.1.4.1\lib\net40\Nancy.Hosting.Self.dll + True + + + ..\Libraries\NetSockets.dll + + + ..\packages\Newtonsoft.Json.9.0.1\lib\net45\Newtonsoft.Json.dll + True + + + + + + + + + + + + + + + True + True + Resources.resx + + + + + + + + + + {a069089a-8962-4607-b2b2-4cf4a371066e} + ShiftOS.Objects + + + + + ResXFileCodeGenerator + Resources.Designer.cs + + + + + + + + + + + \ No newline at end of file diff --git a/ShiftOS.Server/WebAdmin.cs b/ShiftOS.Server/WebAdmin.cs new file mode 100644 index 0000000..cf21181 --- /dev/null +++ b/ShiftOS.Server/WebAdmin.cs @@ -0,0 +1,91 @@ +using Nancy; +using Nancy.Security; +using NetSockets; +using Newtonsoft.Json; +using ShiftOS.Objects; +using System; +using System.Collections.Generic; +using System.IO; +using System.Linq; +using System.Reflection; +using System.Text; +using System.Threading.Tasks; + +namespace ShiftOS.Server +{ + public class WebAdmin : NancyModule + { + private Guid thisGuid { get; set; } + + public WebAdmin() + { + this.RequiresAuthentication(); + + + client = new NetObjectClient(); + + client.OnReceived += (o, a) => + { + var msg = a.Data.Object as ServerMessage; + if (msg.Name == "Welcome") + { + thisGuid = new Guid(msg.Contents); + } + }; + + client.Connect(Program.server.Address.MapToIPv4().ToString(), 13370); + + string template = Properties.Resources.Home; + + Get["/"] = _ => { return GetPage(template, "index.html"); }; + Get["/{page}"] = parameters => + { + return GetPage(template, parameters.page); + }; + } + + public NetObjectClient client = new NetObjectClient(); + + public string GetPage(string template, string page) + { + string pageContents = File.ReadAllText("adm/" + page); + + string page_text = template.Replace("{BODY}", pageContents); + + page_text = page_text.Replace("{IP_ADDR}", client.RemoteHost.ToString()); + page_text = page_text.Replace("{PORT}", client.RemotePort.ToString()); + + return page_text; + } + + public string GrabResource(string page) + { + var type = this.GetType(); + foreach(var property in type.GetProperties(BindingFlags.Public | BindingFlags.Instance)) + { + foreach(var attr in property.GetCustomAttributes(false)) + { + if(attr is PageAttribute) + { + if(page == (attr as PageAttribute).Name) + { + return property.GetGetMethod().Invoke(this, null) as string; + } + } + } + } + + return Properties.Resources.NotFound; + } + } + + public class PageAttribute :Attribute + { + public PageAttribute(string name) + { + Name = name; + } + + public string Name { get; set; } + } +} diff --git a/ShiftOS.Server/packages.config b/ShiftOS.Server/packages.config new file mode 100644 index 0000000..3e07118 --- /dev/null +++ b/ShiftOS.Server/packages.config @@ -0,0 +1,8 @@ + + + + + + + + \ No newline at end of file -- cgit v1.2.3