diff --git a/.gitignore b/.gitignore index 940794e..f281e10 100644 --- a/.gitignore +++ b/.gitignore @@ -1,288 +1,14 @@ -## Ignore Visual Studio temporary files, build results, and -## files generated by popular Visual Studio add-ons. -## -## Get latest from https://github.com/github/gitignore/blob/master/VisualStudio.gitignore +################################################################################ +# This .gitignore file was automatically created by Microsoft(R) Visual Studio. +################################################################################ -# User-specific files -*.suo -*.user -*.userosscache -*.sln.docstates - -# User-specific files (MonoDevelop/Xamarin Studio) -*.userprefs - -# Build results -[Dd]ebug/ -[Dd]ebugPublic/ -[Rr]elease/ -[Rr]eleases/ -x64/ -x86/ -bld/ -[Bb]in/ -[Oo]bj/ -[Ll]og/ - -# Visual Studio 2015 cache/options directory -.vs/ -# Uncomment if you have tasks that create the project's static files in wwwroot -#wwwroot/ - -# MSTest test Results -[Tt]est[Rr]esult*/ -[Bb]uild[Ll]og.* - -# NUNIT -*.VisualState.xml -TestResult.xml - -# Build Results of an ATL Project -[Dd]ebugPS/ -[Rr]eleasePS/ -dlldata.c - -# .NET Core -project.lock.json -project.fragment.lock.json -artifacts/ -**/Properties/launchSettings.json - -*_i.c -*_p.c -*_i.h -*.ilk -*.meta -*.obj -*.pch -*.pdb -*.pgc -*.pgd -*.rsp -*.sbr -*.tlb -*.tli -*.tlh -*.tmp -*.tmp_proj -*.log -*.vspscc -*.vssscc -.builds -*.pidb -*.svclog -*.scc - -# Chutzpah Test files -_Chutzpah* - -# Visual C++ cache files -ipch/ -*.aps -*.ncb -*.opendb -*.opensdf -*.sdf -*.cachefile -*.VC.db -*.VC.VC.opendb - -# Visual Studio profiler -*.psess -*.vsp -*.vspx -*.sap - -# TFS 2012 Local Workspace -$tf/ - -# Guidance Automation Toolkit -*.gpState - -# ReSharper is a .NET coding add-in -_ReSharper*/ -*.[Rr]e[Ss]harper -*.DotSettings.user - -# JustCode is a .NET coding add-in -.JustCode - -# TeamCity is a build add-in -_TeamCity* - -# DotCover is a Code Coverage Tool -*.dotCover - -# Visual Studio code coverage results -*.coverage -*.coveragexml - -# NCrunch -_NCrunch_* -.*crunch*.local.xml -nCrunchTemp_* - -# MightyMoose -*.mm.* -AutoTest.Net/ - -# Web workbench (sass) -.sass-cache/ - -# Installshield output folder -[Ee]xpress/ - -# DocProject is a documentation generator add-in -DocProject/buildhelp/ -DocProject/Help/*.HxT -DocProject/Help/*.HxC -DocProject/Help/*.hhc -DocProject/Help/*.hhk -DocProject/Help/*.hhp -DocProject/Help/Html2 -DocProject/Help/html - -# Click-Once directory -publish/ - -# Publish Web Output -*.[Pp]ublish.xml -*.azurePubxml -# TODO: Comment the next line if you want to checkin your web deploy settings -# but database connection strings (with potential passwords) will be unencrypted -*.pubxml -*.publishproj - -# Microsoft Azure Web App publish settings. Comment the next line if you want to -# checkin your Azure Web App publish settings, but sensitive information contained -# in these scripts will be unencrypted -PublishScripts/ - -# NuGet Packages -*.nupkg -# The packages folder can be ignored because of Package Restore -**/packages/* -# except build/, which is used as an MSBuild target. -!**/packages/build/ -# Uncomment if necessary however generally it will be regenerated when needed -#!**/packages/repositories.config -# NuGet v3's project.json files produces more ignorable files -*.nuget.props -*.nuget.targets - -# Microsoft Azure Build Output -csx/ -*.build.csdef - -# Microsoft Azure Emulator -ecf/ -rcf/ - -# Windows Store app package directories and files -AppPackages/ -BundleArtifacts/ -Package.StoreAssociation.xml -_pkginfo.txt - -# Visual Studio cache files -# files ending in .cache can be ignored -*.[Cc]ache -# but keep track of directories ending in .cache -!*.[Cc]ache/ - -# Others -ClientBin/ -~$* -*~ -*.dbmdl -*.dbproj.schemaview -*.jfm -*.pfx -*.publishsettings -orleans.codegen.cs - -# Since there are multiple workflows, uncomment next line to ignore bower_components -# (https://github.com/github/gitignore/pull/1529#issuecomment-104372622) -#bower_components/ - -# RIA/Silverlight projects -Generated_Code/ - -# Backup & report files from converting an old project file -# to a newer Visual Studio version. Backup files are not needed, -# because we have git ;-) -_UpgradeReport_Files/ -Backup*/ -UpgradeLog*.XML -UpgradeLog*.htm - -# SQL Server files -*.mdf -*.ldf -*.ndf - -# Business Intelligence projects -*.rdl.data -*.bim.layout -*.bim_*.settings - -# Microsoft Fakes -FakesAssemblies/ - -# GhostDoc plugin setting file -*.GhostDoc.xml - -# Node.js Tools for Visual Studio -.ntvs_analysis.dat -node_modules/ - -# Typescript v1 declaration files -typings/ - -# Visual Studio 6 build log -*.plg - -# Visual Studio 6 workspace options file -*.opt - -# Visual Studio 6 auto-generated workspace file (contains which files were open etc.) -*.vbw - -# Visual Studio LightSwitch build output -**/*.HTMLClient/GeneratedArtifacts -**/*.DesktopClient/GeneratedArtifacts -**/*.DesktopClient/ModelManifest.xml -**/*.Server/GeneratedArtifacts -**/*.Server/ModelManifest.xml -_Pvt_Extensions - -# Paket dependency manager -.paket/paket.exe -paket-files/ - -# FAKE - F# Make -.fake/ - -# JetBrains Rider -.idea/ -*.sln.iml - -# CodeRush -.cr/ - -# Python Tools for Visual Studio (PTVS) -__pycache__/ -*.pyc - -# Cake - Uncomment if you are using it -# tools/** -# !tools/packages.config - -# Telerik's JustMock configuration file -*.jmconfig - -# BizTalk build output -*.btp.cs -*.btm.cs -*.odx.cs -*.xsd.cs +/.vs/Bank/v15 +/Bank/obj/Debug +/Client/bin +/Client/obj +/Common/bin +/Common/obj +/keys +/Server/Resources/0x200.d +/Server/bin +/Server/obj diff --git a/Bank.sln b/Bank.sln new file mode 100644 index 0000000..03b8605 --- /dev/null +++ b/Bank.sln @@ -0,0 +1,37 @@ + +Microsoft Visual Studio Solution File, Format Version 12.00 +# Visual Studio 15 +VisualStudioVersion = 15.0.27130.2020 +MinimumVisualStudioVersion = 10.0.40219.1 +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Server", "Server\Server.csproj", "{B458552A-5884-4B27-BA6B-826BC5590106}" +EndProject +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Client", "Client\Client.csproj", "{2236D5D4-7816-4630-8C86-0F0BDD46D7D8}" +EndProject +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Common", "Common\Common.csproj", "{23EB87D4-E310-48C4-A931-0961C83892D7}" +EndProject +Global + GlobalSection(SolutionConfigurationPlatforms) = preSolution + Debug|Any CPU = Debug|Any CPU + Release|Any CPU = Release|Any CPU + EndGlobalSection + GlobalSection(ProjectConfigurationPlatforms) = postSolution + {B458552A-5884-4B27-BA6B-826BC5590106}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {B458552A-5884-4B27-BA6B-826BC5590106}.Debug|Any CPU.Build.0 = Debug|Any CPU + {B458552A-5884-4B27-BA6B-826BC5590106}.Release|Any CPU.ActiveCfg = Release|Any CPU + {B458552A-5884-4B27-BA6B-826BC5590106}.Release|Any CPU.Build.0 = Release|Any CPU + {2236D5D4-7816-4630-8C86-0F0BDD46D7D8}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {2236D5D4-7816-4630-8C86-0F0BDD46D7D8}.Debug|Any CPU.Build.0 = Debug|Any CPU + {2236D5D4-7816-4630-8C86-0F0BDD46D7D8}.Release|Any CPU.ActiveCfg = Release|Any CPU + {2236D5D4-7816-4630-8C86-0F0BDD46D7D8}.Release|Any CPU.Build.0 = Release|Any CPU + {23EB87D4-E310-48C4-A931-0961C83892D7}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {23EB87D4-E310-48C4-A931-0961C83892D7}.Debug|Any CPU.Build.0 = Debug|Any CPU + {23EB87D4-E310-48C4-A931-0961C83892D7}.Release|Any CPU.ActiveCfg = Release|Any CPU + {23EB87D4-E310-48C4-A931-0961C83892D7}.Release|Any CPU.Build.0 = Release|Any CPU + EndGlobalSection + GlobalSection(SolutionProperties) = preSolution + HideSolutionNode = FALSE + EndGlobalSection + GlobalSection(ExtensibilityGlobals) = postSolution + SolutionGuid = {E0FD1BD7-4DD9-46C6-A710-6E5371F96DE7} + EndGlobalSection +EndGlobal diff --git a/Bank/App.config b/Bank/App.config new file mode 100644 index 0000000..731f6de --- /dev/null +++ b/Bank/App.config @@ -0,0 +1,6 @@ + + + + + + \ No newline at end of file diff --git a/Bank/Bank.csproj b/Bank/Bank.csproj new file mode 100644 index 0000000..b8cbc5d --- /dev/null +++ b/Bank/Bank.csproj @@ -0,0 +1,52 @@ + + + + + Debug + AnyCPU + {ACD33936-2A7E-48F0-B7E5-8AA818C309D5} + Exe + Bank + Bank + v4.6.1 + 512 + true + + + AnyCPU + true + full + false + bin\Debug\ + DEBUG;TRACE + prompt + 4 + + + AnyCPU + pdbonly + true + bin\Release\ + TRACE + prompt + 4 + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/Bank/Program.cs b/Bank/Program.cs new file mode 100644 index 0000000..7f59fb7 --- /dev/null +++ b/Bank/Program.cs @@ -0,0 +1,15 @@ +using System; +using System.Collections.Generic; +using System.Linq; +using System.Text; +using System.Threading.Tasks; + +namespace Bank +{ + class Program + { + static void Main(string[] args) + { + } + } +} diff --git a/Bank/Properties/AssemblyInfo.cs b/Bank/Properties/AssemblyInfo.cs new file mode 100644 index 0000000..4887e50 --- /dev/null +++ b/Bank/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("Bank")] +[assembly: AssemblyDescription("")] +[assembly: AssemblyConfiguration("")] +[assembly: AssemblyCompany("")] +[assembly: AssemblyProduct("Bank")] +[assembly: AssemblyCopyright("Copyright © 2018")] +[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("acd33936-2a7e-48f0-b7e5-8aa818c309d5")] + +// 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/BankProject b/BankProject new file mode 160000 index 0000000..e44ae49 --- /dev/null +++ b/BankProject @@ -0,0 +1 @@ +Subproject commit e44ae49cebfda90b5394073c04798a79b1f427e2 diff --git a/Client/App.config b/Client/App.config new file mode 100644 index 0000000..731f6de --- /dev/null +++ b/Client/App.config @@ -0,0 +1,6 @@ + + + + + + \ No newline at end of file diff --git a/Client/Client.csproj b/Client/Client.csproj new file mode 100644 index 0000000..37ff826 --- /dev/null +++ b/Client/Client.csproj @@ -0,0 +1,85 @@ + + + + + Debug + AnyCPU + {2236D5D4-7816-4630-8C86-0F0BDD46D7D8} + Exe + Client + Client + v4.6.1 + 512 + true + + + AnyCPU + true + full + false + bin\Debug\ + DEBUG;TRACE + prompt + 4 + + + AnyCPU + pdbonly + true + bin\Release\ + TRACE + prompt + 4 + + + + + + + + + + + + + + + + + + + True + True + Resources.resx + + + + + + + + + + + + {23eb87d4-e310-48c4-a931-0961c83892d7} + Common + + + + + + Designer + + + Designer + + + + + ResXFileCodeGenerator + Resources.Designer.cs + + + + \ No newline at end of file diff --git a/Client/ConsoleForms.cs b/Client/ConsoleForms.cs new file mode 100644 index 0000000..61467c8 --- /dev/null +++ b/Client/ConsoleForms.cs @@ -0,0 +1,1370 @@ +//#define STRICT_LAYOUT + +using System; +using System.Collections.Generic; +using System.Linq; +using System.Runtime.Serialization; +using System.Text; +using System.Threading.Tasks; +using System.Diagnostics; +using Tofvesson.Crypto; +using System.Runtime.CompilerServices; +using System.Xml; +using System.Reflection; +using Client.Properties; +using Tofvesson.Collections; +using System.Collections.ObjectModel; + +namespace ConsoleForms +{ + [Flags] + public enum Gravity + { + LEFT = 1, + RIGHT = 2, + TOP = 4, + BOTTOM = 8 + } + + public abstract class Padding + { + public abstract int Left(); + public abstract int Right(); + public abstract int Top(); + public abstract int Bottom(); + } + + public sealed class RelativePadding : Padding + { + private readonly float left, right, top, bottom; + + public RelativePadding(float left, float right, float top, float bottom) + { + this.left = Math.Max(1, Math.Min(0, left)); + this.right = Math.Max(1, Math.Min(0, right)); + this.top = Math.Max(1, Math.Min(0, top)); + this.bottom = Math.Max(1, Math.Min(0, bottom)); + } + + public override int Bottom() => (int)Math.Round(Console.WindowHeight * bottom); + public override int Left() => (int)Math.Round(Console.WindowWidth * left); + public override int Right() => (int)Math.Round(Console.WindowWidth * right); + public override int Top() => (int)Math.Round(Console.WindowHeight * top); + } + + public sealed class AbsolutePadding : Padding + { + private readonly int left, right, top, bottom; + + public AbsolutePadding(int left, int right, int top, int bottom) + { + this.left = Math.Max(0, left); + this.right = Math.Max(0, right); + this.top = Math.Max(0, top); + this.bottom = Math.Max(0, bottom); + } + + public override int Bottom() => bottom; + public override int Left() => left; + public override int Right() => right; + public override int Top() => top; + } + + + static class Enums + { + internal static void LayoutCheck(ref Gravity g) + { + if (!IsValidFlag(g)) + { +#if STRICT_LAYOUT + throw new LayoutParameterException(); +#else + Debug.WriteLine($"Invalid layout parameters {{{g}}}:\n{Environment.StackTrace}\n"); + g = 0; +#endif + } + } + internal static bool HasFlag(Gravity value, Gravity flag) => (value & flag) == flag; + internal static bool IsValidFlag(Gravity g) => + !( + (HasFlag(g, Gravity.LEFT) && HasFlag(g, Gravity.RIGHT)) || // Gravity cannot be both LEFT and RIGHT + (HasFlag(g, Gravity.TOP) && HasFlag(g, Gravity.BOTTOM)) // Gravity cannot be both TOP and BOTTOM + ); + } + + // Handles graphics and rendering instrumentation + public sealed class ConsoleController + { + public class KeyEvent + { + public bool ValidEvent { get; set; } + public ConsoleKeyInfo Event { get; } + + internal KeyEvent(ConsoleKeyInfo info) { Event = info; ValidEvent = true; } + } + + public static readonly ConsoleController singleton = new ConsoleController(); + + private readonly List> renderQueue = new List>(); + private int width = Console.WindowWidth, height = Console.WindowHeight; + private CancellationPipe cancel; + private Task resizeListener; + + public bool Dirty + { + get + { + Region occlusion = new Region(); + for (int i = renderQueue.Count - 1; i >= 0; --i) + { + Tuple lParams = renderQueue[i].Item2.ComputeLayoutParams(width, height); + Region test = renderQueue[i].Item1.Occlusion; + if (renderQueue[i].Item1.Dirty && test.Subtract(occlusion).Area > 0) + return true; + else occlusion = occlusion.Add(test); + } + return false; + } + } + + private ConsoleController(bool resizeListener = true) + { + if (resizeListener) EnableResizeListener(); + RegisterListener((w, h) => + { + width = w; + height = h; + Draw(); + }); + + RegisterListener((w1, h1, w2, h2) => Console.Clear()); + } + + public void AddView(View v, bool redraw = true) => AddView(v, LayoutMeta.Centering(v), redraw); + public void AddView(View v, LayoutMeta meta, bool redraw = true) + { + renderQueue.Add(new Tuple(v, meta)); + Draw(false); + } + + public void CloseTop() => CloseView(renderQueue[renderQueue.Count - 1].Item1); + public void CloseView(int idx) => CloseView(renderQueue[idx].Item1); + public void CloseView(View v, bool redraw = true, int maxCloses = -1) + { + if (maxCloses == 0) return; + + Region r = new Region(); + bool needsRedraw = false; + int closed = 0; + for (int i = renderQueue.Count - 1; i >= 0; --i) + if (renderQueue[i].Item1.Equals(v)) + { + Region test = renderQueue[i].Item1.Occlusion; + test.Offset(renderQueue[i].Item2.ComputeLayoutParams(width, height)); + Region removing = test.Subtract(r); + needsRedraw |= removing.Area > 0; + + Region cmp; + for (int j = i - 1; !needsRedraw && j >= 0; --j) + needsRedraw |= (cmp = renderQueue[j].Item1.Occlusion).Subtract(removing).Area != cmp.Area; + + renderQueue.RemoveAt(i); + ClearRegion(removing); + if (++closed == maxCloses) break; + } + if (redraw && needsRedraw) Draw(false); + } + + public void Draw() => Draw(false); + + // downTo allows for partial rendering updates + private void Draw(bool ignoreOcclusion, int downTo = 0) + { + if (downTo < 0) downTo = 0; + if (downTo >= renderQueue.Count) return; + Console.CursorVisible = false; + byte[] occlusionMap = new byte[(renderQueue.Count / 8) + (renderQueue.Count % 8 != 0 ? 1 : 0)]; + Stack> layoutParams = new Stack>(); + + Region occlusion = new Region(); + for (int i = renderQueue.Count - 1; i >= downTo; --i) + { + Tuple lParams = renderQueue[i].Item2.ComputeLayoutParams(width, height); + if (!ignoreOcclusion) + { + Region test = renderQueue[i].Item1.Occlusion; + test.Offset(lParams); + if (test.Subtract(occlusion).Area == 0) + occlusionMap[i / 8] |= (byte)(1 << (i % 8)); + else + { + occlusion = occlusion.Add(test); + layoutParams.Push(lParams); + } + } else layoutParams.Push(lParams); + } + + for (int i = downTo; i < renderQueue.Count; ++i) + if ((occlusionMap[i / 8] & (1 << (i % 8))) == 0) + renderQueue[i].Item1.Draw(layoutParams.Pop()); + } + + public KeyEvent ReadKey(bool redrawOnDirty = true) + { + KeyEvent keyInfo = new KeyEvent(Console.ReadKey(true)); + int lowestDirty = -1; + int count = renderQueue.Count - 1; + for (int i = count; i >= 0; --i) + if (renderQueue[i].Item1.HandleKeyEvent(keyInfo, i == count)) + lowestDirty = i; + if (redrawOnDirty) Draw(false, lowestDirty); + return keyInfo; + } + + public void EnableResizeListener() + { + if (cancel != null) return; + // Set up console window resize listener + cancel = new CancellationPipe(); + resizeListener = new Task(() => ConsoleResizeListener(cancel)); // Start resize listener asynchronously + resizeListener.Start(); + } + + public async void DisableResizeListener() + { + if (cancel == null) return; + cancel.Cancel(); + await resizeListener; + } + + + public delegate void WindowChangeListener(int fromWidth, int fromHeight, int toWidth, int toHeight); + public delegate void WindowChangeCompleteListener(int width, int height); + + private readonly List changeListeners = new List(); + private readonly List completeListeners = new List(); + + public void RegisterListener(WindowChangeListener listener) => changeListeners.Add(listener); + public void RegisterListener(WindowChangeCompleteListener listener) => completeListeners.Add(listener); + public void UnRegisterListener(WindowChangeListener listener) => changeListeners.RemoveAll(p => p == listener); + public void UnRegisterListener(WindowChangeCompleteListener listener) => completeListeners.RemoveAll(p => p == listener); + + private void ConsoleResizeListener(CancellationPipe cancel) + { + int consoleWidth = Console.WindowWidth; + int consoleHeight = Console.WindowHeight; + bool trigger = false; + int trigger_inc = 0; + + while (!cancel.Cancelled) + { + int readWidth = Console.WindowWidth; + int readHeight = Console.WindowHeight; + if (readWidth != consoleWidth || readHeight != consoleHeight) + { + trigger = true; + foreach (var listener in changeListeners) listener(consoleWidth, consoleHeight, readWidth, readHeight); + consoleWidth = readWidth; + consoleHeight = readHeight; + } + else if (trigger && ++trigger_inc >= 5) + { + foreach (var listener in completeListeners) listener(consoleWidth, consoleHeight); + trigger = false; + } + System.Threading.Thread.Sleep(50); + } + } + + private static void ClearRegion(Region r, ConsoleColor clearColor = ConsoleColor.Black) + { + foreach (var rect in r.SubRegions) ClearRegion(rect, clearColor); + } + + private static void ClearRegion(Rectangle rect, ConsoleColor clearColor = ConsoleColor.Black) + { + Console.BackgroundColor = clearColor; + Console.ForegroundColor = ConsoleColor.White; + for (int i = rect.Top; i <= rect.Bottom; ++i) + { + Console.SetCursorPosition(rect.Left, i); + for (int j = rect.Right - rect.Left; j > 0; --j) + Console.Write(' '); + } + } + + private static ViewData DoElementParse(XmlNode el) + { + ViewData data = new ViewData(el.LocalName, el.InnerText); + + if (el.Attributes != null) + foreach (var attr in el.Attributes) + if (attr is XmlAttribute) + data.attributes[((XmlAttribute)attr).Name] = ((XmlAttribute)attr).Value; + + if (el.ChildNodes != null) + foreach (var child in el.ChildNodes) + if (child is XmlNode) data.nestedData.Add(DoElementParse((XmlNode)child)); + + return data; + } + + private static Dictionary>> cache = new Dictionary>>(); + + public static List> LoadResourceViews(string name, bool doCache = true) + { + if (cache.ContainsKey(name)) + return cache[name]; + + PropertyInfo[] properties = typeof(Resources).GetProperties(BindingFlags.NonPublic | BindingFlags.Static); + foreach (var prop in properties) + if (prop.Name.Equals(name) && prop.PropertyType.Equals(typeof(string))) + return LoadViews((string)prop.GetValue(null), doCache ? name : null); + throw new SystemException($"Resource { name } could not be located!"); + } + public static List> LoadViews(string xml, string cacheID = null) + { + if (cacheID != null && cache.ContainsKey(cacheID)) + return cache[cacheID]; + + XmlDocument doc = new XmlDocument(); + doc.LoadXml(xml); + + string ns = doc.FirstChild.NextSibling.Attributes != null ? doc.FirstChild.NextSibling.Attributes.GetNamedItem("xmlns")?.Value ?? "" : ""; + + List> views = new List>(); + + foreach (var child in doc.FirstChild.NextSibling.ChildNodes) + { + if (!(child is XmlNode) || child is XmlComment) continue; + ViewData data = DoElementParse((XmlNode)child); + View load; + Type type; + try { type = Type.GetType(ns + '.' + data.Name, true); } + catch { type = Type.GetType(data.Name, true); } + + ConstructorInfo info = type.GetConstructor(new Type[] { typeof(ViewData) }); + + string id = data.attributes.ContainsKey("id") ? data.attributes["id"] : ""; + + load = (View)info.Invoke(new object[] { data }); + + views.Add(new Tuple(id, load)); + } + + if (cacheID != null) cache[cacheID] = views; + + return views; + } + + public delegate void Runnable(); + public void Popup(string message, long timeout, ConsoleColor borderColor = ConsoleColor.Blue, Runnable onExpire = null) + { + TextBox popup = new TextBox( + new ViewData("ConsoleForms.TextBox") + .SetAttribute("padding_left", 2) + .SetAttribute("padding_right", 2) + .SetAttribute("padding_top", 1) + .SetAttribute("padding_bottom", 1) + .AddNested(new ViewData("Text", message)) // Add message + ) + { + BackgroundColor = ConsoleColor.White, + TextColor = ConsoleColor.Black, + BorderColor = borderColor, + }; + + AddView(popup, LayoutMeta.Centering(popup)); + + new Timer(() => { + CloseView(popup); + onExpire?.Invoke(); + }, timeout).Start(); + + } + } + + public sealed class ContextManager + { + public Context Current { get; private set; } + + public void LoadContext(Context ctx) + { + Current?.OnDestroy(); + Current = ctx; + Current.OnCreate(); + } + + public bool Update(ConsoleController.KeyEvent keypress, bool hasKeypress = true) + => Current?.Update(keypress, hasKeypress) == true; + } + + public abstract class Context + { + protected static readonly ConsoleController controller = ConsoleController.singleton; + protected readonly ReadOnlyCollection> views; + protected readonly ContextManager manager; + + public Context(ContextManager manager, string contextName, bool asResource = true) + { + this.manager = manager; + views = new ReadOnlyCollectionBuilder>(asResource ? ConsoleController.LoadResourceViews(contextName) : ConsoleController.LoadViews(contextName)).ToReadOnlyCollection(); + } + + public virtual bool Update(ConsoleController.KeyEvent keypress, bool hasKeypress = true) + { + if (keypress.ValidEvent && keypress.Event.Key == ConsoleKey.Escape) OnDestroy(); + return controller.Dirty; + } + + public abstract void OnCreate(); // Called when a context is loaded as the primary context of the ConsoleController + public abstract void OnDestroy(); // Called when a context is unloaded + + protected void RegisterSelectListeners(DialogBox.SelectListener listener, params string[] viewNames) + { + foreach(var viewName in viewNames) + { + View v = views.GetNamed(viewName); + if (v != null && v is DialogBox) + ((DialogBox)v).RegisterSelectListener(listener); + } + } + } + + public sealed class ViewData + { + public delegate string TransformAction(ViewData rawValue); + + public string Name { get; } + public string InnerText { get; } + public readonly Dictionary attributes = new Dictionary(); + public readonly List nestedData = new List(); + + public ViewData(string name, string innerText = "") + { + Name = (name ?? "").Replace("\r", ""); + InnerText = (innerText ?? "").Replace("\r", ""); + } + + public ViewData Get(string name) + { + foreach (var data in nestedData) + if (data.Name.Equals(name)) + return data; + return null; + } + + public int TextAsInt(int def = default(int)) => int.TryParse(InnerText, out int p) ? p : def; + public int AttribueAsInt(string name, int def = default(int)) => attributes.ContainsKey(name) && int.TryParse(attributes[name], out int p) ? p : def; + public bool AttribueAsBool(string name, bool def = default(bool)) => attributes.ContainsKey(name) && bool.TryParse(attributes[name], out bool p) ? p : def; + public Tuple[] CollectSub(string name, TransformAction action = null) + { + List> l = new List>(); + foreach (var data in nestedData) + if (data.Name.Equals(name)) + l.Add(new Tuple(data.InnerText, action?.Invoke(data) ?? "")); + return l.ToArray(); + } + public string NestedText(string nestedDataName, string def = "") + { + foreach (var data in nestedData) + if (data.Name.Equals(nestedDataName)) + return data.InnerText; + return def; + } + public int NestedInt(string nestedDataName, int def = default(int)) + { + foreach (var data in nestedData) + if (data.Name.Equals(nestedDataName) && int.TryParse(data.InnerText, out int p)) + return p; + return def; + } + public int NestedAttribute(string nestedName, string attributeName, int def = default(int)) + { + foreach (var data in nestedData) + if (data.Name.Equals(nestedName) && data.attributes.ContainsKey(attributeName) && int.TryParse(data.attributes[attributeName], out int p)) + return p; + return def; + } + public ViewData SetAttribute(string attrName, T value) + { + attributes[attrName] = value == null ? "null" : value.ToString(); + return this; + } + public ViewData AddNested(ViewData nest) + { + nestedData.Add(nest); + return this; + } + + public string GetAttribute(string attr, string def = "") => attributes.ContainsKey(attr) ? attributes[attr] : def; + } + + public static class Extensions + { + public static int CollectiveLength(this ViewData[] data) + { + int len = 0; + foreach (var val in data) + len += val?.InnerText.Length ?? 0; + return len; + } + } + + + public abstract class View + { + protected delegate void EventAction(); + + protected static readonly Padding DEFAULT_PADDING = new AbsolutePadding(0, 0, 0, 0); + + protected readonly Padding padding; + protected readonly Gravity gravity; + protected readonly bool vCenter, hCenter; + protected readonly string back_data; + + public char Border { get; set; } + public bool DrawBorder { get; set; } + public ConsoleColor BorderColor { get; set; } + public int ContentWidth { get; protected set; } + public int ContentHeight { get; protected set; } + public abstract Region Occlusion { get; } + public bool Dirty { get; set; } + + public View(ViewData parameters) + { + this.padding = new AbsolutePadding(parameters.AttribueAsInt("padding_left"), parameters.AttribueAsInt("padding_right"), parameters.AttribueAsInt("padding_top"), parameters.AttribueAsInt("padding_bottom")); + this.gravity = (Gravity) parameters.AttribueAsInt("gravity"); + this.BorderColor = (ConsoleColor)parameters.AttribueAsInt("border", (int)ConsoleColor.Blue); + this.Border = ' '; + DrawBorder = true; + + back_data = parameters.GetAttribute("back"); + + // Do check to ensure that gravity flags are valid + Enums.LayoutCheck(ref gravity); + vCenter = !Enums.HasFlag(gravity, Gravity.LEFT) && !Enums.HasFlag(gravity, Gravity.RIGHT); + hCenter = !Enums.HasFlag(gravity, Gravity.TOP) && !Enums.HasFlag(gravity, Gravity.BOTTOM); + } + + public void Draw(Tuple t) => Draw(t.Item1, t.Item2); + public void Draw(int left, int top) + { + Dirty = false; + if (DrawBorder) _DrawBorder(left, top); + _Draw(left + 1, top); + } + public virtual void _DrawBorder(int left, int top) + { + Console.BackgroundColor = BorderColor; + Console.SetCursorPosition(left, top - 1); + Console.Write(Filler(Border, ContentWidth + 1)); + for(int i = -1; i { }; + var views = ConsoleController.LoadResourceViews(components[0]); + var view = views.GetNamed(components[1]); + return () => + { + if(close) ConsoleController.singleton.CloseView(this); + ConsoleController.singleton.AddView(view); + }; + } + + protected static string Filler(char c, int count) + { + if (count == 0) return ""; + StringBuilder builder = new StringBuilder(count); + for (int i = 0; i < count; ++i) builder.Append(c); + return builder.ToString(); + } + } + + public class TextBox : View + { + protected readonly string[] text; + protected string[] text_render; + protected int maxWidth, maxHeight; + + public int MaxWidth + { + get => maxWidth; + + set + { + maxWidth = value; + text_render = ComputeTextDimensions(text); + Dirty = true; + } + } + public int MaxHeight + { + get => maxHeight; + + set + { + maxHeight = value; + text_render = ComputeTextDimensions(text); + Dirty = true; + } + } + public override Region Occlusion => new Region(new Rectangle(0, -1, ContentWidth + 2, ContentHeight)); + + //public char Border { get; set; } + //public ConsoleColor BorderColor { get; set; } + public ConsoleColor BackgroundColor { get; set; } + public ConsoleColor TextColor { get; set; } + + public TextBox(ViewData parameters) : base(parameters) + { + //BorderColor = (ConsoleColor) parameters.AttribueAsInt("border", (int)ConsoleColor.Blue); + BackgroundColor = (ConsoleColor)parameters.AttribueAsInt("color_background", (int)ConsoleColor.White); + TextColor = (ConsoleColor)parameters.AttribueAsInt("color_text", (int)ConsoleColor.Black); + + Border = ' '; + this.text = parameters.NestedText("Text").Split(' '); + int widest = 0; + foreach (var t in parameters.NestedText("Text").Split('\n')) + if (t.Length > widest) + widest = t.Length; + this.maxWidth = parameters.AttribueAsInt("width") < 1 ? widest : parameters.AttribueAsInt("width"); + this.maxHeight = parameters.AttribueAsInt("height", -1); + + // Compute the layout of the text to be rendered + text_render = ComputeTextDimensions(this.text); + int actualWidth = 0; + foreach (var t in text_render) if (actualWidth < t.Length) actualWidth = t.Length; + ContentWidth = maxWidth + padding.Left() + padding.Right(); + ContentHeight = text_render.Length + padding.Top() + padding.Bottom(); + } + + protected virtual string[] ComputeTextDimensions(string[] text) + { + if (maxHeight == 0) + return new string[0]; + + BoundedList generate = new BoundedList(maxHeight); + + for (int i = 0; i < text.Length; ++i) + { + if (generate.Count == 0) + { + string[] split = Subsplit(text[i], maxWidth); + for (int j = 0; j < split.Length; ++j) + if (!generate.Add(split[j])) + goto Generated; + } + else + { + if (WillSubSplit(text[i], maxWidth)) + { + int startAdd = 0; + string[] split; + if (generate[generate.Count - 1].Length != maxWidth) + { + startAdd = 1; + split = Subsplit(generate[generate.Count - 1] + " " + text[i], maxWidth); + generate[generate.Count - 1] = split[0]; + } + else split = Subsplit(text[i], maxWidth); + for (int j = startAdd; j < split.Length; ++j) + if (!generate.Add(split[j])) + goto Generated; + } + else + { + if (generate[generate.Count - 1].Length + text[i].Length < maxWidth) + generate[generate.Count - 1] += " " + text[i]; + else if (!generate.Add(text[i])) + break; + } + } + } + + Generated: + return generate.ToArray(); + } + + private static string[] Subsplit(string s, int max) + { + int nlCount = 0; + for (int i = 0; i < s.Length; ++i) if (s[i] == '\n') ++nlCount; + + string[] result = new string[((s.Length - nlCount) / max) + nlCount + ((s.Length - nlCount) % max != 0 ? 1 : 0)]; + + int read = 0; + for (int i = 0; i < result.Length; ++i) + { + StringBuilder subCollect = new StringBuilder(); + int idx = read; + int valid = 0; + while (idx < s.Length && valid < max) + { + char c = s[idx]; + subCollect.Append(c); + ++idx; + if (c != '\n') ++valid; + } + string sub = subCollect.ToString(); + if (sub.Contains('\n')) + { + while (sub.Contains('\n')) + { + result[i++] = sub.Substring(0, sub.IndexOf('\n')); + sub = sub.Substring(sub.IndexOf('\n') + 1); + } + if(i ((s.Length / max) + (s.Length % max != 0 ? 1 : 0)) > 1 || s.Contains('\n'); + + protected override void _Draw(int left, int top) + { + DrawEmptyPadding(left, ref top, padding.Top()); + DrawContent(left, ref top); + DrawEmptyPadding(left, ref top, padding.Bottom()); + } + + protected void DrawContent(int left, ref int top) + { + int pl = padding.Left(), pr = padding.Right(); + Console.BackgroundColor = BackgroundColor; + Console.ForegroundColor = TextColor; + for (int i = 0; i < text_render.Length; ++i) + { + Console.SetCursorPosition(left, top++); + Console.Write(Filler(' ', pl) + text_render[i] + Filler(' ', MaxWidth - text_render[i].Length) + Filler(' ', pr)); + } + } + + protected void DrawEmptyPadding(int left, ref int top, int padHeight) + { + int pl = padding.Left(), pr = padding.Right(); + for (int i = padHeight; i > 0; --i) + { + Console.SetCursorPosition(left, top++); + Console.BackgroundColor = BackgroundColor; + Console.Write(Filler(' ', maxWidth + pl + pr)); + } + } + + public override bool HandleKeyEvent(ConsoleController.KeyEvent info, bool inFocus) => base.HandleKeyEvent(info, inFocus); + } + + public class ListView : View + { + + + public ListView(ViewData parameters) : base(parameters) + { + + } + + public override Region Occlusion => throw new NotImplementedException(); + + protected override void _Draw(int left, int top) + { + throw new NotImplementedException(); + } + } + + public class DialogBox : TextBox + { + public delegate void SelectListener(DialogBox view, int selectionIndex, string selection); + + protected readonly ViewData[] options; + protected int select; + protected SelectListener listener; + + public int Select + { + get => select; + set => select = value < 0 ? 0 : value >= options.Length ? options.Length - 1 : value; + } + public override Region Occlusion => new Region(new Rectangle(0, -1, ContentWidth + 2, ContentHeight + 2)); + + public ConsoleColor SelectColor { get; set; } + public ConsoleColor NotSelectColor { get; set; } + public string[] Options { get => options.Transform(d => d.InnerText); } + + private static int ComputeLength(Tuple[] opts) => opts.CollectiveLength(true) + opts.Length - 1; + + public DialogBox(ViewData parameters) : + base(parameters.SetAttribute("width", + Math.Max( + parameters.AttribueAsInt("width") < 1 ? parameters.NestedText("Text").Length : parameters.AttribueAsInt("width"), + ComputeLength(parameters.Get("Options").CollectSub("Option")) + ))) + { + ViewData optionsData = parameters.Get("Options"); + this.options = optionsData.nestedData.Filter(p => p.Name.Equals("Option")).ToArray(); + this.select = parameters.AttribueAsInt("select"); + ContentHeight += 2; + select = select < 0 ? 0 : select >= options.Length ? 0 : select; + SelectColor = (ConsoleColor)parameters.AttribueAsInt("select_color", (int)ConsoleColor.Gray); + NotSelectColor = (ConsoleColor)parameters.AttribueAsInt("unselect_color", (int)ConsoleColor.White); + } + + protected override void _Draw(int left, int top) + { + DrawEmptyPadding(left, ref top, padding.Top()); + base.DrawContent(left, ref top); + DrawEmptyPadding(left, ref top, 1); + DrawOptions(left, ref top); + DrawEmptyPadding(left, ref top, padding.Bottom()); + } + + protected virtual void DrawOptions(int left, ref int top) + { + int pl = padding.Left(), pr = padding.Right(); + Console.SetCursorPosition(left, top++); + + int pad = MaxWidth - options.CollectiveLength() - options.Length + pl + pr; + int lpad = (int)(pad / 2f); + Console.BackgroundColor = BackgroundColor; + Console.Write(Filler(' ', lpad)); + for (int i = 0; i < options.Length; ++i) + { + Console.BackgroundColor = i == select ? SelectColor : NotSelectColor; + Console.Write(options[i].InnerText); + Console.BackgroundColor = BackgroundColor; + Console.Write(' '); + } + Console.Write(Filler(' ', pad - lpad)); + } + + public override bool HandleKeyEvent(ConsoleController.KeyEvent evt, bool inFocus) + { + bool changed = base.HandleKeyEvent(evt, inFocus); + ConsoleKeyInfo info = evt.Event; + if (!evt.ValidEvent || !inFocus) return changed; + switch (info.Key) + { + case ConsoleKey.LeftArrow: + if (select > 0) --select; + break; + case ConsoleKey.RightArrow: + if (select < options.Length - 1) ++select; + break; + case ConsoleKey.Enter: + ParseAction(options[select])(); + listener?.Invoke(this, select, options[select].InnerText); + return changed; + default: + return changed; + } + return true; + } + + public void RegisterSelectListener(SelectListener listener) => this.listener = listener; + } + + public class InputTextBox : TextBox + { + public enum InputType + { + Any, + AlphaNumeric, + Integer, + Decimal, + Alphabet + } + public sealed class InputField + { + public const char hide_char = '*'; + + public string Label { get; private set; } + public int MaxLength { get; private set; } + public bool ShowText { get; set; } + public string Text { get; set; } + public int SelectIndex { get; set; } + public InputType Input { get; set; } + public ConsoleColor TextColor { get; set; } + public ConsoleColor BackgroundColor { get; set; } + public ConsoleColor SelectTextColor { get; set; } + public ConsoleColor SelectBackgroundColor { get; set; } + public string InputTypeString + { + get + { + switch (Input) + { + case InputType.Any: + return "Any"; + case InputType.AlphaNumeric: + return "AlphaNumeric"; + case InputType.Integer: + return "Integer"; + case InputType.Decimal: + return "Decimal"; + case InputType.Alphabet: + return "Alphabet"; + } + throw new SystemException("Invalid system state detected"); + } + + set + { + switch (value.ToLower()) + { + case "alphanumeric": + Input = InputType.AlphaNumeric; + break; + case "integer": + Input = InputType.Integer; + break; + case "decimal": + Input = InputType.Decimal; + break; + case "alphabet": + Input = InputType.Alphabet; + break; + default: + Input = InputType.Any; + break; + } + } + } + internal int RenderStart { get; set; } + + public InputField(string label, int maxLength) + { + TextColor = ConsoleColor.Black; + BackgroundColor = ConsoleColor.DarkGray; + SelectTextColor = ConsoleColor.Black; + SelectBackgroundColor = ConsoleColor.Gray; + Input = InputType.Any; + Label = label; + MaxLength = maxLength; + Text = ""; + } + + public bool IsValidChar(char c) => + (Input == InputType.Any) || + (Input == InputType.AlphaNumeric && c.IsAlphaNumeric()) || + (Input == InputType.Alphabet && c.IsAlphabetical()) || + (Input == InputType.Integer && c.IsNumber()) || + (Input == InputType.Decimal && c.IsDecimal()); + } + + public delegate void SubmissionListener(InputTextBox view); + public delegate bool TextEnteredListener(InputTextBox view, InputField change, ConsoleKeyInfo info); + + public ConsoleColor DefaultBackgroundColor { get; set; } + public ConsoleColor DefaultTextColor { get; set; } + public ConsoleColor DefaultSelectBackgroundColor { get; set; } + public ConsoleColor DefaultSelectTextColor { get; set; } + public InputField[] Inputs { get; private set; } + private int selectedField; + public int SelectedField + { + get => selectedField; + set + { + selectedField = value; + Dirty = true; + } + } + private string[][] splitInputs; + + public SubmissionListener SubmissionsListener { protected get; set; } + public TextEnteredListener InputListener { protected get; set; } + + public InputTextBox(ViewData parameters) : base(parameters) + { + int + sBC = parameters.AttribueAsInt("textfield_select_color", (int)ConsoleColor.Gray), + sTC = parameters.AttribueAsInt("text_select_color", (int)ConsoleColor.Black), + BC = parameters.AttribueAsInt("field_noselect_color", (int)ConsoleColor.DarkGray), + TC = parameters.AttribueAsInt("text_noselect_color", (int)ConsoleColor.Black); + + DefaultBackgroundColor = (ConsoleColor)BC; + DefaultTextColor = (ConsoleColor)TC; + DefaultSelectBackgroundColor = (ConsoleColor)sBC; + DefaultSelectTextColor = (ConsoleColor)sTC; + + List fields = new List(); + foreach (var data in parameters.nestedData.GetFirst(d => d.Name.Equals("Fields")).nestedData) + if (!data.Name.Equals("Field")) continue; + else fields.Add(new InputField(data.InnerText, data.AttribueAsInt("max_length", -1)) + { + ShowText = !data.AttribueAsBool("hide", false), + Text = data.GetAttribute("default"), + InputTypeString = data.GetAttribute("input_type"), + TextColor = (ConsoleColor)data.AttribueAsInt("color_text", TC), + BackgroundColor = (ConsoleColor)data.AttribueAsInt("color_background", BC), + SelectTextColor = (ConsoleColor)data.AttribueAsInt("color_text_select", sTC), + SelectBackgroundColor = (ConsoleColor)data.AttribueAsInt("color_background_select", sBC) + }); + + Inputs = fields.ToArray(); + + int computedSize = 0; + splitInputs = new string[Inputs.Length][]; + for(int i = 0; i< Inputs.Length; ++i) + { + splitInputs[i] = ComputeTextDimensions(Inputs[i].Label.Split(' ')); + computedSize += splitInputs[i].Length; + } + ContentHeight += computedSize + Inputs.Length * 2; + } + + protected override void _Draw(int left, int top) + { + DrawEmptyPadding(left, ref top, padding.Top()); + DrawContent(left, ref top); + DrawInputFields(left, ref top, 1); + DrawEmptyPadding(left, ref top, padding.Bottom()); + } + + protected void DrawInputFields(int left, ref int top, int spaceHeight) + { + int pl = padding.Left(), pr = padding.Right(); + + for (int j = 0; j< Inputs.Length; ++j) + { + DrawEmptyPadding(left, ref top, spaceHeight); + + for (int i = 0; i < splitInputs[j].Length; ++i) + { + Console.SetCursorPosition(left, top++); + Console.BackgroundColor = BackgroundColor; + Console.Write(Filler(' ', pl) + splitInputs[j][i] + Filler(' ', MaxWidth - splitInputs[j][i].Length) + Filler(' ', pr)); + } + Console.SetCursorPosition(left, top++); + + // Draw padding + Console.BackgroundColor = BackgroundColor; + Console.Write(Filler(' ', pl)); + + // Draw field + Console.BackgroundColor = j == selectedField ? Inputs[j].SelectBackgroundColor : Inputs[j].BackgroundColor; + Console.ForegroundColor = j == selectedField ? Inputs[j].SelectTextColor : Inputs[j].TextColor; + Console.Write(Inputs[j].ShowText ? Inputs[j].Text.Substring(Inputs[j].RenderStart, Inputs[j].SelectIndex - Inputs[j].RenderStart) : Filler('*', Inputs[j].SelectIndex - Inputs[j].RenderStart)); + if(j == selectedField) Console.BackgroundColor = ConsoleColor.DarkGray; + Console.Write(Inputs[j].SelectIndex < Inputs[j].Text.Length ? Inputs[j].ShowText ? Inputs[j].Text[Inputs[j].SelectIndex] : '*' : ' '); + if (j == selectedField) Console.BackgroundColor = Inputs[j].SelectBackgroundColor; + int drawn = 0; + if(Inputs[j].SelectIndex < Inputs[j].Text.Length) + Console.Write( + Inputs[j].ShowText ? + Inputs[j].Text.Substring(Inputs[j].SelectIndex + 1, drawn = Math.Min(maxWidth + Inputs[j].SelectIndex - Inputs[j].RenderStart - 1, Inputs[j].Text.Length - Inputs[j].SelectIndex - 1)) : + Filler('*', drawn = Math.Min(maxWidth + Inputs[j].SelectIndex - Inputs[j].RenderStart - 1, Inputs[j].Text.Length - Inputs[j].SelectIndex - 1)) + ); + Console.Write(Filler(' ', maxWidth - 1 - drawn - Inputs[j].SelectIndex + Inputs[j].RenderStart)); + Console.ForegroundColor = ConsoleColor.Black; + + // Draw padding + Console.BackgroundColor = BackgroundColor; + Console.Write(Filler(' ', pr)); + + } + } + + public override bool HandleKeyEvent(ConsoleController.KeyEvent evt, bool inFocus) + { + bool changed = base.HandleKeyEvent(evt, inFocus); + ConsoleKeyInfo info = evt.Event; + if (!evt.ValidEvent || !inFocus || Inputs.Length == 0) return changed; + switch (info.Key) + { + case ConsoleKey.LeftArrow: + if (Inputs[selectedField].SelectIndex > 0) + { + if (Inputs[selectedField].RenderStart == Inputs[selectedField].SelectIndex--) --Inputs[selectedField].RenderStart; + } + else return changed; + break; + case ConsoleKey.RightArrow: + if (Inputs[selectedField].SelectIndex < Inputs[selectedField].Text.Length) + { + if (++Inputs[selectedField].SelectIndex - Inputs[selectedField].RenderStart == maxWidth) ++Inputs[selectedField].RenderStart; + } + else return changed; + break; + case ConsoleKey.Tab: + case ConsoleKey.DownArrow: + if (selectedField < Inputs.Length - 1) ++selectedField; + else return changed; + break; + case ConsoleKey.UpArrow: + if (selectedField > 0) --selectedField; + else return changed; + break; + case ConsoleKey.Backspace: + if (Inputs[selectedField].SelectIndex > 0) + { + if (InputListener?.Invoke(this, Inputs[selectedField], info)==false) break; + string text = Inputs[selectedField].Text; + Inputs[selectedField].Text = text.Substring(0, Inputs[selectedField].SelectIndex - 1); + if(Inputs[selectedField].SelectIndex < text.Length) Inputs[selectedField].Text += text.Substring(Inputs[selectedField].SelectIndex); + if (Inputs[selectedField].RenderStart == Inputs[selectedField].SelectIndex--) --Inputs[selectedField].RenderStart; + } + else return changed; + break; + case ConsoleKey.Delete: + if (Inputs[selectedField].SelectIndex < Inputs[selectedField].Text.Length) + { + if (InputListener?.Invoke(this, Inputs[selectedField], info) == false) break; + string text = Inputs[selectedField].Text; + Inputs[selectedField].Text = text.Substring(0, Inputs[selectedField].SelectIndex); + if (Inputs[selectedField].SelectIndex + 1 < text.Length) Inputs[selectedField].Text += text.Substring(Inputs[selectedField].SelectIndex + 1); + } + else return changed; + break; + case ConsoleKey.Enter: + SubmissionsListener?.Invoke(this); + return changed; + default: + if (info.KeyChar != 0 && info.KeyChar!='\b' && info.KeyChar!='\r' && (Inputs[selectedField].Text.Length < Inputs[selectedField].MaxLength || Inputs[selectedField].MaxLength < 0) && Inputs[selectedField].IsValidChar(info.KeyChar)) + { + if (InputListener?.Invoke(this, Inputs[selectedField], info) == false) break; + Inputs[selectedField].Text = Inputs[selectedField].Text.Substring(0, Inputs[selectedField].SelectIndex) + info.KeyChar + Inputs[selectedField].Text.Substring(Inputs[selectedField].SelectIndex); + if (++Inputs[selectedField].SelectIndex - Inputs[selectedField].RenderStart == maxWidth) ++Inputs[selectedField].RenderStart; + } else return changed; + break; + } + return true; + } + } + + public class LayoutParameterException : SystemException + { + public LayoutParameterException() { } + public LayoutParameterException(string message) : base(message) { } + public LayoutParameterException(string message, Exception innerException) : base(message, innerException) { } + protected LayoutParameterException(SerializationInfo info, StreamingContext context) : base(info, context) { } + } + + + // Computes a Left and Top value for some specified window parameters + public delegate Tuple PositionManager(int screenWidth, int screenHeight); + public sealed class LayoutMeta + { + private readonly PositionManager manager; + public LayoutMeta(PositionManager manager) + { + this.manager = manager; + } + + public Tuple ComputeLayoutParams(int width, int height) => manager(width, height); + + public static LayoutMeta Centering(View view) => new LayoutMeta( + (w, h) => + new Tuple( + SpaceMaths.CenterPad(Console.WindowWidth, view.ContentWidth).Item1, + SpaceMaths.CenterPad(Console.WindowHeight, view.ContentHeight + 1).Item1 + ) + ); + } + + public class Region + { + protected readonly List region = new List(); + + public int Area + { + get + { + int total = 0; + foreach(var rect in region) + total += (rect.Left - rect.Right) * (rect.Top - rect.Bottom); + return total; + } + } + public Rectangle[] SubRegions => region.ToArray(); + + public Region(params Rectangle[] rectangles) + { + if (rectangles.Length > 0) region.Add(rectangles[0]); + for (int i = 1; i < rectangles.Length; ++i) Add(rectangles[i]); + } + + public Region(List region) => this.region.AddRange(region); + public Region(Region r) => this.region.AddRange(r.region); + + public Region Add(Rectangle rect) + { + Region r = new Region(region); + r.IAdd(rect); + return r; + } + + protected void IAdd(Rectangle rect) + { + List recompute = new List(); + foreach (var rectangle in region) recompute.AddRange(rectangle.Subtract(rect)); + recompute.Add(rect); + region.Clear(); + region.AddRange(recompute); + } + + public Region Add(Region region) + { + Region r = new Region(this); + foreach (var rectangle in region.region) r.IAdd(rectangle); + return r; + } + + public Region Subtract(Rectangle rect) + { + Region r = new Region(region); + r.ISubtract(rect); + return r; + } + + protected void ISubtract(Rectangle rect) + { + List recompute = new List(); + foreach (var rectangle in region) recompute.AddRange(rectangle.Subtract(rect)); + region.Clear(); + region.AddRange(recompute); + } + + public Region Subtract(Region region) + { + Region r = new Region(this); + foreach (var rectangle in region.region) r.ISubtract(rectangle); + return r; + } + + public void Offset(Tuple xy) => Offset(xy.Item1, xy.Item2); + public void Offset(int x, int y) + { + foreach (var rect in region) rect.Offset(x, y); + } + } + + public class Rectangle + { + public int Top { get; private set; } + public int Bottom { get; private set; } + public int Left { get; private set; } + public int Right { get; private set; } + public Rectangle(int left, int top, int right, int bottom) + { + Left = left; + Top = top; + Right = right; + Bottom = bottom; + } + + public bool Intersects(Rectangle rect) => ((Left < rect.Right && Right >= rect.Left) || (Left <= rect.Right && Right > rect.Left)) && ((Top > rect.Bottom && Bottom <= rect.Top) || (Top >= rect.Bottom && Bottom < rect.Top)); + public bool Occludes(Rectangle rect) => Top >= rect.Top && Right >= rect.Right && Left >= rect.Left && Bottom >= rect.Bottom; + public Rectangle GetIntersecting(Rectangle rect) + => Intersects(rect) ? + new Rectangle( + Left < rect.Right ? Left : rect.Left, + Bottom < rect.Top ? rect.Top : Top, + Left < rect.Right ? rect.Right : Right, + Bottom < rect.Top ? Bottom : rect.Bottom + ) : + null; + + public Rectangle[] Subtract(Rectangle rect) + { + Rectangle intersect = GetIntersecting(rect); + if (intersect == null || rect.Occludes(this)) return new Rectangle[0]; + Rectangle[] components = new Rectangle[(intersect.Left > Left ? 1 : 0) + (intersect.Right < Right ? 1 : 0) + (intersect.Top > Top ? 1 : 0) + (intersect.Bottom < Bottom ? 1 : 0)]; + int rectangles = 0; + + if(intersect.Left > Left) + components[rectangles++] = new Rectangle(Left, Math.Max(intersect.Top, Top), intersect.Left, Math.Min(intersect.Bottom, Bottom)); + if (intersect.Right < Right) + components[rectangles++] = new Rectangle(intersect.Right, Math.Max(intersect.Top, Top), Left, Math.Min(intersect.Bottom, Bottom)); + if (intersect.Top > Top) + components[rectangles++] = new Rectangle(Math.Min(Left, intersect.Left), Top, Math.Max(Right, intersect.Right), intersect.Top); + if (intersect.Bottom < Bottom) + components[rectangles] = new Rectangle(Math.Min(Left, intersect.Left), intersect.Bottom, Math.Max(Right, intersect.Right), Bottom); + + return components; + } + + public void Offset(Tuple xy) => Offset(xy.Item1, xy.Item2); + public void Offset(int x, int y) + { + Left += x; + Bottom += y; + Right += x; + Top += y; + } + } + + public static class SpaceMaths + { + public static Tuple CenterPad(int maxLength, int contentLength) + { + int pad = maxLength - contentLength; + return new Tuple(pad / 2, pad - (pad / 2)); + } + } + + public sealed class CancellationPipe + { + private bool cancel = false; + public bool Cancelled + { + get => cancel; + set => cancel |= value; + } + + // Redundant + public void Cancel() => Cancelled = true; + } + + public sealed class Timer + { + public delegate void Runnable(); + private readonly long millis; + private readonly Task timer; + + public bool Expired => timer.Status != TaskStatus.Running; + + public Timer(Runnable onExpire, long millis, int resolution = 100) + { + this.millis = CurrentTimeMillis() + millis; + timer = new Task(() => + { + while (CurrentTimeMillis() < this.millis) System.Threading.Thread.Sleep(resolution); + onExpire(); + }); + } + public void Start() => timer.Start(); + public TaskAwaiter GetAwaiter() => timer.GetAwaiter(); + + private static long CurrentTimeMillis() => DateTime.Now.Ticks / 10000; + } +} diff --git a/Client/NetContext.cs b/Client/NetContext.cs new file mode 100644 index 0000000..09ccc9c --- /dev/null +++ b/Client/NetContext.cs @@ -0,0 +1,90 @@ +using ConsoleForms; +using System; +using System.Collections.Generic; +using System.Linq; +using System.Text; +using System.Threading.Tasks; +using Tofvesson.Collections; + +namespace Client +{ + public class NetContext : Context + { + public NetContext(ContextManager manager) : base(manager, "Networking") + { + // Just close when anything is selected and "submitted" + RegisterSelectListeners((s, i, v) => controller.CloseView(s), "EmptyFieldError", "IPError", "PortError", "ConnectionError"); + + ((InputTextBox)views.GetNamed("NetConnect")).SubmissionsListener = i => + { + bool + ip = ParseIP(i.Inputs[0].Text) != null, + port = short.TryParse(i.Inputs[1].Text, out short prt) && prt > 0; + + + if (ip && port) + { + // Connect to server here + BankNetInteractor ita = new BankNetInteractor(i.Inputs[0].Text, prt, false); // Don't do identity check for now + try + { + var t = ita.Connect(); + while (!t.IsCompleted) + if (t.IsCanceled || t.IsFaulted) + { + controller.AddView(views.GetNamed("ConnectError")); + return; + } + } + catch + { + controller.AddView(views.GetNamed("ConnectionError")); + return; + } + manager.LoadContext(new WelcomeContext(manager, ita)); + } + else if (i.Inputs[0].Text.Length == 0 || i.Inputs[1].Text.Length == 0) controller.AddView(views.GetNamed("EmptyFieldError")); + else if (!ip) controller.AddView(views.GetNamed("IPError")); + else controller.AddView(views.GetNamed("PortError")); + }; + } + + public override void OnCreate() + { + controller.AddView(views.GetNamed("NetConnect")); + } + + public override void OnDestroy() + { + foreach (var view in views) + controller.CloseView(view.Item2); + } + + + //int gtrack = 0; + public override bool Update(ConsoleController.KeyEvent keypress, bool hasKeypress = true) + { + /* + var connectBox = (TextBox)views.GetNamed("NetConnect"); + if (++gtrack == 10) + { + connectBox.BorderColor = (ConsoleColor)((int)(connectBox.BorderColor + 1) % 16); + gtrack = 0; + } + + connectBox.Dirty = true; + */ + return base.Update(keypress, hasKeypress); + } + + private static byte[] ParseIP(string ip) + { + if (!ip.ContainsExactly('.', 3)) return null; + string[] vals = ip.Split('.'); + byte[] parts = new byte[4]; + for(int i = 0; i<4; ++i) + if (!byte.TryParse(vals[i], out parts[i])) return null; + return parts; + } + } +} diff --git a/Client/Networking.cs b/Client/Networking.cs new file mode 100644 index 0000000..1f303e2 --- /dev/null +++ b/Client/Networking.cs @@ -0,0 +1,182 @@ +using System; +using System.Collections.Generic; +using System.Linq; +using System.Text; +using System.Threading.Tasks; +using Tofvesson.Crypto; + +namespace Client +{ + public class BankNetInteractor + { + protected static readonly CryptoRandomProvider provider = new CryptoRandomProvider(); + protected static readonly Dictionary changeListeners = new Dictionary(); + + protected Dictionary promises = new Dictionary(); + protected NetClient client; + private bool authenticating = true, authenticated = false; + public bool Authenticating { get => authenticating; } + public bool PeerIsAuthenticated { get => authenticated; } + public RSA AuthenticatedKeys { get; private set; } + public bool IsAlive { get => client.IsAlive; } + + public BankNetInteractor(string address, short port, bool checkIdentity = true) + { + if(checkIdentity) + new Task(() => + { + AuthenticatedKeys = NetClient.CheckServerIdentity(address, port, provider); + authenticating = false; + authenticated = AuthenticatedKeys != null; + }).Start(); + else + { + authenticating = false; + authenticated = false; + } + var addr = System.Net.IPAddress.Parse(address); + client = new NetClient( + new Rijndael128( + Convert.ToBase64String(provider.GetBytes(64)), // 64-byte key (converted to base64) + Convert.ToBase64String(provider.GetBytes(64)) // 64-byte salt (converted to base64) + ), + addr, + port, + MessageRecievedHandler, + ClientConnectionHandler, + 65536); // 64 KiB buffer + } + + public virtual Task Connect() + { + client.Connect(); + Task t = new Task(() => + { + while (!client.IsAlive) System.Threading.Thread.Sleep(125); + }); + t.Start(); + return t; + } + public async virtual Task Disconnect() => await client.Disconnect(); + + public long RegisterListener(OnClientConnectStateChanged stateListener) + { + long tkn; + changeListeners[tkn = GetListenerToken()] = stateListener; + return tkn; + } + + public void UnregisterListener(long tkn) => changeListeners.Remove(tkn); + + protected virtual string MessageRecievedHandler(string msg, Dictionary associated, ref bool keepAlive) + { + string response = HandleResponse(msg, out long pID, out bool err); + if (err || !promises.ContainsKey(pID)) return null; + Promise p = promises[pID]; + promises.Remove(pID); + p.Value = response; + p.HasValue = true; + p.Subscribe?.Invoke(p); + return null; + } + + protected virtual void ClientConnectionHandler(NetClient client, bool connect) + { + foreach (var listener in changeListeners.Values) + listener(client, connect); + } + + public virtual Promise CheckAccountAvailability(string username) + { + if (username.Length > 60) + return new Promise + { + HasValue = true, + Value = "ERROR" + }; + client.Send(CreateCommandMessage("Avail", username, out long pID)); + Promise p = new Promise(); + promises[pID] = p; + return p; + } + + public virtual Promise Authenticate(string username, string password) + { + if (username.Length > 60) + return new Promise + { + HasValue = true, + Value = "ERROR" + }; + client.Send(CreateCommandMessage("Auth", username+":"+password, out long pID)); + Promise p = new Promise(); + promises[pID] = p; + return p; + } + + public virtual Promise Register(string username, string password) + { + if (username.Length > 60) + return new Promise + { + HasValue = true, + Value = "ERROR" + }; + client.Send(CreateCommandMessage("Reg", username + ":" + password, out long pID)); + Promise p = new Promise(); + promises[pID] = p; + return p; + } + + public virtual void Logout(string sessionID) + => client.Send(CreateCommandMessage("Logout", sessionID, out long _)); + + protected long GetNewPromiseUID() + { + long l; + do l = provider.NextLong(); + while (promises.ContainsKey(l)); + return l; + } + + protected long GetListenerToken() + { + long l; + do l = provider.NextLong(); + while (changeListeners.ContainsKey(l)); + return l; + } + + protected static void PostPromise(Promise p, string value) + { + p.Value = value; + p.HasValue = true; + p.Subscribe?.Invoke(p); + } + + protected static string HandleResponse(string response, out long promiseID, out bool error) + { + error = !long.TryParse(response.Substring(0, Math.Max(0, response.IndexOf(':'))), out promiseID); + return response.Substring(Math.Max(0, response.IndexOf(':') + 1)); + } + protected string CreateCommandMessage(string command, string message, out long promiseID) => command + ":" + (promiseID = GetNewPromiseUID()) + ":" + message; + } + + public delegate void Event(Promise p); + public class Promise + { + private Event evt; + public string Value { get; internal set; } + public bool HasValue { get; internal set; } + public Event Subscribe + { + get => evt; + set + { + evt = value; + if (HasValue) + evt(this); + } + } + } +} diff --git a/Client/Networking.xml b/Client/Networking.xml new file mode 100644 index 0000000..e94b812 --- /dev/null +++ b/Client/Networking.xml @@ -0,0 +1,85 @@ + + + + + + Server IP: + Port: + + Server connection configuration + + + + + + + + The identity of the server could not be verified. Continue? + + + + + + + Connecting to server... + + + + + + + + One or more required input field is empty + + + + + + + The supplied IP-address is not valid + + + + + + + The supplied port is not valid + + + + + + + Could not connect to server + + \ No newline at end of file diff --git a/Client/Program.cs b/Client/Program.cs new file mode 100644 index 0000000..988690d --- /dev/null +++ b/Client/Program.cs @@ -0,0 +1,62 @@ +using System; +using System.Collections.Generic; +using System.IO; +using System.Linq; +using System.Runtime.CompilerServices; +using System.Runtime.InteropServices; +using System.Text; +using System.Threading.Tasks; +using Client; +using Client.Properties; +using Common; +using Tofvesson.Collections; + +namespace ConsoleForms +{ + class Program + { + public static TextWriter DebugStream = new DebugAdapterWriter(); + private static ConsoleController controller = ConsoleController.singleton; + + static void Main(string[] args) + { + // Set up timestamps in debug output + DebugStream = new TimeStampWriter(DebugStream, "HH:mm:ss.fff"); + + Padding p = new AbsolutePadding(2, 2, 1, 1); + + Console.CursorVisible = false; + Console.Title = "Tofvesson Enterprises"; // Set console title + + // Start with the networking context + ContextManager manager = new ContextManager(); + + manager.LoadContext(new NetContext(manager)); + + // Start input listener loop. Graphics happen here too (triggered by keystrokes) + ConsoleController.KeyEvent info = new ConsoleController.KeyEvent(default(ConsoleKeyInfo)) + { + ValidEvent = false + }; + bool first = true; + do + { + if (first) first = false; + else info = controller.ReadKey(); + + bool b = manager.Update(info), haskey = false; + while (b) + { + System.Threading.Thread.Sleep(25); + haskey = _kbhit() != 0; + if (haskey) info = controller.ReadKey(false); + b = manager.Update(info, haskey); + controller.Draw(); + } + } while (!info.ValidEvent || info.Event.Key != ConsoleKey.Escape); + } + + [DllImport("msvcrt")] + public static extern int _kbhit(); + } +} diff --git a/Client/Properties/AssemblyInfo.cs b/Client/Properties/AssemblyInfo.cs new file mode 100644 index 0000000..9f15b87 --- /dev/null +++ b/Client/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("Client")] +[assembly: AssemblyDescription("")] +[assembly: AssemblyConfiguration("")] +[assembly: AssemblyCompany("")] +[assembly: AssemblyProduct("Client")] +[assembly: AssemblyCopyright("Copyright © 2018")] +[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("2236d5d4-7816-4630-8c86-0f0bdd46d7d8")] + +// 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/Client/Properties/Resources.Designer.cs b/Client/Properties/Resources.Designer.cs new file mode 100644 index 0000000..7f3fea7 --- /dev/null +++ b/Client/Properties/Resources.Designer.cs @@ -0,0 +1,149 @@ +//------------------------------------------------------------------------------ +// +// 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 Client.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", "15.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("Client.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 resource of type System.Byte[]. + /// + internal static byte[] e_0x200 { + get { + object obj = ResourceManager.GetObject("e_0x200", resourceCulture); + return ((byte[])(obj)); + } + } + + /// + /// Looks up a localized resource of type System.Byte[]. + /// + internal static byte[] n_0x200 { + get { + object obj = ResourceManager.GetObject("n_0x200", resourceCulture); + return ((byte[])(obj)); + } + } + + /// + /// Looks up a localized string similar to <?xml version="1.0" encoding="utf-8" ?> + ///<Elements xmlns="ConsoleForms"> + /// <!-- Networking context --> + /// <InputTextBox id="NetConnect" + /// padding_left="2" + /// padding_right="2" + /// padding_top="1" + /// padding_bottom="1"> + /// <Fields> + /// <Field input_type="decimal" max_length="15">Server IP:</Field> + /// <Field default="80" input_type="integer" max_length="5">Port:</Field> + /// </Fields> + /// <Text>Server connection configuration</Text> + /// </Input [rest of string was truncated]";. + /// + internal static string Networking { + get { + return ResourceManager.GetString("Networking", resourceCulture); + } + } + + /// + /// Looks up a localized string similar to <?xml version="1.0" encoding="utf-8" ?> + ///<Elements xmlns="ConsoleForms"> + /// <DialogBox id="Success" + /// padding_left="2" + /// padding_right="2" + /// padding_top="1" + /// padding_bottom="1"> + /// <Options> + /// <Option>Quit</Option> + /// </Options> + /// <Text>Login succeeded!</Text> + /// </DialogBox> + ///</Elements>. + /// + internal static string Session { + get { + return ResourceManager.GetString("Session", resourceCulture); + } + } + + /// + /// Looks up a localized string similar to <?xml version="1.0" encoding="utf-8" ?> + ///<Elements xmlns="ConsoleForms"> + /// + /// <!-- Welcome screen --> + /// <DialogBox id="WelcomeScreen" + /// padding_left="2" + /// padding_right="2" + /// padding_top="1" + /// padding_bottom="1" + /// width="42"> + /// <Options> + /// <Option event="Setup:Login" close="true">Login</Option> + /// <Option event="Setup:Register" close="true">Register</Option> + /// </Options> + /// <Text>Welcome to the Tofvesson banking system! To conti [rest of string was truncated]";. + /// + internal static string Setup { + get { + return ResourceManager.GetString("Setup", resourceCulture); + } + } + } +} diff --git a/Client/Properties/Resources.resx b/Client/Properties/Resources.resx new file mode 100644 index 0000000..d9b37cf --- /dev/null +++ b/Client/Properties/Resources.resx @@ -0,0 +1,136 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + 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\0x200.e;System.Byte[], mscorlib, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089 + + + ..\Networking.xml;System.String, mscorlib, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089;utf-8 + + + ..\Resources\0x200.n;System.Byte[], mscorlib, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089 + + + ..\Setup.xml;System.String, mscorlib, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089;utf-8 + + + ..\Session.xml;System.String, mscorlib, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089;utf-8 + + \ No newline at end of file diff --git a/Client/Resources/0x200.e b/Client/Resources/0x200.e new file mode 100644 index 0000000..c4ae053 Binary files /dev/null and b/Client/Resources/0x200.e differ diff --git a/Client/Resources/0x200.n b/Client/Resources/0x200.n new file mode 100644 index 0000000..1ae6fe8 Binary files /dev/null and b/Client/Resources/0x200.n differ diff --git a/Client/Session.xml b/Client/Session.xml new file mode 100644 index 0000000..f8e3802 --- /dev/null +++ b/Client/Session.xml @@ -0,0 +1,13 @@ + + + + + + + Login succeeded! + + \ No newline at end of file diff --git a/Client/SessionContext.cs b/Client/SessionContext.cs new file mode 100644 index 0000000..7ce7f3c --- /dev/null +++ b/Client/SessionContext.cs @@ -0,0 +1,39 @@ +using ConsoleForms; +using System; +using System.Collections.Generic; +using System.Linq; +using System.Text; +using System.Threading.Tasks; +using Tofvesson.Collections; + +namespace Client +{ + public sealed class SessionContext : Context + { + private readonly BankNetInteractor interactor; + private readonly string sessionID; + + public SessionContext(ContextManager manager, BankNetInteractor interactor, string sessionID) : base(manager, "Session") + { + this.interactor = interactor; + this.sessionID = sessionID; + + ((DialogBox)views.GetNamed("Success")).RegisterSelectListener((v, i, s) => + { + interactor.Logout(sessionID); + manager.LoadContext(new NetContext(manager)); + }); + } + + public override void OnCreate() + { + controller.AddView(views.GetNamed("Success")); + } + + public override void OnDestroy() + { + controller.CloseView(views.GetNamed("Success")); + interactor.Disconnect(); + } + } +} diff --git a/Client/Setup.xml b/Client/Setup.xml new file mode 100644 index 0000000..131b4ed --- /dev/null +++ b/Client/Setup.xml @@ -0,0 +1,128 @@ + + + + + + + + + + Welcome to the Tofvesson banking system! To continue, press [ENTER] To go back, press [ESCAPE] + + + + + + Username: + Password: + Repeat password: + + Register Account + + + + Registering... + + + + + + + An account with this username already exists! + + + + + + + The entered passwords don't match! + + + + + + + + The password you have supplied has been deemed to be weak. Are you sure you want to continue? + + + + + + Username: + Password: + + Log in + + + + Authenticating... + + + + + + + + One or more required input field is empty + + + + + + + The given username or password was incorrect + + \ No newline at end of file diff --git a/Client/WelcomeContext.cs b/Client/WelcomeContext.cs new file mode 100644 index 0000000..b4d0b58 --- /dev/null +++ b/Client/WelcomeContext.cs @@ -0,0 +1,159 @@ +using ConsoleForms; +using System; +using System.Collections.Generic; +using System.Linq; +using System.Text; +using System.Threading.Tasks; +using Tofvesson.Collections; + +namespace Client +{ + public sealed class WelcomeContext : Context + { + private readonly BankNetInteractor interactor; + private long token; + private Promise promise; + private bool forceDestroy = true; + + public WelcomeContext(ContextManager manager, BankNetInteractor connection) : base(manager, "Setup") + { + this.interactor = connection; + + // Prepare events and stuff + + // Just close when anything is selected and "submitted" + RegisterSelectListeners((s, i, v) => controller.CloseView(s), "DuplicateAccountError", "EmptyFieldError", "IPError", "PortError", "AuthError", "PasswordMismatchError"); + + + ((InputTextBox)views.GetNamed("Login")).SubmissionsListener = i => + { + bool success = true; + + foreach (var input in i.Inputs) + { + if (input.Text.Length == 0) + { + success = false; + input.SelectBackgroundColor = ConsoleColor.Red; + input.BackgroundColor = ConsoleColor.DarkRed; + } + } + + if (success) + { + // Authenticate against server here + controller.AddView(views.GetNamed("AuthWait")); + promise = interactor.Authenticate(i.Inputs[0].Text, i.Inputs[1].Text); + promise.Subscribe = + response => + { + controller.CloseView(views.GetNamed("AuthWait")); + if (response.Value.Equals("ERROR")) + controller.AddView(views.GetNamed("AuthError")); + else + { + forceDestroy = false; + manager.LoadContext(new SessionContext(manager, interactor, response.Value)); + } + }; + } + else controller.AddView(views.GetNamed("EmptyFieldError")); + }; + + // For a smooth effect + ((InputTextBox)views.GetNamed("Login")).InputListener = (v, c, i) => + { + c.BackgroundColor = v.DefaultBackgroundColor; + c.SelectBackgroundColor = v.DefaultSelectBackgroundColor; + return true; + }; + + ((InputTextBox)views.GetNamed("Register")).SubmissionsListener = i => + { + bool success = true, mismatch = false; + + foreach (var input in i.Inputs) + { + if (input.Text.Length == 0) + { + success = false; + input.SelectBackgroundColor = ConsoleColor.Red; + input.BackgroundColor = ConsoleColor.DarkRed; + } + } + + mismatch = !i.Inputs[1].Text.Equals(i.Inputs[2].Text); + if (success && !mismatch) + { + void a() + { + controller.AddView(views.GetNamed("RegWait")); + promise = interactor.Register(i.Inputs[0].Text, i.Inputs[1].Text); + promise.Subscribe = + response => + { + controller.CloseView(views.GetNamed("RegWait")); + if (response.Value.Equals("ERROR")) + controller.AddView(views.GetNamed("DuplicateAccountError")); + else + { + forceDestroy = false; + manager.LoadContext(new SessionContext(manager, interactor, response.Value)); + } + }; + } + + if (i.Inputs[1].Text.Length < 5 || i.Inputs[1].Text.StartsWith("asdfasdf") || i.Inputs[1].Text.StartsWith("asdf1234")) + { + var warning = (DialogBox)views.GetNamed("WeakPasswordWarning"); + warning.RegisterSelectListener((wrn, idx, sel) => + { + controller.CloseView(warning); + if (idx == 0) a(); + }); + controller.AddView(warning); + } + else a(); + } + else if (mismatch) controller.AddView(views.GetNamed("PasswordMismatchError")); + else controller.AddView(views.GetNamed("EmptyFieldError")); + }; + + ((InputTextBox)views.GetNamed("Register")).InputListener = (v, c, i) => + { + c.BackgroundColor = v.DefaultBackgroundColor; + c.SelectBackgroundColor = v.DefaultSelectBackgroundColor; + return true; + }; + } + + public override void OnCreate() + { + token = interactor.RegisterListener((c, s) => + { + if(!s) controller.Popup("The connection to the server was severed! ", 4500, ConsoleColor.DarkRed, () => manager.LoadContext(new NetContext(manager))); + }); + + // Add the initial view + controller.AddView(views.GetNamed("WelcomeScreen")); + } + + public override void OnDestroy() + { + // TODO: Save state + + + // Close views + foreach (var view in views) + controller.CloseView(view.Item2); + + // Unsubscribe from events + if (promise != null && !promise.HasValue) promise.Subscribe = null; + + // Stop listening + interactor.UnregisterListener(token); + + if (forceDestroy) interactor.Disconnect(); + } + } +} diff --git a/Common/AES.cs b/Common/AES.cs new file mode 100644 index 0000000..e6c0934 --- /dev/null +++ b/Common/AES.cs @@ -0,0 +1,668 @@ +using System; +using System.Collections.Generic; +using System.IO; +using System.Text; + +namespace Tofvesson.Crypto +{ + public class Rijndael128 : BlockCipher + { + protected readonly byte[] roundKeys; + protected readonly byte[] key; + + public Rijndael128(string key, string salt = "PlsNoRainbowz") : base(16) + { + // Derive a proper key + var t = DeriveKey(key, salt); + this.key = t.Item1; + + // Expand the derived key + roundKeys = KeySchedule(this.key, BitMode.Bit128); + } + protected Rijndael128(byte[] key) : base(16) + { + this.key = key; + + // Expand the derived key + roundKeys = KeySchedule(this.key, BitMode.Bit128); + } + + + // Encrypt/Decrypt a string by just converting it to bytes and passing it along to the byte-based encryption/decryption methods + public byte[] EncryptString(string message) => Encrypt(Encoding.UTF8.GetBytes(message)); + public string DecryptString(byte[] message, int length) => new string(Encoding.UTF8.GetChars(Decrypt(message, length, false))).Substring(0, length); + + // Encrypt a message (this one just splits the message into blocks and passes it along) + public override byte[] Encrypt(byte[] message) + { + byte[] result = new byte[message.Length + ((16 - (message.Length % 16))%16)]; + Array.Copy(message, result, message.Length); + for(int i = 0; i Decrypt(ciphertext, -1, false); + public byte[] Decrypt(byte[] message, int messageLength) => Decrypt(message, messageLength, true); + protected byte[] Decrypt(byte[] message, int messageLength, bool doTruncate) + { + if (message.Length % 16 != 0) throw new SystemException("Invalid encrypted message length!"); + byte[] result = new byte[message.Length]; + Array.Copy(message, result, message.Length); + for (int i = 0; i < result.Length / 16; ++i) + Array.Copy(AES128_Decrypt(result.SubArray(i * 16, i * 16 + 16)), 0, result, i * 16, 16); + return doTruncate ? result.SubArray(0, messageLength) : result; + } + + // The actual AES encryption implementation + protected virtual byte[] AES128_Encrypt(byte[] input) + { + // The "state" is the name given the the 4x4 matrix that AES encrypts. The state is known as the "state" no matter what stage of AES it has gone through or how many left it has + byte[] state = new byte[16]; + Array.Copy(input, state, 16); + + // Initial round. Just just xor the key for this round input the input + state = AddRoundKey(state, roundKeys, 0); + + // Rounds 1 - 9 + for (int rounds = 1; rounds < 10; ++rounds) + { + state = ShiftRows(SubBytes(state, false)); // Shift the rows of the column-major matrix + if (rounds != 9) state = MixColumns(state, true); // Mix the columns (gonna be honest, I don't remember what this does, but it has something to do with galois fields, so just check Galois2 out) + state = AddRoundKey(state, roundKeys, rounds * 16); // Xor the key into the mess + } + + // Now this matrix is encrypted! + return state; + } + + // Literally just the inverse functions of the Encrypt-process run in reverse. + protected virtual byte[] AES128_Decrypt(byte[] input) + { + byte[] state = new byte[16]; + Array.Copy(input, state, 16); + + for (int rounds = 9; rounds > 0; --rounds) + { + state = AddRoundKey(state, roundKeys, rounds * 16); + if (rounds != 9) state = MixColumns(state, false); + state = SubBytes(UnShiftRows(state), true); + } + + return AddRoundKey(state, roundKeys, 0); + } + + // Save the key to a file + public void Save(string baseName, bool force = false) + { + if (force || !File.Exists(baseName + ".key")) File.WriteAllBytes(baseName + ".key", key); + } + + // Load the key from a file (gonna be honest, I think I just copy-pasted this from the RSA file and renamed some stuff) + public static Rijndael128 Load(string baseName) + { + if (!File.Exists(baseName + ".key")) throw new SystemException("Required files could not be located"); + return new Rijndael128(File.ReadAllBytes(baseName + ".key")); + } + + // De/-serializes the key (the method is just here for compatibility) + public byte[] Serialize() => Support.SerializeBytes(new byte[][] { key }); + public static Rijndael128 Deserialize(byte[] message, out int read) + { + byte[][] output = Support.DeserializeBytes(message, 1); + read = output[0].Length + 8; + return new Rijndael128(output[0]); + } + + + // Internal methods for encryption :) + private static uint KSchedCore(uint input, int iteration) + { + input = Rotate(input); + byte[] bytes = Support.WriteToArray(new byte[4], input, 0); + for (int i = 0; i < bytes.Length; ++i) bytes[i] = SBox(bytes[i]); + bytes[bytes.Length - 1] ^= RCON(iteration); + return (uint)Support.ReadInt(bytes, 0); + } + + // Rijndael key schedule: implemented for the three common implementations because I'm thorough or something + public enum BitMode { Bit128, Bit192, Bit256 } + private static byte[] KeySchedule(byte[] key, BitMode mode) + { + int n = mode == BitMode.Bit128 ? 16 : mode == BitMode.Bit192 ? 24 : 32; + int b = mode == BitMode.Bit128 ? 176 : mode == BitMode.Bit192 ? 208 : 240; + + byte[] output = new byte[b]; + Array.Copy(key, output, n); + + int rcon_iter = 1; + + int accruedBytes = n; + while (accruedBytes < b) + { + // Generate 4 new bytes of extended key + byte[] t = Support.WriteToArray(new byte[4], KSchedCore((uint)Support.ReadInt(output, accruedBytes - 4), rcon_iter), 0); + ++rcon_iter; + for (int i = 0; i < 4; ++i) t[i] ^= output[accruedBytes - n + i]; + Array.Copy(t, 0, output, accruedBytes, 4); + accruedBytes += 4; + + // Generate 12 new bytes of extended key + for (int i = 0; i < 3; ++i) + { + Array.Copy(output, accruedBytes - 4, t, 0, 4); + for (int j = 0; j < 4; ++j) t[j] ^= output[accruedBytes - n + j]; + Array.Copy(t, 0, output, accruedBytes, 4); + accruedBytes += 4; + } + + // Special processing for 256-bit key schedule + if (mode == BitMode.Bit256) + { + Array.Copy(output, accruedBytes - 4, t, 0, 4); + for (int j = 0; j < 4; ++j) t[j] = (byte)(SBox(t[j]) ^ output[accruedBytes - n + j]); + Array.Copy(t, 0, output, accruedBytes, 4); + accruedBytes += 4; + } + // Special processing for 192-bit key schedule + if (mode != BitMode.Bit128) + for (int i = mode == BitMode.Bit192 ? 1 : 2; i >= 0; --i) + { + Array.Copy(output, accruedBytes - 4, t, 0, 4); + for (int j = 0; j < 4; ++j) t[j] ^= output[accruedBytes - n + j]; + Array.Copy(t, 0, output, accruedBytes, 4); + accruedBytes += 4; + } + } + + return output; + } + + // MixColumns matrix basis. Used for multiplication over the rijndael field + private static readonly byte[] mix_matrix = new byte[] { 2, 3, 1, 1 }; + private static readonly byte[] unmix_matrix = new byte[] { 14, 11, 13, 9 }; + + /// + /// Rijndael substitution step in the encryption (first thing that happens). This supplies confusion for the algorithm + /// + /// The value (most likely from the AES state) that should be substituted + /// The substituted byte + private static byte SBox(byte b) => Affine(new Galois2(new byte[] { b }).InvMul().ToByteArray()[0]); + + // Inverse SBox-function + private static byte ISBox(byte b) => new Galois2(new byte[] { Rffine(b) }).InvMul().ToByteArray()[0]; + + // Replaces GF(2^8) matrix multiplication for the affine and reverse affine functions + private static byte Affine(byte value) => (byte)(value ^ Rot(value, 1) ^ Rot(value, 2) ^ Rot(value, 3) ^ Rot(value, 4) ^ 0b0110_0011); + private static byte Rffine(byte value) => (byte)(Rot(value, 1) ^ Rot(value, 3) ^ Rot(value, 6) ^ 0b0000_0101); + + // Rotate bits + private static byte Rot(byte value, int by) => (byte)((value << by) | (value >> (8 - by))); + + private delegate byte SBOXFunc(byte b); + private static byte[] SubBytes(byte[] state, bool reverse) + { + SBOXFunc v; + if (reverse) v = ISBox; + else v = SBox; + for (int i = 0; i < state.Length; ++i) state[i] = v(state[i]); + return state; + } + + // The AES state is a column-major 4x4 matrix (for AES-128). Demonstrated below are the decimal indices, as would be represented in the state: + // 00 04 08 12 + // 01 05 09 13 + // 02 06 10 14 + // 03 07 11 15 + + // Shiftrows applied to state above: + // 00 04 08 12 - No change + // 05 09 13 01 - Shifted 1 to the left + // 10 14 02 06 - Shifted 2 to the left + // 15 03 07 11 - Shifted 3 to the left + + /// + /// Shifts the rows of the column-major matrix + /// + /// + /// The shifted matrix + public static byte[] ShiftRows(byte[] state) + { + for (int i = 1; i < 4; ++i) + { + uint value = GetRow(state, i); + for (int j = 0; j < i; ++j) value = Rotate(value); + WriteToRow(value, state, i); + } + return state; + } + + // Reverse ShiftRows + private static byte[] UnShiftRows(byte[] state) + { + for (int i = 1; i < 4; ++i) + { + uint value = GetRow(state, i); + for (int j = 3; j >= i; --j) value = Rotate(value); + WriteToRow(value, state, i); + } + return state; + } + + // Helper method, really + private static void WriteToRow(uint value, byte[] to, int row) + { + to[row] = (byte)(value & 255); + to[row + 4] = (byte)((value >> 8) & 255); + to[row + 8] = (byte)((value >> 16) & 255); + to[row + 12] = (byte)((value >> 24) & 255); + } + + // Boring helper method + private static uint GetRow(byte[] from, int row) => (uint)(from[row] | (from[row + 4] << 8) | (from[row + 8] << 16) | (from[row + 12] << 24)); + + /// + /// MixColumns adds diffusion to the algorithm. Performs matrix multiplication under GF(2^8) with the irreducible prime 0x11B (x^8 + x^4 + x^3 + x + 1) + /// + /// + /// A matrix-multiplied and limited state (mixed) + private static byte[] MixColumns(byte[] state, bool mix) + { + byte[] res = new byte[16]; + byte[] rowGenerator = mix ? mix_matrix : unmix_matrix; + + // Simplified matrix multiplication under GF(2^8) + for (int i = 0; i < 4; ++i) + { + for (int j = 0; j < 4; ++j) + { + for (int k = 0; k < 4; ++k) + { + int idx = 4 - j; + Galois2 g = Galois2.FromValue(state[k + i * 4]); + res[j + i * 4] ^= g.Multiply(Galois2.FromValue(rowGenerator[(k + idx) % 4])).ToByteArray()[0]; + //int r = ((state[k + i * 4] * (mix_matrix[(k + idx) % 4] & 1)) ^ ((state[k + i * 4] << 1) * ((mix_matrix[(k + idx) % 4]>>1)&1))); + //if (r > 0b100011011) r ^= 0b100011011; + //res[j + i * 4] ^= (byte) r; + } + } + } + return res; + } + + /// + /// Introduces the subkey for this round to the state + /// + /// The state to introduce the roundkey to + /// The subkey + /// The state where the roundkey has been added + private static byte[] AddRoundKey(byte[] state, byte[] subkey, int offset) + { + for (int i = 0; i < state.Length; ++i) state[i] ^= subkey[i + offset]; + return state; + } + + /// + /// Rotate bits to the left by 8 bits. This means that, for example, "0F AB 09 16" becomes "AB 09 16 0F" + /// + /// + /// Rotated value + private static uint Rotate(uint i) => (uint)(((i >> 24) & 255) | ((i << 8) & ~255)); + + /// + /// KDF for a given input string. + /// + /// Input string to derive key from + /// A key and an IV + private static Tuple DeriveKey(string message, string salt) + { + byte[] key = KDF.PBKDF2(KDF.HMAC_SHA1, Encoding.UTF8.GetBytes(message), salt.ToUTF8Bytes(), 4096, 16); // Generate a 16-byte (128-bit) key from salt over 4096 iterations of HMAC-SHA1 + return new Tuple(key, salt.ToUTF8Bytes()); + } + + private static byte RCON(int i) => i <= 0 ? (byte)0x8d : new Galois2(i - 1).ToByteArray()[0]; + } + + + + // If you genuinely care what this does, to which I would under regular circumstances call you a nerd but alas I researched this, soooooo here: + // https://en.wikipedia.org/wiki/Finite_field_arithmetic + // http://www.cs.utsa.edu/~wagner/laws/FFM.html + // https://www.wolframalpha.com/examples/math/algebra/finite-fields/ + // https://www.youtube.com/watch?v=x1v2tX4_dkQ + // That YouTube link explains it the best imo, but the others really help and solidify the concept. + // Essentially, if you're genuinely going to torture yourself by trying to learn this stuff, at least do yourself a favour and start with the video. + // Also, I'm not gonna comment the code down there because it's really annoying and it's late and I have a headache and ooooooooohhhhh I don't want to spend more time than I already have on that stuff. + // You get the comments that I put there when I made this; no more than that! Honestly, they're still plenty so just make do! + /// + /// Object representation of a Galois Field with characteristic 2 + /// + public class Galois2 + { + private static readonly byte[] ZERO = new byte[1] { 0 }; + private static readonly byte[] ONE = new byte[1] { 1 }; + + public static byte[] RijndaelIP + { get { return new byte[] { 0b0001_1011, 0b0000_0001 }; } } + + protected readonly byte[] value; + protected readonly byte[] ip; + + /// + /// Create a new Galois2 instance representing the given polynomial using the given irreducible polynomial. The given value will be reduced if possible + /// + /// Value to represent + /// Irreducible polynomial + public Galois2(byte[] value, byte[] ip) + { + this.value = _ClipZeroes(_FieldMod(value, this.ip = ip)); + } + + public Galois2(int pow, byte[] ip) : this(_FlipBit(new byte[0], pow), ip) + { } + + public Galois2(byte[] value) : this(value, RijndaelIP) + { } + + public Galois2(int pow) : this(pow, RijndaelIP) + { } + + public static Galois2 FromValue(int value, byte[] ip) => new Galois2(Support.WriteToArray(new byte[4], value, 0), ip); + public static Galois2 FromValue(int value) => FromValue(value, Galois2.RijndaelIP); + + public Galois2 Multiply(Galois2 factor) => new Galois2(_Mul(value, factor.value), ip); + public Galois2 Add(Galois2 val) => new Galois2(_Add(value, val.value), ip); + public Galois2 Subtract(Galois2 val) => new Galois2(_Sub(value, val.value), ip); + public Galois2 XOR(Galois2 val) => new Galois2(_XOR(value, val.value), ip); + + /// + /// Perform inverse multiplication on this Galois2 object. This is done by performing the extended euclidean algorithm (two-variable linear diophantine equations). + /// + /// + /// + public Galois2 InvMul() + { + if (_ArraysEquals(value, ZERO)) return FromValue(0, ip); + Stack factors = new Stack(); + byte[] val = value; + byte[] mod = ip; + ModResult res; + while (!_ArraysEquals((res = _Mod(val, mod)).rem, ZERO)) + { + factors.Push(res.div); + val = mod; + mod = res.rem; + } + + // Values are not coprime. There is no solution! + if (!_ArraysEquals(mod, ONE)) return new Galois2(new byte[0], ip); + + byte[] useful = new byte[1] { 1 }; + byte[] theOtherOne = factors.Pop(); + byte[] tmp; + while (factors.Count > 0) + { + tmp = theOtherOne; + theOtherOne = _Add(useful, _Mul(theOtherOne, factors.Pop())); + useful = tmp; + } + return new Galois2(useful, ip); + } + + public byte[] ToByteArray() => (byte[])value.Clone(); + public override string ToString() + { + StringBuilder builder = new StringBuilder(); + for (int i = _GetFirstSetBit(value); i >= 0; --i) + if (_BitAt(value, i)) + builder.Append("x^").Append(i).Append(" + "); + if (builder.Length == 0) builder.Append("0 "); + else builder.Remove(builder.Length - 2, 2); + builder.Append("(mod "); + int j; + for(int i = j = _GetFirstSetBit(ip); i>=0; --i) + if (_BitAt(ip, i)) + builder.Append("x^").Append(i).Append(" + "); + if (j == -1) builder.Append('0'); + else builder.Remove(builder.Length - 3, 3); + + return builder.Append(')').ToString(); + } + + // Overrides + public override bool Equals(object obj) + { + if (obj == null || !(obj is Galois2 || obj is byte[])) return false; + + byte[] val = obj is Galois2 ? ((Galois2)obj).value : (byte[])obj; + + bool cmp = val.Length > value.Length; + byte[] bigger = cmp ? val : value; + byte[] smaller = cmp ? value : val; + for (int i = bigger.Length - 1; i >= 0; --i) + if (i >= smaller.Length) + { + if (bigger[i] != 0) return false; + } + else if (bigger[i] != smaller[i]) return false; + + // If the value supplied was a byte array, ignore the irreducible prime, otherwise, make sure the irreducible primes are the same + return obj is byte[] || ((Galois2)obj).ip.Equals(ip); + } + + public override int GetHashCode() + { + var hashCode = -579181322; + hashCode = hashCode * -1521134295 + EqualityComparer.Default.GetHashCode(value); + hashCode = hashCode * -1521134295 + EqualityComparer.Default.GetHashCode(ip); + return hashCode; + } + + + // Just internal stuff from here on out boiiiiiiis! + + + + protected static bool _ArraysEquals(byte[] v1, byte[] v2) + { + bool cmp = v1.Length > v2.Length; + byte[] bigger = cmp ? v1 : v2; + byte[] smaller = cmp ? v2 : v1; + for (int i = bigger.Length - 1; i >= 0; --i) + if (i >= smaller.Length) + { + if (bigger[i] != 0) return false; + } + else if (bigger[i] != smaller[i]) return false; + return true; + } + + // Internal methods for certain calculations + protected static byte[] _FieldMod(byte[] applyTo, byte[] fieldIP) + { + byte[] CA_l; + int fsb = _GetFirstSetBit(fieldIP); + while (_GetFirstSetBit(applyTo) >= fsb) // In GF(2^8), polynomials may not exceed x^7. This means that a value containing a bit representing x^8 or higher is invalid + { + CA_l = _GetFirstSetBit(applyTo) >= _GetFirstSetBit(fieldIP) ? _Align((byte[])fieldIP.Clone(), applyTo) : fieldIP; + byte[] res = new byte[CA_l.Length]; + for (int i = 0; i < CA_l.Length; ++i) res[i] = (byte)(applyTo[i] ^ CA_l[i]); + applyTo = _ClipZeroes(res); + } + return applyTo; + } + + + /// + /// Remove preceding zero-bytes + /// + /// Value to remove preceding zeroes from + /// Truncated value (if truncation was necessary) + protected static byte[] _ClipZeroes(byte[] val) + { + int i = 0; + for (int j = val.Length - 1; j >= 0; --j) if (val[j] != 0) { i = j; break; } + byte[] res = new byte[i + 1]; + Array.Copy(val, res, res.Length); + return res; + } + + + + /// + /// Get the bit index of the highest bit. This will get the value of the exponent, i.e. index 8 represents x^8 + /// + /// Value to get the highest set bit from + /// Index of the highest set bit. -1 if no bits are set + protected static int _GetFirstSetBit(byte[] b) + { + for (int i = (b.Length * 8) - 1; i >= 0; --i) + if (b[i / 8] == 0) i -= i % 8; // Speeds up searches through blank bytes + else if ((b[i / 8] & (1 << (i % 8))) != 0) + return i; + return -1; + } + + /// + /// Get the state of a bit in the supplied value. + /// + /// Value to get bit from + /// Bit index to get bit from. (Not byte index) + /// + protected static bool _BitAt(byte[] value, int index) => (value[index / 8] & (1 << (index % 8))) != 0; + + protected static byte _ShiftedBitmask(int start) + { + byte res = 0; + for (int i = start; i > 0; --i) res = (byte)((res >> 1) | 128); + return res; + } + + + protected static byte[] _Align(byte[] value, byte[] to) => _SHL(value, _GetFirstSetBit(to) - _GetFirstSetBit(value)); + protected static bool _NeedsAlignment(byte[] value, byte[] comp) => _GetFirstSetBit(value) > _GetFirstSetBit(comp); + protected static bool _GT(byte[] v1, byte[] v2, bool eq) + { + byte[] bigger = v1.Length > v2.Length ? v1 : v2; + byte[] smaller = v1.Length > v2.Length ? v2 : v1; + for (int i = bigger.Length - 1; i >= 0; --i) + if (i >= smaller.Length && bigger[i] != 0) + return bigger == v1; + else if (i < smaller.Length && bigger[i] != smaller[i]) + return (bigger[i] > smaller[i]) ^ (bigger != v1); + return eq; + } + + + + /// + /// Shifts bit in the array by 'shift' bits to the left. This means that 0b0010_0000_1000_1111 shited by 2 becomes 0b1000_0010_0011_1100. + /// Note: A shift of 0 just acts like a slow value.Clone() + /// + /// + /// + /// + protected static byte[] _SHL(byte[] value, int shift) + { + int set = shift / 8; + int sub = shift % 8; + byte bm = _ShiftedBitmask(sub); + byte ibm = (byte)~bm; + byte carry = 0; + int fsb1 = _GetFirstSetBit(value); + if (fsb1 == -1) return value; + byte fsb = (byte)(fsb1 % 8); + byte[] create = new byte[value.Length + set + (fsb + sub >= 7 ? 1 : 0)]; + for (int i = set; i - set < value.Length; ++i) + { + create[i] = (byte)(((value[i - set] & ibm) << sub) | carry); + carry = (byte)((value[i - set] & bm) >> (8 - sub)); + } + create[create.Length - 1] |= carry; + return create; + } + + + /// + /// Flips the bit at the given binary index in the supplied value. For example, flipping bit 5 in the number 0b0010_0011 would result in 0b0000_0011, whereas flipping index 7 would result in 0b1010_0011. + /// + /// Value to manipulate bits of + /// Index (in bits) of the bit to flip. + /// An array (may be the same object as the one given) with a bit flipped. + protected static byte[] _FlipBit(byte[] value, int bitIndex) + { + if (bitIndex >= value.Length * 8) + { + byte[] intermediate = new byte[(bitIndex / 8) + 1]; + Array.Copy(value, intermediate, value.Length); + value = intermediate; + } + value[bitIndex / 8] ^= (byte)(1 << (bitIndex % 8)); + return value; + } + + + + + // Addition, Subtraction and XOR are all equivalent under GF(2^8) due to the modular nature of the field + protected static byte[] _Add(byte[] v1, byte[] v2) => _XOR(v1, v2); + protected static byte[] _Sub(byte[] v1, byte[] v2) => _XOR(v1, v2); + protected static byte[] _XOR(byte[] v1, byte[] v2) + { + bool size = v1.Length > v2.Length; + byte[] bigger = size ? v1 : v2; + byte[] smaller = size ? v2 : v1; + byte[] res = new byte[bigger.Length]; + Array.Copy(bigger, res, bigger.Length); + for (int i = 0; i < smaller.Length; ++i) res[i] ^= smaller[i]; + return _ClipZeroes(res); + } + + /// + /// Perform polynomial multiplication under a galois field with characteristic 2 + /// + /// Factor to multiply + /// Factor to multiply other value by + /// The product of the multiplication + protected static byte[] _Mul(byte[] value, byte[] by) + { + byte[] result = new byte[0]; + for (int i = _GetFirstSetBit(by); i >= 0; --i) + if (_BitAt(by, i)) + result = _Add(result, _SHL(value, i)); + return result; + } + + /// + /// Performs modulus on a given value by a certain value (mod) over a Galois Field with characteristic 2. This method performs both modulus and division. + /// + /// Value to perform modular aithmetic on + /// Modular value + /// The result of the polynomial division and the result of the modulus + protected static ModResult _Mod(byte[] value, byte[] mod) + { + byte[] divRes = new byte[1]; + while (_GT(value, mod, true)) + { + divRes = _FlipBit(divRes, _GetFirstSetBit(value) - _GetFirstSetBit(mod)); // Notes the bit shift in the division tracker + value = _Sub(value, _Align(mod, value)); + } + return new ModResult(divRes, value); + } + + /// + /// Used to store the result of a polynomial division/modulus in GF(2^m) + /// + protected struct ModResult + { + public ModResult(byte[] div, byte[] rem) + { + this.div = div; + this.rem = rem; + } + public byte[] div; + public byte[] rem; + } + } +} diff --git a/Common/AccountInfo.cs b/Common/AccountInfo.cs new file mode 100644 index 0000000..bd2bd17 --- /dev/null +++ b/Common/AccountInfo.cs @@ -0,0 +1,116 @@ +using System; +using System.Collections.Generic; +using System.Linq; +using System.Text; +using System.Threading.Tasks; +using System.Xml; +using Tofvesson.Crypto; + +namespace Common +{ + public class User + { + public bool ProblematicTransactions { get; private set; } + public string Name { get; private set; } + public long Balance { get; set; } + public bool IsAdministrator { get; set; } + public string PasswordHash { get; private set; } + public string Salt { get; private set; } + public List History { get; } + private User() + { + Name = ""; + History = new List(); + } + + public User(string name, string passHash, string salt, long balance, bool generatePass = false, List transactionHistory = null, bool admin = false) + : this(name, passHash, Encoding.UTF8.GetBytes(salt), balance, generatePass, transactionHistory, admin) + { } + + public User(string name, string passHash, byte[] salt, long balance, bool generatePass = false, List transactionHistory = null, bool admin = false) + { + History = transactionHistory ?? new List(); + Balance = balance; + Name = name; + IsAdministrator = admin; + Salt = Convert.ToBase64String(salt); + PasswordHash = generatePass ? Convert.ToBase64String(KDF.PBKDF2(KDF.HMAC_SHA1, Encoding.UTF8.GetBytes(passHash), Encoding.UTF8.GetBytes(Salt), 8192, 320)) : passHash; + } + + public bool Authenticate(string password) + => Convert.ToBase64String(KDF.PBKDF2(KDF.HMAC_SHA1, Encoding.UTF8.GetBytes(password), Encoding.UTF8.GetBytes(Salt), 8192, 320)).Equals(PasswordHash); + + public User AddTransaction(Transaction tx) + { + History.Add(tx); + return this; + } + + public Transaction CreateTransaction(User recipient, long amount, string message = null) => new Transaction(this.Name, recipient.Name, amount, message); + + public override bool Equals(object obj) => obj is User && ((User)obj).Name.Equals(Name); + + public override int GetHashCode() + => 539060726 + EqualityComparer.Default.GetHashCode(Name); + + /* + public string Serialize() + { + + } + + public static User Deserialize(string ser) + { + + } + */ + } + + public class Transaction + { + public string from; + public string to; + public long amount; + public string meta; + + public Transaction(string from, string to, long amount, string meta) + { + this.from = from; + this.to = to; + this.amount = amount; + this.meta = meta; + } + + public string Serialize() + { + XmlDocument doc = new XmlDocument(); + XmlElement el = doc.CreateElement("Transaction"); + + XmlElement to = doc.CreateElement("To"); + to.InnerText = this.to; + + XmlElement from = doc.CreateElement("From"); + from.InnerText = this.from; + + XmlElement amount = doc.CreateElement("Balance"); + amount.InnerText = amount.ToString(); + + el.AppendChild(to).AppendChild(from).AppendChild(amount); + if (meta != null) + { + XmlElement msg = doc.CreateElement("Meta"); + msg.InnerText = meta; + el.AppendChild(msg); + } + + return el.ToString(); + } + + public static Transaction Deserialize(string ser) + { + XmlDocument doc = new XmlDocument(); + doc.LoadXml(ser); + return null; + } + } +} diff --git a/Common/App.config b/Common/App.config new file mode 100644 index 0000000..731f6de --- /dev/null +++ b/Common/App.config @@ -0,0 +1,6 @@ + + + + + + \ No newline at end of file diff --git a/Common/CBC.cs b/Common/CBC.cs new file mode 100644 index 0000000..f35e0f4 --- /dev/null +++ b/Common/CBC.cs @@ -0,0 +1,227 @@ +using System; +using System.Linq; +using System.Reflection; + +namespace Tofvesson.Crypto +{ + // Specifies the basic structure of a block cipher + public abstract class BlockCipher + { + public Int32 BlockSize { get; private set; } + + public BlockCipher(int blockSize) { this.BlockSize = blockSize; } + + public abstract byte[] Encrypt(byte[] message); + public abstract byte[] Decrypt(byte[] ciphertext); + } + + // Base structure of a cipher block chaining algorithm + // For an in-depth explanation of what this is (as well as visuals of the implementations that are a bit futher down), visit https://en.wikipedia.org/wiki/Block_cipher_mode_of_operation + public abstract class GenericCBC : BlockCipher + { + private static readonly byte[] splitter = ":".ToUTF8Bytes(); + + + private readonly byte[] iv_e; + public byte[] IV { get => (byte[])iv_e.Clone(); } + + protected readonly byte[] currentIV_e; + protected readonly byte[] currentIV_d; + protected readonly BlockCipher cipher; + + public GenericCBC(BlockCipher cipher, RandomProvider provider) : base(cipher.BlockSize) + { + this.cipher = cipher; + + // Generate initialization vector and set it as the current iv + iv_e = provider.GetBytes(new byte[cipher.BlockSize]); + currentIV_e = new byte[cipher.BlockSize]; + currentIV_d = new byte[cipher.BlockSize]; + Array.Copy(iv_e, currentIV_e, iv_e.Length); + Array.Copy(iv_e, currentIV_d, iv_e.Length); + } + + public GenericCBC(BlockCipher cipher, byte[] iv_e) : base(cipher.BlockSize) + { + this.iv_e = iv_e; + this.cipher = cipher; + + currentIV_e = new byte[cipher.BlockSize]; + currentIV_d = new byte[cipher.BlockSize]; + Array.Copy(iv_e, currentIV_e, iv_e.Length); + Array.Copy(iv_e, currentIV_d, iv_e.Length); + } + + // Separate a given messae into blocks for processing + protected byte[][] SplitBlocks(byte[] message) + { + byte[][] blocks = new byte[(message.Length / cipher.BlockSize) + (message.Length % cipher.BlockSize == 0 ? 0 : 1)][]; + for (int i = 0; i < blocks.Length; ++i) + { + blocks[i] = message.SubArray(i * cipher.BlockSize, Math.Min((i + 1) * cipher.BlockSize, message.Length)); + if (blocks[i].Length != cipher.BlockSize) + { + byte[] res = new byte[cipher.BlockSize]; + Array.Copy(blocks[i], res, blocks[i].Length); + blocks[i] = res; + } + } + return blocks; + } + + // Recombine blocks that have been split back into a single string of bytes + protected byte[] CollectBlocks(byte[][] result) + { + byte[] collected = new byte[result.Length * cipher.BlockSize]; + for (int i = 0; i < result.Length; ++i) Array.Copy(result[i], 0, collected, cipher.BlockSize * i, cipher.BlockSize); + return collected; + } + + // Resets the state of this CBC instance + public virtual void Reset() + { + Array.Copy(iv_e, currentIV_e, iv_e.Length); + Array.Copy(iv_e, currentIV_d, iv_e.Length); + } + } + + /// + /// Standard cipher block chaining implementation (not recommended, but available nonetheless) + /// + public sealed class CBC : GenericCBC + { + public CBC(BlockCipher cipher, RandomProvider provider) : base(cipher, provider) + { } + + public CBC(BlockCipher cipher, byte[] iv_e) : base(cipher, iv_e) + { } + + + // This entire method is pretty self-explanatory. All you need to know is: currentIV_e represents the IV currently being used for encryption + public override byte[] Encrypt(byte[] message) + { + byte[][] blocks = SplitBlocks(message); + + for (int i = 0; i < blocks.Length; ++i) + { + byte[] enc_result = cipher.Encrypt(blocks[i].XOR(currentIV_e)); + Array.Copy(enc_result, currentIV_e, enc_result.Length); + Array.Copy(enc_result, blocks[i], cipher.BlockSize); + } + + return CollectBlocks(blocks); + } + + public override byte[] Decrypt(byte[] ciphertext) + { + // Split ciphertext into encrypted blocks + byte[][] blocks = SplitBlocks(ciphertext); + + for(int i = 0; i : IEnumerable + { + protected const float GROW_FACTOR = 1.75f; + protected const float SHRINK_FACTOR = 1.25f; + + protected readonly int maxCapacity; + protected T[] values; + public int Count { get; private set; } + + public T this[int i] + { + get => ElementAt(i); + set + { + DoRangeCheck(i); + values[i] = value; + } + } + + public BoundedList(int maxCapacity = -1, int initialCapacity = 10) + { + this.maxCapacity = maxCapacity < 0 ? -1 : maxCapacity; + values = new T[maxCapacity == -1 ? Max(initialCapacity, 0) : Min(maxCapacity, Max(initialCapacity, 0))]; + } + + private static int Min(int i1, int i2) => i1 > i2 ? i2 : i1; + private static int Max(int i1, int i2) => i1 > i2 ? i1 : i2; + + public BoundedList(int maxCapacity, IEnumerable collection) : this(maxCapacity, collection.Count()) + { + int track = 0; + IEnumerator enumerator = collection.GetEnumerator(); + while(enumerator.MoveNext() && track < maxCapacity) + { + Add(enumerator.Current); + ++track; + } + } + + public virtual bool Add(T t) + { + if (Count == maxCapacity) return false; + if (Count == values.Length) Resize(Count * GROW_FACTOR); + values[Count] = t; + ++Count; + return true; + } + + public virtual bool Remove(T t) => RemoveIf(t1 => (t == null && t1 == null) || (t != null && t.Equals(t1))) > 0; + + public int RemoveIf(Predicate p) + { + int removed = 0; + for (int c = 0; c < Count; ++c) + if (p(values[c])) + { + _RemoveAt(c); + ++removed; + } + if (values.Length >= Count * SHRINK_FACTOR) Resize(Count * SHRINK_FACTOR); + return removed; + } + + public virtual void RemoveAt(int i) + { + _RemoveAt(i); + if (values.Length >= Count * SHRINK_FACTOR) Resize(Count * SHRINK_FACTOR); + } + + public virtual T ElementAt(int i) + { + DoRangeCheck(i); + return values[i]; + } + + public virtual T[] ToArray() + { + T[] t = new T[Count]; + Array.Copy(values, t, Count); + return t; + } + + protected virtual void _RemoveAt(int i) + { + DoRangeCheck(i); + for (int j = i + 1; j < Count; ++j) values[j - 1] = values[j]; + values[Count - 1] = default(T); // Don't keep references in case GC needs to claim the object + --Count; + } + + protected virtual void Resize(float targetSize_f) + { + int targetSize = maxCapacity == -1 ? Math.Max((int)Math.Round(targetSize_f), 0) : Math.Max(0, Math.Min((int)Math.Round(targetSize_f), maxCapacity)); + T[] surrogate = new T[targetSize]; + Array.Copy(values, surrogate, Math.Min(targetSize, Count)); + values = surrogate; + } + + protected void DoRangeCheck(int i) + { + if (i < 0 || i >= Count) throw new IndexOutOfRangeException(); + } + + public virtual IEnumerator GetEnumerator() => new BoundedListEnumerator(this); + + IEnumerator IEnumerable.GetEnumerator() => GetEnumerator(); + + private sealed class BoundedListEnumerator : IEnumerator + { + public K Current => list.values[current]; + object IEnumerator.Current => Current; + + private readonly BoundedList list; + private int current = -1; + + public BoundedListEnumerator(BoundedList list) + { + this.list = list; + } + + public void Dispose() { } + + public bool MoveNext() => ++current < list.Count; + + public void Reset() => current = 0; + } + } + + public class EvictionList : BoundedList + { + public EvictionList(int maxCapacity = -1, int initialCapacity = 10) : base(maxCapacity, initialCapacity) + { } + + public override bool Add(T t) + { + if (Count == maxCapacity) RemoveAt(0); + return base.Add(t); + } + } + + public static class Collections + { + public static M Get(this Dictionary dict, K key) + { + object d; + if (dict.ContainsKey(key) && dict[key] != null && dict[key] is M) d = dict[key]; + else d = default(M); + return (M)d; + } + public static Dictionary Replace(this Dictionary dict, K key, T replace) + { + dict[key] = replace; + return dict; + } + public static int CollectiveLength(this Tuple[] values, bool first) + { + int len = 0; + foreach (var val in values) + len += (first ? val.Item1 : val.Item2)?.Length ?? 0; + return len; + } + public static T[] Collect(this Tuple[] values, bool first) + { + T[] collect = new T[values.Length]; + for (int i = 0; i < values.Length; ++i) collect[i] = (first ? values[i].Item1 : values[i].Item2); + return collect; + } + public static V GetNamed(this List> list, T key) + { + foreach (var element in list) + if (element != null && ObjectEquals(key, element.Item1)) + return element.Item2; + return default(V); + } + public static V GetNamed(this ReadOnlyCollection> list, T key) + { + foreach (var element in list) + if (element != null && ObjectEquals(key, element.Item1)) + return element.Item2; + return default(V); + } + + public static V GetFirst(this List v, Predicate p) + { + foreach (var v1 in v) + if (p(v1)) + return v1; + return default(V); + } + + public static bool ContainsExactly(this string s, char c, int count = 1) + { + int ctr = 0; + for (int i = 0; i < s.Length; ++i) + if (s[i] == c && ++ctr > count) + return false; + return ctr == count; + } + + public static bool ObjectEquals(object o1, object o2) => (o1==null && o2==null) || (o1!=null && o1.Equals(o2)); + + public static int CollectiveLength(this Tuple[] t, bool redundant = true) + { + int len = 0; + foreach (var val in t) + len += val?.Item1?.Length ?? 0; + return len; + } + + public static int CollectiveLength(this Tuple[] t) + { + int len = 0; + foreach (var val in t) + len += val?.Item2?.Length ?? 0; + return len; + } + + public static List Filter(this List t, Predicate p) + { + List l1 = new List(); + foreach (var l in t) + if (p(l)) + l1.Add(l); + return l1; + } + + public delegate T Transformation(V v); + public static List Transform(this List l, Transformation t) + { + List l1 = new List(); + foreach (var l2 in l) + l1.Add(t(l2)); + return l1; + } + public static T[] Transform(this V[] l, Transformation t) + { + T[] l1 = new T[l.Length]; + for (int i = 0; i < l.Length; ++i) + l1[i] = t(l[i]); + return l1; + } + } +} diff --git a/Common/Common.csproj b/Common/Common.csproj new file mode 100644 index 0000000..f5547d8 --- /dev/null +++ b/Common/Common.csproj @@ -0,0 +1,94 @@ + + + + + Debug + AnyCPU + {23EB87D4-E310-48C4-A931-0961C83892D7} + Library + Common + Common + v4.6.1 + 512 + true + publish\ + true + Disk + false + Foreground + 7 + Days + false + false + true + 0 + 1.0.0.%2a + false + false + true + + + AnyCPU + true + full + false + bin\Debug\ + DEBUG;TRACE + prompt + 4 + false + + + AnyCPU + pdbonly + true + bin\Release\ + TRACE + prompt + 4 + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + False + Microsoft .NET Framework 4.6.1 %28x86 and x64%29 + true + + + False + .NET Framework 3.5 SP1 + false + + + + \ No newline at end of file diff --git a/Common/Common.csproj.user b/Common/Common.csproj.user new file mode 100644 index 0000000..7937b01 --- /dev/null +++ b/Common/Common.csproj.user @@ -0,0 +1,13 @@ + + + + publish\ + + + + + + en-US + false + + \ No newline at end of file diff --git a/Common/KDF.cs b/Common/KDF.cs new file mode 100644 index 0000000..f61f51b --- /dev/null +++ b/Common/KDF.cs @@ -0,0 +1,72 @@ +using System; +using System.Collections.Generic; +using System.Linq; +using System.Text; +using System.Threading.Tasks; + +namespace Tofvesson.Crypto +{ + // Class for key derivation + public static class KDF + { + public delegate byte[] HashFunction(byte[] message); + + // Hash-based Message Authentication Codes: generates a code for verifying the sender of a message and the like + public static byte[] HMAC(byte[] key, byte[] message, HashFunction func, int blockSizeBytes) + { + if (key.Length > blockSizeBytes) key = func(key); + else if (key.Length < blockSizeBytes) + { + byte[] b = new byte[blockSizeBytes]; + Array.Copy(key, b, key.Length); + key = b; + } + + byte[] o_key_pad = new byte[blockSizeBytes]; // Outer padding + byte[] i_key_pad = new byte[blockSizeBytes]; // Inner padding + for (int i = 0; i < blockSizeBytes; ++i) + { + // Combine padding with key + o_key_pad[i] = (byte)(key[i] ^ 0x5c); + i_key_pad[i] = (byte)(key[i] ^ 0x36); + } + return func(Support.Concatenate(o_key_pad, func(Support.Concatenate(message, i_key_pad)))); + } + + // Perform HMAC using SHA1 + public static byte[] HMAC_SHA1(byte[] key, byte[] message) => HMAC(key, message, SHA.SHA1, 20); + + // Pseudorandom function delegate + public delegate byte[] PRF(byte[] key, byte[] salt); + + // Password-Based Key Derivation Function 2. Used to generate "pseudorandom" keys from a given password and salt using a certain PRF applied a certain amount of times (iterations). + // dklen specified the "derived key length" in bytes. It is recommended to use a high number for the iterations variable (somewhere around 4096 is the standard for SHA1 currently) + public static byte[] PBKDF2(PRF function, byte[] password, byte[] salt, int iterations, int dklen) + { + byte[] dk = new byte[0]; // Create a placeholder for the derived key + uint iter = 1; // Track the iterations + while (dk.Length < dklen) + { + // F-function + // The F-function (PRF) takes the amount of iterations performed in the opposite endianness format from what C# uses, so we have to swap the endianness + byte[] u = function(password, Support.Concatenate(salt, Support.WriteToArray(new byte[4], Support.SwapEndian(iter), 0))); + byte[] ures = new byte[u.Length]; + Array.Copy(u, ures, u.Length); + for(int i = 1; i associations, ref bool stayAlive); + public delegate void OnClientConnectStateChanged(NetClient client, bool connect); + public sealed class NetServer + { + private readonly short port; + private readonly object state_lock = new object(); + private readonly List clients = new List(); + private readonly OnMessageRecieved callback; + private readonly OnClientConnectStateChanged onConn; + private readonly IPAddress ipAddress; + private Socket listener; + private readonly RSA crypto; + private readonly byte[] ser_cache; + private readonly int bufSize; + + private bool state_running = false; + private Thread listenerThread; + + + public int Count + { + get + { + return clients.Count; + } + } + + public bool Running + { + get + { + lock (state_lock) return state_running; + } + + private set + { + lock (state_lock) state_running = value; + } + } + + public NetServer(RSA crypto, short port, OnMessageRecieved callback, OnClientConnectStateChanged onConn, int bufSize = 16384) + { + this.callback = callback; + this.onConn = onConn; + this.bufSize = bufSize; + this.crypto = crypto; + this.port = port; + this.ser_cache = crypto.Serialize(); // Keep this here so we don't wastefully re-serialize every time we get a new client + + IPHostEntry ipHostInfo = Dns.GetHostEntry(Dns.GetHostName()); + this.ipAddress = ipHostInfo.GetIPV4(); + if (ipAddress == null) + ipAddress = IPAddress.Parse("127.0.0.1"); // If there was no IPv4 result in dns lookup, use loopback address + } + + public void StartListening() + { + bool isAlive = false; + object lock_await = new object(); + if(!Running && (listenerThread==null || !listenerThread.IsAlive)) + { + Running = true; + listenerThread = new Thread(() => + { + + this.listener = new Socket(ipAddress.AddressFamily, SocketType.Stream, ProtocolType.Tcp) + { + Blocking = false // When calling Accept() with no queued sockets, listener throws an exception + }; + IPEndPoint localEndPoint = new IPEndPoint(ipAddress, port); + listener.Bind(localEndPoint); + listener.Listen(100); + + byte[] buffer = new byte[bufSize]; + lock (lock_await) isAlive = true; + Stopwatch limiter = new Stopwatch(); + while (Running) + { + limiter.Start(); + // Accept clients + try + { + Socket s = listener.Accept(); + s.Blocking = false; + clients.Add(new ClientStateObject(new NetClient(s, crypto, callback, onConn), buffer)); + } + catch (Exception) + { + if(clients.Count==0) + Thread.Sleep(25); // Wait a bit before trying to accept another client + } + + // Update clients + foreach (ClientStateObject cli in clients.ToArray()) + // Ensure we are still connected to client + if (!(cli.IsConnected() && !cli.Update())) + { + cli.client.onConn(cli.client, false); + clients.Remove(cli); + continue; + } + limiter.Stop(); + if (limiter.ElapsedMilliseconds < 125) Thread.Sleep(250); // If loading data wasn't heavy, take a break + limiter.Reset(); + } + }) + { + Priority = ThreadPriority.Highest, + Name = $"NetServer-${port}" + }; + listenerThread.Start(); + } + + bool rd; + do + { + Thread.Sleep(25); + lock (lock_await) rd = isAlive; + } while (!rd); + } + + public Task StopRunning() + { + Running = false; + + return new TaskFactory().StartNew(() => + { + listenerThread.Join(); + return null; + }); + } + + private class ClientStateObject + { + internal NetClient client; + private bool hasCrypto = false; // Whether or not encrypted communication has been etablished + private Queue buffer = new Queue(); // Incoming data buffer + private int expectedSize = 0; // Expected size of next message + private readonly byte[] buf; + + public ClientStateObject(NetClient client, byte[] buf) + { + this.client = client; + this.buf = buf; + } + + public bool Update() + { + bool stop = client.SyncListener(ref hasCrypto, ref expectedSize, out bool read, buffer, buf); + return stop; + } + public bool IsConnected() => client.IsConnected; + } + } + + public class NetClient + { + private static readonly RandomProvider rp = new CryptoRandomProvider(); + + // Thread state lock for primitive values + private readonly object state_lock = new object(); + + // Primitive state values + private bool state_running = false; + + // Socket event listener + private Thread eventListener; + + // Communication parameters + protected readonly Queue messageBuffer = new Queue(); + public readonly Dictionary assignedValues = new Dictionary(); + protected readonly OnMessageRecieved handler; + protected internal readonly OnClientConnectStateChanged onConn; + protected readonly IPAddress target; + protected readonly int bufSize; + protected readonly RSA decrypt; + protected internal long lastComm = DateTime.Now.Ticks; // Latest comunication event (in ticks) + public RSA RemoteCrypto { get => decrypt; } + + // Connection to peer + protected Socket Connection { get; private set; } + + // State/connection parameters + protected Rijndael128 Crypto { get; private set; } + protected GenericCBC CBC { get; private set; } + public short Port { get; } + protected bool Running + { + get + { + lock (state_lock) return state_running; + } + private set + { + lock (state_lock) state_running = value; + } + } + + protected internal bool IsConnected + { + get + { + return Connection != null && Connection.Connected && !(Connection.Poll(1, SelectMode.SelectRead) && Connection.Available == 0); + } + } + + public bool IsAlive + { + get + { + return Running || (Connection != null && Connection.Connected) || (eventListener != null && eventListener.IsAlive); + } + } + + protected bool ServerSide { get; private set; } + + + public NetClient(Rijndael128 crypto, IPAddress target, short port, OnMessageRecieved handler, OnClientConnectStateChanged onConn, int bufSize = 16384) + { +#pragma warning disable CS0618 // Type or member is obsolete + if (target.AddressFamily==AddressFamily.InterNetwork && target.Address == 16777343) +#pragma warning restore CS0618 // Type or member is obsolete + { + IPAddress addr = Dns.GetHostEntry(Dns.GetHostName()).GetIPV4(); + if (addr != null) target = addr; + } + this.target = target; + Crypto = crypto; + if(crypto!=null) CBC = new PCBC(crypto, rp); + this.bufSize = bufSize; + this.handler = handler; + this.onConn = onConn; + Port = port; + ServerSide = false; + } + + internal NetClient(Socket sock, RSA crypto, OnMessageRecieved handler, OnClientConnectStateChanged onConn) + : this(null, ((IPEndPoint)sock.RemoteEndPoint).Address, (short) ((IPEndPoint)sock.RemoteEndPoint).Port, handler, onConn, -1) + { + decrypt = crypto; + Connection = sock; + Running = true; + ServerSide = true; + + // Initiate crypto-handshake by sending public keys + Connection.Send(NetSupport.WithHeader(crypto.Serialize())); + } + + public virtual void Connect() + { + if (ServerSide) throw new SystemException("Serverside socket cannot connect to a remote peer!"); + NetSupport.DoStateCheck(IsAlive || (eventListener != null && eventListener.IsAlive), false); + Connection = new Socket(SocketType.Stream, ProtocolType.Tcp); + Connection.Connect(target, Port); + Running = true; + eventListener = new Thread(() => + { + bool cryptoEstablished = false; + int mLen = 0; + Queue ibuf = new Queue(); + byte[] buffer = new byte[bufSize]; + Stopwatch limiter = new Stopwatch(); + while (Running) + { + limiter.Start(); + if (SyncListener(ref cryptoEstablished, ref mLen, out bool _, ibuf, buffer)) + break; + if (cryptoEstablished && DateTime.Now.Ticks >= lastComm + (5 * TimeSpan.TicksPerSecond)) + try + { + Connection.Send(NetSupport.WithHeader(new byte[0])); // Send a test packet. (Will just send an empty header to the peer) + lastComm = DateTime.Now.Ticks; + } + catch + { + break; // Connection died + } + limiter.Stop(); + if (limiter.ElapsedMilliseconds < 125) Thread.Sleep(250); // If loading data wasn't heavy, take a break + limiter.Reset(); + } + if (ibuf.Count != 0) Debug.WriteLine("Client socket closed with unread data!"); + onConn(this, false); + }) + { + Priority = ThreadPriority.Highest, + Name = $"NetClient-${target}:${Port}" + }; + eventListener.Start(); + } + + protected internal bool SyncListener(ref bool cryptoEstablished, ref int mLen, out bool acceptedData, Queue ibuf, byte[] buffer) + { + if (cryptoEstablished) + { + lock (messageBuffer) + { + foreach (byte[] message in messageBuffer) Connection.Send(NetSupport.WithHeader(message)); + if(messageBuffer.Count > 0) lastComm = DateTime.Now.Ticks; + messageBuffer.Clear(); + } + } + if (acceptedData = Connection.Available > 0) + { + int read = Connection.Receive(buffer); + ibuf.EnqueueAll(buffer, 0, read); + if (read > 0) lastComm = DateTime.Now.Ticks; + } + if (mLen == 0 && ibuf.Count >= 4) + mLen = Support.ReadInt(ibuf.Dequeue(4), 0); + if (mLen != 0 && ibuf.Count >= mLen) + { + // Got a full message. Parse! + byte[] message = ibuf.Dequeue(mLen); + lastComm = DateTime.Now.Ticks; + + if (!cryptoEstablished) + { + if (ServerSide) + { + var nonceText = new string(Encoding.UTF8.GetChars(message)); + byte[] sign; + if(nonceText.StartsWith("Nonce:") && BigInteger.TryParse(nonceText.Substring(6), out BigInteger parse) && (sign=parse.ToByteArray()).Length <= 512) + { + Connection.Send(NetSupport.WithHeader(decrypt.Encrypt(parse.ToByteArray(), null, true))); + Disconnect(); + return true; + } + + if (Crypto == null) + { + byte[] m = decrypt.Decrypt(message); + if (m.Length == 0) return false; + Crypto = Rijndael128.Deserialize(m, out int _); + } + else + { + byte[] m = decrypt.Decrypt(message); + if (m.Length == 0) return false; + CBC = new PCBC(Crypto, m); + onConn(this, true); + } + } + else + { + // Reconstruct RSA object from remote public keys and use it to encrypt our serialized AES key/iv + RSA asymm = RSA.Deserialize(message, out int _); + Connection.Send(NetSupport.WithHeader(asymm.Encrypt(Crypto.Serialize()))); + Connection.Send(NetSupport.WithHeader(asymm.Encrypt(CBC.IV))); + onConn(this, true); + } + if (CBC != null) + cryptoEstablished = true; + } + else + { + // Decrypt the incoming message + byte[] read = Crypto.Decrypt(message); + + // Read the decrypted message length + int mlenInner = Support.ReadInt(read, 0); + if (mlenInner == 0) return false; // Got a ping packet + + // Send the message to the handler and get a response + bool live = true; + string response = handler(read.SubArray(4, 4+mlenInner).ToUTF8String(), assignedValues, ref live); + + // Send the response (if given one) and drop the connection if the handler tells us to + if (response != null) Connection.Send(NetSupport.WithHeader(Crypto.Encrypt(NetSupport.WithHeader(response.ToUTF8Bytes())))); + if (!live) + { + Running = false; + try + { + Connection.Close(); + } + catch (Exception) { } + return true; + } + } + + // Reset expexted message length + mLen = 0; + } + return false; + } + + /// + /// Disconnect from server + /// + /// + public virtual async Task Disconnect() + { + NetSupport.DoStateCheck(IsAlive, true); + Running = false; + + + return await new TaskFactory().StartNew(() => { eventListener.Join(); return null; }); + } + + // Methods for sending data to the server + public bool TrySend(string message) => TrySend(Encoding.UTF8.GetBytes(message)); + public bool TrySend(byte[] message) + { + try + { + Send(message); + return true; + } + catch (InvalidOperationException) { return false; } + } + public virtual void Send(string message) => Send(Encoding.UTF8.GetBytes(message)); + public virtual void Send(byte[] message) { + NetSupport.DoStateCheck(IsAlive, true); + lock (messageBuffer) messageBuffer.Enqueue(Crypto.Encrypt(NetSupport.WithHeader(message))); + } + + public static RSA CheckServerIdentity(string host, short port, RandomProvider provider, long timeout = 10000) + { + Socket sock = new Socket(SocketType.Stream, ProtocolType.Tcp) + { + ReceiveTimeout = 5000, + SendTimeout = 5000 + }; + sock.Blocking = false; + sock.Connect(host, port); + List read = new List(); + byte[] buf = new byte[1024]; + + if (!Read(sock, read, buf, timeout)) return null; + read.RemoveRange(0, 4); + RSA remote; + try + { + remote = RSA.Deserialize(read.ToArray(), out int _); + } + catch { return null; } + BigInteger cmp; + sock.Send(NetSupport.WithHeader(Encoding.UTF8.GetBytes("Nonce:"+(cmp=BigInteger.Abs(new BigInteger(provider.GetBytes(128))))))); + Thread.Sleep(250); // Give the server ample time to compute the signature + read.Clear(); + if (!Read(sock, read, buf, timeout)) return null; + read.RemoveRange(0, 4); + try + { + if (!cmp.Equals(new BigInteger(remote.Encrypt(read.ToArray())))) return null; + } + catch { return null; } + return remote; // Passed signature check + } + + private static bool Read(Socket sock, List read, byte[] buf, long timeout) + { + Stopwatch sw = new Stopwatch(); + int len = -1; + sw.Start(); + while ((len == -1 || read.Count < 4) && (sw.ElapsedTicks / 10000) < timeout) + { + if (len == -1 && read.Count > 4) + len = Support.ReadInt(read, 0); + + try + { + int r = sock.Receive(buf); + read.AddRange(buf.SubArray(0, r)); + } + catch { } + } + sw.Stop(); + return read.Count - 4 == len && len>0; + } + } + + // Helper methods. WithHeader() should really just be in Support.cs + public static class NetSupport + { + public static byte[] WithHeader(string message) => WithHeader(Encoding.UTF8.GetBytes(message)); + public static byte[] WithHeader(byte[] message) + { + byte[] nmsg = new byte[message.Length + 4]; + Support.WriteToArray(nmsg, message.Length, 0); + Array.Copy(message, 0, nmsg, 4, message.Length); + return nmsg; + } + + public static byte[] FromHeaded(byte[] msg, int offset) => msg.SubArray(offset + 4, offset + 4 + Support.ReadInt(msg, offset)); + + internal static void DoStateCheck(bool state, bool target) { + if (state != target) throw new InvalidOperationException("Bad state!"); + } + } +} diff --git a/Common/Padding.cs b/Common/Padding.cs new file mode 100644 index 0000000..2c33aa6 --- /dev/null +++ b/Common/Padding.cs @@ -0,0 +1,376 @@ +using System; +using System.Collections.Generic; +using System.IO; +using System.Text; +using System.Text.RegularExpressions; +using System.Xml; + +namespace Tofvesson.Crypto +{ + public interface CryptoPadding + { + byte[] Pad(byte[] message); + byte[] Unpad(byte[] message); + PaddingIdentifier GetParameters(); + } + + public sealed class PaddingIdentifier + { + private readonly Dictionary> attributes = new Dictionary>(); + private readonly Dictionary nests = new Dictionary(); + + public string Name { get; private set; } + + public List AttributeKeys + { + get + { + List keys = new List(); + keys.AddRange(attributes.Keys); + return keys; + } + } + + public List NestedKeys + { + get + { + List keys = new List(); + keys.AddRange(nests.Keys); + return keys; + } + } + + public PaddingIdentifier(string name) { this.Name = name; } + public void AddAttribute(string attr, byte[] data) => attributes.Add(attr, new Tuple(ParameterTypes.BYTES, Support.ArrayToString(data))); + public void AddAttribute(string attr, int data) => attributes.Add(attr, new Tuple(ParameterTypes.NUMBER, data.ToString())); + public void AddAttribute(string attr, PaddingIdentifier data) => nests.Add(attr, data); + + public Tuple GetAttribute(string key) + { + if (attributes.ContainsKey(key)) return attributes[key]; + return null; + } + + public PaddingIdentifier GetNested(string key) + { + if (nests.ContainsKey(key)) return nests[key]; + return null; + } + + public XmlElement Compile(XmlDocument writeTo) + { + XmlElement root = writeTo.CreateElement(Name); + foreach (string key in attributes.Keys) + { + XmlElement attr = writeTo.CreateElement(key); + attr.SetAttribute("type", attributes[key].Item1.ToString()); + attr.InnerText = attributes[key].Item2; + root.AppendChild(attr); + } + + foreach (string key in nests.Keys) root.AppendChild(nests[key].Compile(writeTo)); + return root; + } + } + + public enum ParameterTypes { NUMBER, BYTES, NESTED } + + public sealed class RandomLengthPadding : CryptoPadding + { + private const ushort DEFAULT_MAX = 12; + + private readonly RandomProvider provider; + private readonly byte[] delimiter; + private readonly ushort maxLen; + + public RandomLengthPadding(RandomProvider provider, byte[] delimiter, ushort maxLen = DEFAULT_MAX) + { + this.provider = provider; + this.delimiter = delimiter; + this.maxLen = maxLen; + } + + public RandomLengthPadding(byte[] delimiter, ushort maxLen = DEFAULT_MAX) + : this(new RegularRandomProvider(), delimiter, maxLen) + { } + + + public byte[] Pad(byte[] message) + { + // Generate padding + byte[] prepadding = GenerateSequence(); + byte[] postpadding = GenerateSequence(); + + // Allocate output array + byte[] result = new byte[message.Length + prepadding.Length + postpadding.Length + delimiter.Length * 2]; + + // Assemble padding + int index = 0; + Array.Copy(prepadding, 0, result, 0, -(index - (index += prepadding.Length))); + Array.Copy(delimiter, 0, result, index, -(index - (index += delimiter.Length))); + Array.Copy(message, 0, result, index, -(index - (index += message.Length))); + Array.Copy(delimiter, 0, result, index, -(index - (index += delimiter.Length))); + Array.Copy(postpadding, 0, result, index, -(index - (index += postpadding.Length))); + + return result; + } + + public byte[] Unpad(byte[] message) + { + int index = Support.ArrayContains(message, delimiter); + if (index == -1) throw new InvalidPaddingException("Preceding delimiter could not be found"); + byte[] result_stage1 = new byte[message.Length - 1 - index]; + Array.Copy(message, index + 1, result_stage1, 0, message.Length - 1 - index); + index = Support.ArrayContains(result_stage1, delimiter, false); + if (index == -1) throw new InvalidPaddingException("Trailing delimeter could not be found"); + byte[] result_stage2 = new byte[index]; + Array.Copy(result_stage1, 0, result_stage2, 0, index); + return result_stage2; + } + + private byte[] GenerateSequence() + { + // Generate between 0 and maxLen random bytes to be used as padding + byte[] padding = provider.GetBytes(provider.NextUShort((ushort)(maxLen + 1))); + + // Remove instances of the delimiter sequence from the padding + int idx; + while ((idx = Support.ArrayContains(padding, delimiter)) != -1) + foreach (byte val in provider.GetBytes(delimiter.Length)) + padding[idx++] = val; + return padding; + } + + public PaddingIdentifier GetParameters() + { + PaddingIdentifier id = new PaddingIdentifier("R"); + id.AddAttribute("delimiter", delimiter); + id.AddAttribute("maxLen", maxLen); + return id; + } + } + + public sealed class IncrementalPadding : CryptoPadding + { + private const int DEFAULT_INCREMENT = 12; + + private readonly RandomProvider provider; + private readonly int increments; + private readonly int determiner; + + + public IncrementalPadding(RandomProvider provider, int determiner, int increments = DEFAULT_INCREMENT) + { + this.provider = provider; + this.increments = increments * determiner; + this.determiner = determiner; + if (increments < 0) throw new InvalidPaddingException("Increments cannot be negative!"); + if (determiner <= 1) throw new InvalidPaddingException("Determiner must be a positive value larger than 1!"); + if (increments * determiner < 0) throw new InvalidPaddingException("Increment-Delimiter pair is too large!"); + } + + public byte[] Pad(byte[] message) + { + if (message.Length % determiner != 0) + { + byte[] result = new byte[message.Length + increments]; + Array.Copy(message, result, message.Length); + Array.Copy(provider.GetBytes(increments), 0, result, message.Length, increments); + return result; + } + else return message; + } + + public byte[] Unpad(byte[] message) + { + if (message.Length % determiner == 0) return message; + byte[] result = new byte[message.Length - increments]; + Array.Copy(message, result, result.Length); + return result; + } + + public PaddingIdentifier GetParameters() + { + PaddingIdentifier id = new PaddingIdentifier("I"); + id.AddAttribute("increments", increments / determiner); + id.AddAttribute("determiner", determiner); + return id; + } + } + + public sealed class SequentialPadding : CryptoPadding + { + private readonly List pads = new List(); + + public SequentialPadding WithPadding(CryptoPadding padding) + { + pads.Add(padding); + return this; + } + + public byte[] Pad(byte[] message) + { + for (int i = 0; i < pads.Count; ++i) message = pads[i].Pad(message); + return message; + } + + public byte[] Unpad(byte[] message) + { + for (int i = pads.Count - 1; i >= 0; --i) message = pads[i].Unpad(message); + return message; + } + + public PaddingIdentifier GetParameters() + { + PaddingIdentifier id = new PaddingIdentifier("S"); + for (int i = 0; i < pads.Count; ++i) id.AddAttribute(i.ToString(), pads[i].GetParameters()); + return id; + } + } + + public sealed class PassthroughPadding : CryptoPadding + { + public byte[] Pad(byte[] message) => message; + public byte[] Unpad(byte[] message) => message; + public PaddingIdentifier GetParameters() => new PaddingIdentifier("P"); + } + + public static class PaddingSupport + { + private static readonly Regex byteFinder = new Regex("(\\d{0,3})[,\\]]"); + + + public static string SerializePadding(CryptoPadding padding) + { + XmlDocument doc = new XmlDocument(); + doc.AppendChild(padding.GetParameters().Compile(doc)); + + string output; + using (var stream = new MemoryStream()) + { + XmlTextWriter writer = new XmlTextWriter(stream, Encoding.UTF8); + doc.WriteTo(writer); + writer.Flush(); + stream.Position = 0; + using (var reader = new StreamReader(stream)) + { + output = reader.ReadToEnd(); + } + } + return output; + } + + // WIP + public static string NetSerialize(CryptoPadding padding) => NetSerialize(padding.GetParameters(), new StringBuilder()).ToString(); + public static string NetSerialize(PaddingIdentifier padding) => NetSerialize(padding, new StringBuilder()).ToString(); + public static StringBuilder NetSerialize(PaddingIdentifier id, StringBuilder builder) + { + builder.Append(id.Name).Append('{'); + foreach (string key in id.AttributeKeys) builder.Append(id.GetAttribute(key).Item2).Append(','); + foreach (string key in id.NestedKeys) NetSerialize(id.GetNested(key), builder).Append(','); + if (id.AttributeKeys.Count > 0 || id.NestedKeys.Count > 0) builder.Remove(builder.Length - 1, 1); // Remove last ',' + builder.Append('}'); + return builder; + } + + // Works but is really large + public static CryptoPadding DeserializePadding(string ser) => DeserializePadding(ser, new DummyRandomProvider()); + public static CryptoPadding DeserializePadding(string ser, RandomProvider provider) + { + XmlDocument doc = new XmlDocument(); + doc.LoadXml(ser); + XmlNodeList lst = doc.ChildNodes; + if (lst.Count != 1) throw new XMLCryptoParseException("Cannot have more than one root node!"); + return ParseNode(lst.Item(0), provider); + } + + + private static CryptoPadding ParseNode(XmlNode el, RandomProvider provider) + { + XmlNodeList lst; + switch (el.Name) + { + case "P": + return new PassthroughPadding(); + case "S": + { + SequentialPadding seq = new SequentialPadding(); + if (el.HasChildNodes) + { + lst = el.ChildNodes; + foreach (XmlNode subNode in lst) seq.WithPadding(ParseNode(subNode, provider)); + } + return seq; + } + case "I": + { + if (el.HasChildNodes && (lst = el.ChildNodes).Count == 2) + { + int increments; + if (!TryParseNumberNode("increments", lst, out increments)) + throw new XMLCryptoParseException("Invalid parameter supplied"); + int determiner; + if (!TryParseNumberNode("determiner", lst, out determiner)) + throw new XMLCryptoParseException("Invalid parameter supplied"); + return new IncrementalPadding(provider, determiner, increments); + } + else throw new XMLCryptoParseException("No parameters supplied"); + } + case "R": + { + if (el.HasChildNodes && (lst = el.ChildNodes).Count == 2) + { + byte[] delimiter = TryParseByteNode("delimiter", lst); + if (delimiter == null) throw new XMLCryptoParseException("Invalid parameter supplied"); + int maxLen; + if (!TryParseNumberNode("maxLen", lst, out maxLen)) + throw new XMLCryptoParseException("Invalid parameter supplied"); + return new RandomLengthPadding(provider, delimiter, (ushort)maxLen); + } + else throw new XMLCryptoParseException("No parameters supplied"); + } + default: + throw new XMLCryptoParseException($"Unrecognized padding algorithm \"{el.Name}\""); + + } + } + + private static bool TryParseNumberNode(string name, XmlNodeList from, out int val) + { + XmlNode node = Support.ContainsNamedNode(name, from); + val = 0; + return node != null && int.TryParse(node.InnerText, out val); + } + + public static byte[] TryParseByteNode(string name, XmlNodeList from) + { + XmlNode node = Support.ContainsNamedNode(name, from); + if (node == null) return null; + List collect = new List(); + Match m = byteFinder.Match(node.InnerText); + while (m.Success) + { + collect.Add(byte.Parse(m.Groups[1].Value)); + m = m.NextMatch(); + } + return collect.ToArray(); + } + + + public class XMLCryptoParseException : SystemException + { + public XMLCryptoParseException() { } + public XMLCryptoParseException(string message) : base(message) { } + public XMLCryptoParseException(string message, Exception innerException) : base(message, innerException) { } + } + } + + // Exception related to padding errors + public class InvalidPaddingException : SystemException + { + public InvalidPaddingException() { } + public InvalidPaddingException(string message) : base(message) { } + public InvalidPaddingException(string message, Exception innerException) : base(message, innerException) { } + } +} diff --git a/Common/Properties/AssemblyInfo.cs b/Common/Properties/AssemblyInfo.cs new file mode 100644 index 0000000..9a1fdb7 --- /dev/null +++ b/Common/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("Common")] +[assembly: AssemblyDescription("")] +[assembly: AssemblyConfiguration("")] +[assembly: AssemblyCompany("")] +[assembly: AssemblyProduct("Common")] +[assembly: AssemblyCopyright("Copyright © 2018")] +[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("23eb87d4-e310-48c4-a931-0961c83892d7")] + +// 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/Common/RSA.cs b/Common/RSA.cs new file mode 100644 index 0000000..92859a7 --- /dev/null +++ b/Common/RSA.cs @@ -0,0 +1,182 @@ +using System; +using System.Collections.Generic; +using System.IO; +using System.Linq; +using System.Numerics; +using System.Text; + +namespace Tofvesson.Crypto +{ + public class RSA + { + private static readonly PassthroughPadding NO_PADDING = new PassthroughPadding(); + private static readonly Encoding DEFAULT_ENCODING = Encoding.UTF8; + + private readonly RandomProvider provider = new CryptoRandomProvider(); + + private readonly BigInteger e; + private readonly BigInteger n; + private readonly BigInteger d; + + public bool CanEncrypt { get; private set; } + public bool CanDecrypt { get; private set; } + + public RSA(int byteSize, int margin, int threads, int certainty) + { + // Choose primes + BigInteger p = Support.GeneratePrime(threads, byteSize, margin, certainty, provider); + BigInteger q = Support.GeneratePrime(threads, byteSize, margin, certainty, provider); + + + // For optimization + BigInteger p_1 = p - 1; + BigInteger q_1 = q - 1; + + // Calculate needed values + n = p * q; + BigInteger lcm = (p_1 * q_1) / Support.GCD(p_1, q_1); + + // Generate e such that is is less than and coprime to lcm + do + { + e = RandomSupport.GenerateBoundedRandom(lcm, provider); + } while (e == lcm || Support.GCD(e, lcm) != 1); + + // Generate the modular multiplicative inverse + d = Support.Dio(e, lcm).Key + lcm; + CanEncrypt = true; + CanDecrypt = true; + } + + // Load necessary values from files + public RSA(string e_file, string n_file, string d_file) : this(File.ReadAllBytes(e_file), File.ReadAllBytes(n_file), File.ReadAllBytes(d_file)) + { } + + public RSA(byte[] e, byte[] n, byte[] d = null) + { + this.e = new BigInteger(e); + this.n = new BigInteger(n); + this.d = new BigInteger(d ?? new byte[0]); + CanEncrypt = true; + CanDecrypt = d!=null; + } + + // Create a shallow copy of the given object because it's really just wasteful to perform a deep copy + // unless the person modifying this code is a madman, in which case I highly doubt they'd willfully leave this code alone anyway... + public RSA(RSA copy) + { + e = copy.e; + n = copy.n; + d = copy.d; + CanEncrypt = copy.CanEncrypt; + CanDecrypt = copy.CanDecrypt; + } + + // Create a "remote" instance of the rsa object. This means that we do not know the private exponent + private RSA(byte[] e, byte[] n) + { + this.e = new BigInteger(e); + this.n = new BigInteger(n); + this.d = BigInteger.Zero; + CanEncrypt = true; + CanDecrypt = false; + } + + // Encrypt (duh) + public byte[] EncryptString(string message, Encoding encoding = null, CryptoPadding padding = null, bool sign = false) => Encrypt((encoding ?? DEFAULT_ENCODING).GetBytes(message), padding, sign); + public byte[] Encrypt(byte[] message, CryptoPadding padding = null, bool sign = false) + { + // Apply dynamic padding + message = (padding ?? NO_PADDING).Pad(message); + + // Apply fixed padding + byte[] b1 = new byte[message.Length + 1]; + Array.Copy(message, b1, message.Length); + b1[message.Length] = 1; + message = b1; + + // Represent message as a number + BigInteger m = new BigInteger(message); + + // Encrypt message + BigInteger cryptomessage = Support.ModExp(m, sign ? d : e, n); + + // Convert encrypted message back to bytes + return cryptomessage.ToByteArray(); + } + + // Decrypt (duh) + public string DecryptString(byte[] message, Encoding encoding = null, CryptoPadding padding = null, bool checkSign = false) => new string((encoding ?? DEFAULT_ENCODING).GetChars(Decrypt(message, padding, checkSign))); + public byte[] Decrypt(byte[] message, CryptoPadding padding = null, bool checkSign = false) + { + + // Reinterpret encrypted message as a number + BigInteger cryptomessage = new BigInteger(message); + + // Reverse encryption + message = Support.ModExp(cryptomessage, checkSign ? e : d, n).ToByteArray(); + + // Remove fixed padding + byte[] b1 = new byte[message.Length - 1]; + Array.Copy(message, b1, message.Length - 1); + message = b1; + + // Remove dynamic padding + message = (padding ?? NO_PADDING).Unpad(message); + + return message; + } + + // Gives you the public key + public byte[] GetPubK() => e.ToByteArray(); + + // Save this RSA instance to correspondingly named files + public void Save(string fileNameBase, bool force = false) + { + if (force || !File.Exists(fileNameBase + ".e")) File.WriteAllBytes(fileNameBase + ".e", e.ToByteArray()); + if (force || !File.Exists(fileNameBase + ".n")) File.WriteAllBytes(fileNameBase + ".n", n.ToByteArray()); + if (force || !File.Exists(fileNameBase + ".d")) File.WriteAllBytes(fileNameBase + ".d", d.ToByteArray()); + } + + // Serialize (for public key distribution) + public byte[] Serialize() => Support.SerializeBytes(new byte[][] { e.ToByteArray(), n.ToByteArray() }); + + // Deserialize RSA data (for key distribution (but the other end (how many parentheses deep can I go?))) + public static RSA Deserialize(byte[] function, out int read) + { + byte[][] rd = Support.DeserializeBytes(function, 2); + read = rd[0].Length + rd[1].Length + 8; + return new RSA(rd[0], rd[1]); + } + + // Check if the data we want to convert into an RSA-instance will cause a crash if we try to parse it + public static bool CanDeserialize(IEnumerable data) + { + try + { + int size = Support.ReadInt(data, 0), size2; + if (size >= data.Count() - 8) return false; + size2 = Support.ReadInt(data, 4 + size); + if (size2 > data.Count() - size - 8) return false; + return true; + } + catch (Exception) { } + return false; + } + + // Safely attempt to load RSA keys from files + public static RSA TryLoad(string fileNameBase) => TryLoad(fileNameBase + ".e", fileNameBase + ".n", fileNameBase + ".d"); + public static RSA TryLoad(string e_file, string n_file, string d_file) + { + try + { + return new RSA(e_file, n_file, d_file); + } + catch (Exception) { } + return null; + } + + public override bool Equals(object obj) + => obj is RSA && ((RSA)obj).CanDecrypt == CanDecrypt && ((RSA)obj).e.Equals(e) && ((RSA)obj).n.Equals(n) && (!CanDecrypt || ((RSA)obj).d.Equals(d)); + } +} diff --git a/Common/SHA.cs b/Common/SHA.cs new file mode 100644 index 0000000..31b5093 --- /dev/null +++ b/Common/SHA.cs @@ -0,0 +1,90 @@ +using System; +using System.Collections.Generic; +using System.Linq; +using System.Text; +using System.Threading.Tasks; + +namespace Tofvesson.Crypto +{ + /// + /// Secure Hashing Alorithm implementations + /// + public static class SHA + { + public static byte[] SHA1(byte[] message) + { + // Initialize buffers + uint h0 = 0x67452301; + uint h1 = 0xEFCDAB89; + uint h2 = 0x98BADCFE; + uint h3 = 0x10325476; + uint h4 = 0xC3D2E1F0; + + // Pad message + int ml = message.Length + 1; + byte[] msg = new byte[ml + ((960 - (ml*8 % 512)) % 512)/8 + 8]; + Array.Copy(message, msg, message.Length); + msg[message.Length] = 0x80; + long len = message.Length * 8; + for (int i = 0; i < 8; ++i) msg[msg.Length - 1 - i] = (byte)((len >> (i*8)) & 255); + //Support.WriteToArray(msg, message.Length * 8, msg.Length - 8); + //for (int i = 0; i <4; ++i) msg[msg.Length - 5 - i] = (byte)(((message.Length*8) >> (i * 8)) & 255); + + int chunks = msg.Length / 64; + + // Perform hashing for each 512-bit block + for(int i = 0; i + t < 20 ? (b & c) | ((~b) & d) : + t < 40 ? b ^ c ^ d : + t < 60 ? (b & c) | (b & d) | (c & d) : + /*t<80*/ b ^ c ^ d; + + private static uint K(int t) => + t < 20 ? 0x5A827999 : + t < 40 ? 0x6ED9EBA1 : + t < 60 ? 0x8F1BBCDC : + /*t<80*/ 0xCA62C1D6 ; + + private static uint Rot(uint val, int by) => (val << by) | (val >> (32 - by)); + } +} diff --git a/Common/Streams.cs b/Common/Streams.cs new file mode 100644 index 0000000..ef2bd82 --- /dev/null +++ b/Common/Streams.cs @@ -0,0 +1,57 @@ +using System; +using System.Collections.Generic; +using System.IO; +using System.Linq; +using System.Text; +using System.Threading.Tasks; +using System.Diagnostics; + +namespace Common +{ + public sealed class TimeStampWriter : TextWriter + { + private readonly DateTime time = DateTime.Now; + private readonly string dateFormat; + private readonly TextWriter underlying; + private bool triggered; + + public TimeStampWriter(TextWriter underlying, string dateFormat, bool emulateNL = true) + { + this.dateFormat = dateFormat; + this.underlying = underlying; + triggered = emulateNL; + } + + public TimeStampWriter(TextWriter underlying, string dateFormat, IFormatProvider formatProvider, bool emulateNL = true) : base(formatProvider) + { + this.dateFormat = dateFormat; + this.underlying = underlying; + triggered = emulateNL; + } + + public override Encoding Encoding => underlying.Encoding; + + public override void Write(char value) + { + if (triggered) + { + StringBuilder s = new StringBuilder(); + s.Append('[').Append(time.ToString(dateFormat)).Append("] "); + foreach (var c in s.ToString()) underlying.Write(c); + } + underlying.Write(value); + triggered = value == '\n'; + } + } + + // A TextWriter wrapper for the Debug output + public sealed class DebugAdapterWriter : TextWriter + { + public override Encoding Encoding => throw new NotImplementedException(); + + public override void Write(char value) + { + Debug.Write(value); + } + } +} diff --git a/Common/Support.cs b/Common/Support.cs new file mode 100644 index 0000000..d8f5124 --- /dev/null +++ b/Common/Support.cs @@ -0,0 +1,582 @@ +using System; +using System.Collections.Generic; +using System.IO; +using System.Linq; +using System.Net; +using System.Numerics; +using System.Reflection; +using System.Security.Cryptography; +using System.Text; +using System.Threading.Tasks; +using System.Xml; + +namespace Tofvesson.Crypto +{ + + // Just a ton of support methods to make life easier. Almost aboslutely nothing of notable value here + // Honestly, just continue on to the next file of whatever, unless you have some unbearable desire to give yourself a headache and be completely disappointed by the end of reading this + public static class Support + { + // -- Math -- + public static BigInteger Invert(BigInteger b) + { + byte[] arr = b.ToByteArray(); + for (int i = 0; i < arr.Length; ++i) arr[i] ^= 255; + BigInteger integer = new BigInteger(arr); + integer += 1; + return integer; + } + + public static BigInteger ModExp(BigInteger b, BigInteger e, BigInteger m) + { + int count = e.ToByteArray().Length * 8; + BigInteger result = BigInteger.One; + b = b % m; + while (count>0) + { + if (e % 2 != 0) result = (result * b) % m; + b = (b * b) % m; + e >>= 1; + --count; + } + return result; + } + + /// + /// Uses the fermat test a given amount of times to test whether or not a supplied interger is probably prime. + /// + /// Value to test primality of + /// Random provider used to generate values to test b against + /// How many times the test should be performed. More iterations means higher certainty, but at the cost of performance! + /// Whether or not the given value is probably prime or not + public static bool IsProbablePrime(BigInteger b, RandomProvider provider, int certainty) + { + BigInteger e = b - 1; + byte[] b1 = b.ToByteArray(); + byte last = b1[b1.Length-1]; + int len = b1.Length - 1; + for (int i = 0; i < certainty; ++i) + { + byte[] gen = new byte[provider.NextInt(len)+1]; + provider.GetBytes(gen); + if (last != 0 && gen.Length==len+1) gen[gen.Length - 1] %= last; + else gen[gen.Length - 1] &= 127; + + BigInteger test = new BigInteger(gen); + if (ModExp(test, e, b) != 1) return false; + } + return true; + } + + /// + /// Calculate the greatest common divisor for two values. + /// + /// First value + /// Second value + /// The greatest common divisor + public static BigInteger GCD(BigInteger b1, BigInteger b2) + { + BigInteger tmp; + while ((tmp = b1 % b2) != 0) + { + b1 = b2; + b2 = tmp; + } + return b2; + } + + public static int CollectiveLength(this string[] s) + { + int i = 0; + foreach (var s1 in s) i += s1.Length; + return i; + } + + /// + /// Linear diophantine equations. Calculates the modular multiplicative inverse for a given value and a given modulus. + /// For: ax + by = 1 + /// Where 'a' and 'b' are known factors + /// + /// First known factor (a) + /// Second known factor (b) + /// A pair of factors that fulfill the aforementioned equations (if possible), where Item1 corresponds to 'x' and Item2 corresponds to 'y'. If the two supplied known factors are not coprime, both factors will be 0 + public static KeyValuePair Dio(BigInteger in1, BigInteger in2) + { + // Euclidean algorithm + BigInteger tmp; + var i1 = in1; + var i2 = in2; + if (i1 <= BigInteger.Zero || i2 <= BigInteger.Zero || i1 == i2 || i1 % i2 == BigInteger.Zero || i2 % i1 == BigInteger.Zero) + { + return new KeyValuePair(BigInteger.Zero, BigInteger.Zero); + } + var minusOne = new BigInteger(-1); + var e_m = new BigInteger(-1L); + var collect = new Stack(); + while ((e_m = i1 % i2) != BigInteger.Zero) + { + collect.Push(i1 / i2 * minusOne); + i1 = i2; + i2 = e_m; + } + + // There are no solutions because 'a' and 'b' are not coprime + if (i2 != BigInteger.One) + return new KeyValuePair(BigInteger.Zero, BigInteger.Zero); + + + // Extended euclidean algorithm + var restrack_first = BigInteger.One; + var restrack_second = collect.Pop(); + + while (collect.Count > 0) + { + tmp = restrack_second; + restrack_second = restrack_first + restrack_second * collect.Pop(); + restrack_first = tmp; + } + return new KeyValuePair(restrack_first, restrack_second); + } + + /// + /// Generate a prime number using with a given approximate length and byte length margin + /// + /// How many threads to use to generate primes + /// The byte array length around which the prime generator will select lengths + /// Allowed deviation of byte length from approximateByteCount + /// How many iterations of the fermat test should be run to test primailty for each generated number + /// Random provider that will be used to generate random primes + /// A prime number that is aproximately approximateByteCount long + public static BigInteger GeneratePrime(int threads, int approximateByteCount, int byteMargin, int certainty, RandomProvider provider) + { + var found = false; + BigInteger result = BigInteger.Zero; + for(int i = 0; i + { + char left = '\0'; + byte rand = 0; + BigInteger b = BigInteger.Zero; + while (!found) + { + if (left == 0) + { + rand = provider.GetBytes(1)[0]; + left = (char)8; + } + + byte[] b1 = provider.GetBytes(approximateByteCount + (provider.GetBytes(1)[0] % byteMargin) * (rand % 2 == 1 ? 1 : -1)); + b1[0] |= 1; // Always odd + b1[b1.Length - 1] &= 127; // Always positive + b = new BigInteger(b1); + rand >>= 1; + --left; + if (IsProbablePrime(b, provider, certainty)) + { + found = true; + result = b; + } + } + }); + while (!found) System.Threading.Thread.Sleep(125); + return result; + } + + + + // -- Net -- + /// + /// Finds an IPv4a address in the address list. + /// + /// IPHostEntry to get the address from + /// An IPv4 address if available, otherwise null + public static IPAddress GetIPV4(this IPHostEntry entry) + { + foreach (IPAddress addr in entry.AddressList) + if (addr.AddressFamily == System.Net.Sockets.AddressFamily.InterNetwork) + return addr; + return null; + } + + + // -- Arrays/Collections -- + /// + /// Pad or truncate this array to the specified length. Padding is performed by filling the new indicies with 0's. Truncation removes bytes from the end. + /// + /// The array type + /// The array to resize + /// Target length + /// A resized array + public static T[] ToLength(this T[] t, int length) + { + var t1 = new T[length]; + Array.Copy(t, t1, Math.Min(length, t.Length)); + return t1; + } + + /// + /// Reads a serialized 32-bit integer from the byte collection + /// + /// + /// + /// + public static int ReadInt(IEnumerable data, int offset) + { + int result = 0; + for (int i = 0; i < 4; ++i) + result |= data.ElementAt(i + offset) << (i * 8); + return result; + } + + public static int ArrayContains(byte[] b, byte[] seq, bool fromStart = true) + { + int track = 0; + for (int i = fromStart ? 0 : b.Length - 1; (fromStart && i < b.Length) || (!fromStart && i >= 0); i+=fromStart?1:-1) + if (b[i] == seq[fromStart?track:seq.Length - 1 - track]) + { + if (++track == seq.Length) return i; + } + else track = 0; + return -1; + } + + public static byte[] WriteToArray(byte[] target, int data, int offset) + { + for (int i = 0; i < 4; ++i) + target[i + offset] = (byte)((data >> (i * 8))&255); + return target; + } + + public static byte[] WriteContiguous(byte[] target, int offset, params int[] data) + { + for (int i = 0; i < data.Length; ++i) WriteToArray(target, data[i], offset + i * 4); + return target; + } + + public static byte[] WriteToArray(byte[] target, uint data, int offset) + { + for (int i = 0; i < 4; ++i) + target[i + offset] = (byte)((data >> (i * 8)) & 255); + return target; + } + + public static byte[] WriteContiguous(byte[] target, int offset, params uint[] data) + { + for (int i = 0; i < data.Length; ++i) WriteToArray(target, data[i], offset + i * 4); + return target; + } + + public static byte[] Concatenate(params byte[][] bytes) + { + int alloc = 0; + foreach (byte[] b in bytes) alloc += b.Length; + byte[] result = new byte[alloc]; + alloc = 0; + for(int i = 0; i(IEnumerable source, int sourceOffset, T[] destination, int offset, int length) + { + for (int i = 0; i < length; ++i) destination[i + offset] = source.ElementAt(i+sourceOffset); + } + + public static string ArrayToString(byte[] array) + { + StringBuilder builder = new StringBuilder().Append('['); + for (int i = 0; i < array.Length; ++i) + { + builder.Append(array[i]); + if (i != array.Length - 1) builder.Append(", "); + } + return builder.Append(']').ToString(); + } + + public static bool ArraysEqual(T[] t1, T[] t2) + { + if (t1 == t2) return true; + else if (t1 == null) return false; + else if (t1.Length != t2.Length) return false; + for (int i = 0; i < t1.Length; ++i) + if (!ObjectsEqual(t1[i], t2[i])) + return false; + return true; + } + + public static bool ObjectsEqual(object o1, object o2) => (o1 == null && o2 == null) || (o1 != null && o1.Equals(o2)); + + public static void EnqueueAll(this Queue q, IEnumerable items, int offset, int length) + { + for (int i = 0; i < length; ++i) q.Enqueue(items.ElementAt(i+offset)); + } + public static T[]Dequeue(this Queue q, int count) + { + T[] t = new T[count]; + for (int i = 0; i < count; ++i) t[i] = q.Dequeue(); + return t; + } + + public static byte[] SerializeBytes(byte[][] bytes) + { + int collectSize = 0; + for (int i = 0; i < bytes.Length; ++i) collectSize += bytes[i].Length; + byte[] output = new byte[collectSize + 4*bytes.Length]; + collectSize = 0; + for(int i = 0; i message.Length - offset - 4 || (i!=messageCount-1 && size==message.Length-offset-4)) + throw new IndexOutOfRangeException("Attempted to read more bytes than are available"); + offset += 4; + output[i] = new byte[size]; + Array.Copy(message, offset, output[i], 0, size); + offset += size; + } + return output; + } + + public static T[] SubArray(this T[] array, int start, int end) + { + T[] res = new T[end-start]; + for (int i = start; i < end; ++i) res[i - start] = array[i]; + return res; + } + + public static byte[] XOR(this byte[] array, byte[] xor) + { + for (int i = Math.Min(array.Length, xor.Length) - 1; i >= 0; --i) array[i] ^= xor[i]; + return array; + } + + public static string ToUTF8String(this byte[] b) => new string(Encoding.UTF8.GetChars(b)); + public static byte[] ToUTF8Bytes(this string s) => Encoding.UTF8.GetBytes(s); + + + // -- Misc -- + // Allows deconstruction when iterating over a collection of Tuples + public static void Deconstruct(this Tuple tuple, out T1 key, out T2 value) + { + key = tuple.Item1; + value = tuple.Item2; + } + public static XmlNode ContainsNamedNode(string name, XmlNodeList lst) + { + for (int i = lst.Count - 1; i >= 0; --i) + if (lst.Item(i).Name.Equals(name)) + return lst.Item(i); + return null; + } + public static bool IsNumber(this char c) => c > 47 && c < 58; + public static bool IsAlphabetical(this char c) => (c > 64 && c < 91) || (c > 96 && c < 123); + public static bool IsAlphaNumeric(this char c) => c.IsNumber() || c.IsAlphabetical(); + public static bool IsDecimal(this char c) => c == '.' || c.IsNumber(); + + // Swap endianness of a given integer + public static uint SwapEndian(uint value) => (uint)(((value >> 24) & (255 << 0)) | ((value >> 8) & (255 << 8)) | ((value << 8) & (255 << 16)) | ((value << 24) & (255 << 24))); + + public static string ToHexString(byte[] value) + { + StringBuilder builder = new StringBuilder(); + foreach(byte b in value) + { + builder.Append((char)((((b >> 4) < 10) ? 48 : 87) + (b >> 4))); + builder.Append((char)((((b & 15) < 10) ? 48 : 87) + (b & 15))); + } + return builder.ToString(); + } + + public static bool ReadYNBool(this TextReader reader, string nonDefault) => reader.ReadLine().ToLower().Equals(nonDefault); + + public static string SerializeStrings(string[] data) + { + StringBuilder builder = new StringBuilder(); + foreach (var datum in data) builder.Append(datum.Replace("&", "&").Replace("\n", "&nl;")).Append("&nm;"); + if (builder.Length > 0) builder.Remove(builder.Length - 4, 4); + return builder.ToString(); + } + + public static string[] DeserializeString(string message) + { + List collect = new List(); + const string target = "&nm;"; + int found = 0; + int prev = 0; + for(int i = 0; i= pastFirst) + return i; + return -1; + } + } + + + public abstract class RandomProvider + { + public abstract byte[] GetBytes(int count); + public abstract byte[] GetBytes(byte[] buffer); + + // Randomly generates a shortinteger bounded by the supplied integer. If bounding value is <= 0, it will be ignored + public ushort NextUShort(ushort bound = 0) + { + byte[] raw = GetBytes(2); + ushort result = 0; + for (byte s = 0; s < 2; ++s) + { + result <<= 8; + result |= raw[s]; + } + return (ushort)(bound > 0 ? result % bound : result); + } + + // Randomly generates an integer bounded by the supplied integer. If bounding value is <= 0, it will be ignored + public uint NextUInt(uint bound = 0) + { + byte[] raw = GetBytes(4); + uint result = 0; + for (byte s = 0; s < 4; ++s) + { + result <<= 8; + result |= raw[s]; + } + return bound > 0 ? result % bound : result; + } + + // Randomly generates a long integer bounded by the supplied integer. If bounding value is <= 0, it will be ignored + public ulong NextULong(ulong bound = 0) + { + byte[] raw = GetBytes(8); + ulong result = 0; + for (byte s = 0; s < 8; ++s) + { + result <<= 8; + result |= raw[s]; + } + return bound > 0 ? result % bound : result; + } + + public char NextChar(bool alphanumeric = false) + { + char c = (char) GetBytes(1)[0]; + if (alphanumeric) + { + c %= (char)62; + c += (char)(c < 10 ? 48 : c < 36 ? 55 : 61); + } + return c; + } + + public string NextString(int length) + { + byte[] b = GetBytes(length); + StringBuilder builder = new StringBuilder(length); + foreach(var b1 in b) + { + char c = (char)(b1%62); + builder.Append((char)(c+(c < 10 ? 48 : c < 36 ? 55 : 61))); + } + return builder.ToString(); + } + + public short NextShort(short bound = 0) => (short)NextUInt((ushort)bound); + public int NextInt(int bound = 0) => (int)NextUInt((uint)bound); + public long NextLong(long bound = 0) => (long)NextULong((ulong)bound); + } + + public sealed class RegularRandomProvider : RandomProvider + { + private Random rand; + public RegularRandomProvider(Random rand) { this.rand = rand; } + public RegularRandomProvider() : this(new Random(Environment.TickCount)) {} + + // Copy our random reference to the other provider: share a random object + public void share(RegularRandomProvider provider) => provider.rand = this.rand; + + public override byte[] GetBytes(int count) => GetBytes(new byte[count]); + + public override byte[] GetBytes(byte[] buffer) + { + rand.NextBytes(buffer); + return buffer; + } + } + + public sealed class CryptoRandomProvider : RandomProvider + { + private RNGCryptoServiceProvider rand; + public CryptoRandomProvider(RNGCryptoServiceProvider rand) { this.rand = rand; } + public CryptoRandomProvider() : this(new RNGCryptoServiceProvider()) { } + + // Copy our random reference to the other provider: share a random object + public void share(CryptoRandomProvider provider) => provider.rand = this.rand; + + public override byte[] GetBytes(int count) => GetBytes(new byte[count]); + + public override byte[] GetBytes(byte[] buffer) + { + rand.GetBytes(buffer); + return buffer; + } + } + + public sealed class DummyRandomProvider : RandomProvider + { + public override byte[] GetBytes(int count) => new byte[count]; + + public override byte[] GetBytes(byte[] buffer) + { + for (int i = 0; i < buffer.Length; ++i) buffer[i] = 0; + return buffer; + } + } + + public static class RandomSupport + { + public static BigInteger GenerateBoundedRandom(BigInteger max, RandomProvider provider) + { + byte[] b = max.ToByteArray(); + byte maxLast = b[b.Length - 1]; + provider.GetBytes(b); + if (maxLast != 0) b[b.Length - 1] %= maxLast; + b[b.Length - 1] |= 127; + return new BigInteger(b); + } + } +} diff --git a/Server/App.config b/Server/App.config new file mode 100644 index 0000000..731f6de --- /dev/null +++ b/Server/App.config @@ -0,0 +1,6 @@ + + + + + + \ No newline at end of file diff --git a/Server/Database.cs b/Server/Database.cs new file mode 100644 index 0000000..cf5ff6d --- /dev/null +++ b/Server/Database.cs @@ -0,0 +1,542 @@ +using System; +using System.Collections.Generic; +using System.IO; +using System.Linq; +using System.Text; +using System.Threading.Tasks; +using System.Xml; +using Tofvesson.Crypto; +using Tofvesson.Collections; + +namespace Server +{ + public sealed class Database + { + private static readonly RandomProvider random = new RegularRandomProvider(); + public string[] MasterEntry { get; } + public string DatabaseName { get; } + public bool LoadFull { get; set; } + + // Cached changes + private readonly List changeList = new List(); + private readonly List toRemove = new List(); + private readonly EvictionList loadedUsers = new EvictionList(40); + + + public Database(string dbName, string master, bool loadFullDB = false) + { + dbName += ".xml"; + MasterEntry = master.Split('/'); + if (!File.Exists(dbName)) + { + FileStream strm = File.Create(dbName); + byte[] b; + strm.Write(b = Encoding.UTF8.GetBytes($"\n"), 0, b.Length); + + // Generate root element for users + for (int i = 0; i < MasterEntry.Length; ++i) strm.Write(b = Encoding.UTF8.GetBytes($"<{MasterEntry[i]}>"), 0, b.Length); + for (int i = MasterEntry.Length - 1; i >= 0; --i) strm.Write(b = Encoding.UTF8.GetBytes($""), 0, b.Length); + strm.Close(); + } + DatabaseName = dbName; + LoadFull = loadFullDB; + } + + // Flush before deletion + ~Database() { Flush(false); } + + // UpdateUser is just another name for AddUser + public void UpdateUser(User entry) => AddUser(entry, true); + public void AddUser(User entry) => AddUser(entry, true); + private void AddUser(User entry, bool withFlush) + { + entry = ToEncoded(entry); + for (int i = 0; i < loadedUsers.Count; ++i) + if (entry.Equals(loadedUsers[i])) + loadedUsers[i] = entry; + + for (int i = toRemove.Count - 1; i >= 0; --i) + if (toRemove[i].Equals(entry.Name)) + toRemove.RemoveAt(i); + + for (int i = 0; i < changeList.Count; ++i) + if (changeList[i].Equals(entry.Name)) + return; + + changeList.Add(entry); + + if(withFlush) Flush(true); + } + + public void RemoveUser(User entry) => RemoveUser(entry, true); + private void RemoveUser(User entry, bool withFlush) + { + entry = ToEncoded(entry); + for (int i = 0; i < loadedUsers.Count; ++i) + if (entry.Equals(loadedUsers[i])) + loadedUsers.RemoveAt(i); + + for (int i = changeList.Count - 1; i >= 0; --i) + if (changeList[i].Equals(entry.Name)) + changeList.RemoveAt(i); + + for (int i = toRemove.Count - 1; i >= 0; --i) + if (toRemove[i].Equals(entry.Name)) + return; + + toRemove.Add(entry); + + if(withFlush) Flush(true); + } + + // Triggers a forceful flush + public void Flush() => Flush(false); + + // Permissive (cache-dependent) flush + private void Flush(bool optional) + { + if(!(optional || changeList.Count > 30 || toRemove.Count > 30)) return; // No need to flush + string temp = GenerateTempFileName("tmp_", ".xml"); + using(var writer = XmlWriter.Create(temp)) + { + using(var reader = XmlReader.Create(DatabaseName)) + { + int masterDepth = 0; + bool trigger = false, wn = false, recent = false; + while (wn || reader.Read()) + { + wn = false; + if (trigger) + { + foreach (var user in changeList) + WriteUser(writer, user); + + bool wroteNode = false; + while ((wroteNode || reader.Name.Equals("User") || reader.Read()) && reader.NodeType != XmlNodeType.EndElement) + { + wroteNode = false; + if (reader.Name.Equals("User")) + { + User u = User.Parse(ReadEntry(reader), this); + if (u != null) + { + bool shouldWrite = true; + foreach (var toChange in changeList) + if (toChange.Name.Equals(u.Name)) + { + shouldWrite = false; + break; + } + if (shouldWrite) + foreach (var remove in toRemove) + if (remove.Name.Equals(u.Name)) + { + shouldWrite = false; + break; + } + if (shouldWrite) WriteUser(writer, u); + } + } + else + { + wroteNode = true; + writer.WriteNode(reader, true); + } + } + trigger = false; + recent = true; + writer.WriteEndElement(); + toRemove.Clear(); + changeList.Clear(); + } + if (masterDepth != MasterEntry.Length && reader.Name.Equals(MasterEntry[masterDepth])) + { + trigger = reader.NodeType == XmlNodeType.Element && ++masterDepth == MasterEntry.Length; + reader.MoveToContent(); + writer.WriteStartElement(MasterEntry[masterDepth - 1]); + } + else if (masterDepth == MasterEntry.Length && recent) + { + if(masterDepth!=1) writer.WriteEndElement(); + recent = false; + } + else + { + wn = true; + writer.WriteNode(reader, true); + } + } + } + writer.Flush(); + } + + File.Delete(DatabaseName); + File.Move(temp, DatabaseName); + } + + private static void WriteUser(XmlWriter writer, User u) + { + writer.WriteStartElement("User"); + if (u.IsAdministrator) writer.WriteAttributeString("admin", "", "true"); + writer.WriteElementString("Name", u.Name); + writer.WriteElementString("Balance", u.Balance.ToString()); + writer.WriteElementString("Password", u.PasswordHash); + writer.WriteElementString("Salt", u.Salt); + foreach (var tx in u.History) + { + writer.WriteStartElement("Transaction"); + writer.WriteElementString(tx.to.Equals(u.Name) ? "From" : "To", tx.to.Equals(u.Name) ? tx.from : tx.to); + writer.WriteElementString("Balance", tx.amount.ToString()); + if (tx.meta != null && tx.meta.Length != 0) writer.WriteElementString("Meta", tx.meta); + writer.WriteEndElement(); + } + writer.WriteEndElement(); + } + + private static string GenerateTempFileName(string prefix, string suffix) + { + string s; + do s = prefix + random.NextString((Math.Abs(random.NextInt())%16)+8) + suffix; + while (File.Exists(s)); + return s; + } + + private Entry ReadEntry(XmlReader reader) + { + Entry e = new Entry(reader.Name); + if (reader.HasAttributes) + { + reader.MoveToAttribute(0); + do e.Attributes.Add(reader.Name, reader.Value); + while (reader.MoveToNextAttribute()); + } + reader.MoveToContent(); + while (reader.Read() && reader.NodeType != XmlNodeType.EndElement) + { + if (reader.NodeType == XmlNodeType.Element) + using (var subRead = reader.ReadSubtree()) + { + SkipSpaces(subRead); + e.NestedEntries.Add(ReadEntry(subRead)); + } + else if (reader.NodeType == XmlNodeType.Text) e.Text = reader.Value; + } + reader.Read(); + return e; + } + + public User GetUser(string name) => FirstUser(u => u.Name.Equals(name)); + public User FirstUser(Predicate p) + { + User u; + foreach (var entry in loadedUsers) + if (p(u=FromEncoded(entry))) + return u; + + foreach (var entry in changeList) + if (p(u=FromEncoded(entry))) + { + if (!loadedUsers.Contains(entry)) loadedUsers.Add(u); + return u; + } + + using (var reader = XmlReader.Create(DatabaseName)) + { + if (!Traverse(reader, MasterEntry)) return null; + while (reader.Read() && reader.NodeType != XmlNodeType.EndElement) + { + if (reader.Name.Equals("User")) + { + User n = User.Parse(ReadEntry(reader), this); + if (n != null && p(n=FromEncoded(n))) + { + if (!loadedUsers.Contains(n)) loadedUsers.Add(n); + return n; + } + } + } + } + return null; + } + + public bool AddTransaction(string sender, string recipient, long amount, string message = null) + { + User from = FirstUser(u => u.Name.Equals(sender)); + User to = FirstUser(u => u.Name.Equals(recipient)); + + if (to == null || (from == null && !to.IsAdministrator)) return false; + + Transaction tx = new Transaction(from == null ? "System" : from.Name, to.Name, amount, message); + to.History.Add(tx); + AddUser(to); + if (from != null) + { + from.History.Add(tx); + AddUser(from); + } + return true; + } + + public User[] Users(Predicate p) + { + List l = new List(); + User u; + foreach (var entry in changeList) + if (p(u=FromEncoded(entry))) + l.Add(entry); + + using (var reader = XmlReader.Create(DatabaseName)) + { + if (!Traverse(reader, MasterEntry)) return null; + + while (SkipSpaces(reader) && reader.NodeType != XmlNodeType.EndElement) + { + if (reader.NodeType == XmlNodeType.EndElement) break; + User e = User.Parse(ReadEntry(reader), this); + if (e!=null && p(e=FromEncoded(e))) l.Add(e); + } + } + return l.ToArray(); + } + + public bool ContainsUser(string user) => FirstUser(u => u.Name.Equals(user)) != null; + public bool ContainsUser(User user) => FirstUser(u => u.Name.Equals(user.Name)) != null; + + private bool Traverse(XmlReader reader, params string[] downTo) + { + for(int i = 0; i Convert.ToBase64String(Encoding.UTF8.GetBytes(s)); + private static string Decode(string s) => Convert.FromBase64String(s).ToUTF8String(); + + internal class Entry + { + public string Text { get; set; } + public string Name { get; set; } + public Dictionary Attributes { get; } // Transient properties in comparison + public List NestedEntries { get; } // Semi-transient comparison properties + + public Entry(string name, string text = "") + { + Name = name; + Text = text; + Attributes = new Dictionary(); + NestedEntries = new List(); + } + public Entry GetNestedEntry(Predicate p) + { + foreach (var entry in NestedEntries) + if (p(entry)) + return entry; + return null; + } + + public bool BoolAttribute(string attrName, bool def = false, bool ignoreCase = true) + => Attributes.ContainsKey(attrName) && bool.TryParse(ignoreCase ? Attributes[attrName].ToLower() :Attributes[attrName], out bool b) ? b : def; + + public Entry AddNested(Entry e) + { + NestedEntries.Add(e); + return this; + } + + public Entry AddAttribute(string key, string value) + { + Attributes[key] = value; + return this; + } + + public override bool Equals(object obj) + { + if (!(obj is Entry)) return false; + Entry cmp = (Entry)obj; + if (cmp.Attributes.Count != Attributes.Count || cmp.NestedEntries.Count != NestedEntries.Count || !Text.Equals(cmp.Text) || !Name.Equals(cmp.Name)) return false; + foreach(var entry in NestedEntries) + { + if (entry.BoolAttribute("omit")) goto Next; + foreach (var cmpEntry in cmp.NestedEntries) + if (cmpEntry.BoolAttribute("omit")) continue; + else if (cmpEntry.Equals(entry)) goto Next; + return false; + Next: { } + } + + return true; + } + } + + public class User + { + public bool ProblematicTransactions { get; internal set; } + public string Name { get; internal set; } + public long Balance { get; set; } + public bool IsAdministrator { get; set; } + public string PasswordHash { get; internal set; } + public string Salt { get; internal set; } + public List History { get; } + private User() + { + Name = ""; + History = new List(); + } + + public User(User copy) : this() + { + this.ProblematicTransactions = copy.ProblematicTransactions; + this.Name = copy.Name; + this.Balance = copy.Balance; + this.IsAdministrator = copy.IsAdministrator; + this.PasswordHash = copy.PasswordHash; + this.Salt = copy.Salt; + this.History.AddRange(copy.History); + } + + public User(string name, string passHash, string salt, long balance, bool generatePass = false, List transactionHistory = null, bool admin = false) + : this(name, passHash, Encoding.UTF8.GetBytes(salt), balance, generatePass, transactionHistory, admin) + { } + + public User(string name, string passHash, byte[] salt, long balance, bool generatePass = false, List transactionHistory = null, bool admin = false) + { + History = transactionHistory ?? new List(); + Balance = balance; + Name = name; + IsAdministrator = admin; + Salt = Convert.ToBase64String(salt); + PasswordHash = generatePass ? Convert.ToBase64String(KDF.PBKDF2(KDF.HMAC_SHA1, Encoding.UTF8.GetBytes(passHash), Encoding.UTF8.GetBytes(Salt), 8192, 320)) : passHash; + } + + public bool Authenticate(string password) + => Convert.ToBase64String(KDF.PBKDF2(KDF.HMAC_SHA1, Encoding.UTF8.GetBytes(password), Encoding.UTF8.GetBytes(Salt), 8192, 320)).Equals(PasswordHash); + + public User AddTransaction(Transaction tx) + { + History.Add(tx); + return this; + } + + private Entry Serialize() + { + Entry root = new Entry("User") + .AddNested(new Entry("Name", Name)) + .AddNested(new Entry("Balance", Balance.ToString()).AddAttribute("omit", "true")); + foreach (var transaction in History) + { + Entry tx = + new Entry("Transaction") + .AddAttribute("omit", "true") + .AddNested(new Entry(transaction.to.Equals(Name) ? "From" : "To", transaction.to.Equals(Name) ? transaction.from : transaction.to)) + .AddNested(new Entry("Balance", transaction.amount.ToString())); + if (transaction.meta != null) tx.AddNested(new Entry("Meta", transaction.meta)); + root.AddNested(tx); + } + return root; + } + + internal static User Parse(Entry e, Database db) + { + if (!e.Name.Equals("User")) return null; + User user = new User(); + foreach (var entry in e.NestedEntries) + { + if (entry.Name.Equals("Name")) user.Name = entry.Text; + else if (entry.Name.Equals("Balance")) user.Balance = long.TryParse(entry.Text, out long l) ? l : 0; + else if (entry.Name.Equals("Transaction")) + { + string from = null; + string to = null; + long amount = -1; + string meta = ""; + foreach (var e1 in entry.NestedEntries) + { + if (e1.Name.Equals("To")) to = e1.Text; + else if (e1.Name.Equals("From")) from = e1.Text; + else if (e1.Name.Equals("Balance")) amount = long.TryParse(e1.Text, out amount) ? amount : 0; + else if (e1.Name.Equals("Meta")) meta = e1.Text; + } + if ((from == null && to == null) || (from != null && to != null) || amount <= 0) user.ProblematicTransactions = true; + else user.History.Add(new Transaction(from, to, amount, meta)); + } + else if (entry.Name.Equals("Password")) user.PasswordHash = entry.Text; + else if (entry.Name.Equals("Salt")) user.Salt = entry.Text; + } + if (user.Name == null || user.Name.Length == 0 || user.PasswordHash == null || user.Salt == null || user.PasswordHash.Length==0 || user.Salt.Length==0) return null; + if (user.Balance < 0) user.Balance = 0; + + // Populate transaction names + foreach (var transaction in user.History) + if (transaction.from == null) transaction.from = user.Name; + else if (transaction.to == null) transaction.to = user.Name; + + return user; + } + + public Transaction CreateTransaction(User recipient, long amount, string message = null) => new Transaction(this.Name, recipient.Name, amount, message); + + public override bool Equals(object obj) => obj is User && ((User)obj).Name.Equals(Name); + + public override int GetHashCode() + { + return 539060726 + EqualityComparer.Default.GetHashCode(Name); + } + } + + public class Transaction + { + public string from; + public string to; + public long amount; + public string meta; + + public Transaction(string from, string to, long amount, string meta) + { + this.from = from; + this.to = to; + this.amount = amount; + this.meta = meta; + } + + public User GetTxToUser(Database db) => db.FirstUser(u => u.Name.Equals(to)); + public User GetTxFromUser(Database db) => db.FirstUser(u => u.Name.Equals(from)); + } + } +} diff --git a/Server/Program.cs b/Server/Program.cs new file mode 100644 index 0000000..a511251 --- /dev/null +++ b/Server/Program.cs @@ -0,0 +1,122 @@ +using Common; +using Server.Properties; +using System; +using System.Collections.Generic; +using System.Linq; +using System.Text; +using System.Threading.Tasks; +using Tofvesson.Crypto; + +namespace Server +{ + class Program + { + static void Main(string[] args) + { + Console.SetError(new TimeStampWriter(Console.Error, "HH:mm:ss.fff")); + Console.SetOut(new TimeStampWriter(Console.Out, "HH:mm:ss.fff")); + + SessionManager manager = new SessionManager(120 * TimeSpan.TicksPerSecond, 20); + + Database db = new Database("BankDB", "Resources"); + + //Database.User me = db.GetUser("Gabriel Tofvesson");//new Database.User("Gabriel Tofvesson", "Hello, World", "NoRainbow", 1337, true, null, true); + + + CryptoRandomProvider random = new CryptoRandomProvider(); + RSA rsa = null;// new RSA(Resources.e_0x200, Resources.n_0x200, Resources.d_0x200); + if (rsa == null) + { + Console.ForegroundColor = ConsoleColor.Red; + Console.Error.WriteLine("No RSA keys available! Server identity will not be verifiable!"); + Console.ForegroundColor = ConsoleColor.Gray; + Console.WriteLine("Generating session-specific RSA-keys..."); + rsa = new RSA(64, 8, 8, 5); + Console.WriteLine("Done!"); + } + + NetServer server = new NetServer( + rsa, + 80, + (string r, Dictionary associations, ref bool s) => + { + string[] cmd = ParseCommand(r, out long id); + + // Perform a signature verification by signing a nonce + switch (cmd[0]) + { + case "Auth": + { + int idx = cmd[1].IndexOf(':'); + if (idx == -1) return GenerateResponse(id, "ERROR"); + string user = cmd[1].Substring(0, idx); + string pass = cmd[1].Substring(idx + 1); + Database.User usr = db.GetUser(user); + if (usr == null || !usr.Authenticate(pass)) + { + Console.WriteLine("Authentcation failure for user: "+user); + return GenerateResponse(id, "ERROR"); + } + + string sess = manager.GetSession(usr, "ERROR"); + Console.WriteLine("Authentication success for user: "+user+"\nSession: "+sess); + associations["session"] = sess; + return GenerateResponse(id, sess); + } + case "Logout": + manager.Expire(cmd[1]); + Console.WriteLine("Prematurely expired session: "+cmd[1]); + break; + case "Reg": + { + int idx = cmd[1].IndexOf(':'); + if (idx == -1) return GenerateResponse(id, "ERROR"); + string user = cmd[1].Substring(0, idx); + string pass = cmd[1].Substring(idx + 1); + if (db.ContainsUser(user)) return GenerateResponse(id, "ERROR"); + Database.User u = new Database.User(user, pass, random.GetBytes(Math.Abs(random.NextShort() % 60) + 20), 0, true); + db.AddUser(u); + string sess = manager.GetSession(u, "ERROR"); + Console.WriteLine("Registered account: " + u.Name + "\nSession: "+sess); + associations["session"] = sess; + return GenerateResponse(id, sess); + } + default: + return GenerateResponse(id, "ERROR"); + } + + return null; + }, + (c, b) => + { + Console.WriteLine($"Client has {(b ? "C" : "Disc")}onnected"); + //if(!b && c.assignedValues.ContainsKey("session")) + // manager.Expire(c.assignedValues["session"]); + }); + server.StartListening(); + + Console.ReadLine(); + + server.StopRunning(); + } + + private static string[] ParseCommand(string cmd, out long id) + { + int idx = cmd.IndexOf(':'), idx1; + string sub; + if (idx == -1 || !(sub = cmd.Substring(idx + 1)).Contains(':') || !long.TryParse(sub.Substring(0, idx1 = sub.IndexOf(':')), out id)) + { + id = 0; + return null; + } + return new string[] { cmd.Substring(0, idx), sub.Substring(idx1 + 1) }; + } + + private static string GenerateResponse(long id, bool b) => GenerateResponse(id, b.ToString()); + private static string GenerateResponse(long id, int b) => GenerateResponse(id, b.ToString()); + private static string GenerateResponse(long id, long b) => GenerateResponse(id, b.ToString()); + private static string GenerateResponse(long id, float b) => GenerateResponse(id, b.ToString()); + private static string GenerateResponse(long id, double b) => GenerateResponse(id, b.ToString()); + private static string GenerateResponse(long id, string response) => id + ":" + response; + } +} diff --git a/Server/Properties/AssemblyInfo.cs b/Server/Properties/AssemblyInfo.cs new file mode 100644 index 0000000..71ce435 --- /dev/null +++ b/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("Server")] +[assembly: AssemblyDescription("")] +[assembly: AssemblyConfiguration("")] +[assembly: AssemblyCompany("")] +[assembly: AssemblyProduct("Server")] +[assembly: AssemblyCopyright("Copyright © 2018")] +[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("b458552a-5884-4b27-ba6b-826bc5590106")] + +// 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/Server/Properties/Resources.Designer.cs b/Server/Properties/Resources.Designer.cs new file mode 100644 index 0000000..2f2759b --- /dev/null +++ b/Server/Properties/Resources.Designer.cs @@ -0,0 +1,93 @@ +//------------------------------------------------------------------------------ +// +// 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 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", "15.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("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 resource of type System.Byte[]. + /// + internal static byte[] d_0x200 { + get { + object obj = ResourceManager.GetObject("d_0x200", resourceCulture); + return ((byte[])(obj)); + } + } + + /// + /// Looks up a localized resource of type System.Byte[]. + /// + internal static byte[] e_0x200 { + get { + object obj = ResourceManager.GetObject("e_0x200", resourceCulture); + return ((byte[])(obj)); + } + } + + /// + /// Looks up a localized resource of type System.Byte[]. + /// + internal static byte[] n_0x200 { + get { + object obj = ResourceManager.GetObject("n_0x200", resourceCulture); + return ((byte[])(obj)); + } + } + } +} diff --git a/Server/Properties/Resources.resx b/Server/Properties/Resources.resx new file mode 100644 index 0000000..3882dfc --- /dev/null +++ b/Server/Properties/Resources.resx @@ -0,0 +1,130 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + 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\0x200.d;System.Byte[], mscorlib, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089 + + + ..\Resources\0x200.e;System.Byte[], mscorlib, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089 + + + ..\Resources\0x200.n;System.Byte[], mscorlib, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089 + + \ No newline at end of file diff --git a/Server/Resources/0x200.e b/Server/Resources/0x200.e new file mode 100644 index 0000000..c4ae053 Binary files /dev/null and b/Server/Resources/0x200.e differ diff --git a/Server/Resources/0x200.n b/Server/Resources/0x200.n new file mode 100644 index 0000000..1ae6fe8 Binary files /dev/null and b/Server/Resources/0x200.n differ diff --git a/Server/Server.csproj b/Server/Server.csproj new file mode 100644 index 0000000..3fbdbe6 --- /dev/null +++ b/Server/Server.csproj @@ -0,0 +1,74 @@ + + + + + Debug + AnyCPU + {B458552A-5884-4B27-BA6B-826BC5590106} + Exe + Server + Server + v4.6.1 + 512 + true + + + AnyCPU + true + full + false + bin\Debug\ + DEBUG;TRACE + prompt + 4 + + + AnyCPU + pdbonly + true + bin\Release\ + TRACE + prompt + 4 + + + + + + + + + + + + + + + + + True + True + Resources.resx + + + + + + + + + + + + {23eb87d4-e310-48c4-a931-0961c83892d7} + Common + + + + + ResXFileCodeGenerator + Resources.Designer.cs + + + + \ No newline at end of file diff --git a/Server/SessionManager.cs b/Server/SessionManager.cs new file mode 100644 index 0000000..d71247e --- /dev/null +++ b/Server/SessionManager.cs @@ -0,0 +1,120 @@ +using System; +using System.Collections.Generic; +using Tofvesson.Crypto; + +namespace Server +{ + public sealed class SessionManager + { + private static readonly RandomProvider random = new RegularRandomProvider(); + private readonly List sessions = new List(); + private readonly long timeout; + private readonly int sidLength; + + public SessionManager(long timeout, int sidLength = 10) + { + this.timeout = timeout; + this.sidLength = sidLength < 10 ? 10 : sidLength; + } + + public string GetSession(Database.User user, string invalidSID) + { + Update(); + for (int i = 0; i < sessions.Count; ++i) + if (sessions[i].user.Equals(user)) + return sessions[i].sessionID; + + Session s = new Session + { + sessionID = GenerateRandomSID(invalidSID), + user = user, + expiry = DateTime.Now.Ticks + timeout + }; + sessions.Add(s); + return s.sessionID; + } + + public bool Refresh(Database.User user) + { + Update(); + for (int i = sessions.Count - 1; i >= 0; --i) + if (sessions[i].user.Equals(user)) + { + Session s = sessions[i]; + s.expiry = DateTime.Now.Ticks + timeout; + sessions[i] = s; + return true; + } + return false; + } + + public void Expire(Database.User user) + { + Update(); + for (int i = sessions.Count - 1; i >= 0; --i) + if (sessions[i].user.Equals(user)) + { + sessions.RemoveAt(i); + return; + } + return; + } + + public bool Refresh(string sid) + { + Update(); + for (int i = sessions.Count - 1; i >= 0; --i) + if (sessions[i].sessionID.Equals(sid)) + { + Session s = sessions[i]; + s.expiry = DateTime.Now.Ticks + timeout; + sessions[i] = s; + return true; + } + return false; + } + + public void Expire(string sid) + { + Update(); + for (int i = sessions.Count - 1; i >= 0; --i) + if (sessions[i].sessionID.Equals(sid)) + { + sessions.RemoveAt(i); + return; + } + return; + } + + public bool CheckSession(string sid, Database.User user) + { + foreach (var session in sessions) + if (session.sessionID.Equals(sid) && session.user.Equals(user)) + return true; + + return false; + } + + public void Update() + { + for(int i = sessions.Count - 1; i>=0; --i) + if (sessions[i].expiry < DateTime.Now.Ticks) + sessions.RemoveAt(i); + } + + private string GenerateRandomSID(string invalid) + { + string res; + do res = random.NextString(sidLength); + while (res.Equals(invalid)); + return res; + } + } + + public struct Session + { + public string sessionID; + public Database.User user; + public long expiry; // Measured in ticks + } +}