Major changes
Refactorings: * BinaryCollector -> BitWriter * BinaryDistributor -> BitReader Additions: * Output class for making serverside output pretty and more readable * Better RSA keys (private keys withheld) Changes: * Minor changes to all views and their rendering * Added corrective resizing to resize listener to prevent errant window sizes * Removed "default" language in favour of a purely priority-based system * NetContext now attempts to verify server identity before continuing to next context * Simplified common operations in Context * Minor updates to some layouts * Completed translations for english and swedish * Promise system now supports internal processing before notifying original caller * Bank interactor methods are now async * Added support for multiple accounts per user (separate repositories for money) * Removed test code from client program * Updated Database to support multiple accounts * Reimplemented RSA on the server side purely as an identity verification system on top of the networking layer (rather than part of the layer) * Added Account management endpoints * Added full support for System-sourced transactions * Added Account availability endpoint * Added verbose error responses
This commit is contained in:
parent
308639da5f
commit
100f5a32be
3
.gitignore
vendored
3
.gitignore
vendored
@ -12,3 +12,6 @@
|
||||
/Server/Resources/0x200.d
|
||||
/Server/bin
|
||||
/Server/obj
|
||||
/Server/Resources/0x100.d
|
||||
/Server/Resources/0x100.e
|
||||
/Server/Resources/0x100.n
|
||||
|
3
Bank.sln
3
Bank.sln
@ -10,6 +10,9 @@ EndProject
|
||||
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Common", "Common\Common.csproj", "{23EB87D4-E310-48C4-A931-0961C83892D7}"
|
||||
EndProject
|
||||
Global
|
||||
GlobalSection(Performance) = preSolution
|
||||
HasPerformanceSessions = true
|
||||
EndGlobalSection
|
||||
GlobalSection(SolutionConfigurationPlatforms) = preSolution
|
||||
Debug|Any CPU = Debug|Any CPU
|
||||
Release|Any CPU = Release|Any CPU
|
||||
|
114
Client.psess
Normal file
114
Client.psess
Normal file
@ -0,0 +1,114 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<VSPerformanceSession Version="1.00">
|
||||
<Options>
|
||||
<Solution>Bank.sln</Solution>
|
||||
<CollectionMethod>Sampling</CollectionMethod>
|
||||
<AllocationMethod>None</AllocationMethod>
|
||||
<AddReport>true</AddReport>
|
||||
<ResourceBasedAnalysisSelected>true</ResourceBasedAnalysisSelected>
|
||||
<UniqueReport>Timestamp</UniqueReport>
|
||||
<SamplingMethod>Cycles</SamplingMethod>
|
||||
<CycleCount>10000000</CycleCount>
|
||||
<PageFaultCount>10</PageFaultCount>
|
||||
<SysCallCount>10</SysCallCount>
|
||||
<SamplingCounter Name="" ReloadValue="00000000000f4240" DisplayName="" />
|
||||
<RelocateBinaries>false</RelocateBinaries>
|
||||
<HardwareCounters EnableHWCounters="false" />
|
||||
<EtwSettings />
|
||||
<PdhSettings>
|
||||
<PdhCountersEnabled>false</PdhCountersEnabled>
|
||||
<PdhCountersRate>500</PdhCountersRate>
|
||||
<PdhCounters>
|
||||
<PdhCounter>\Memory\Pages/sec</PdhCounter>
|
||||
<PdhCounter>\PhysicalDisk(_Total)\Avg. Disk Queue Length</PdhCounter>
|
||||
<PdhCounter>\Processor(_Total)\% Processor Time</PdhCounter>
|
||||
</PdhCounters>
|
||||
</PdhSettings>
|
||||
</Options>
|
||||
<ExcludeSmallFuncs>true</ExcludeSmallFuncs>
|
||||
<InteractionProfilingEnabled>false</InteractionProfilingEnabled>
|
||||
<JScriptProfilingEnabled>false</JScriptProfilingEnabled>
|
||||
<PreinstrumentEvent>
|
||||
<InstrEventExclude>false</InstrEventExclude>
|
||||
</PreinstrumentEvent>
|
||||
<PostinstrumentEvent>
|
||||
<InstrEventExclude>false</InstrEventExclude>
|
||||
</PostinstrumentEvent>
|
||||
<Binaries>
|
||||
<ProjBinary>
|
||||
<Path>Client\obj\Release\Client.exe</Path>
|
||||
<ArgumentTimestamp>01/01/0001 00:00:00</ArgumentTimestamp>
|
||||
<Instrument>true</Instrument>
|
||||
<Sample>true</Sample>
|
||||
<ExternalWebsite>false</ExternalWebsite>
|
||||
<InteractionProfilingEnabled>false</InteractionProfilingEnabled>
|
||||
<IsLocalJavascript>false</IsLocalJavascript>
|
||||
<IsWindowsStoreApp>false</IsWindowsStoreApp>
|
||||
<IsWWA>false</IsWWA>
|
||||
<LaunchProject>true</LaunchProject>
|
||||
<OverrideProjectSettings>false</OverrideProjectSettings>
|
||||
<LaunchMethod>Executable</LaunchMethod>
|
||||
<ExecutablePath>Client\bin\Release\Client.exe</ExecutablePath>
|
||||
<StartupDirectory>Client\bin\Release\</StartupDirectory>
|
||||
<Arguments>
|
||||
</Arguments>
|
||||
<NetAppHost>IIS</NetAppHost>
|
||||
<NetBrowser>InternetExplorer</NetBrowser>
|
||||
<ExcludeSmallFuncs>true</ExcludeSmallFuncs>
|
||||
<JScriptProfilingEnabled>false</JScriptProfilingEnabled>
|
||||
<PreinstrumentEvent>
|
||||
<InstrEventExclude>false</InstrEventExclude>
|
||||
</PreinstrumentEvent>
|
||||
<PostinstrumentEvent>
|
||||
<InstrEventExclude>false</InstrEventExclude>
|
||||
</PostinstrumentEvent>
|
||||
<ProjRef>{2236D5D4-7816-4630-8C86-0F0BDD46D7D8}|Client\Client.csproj</ProjRef>
|
||||
<ProjPath>Client\Client.csproj</ProjPath>
|
||||
<ProjName>Client</ProjName>
|
||||
</ProjBinary>
|
||||
<ProjBinary>
|
||||
<Path>Server\obj\Release\Server.exe</Path>
|
||||
<ArgumentTimestamp>01/01/0001 00:00:00</ArgumentTimestamp>
|
||||
<Instrument>true</Instrument>
|
||||
<Sample>true</Sample>
|
||||
<ExternalWebsite>false</ExternalWebsite>
|
||||
<InteractionProfilingEnabled>false</InteractionProfilingEnabled>
|
||||
<IsLocalJavascript>false</IsLocalJavascript>
|
||||
<IsWindowsStoreApp>false</IsWindowsStoreApp>
|
||||
<IsWWA>false</IsWWA>
|
||||
<LaunchProject>true</LaunchProject>
|
||||
<OverrideProjectSettings>false</OverrideProjectSettings>
|
||||
<LaunchMethod>Executable</LaunchMethod>
|
||||
<ExecutablePath>Server\bin\Release\Server.exe</ExecutablePath>
|
||||
<StartupDirectory>Server\bin\Release\</StartupDirectory>
|
||||
<Arguments>
|
||||
</Arguments>
|
||||
<NetAppHost>IIS</NetAppHost>
|
||||
<NetBrowser>InternetExplorer</NetBrowser>
|
||||
<ExcludeSmallFuncs>true</ExcludeSmallFuncs>
|
||||
<JScriptProfilingEnabled>false</JScriptProfilingEnabled>
|
||||
<PreinstrumentEvent>
|
||||
<InstrEventExclude>false</InstrEventExclude>
|
||||
</PreinstrumentEvent>
|
||||
<PostinstrumentEvent>
|
||||
<InstrEventExclude>false</InstrEventExclude>
|
||||
</PostinstrumentEvent>
|
||||
<ProjRef>{B458552A-5884-4B27-BA6B-826BC5590106}|Server\Server.csproj</ProjRef>
|
||||
<ProjPath>Server\Server.csproj</ProjPath>
|
||||
<ProjName>Server</ProjName>
|
||||
</ProjBinary>
|
||||
</Binaries>
|
||||
<Reports>
|
||||
<Report>
|
||||
<Path>Client180412.vspx</Path>
|
||||
</Report>
|
||||
</Reports>
|
||||
<Launches>
|
||||
<ProjBinary>
|
||||
<Path>:PB:{2236D5D4-7816-4630-8C86-0F0BDD46D7D8}|Client\Client.csproj</Path>
|
||||
</ProjBinary>
|
||||
<ProjBinary>
|
||||
<Path>:PB:{B458552A-5884-4B27-BA6B-826BC5590106}|Server\Server.csproj</Path>
|
||||
</ProjBinary>
|
||||
</Launches>
|
||||
</VSPerformanceSession>
|
@ -80,6 +80,8 @@
|
||||
</ItemGroup>
|
||||
<ItemGroup>
|
||||
<None Include="App.config" />
|
||||
<None Include="Resources\0x100.e" />
|
||||
<None Include="Resources\0x100.n" />
|
||||
<None Include="Resources\0x200.e" />
|
||||
<None Include="Resources\0x200.n" />
|
||||
</ItemGroup>
|
||||
|
@ -49,6 +49,12 @@ namespace Client.ConsoleForms
|
||||
if (resizeListener) EnableResizeListener();
|
||||
RegisterListener((w, h) =>
|
||||
{
|
||||
// Corrective resizing to prevent rendering issues
|
||||
if (w < 20 || h < 20)
|
||||
{
|
||||
Console.SetWindowSize(Math.Max(w, 60), Math.Max(h, 40));
|
||||
return;
|
||||
}
|
||||
width = w;
|
||||
height = h;
|
||||
Draw();
|
||||
@ -56,6 +62,9 @@ namespace Client.ConsoleForms
|
||||
|
||||
RegisterListener((w1, h1, w2, h2) =>
|
||||
{
|
||||
// Corrective resizing to prevent rendering issues
|
||||
if (w2 < 20 || h2 < 20)
|
||||
Console.SetWindowSize(Math.Max(w2, 60), Math.Max(h2, 40));
|
||||
Console.BackgroundColor = ConsoleColor.Black;
|
||||
Console.Clear();
|
||||
});
|
||||
|
@ -53,5 +53,16 @@ namespace Client.ConsoleForms
|
||||
((DialogView)v).RegisterSelectListener(listener);
|
||||
}
|
||||
}
|
||||
protected void Show(View v) => controller.AddView(v);
|
||||
protected void Show(string viewID) => controller.AddView(views.GetNamed(viewID));
|
||||
protected T GetView<T>(string viewID) where T : View => (T) views.GetNamed(viewID);
|
||||
protected View GetView(string viewID) => views.GetNamed(viewID);
|
||||
protected void Hide(string viewID) => controller.CloseView(views.GetNamed(viewID));
|
||||
protected void Hide(View v) => controller.CloseView(v);
|
||||
protected void HideAll()
|
||||
{
|
||||
foreach (var viewEntry in views)
|
||||
Hide(viewEntry.Item2);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -21,7 +21,7 @@ namespace Client.ConsoleForms.Graphics
|
||||
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 override Region Occlusion => new Region(new Rectangle(-1, -1, ContentWidth + 4, ContentHeight + 2));
|
||||
|
||||
public ConsoleColor SelectColor { get; set; }
|
||||
public ConsoleColor NotSelectColor { get; set; }
|
||||
@ -33,7 +33,7 @@ namespace Client.ConsoleForms.Graphics
|
||||
base(parameters.SetAttribute("width",
|
||||
Math.Max(
|
||||
parameters.AttribueAsInt("width") < 1 ? parameters.NestedText("Text").Length : parameters.AttribueAsInt("width"),
|
||||
ComputeLength(parameters.Get("Options").CollectSub("Option"))
|
||||
ComputeLength(parameters.Get("Options")?.CollectSub("Option") ?? new Tuple<string, string>[0])
|
||||
)), lang)
|
||||
{
|
||||
ViewData optionsData = parameters.Get("Options");
|
||||
|
@ -157,7 +157,7 @@ namespace Client.ConsoleForms.Graphics
|
||||
computedSize += splitInputs[i].Length;
|
||||
}
|
||||
ContentHeight += computedSize + Inputs.Length * 2;
|
||||
++ContentWidth; // Idk, it works, though...
|
||||
//++ContentWidth; // Idk, it works, though...
|
||||
}
|
||||
|
||||
protected override void _Draw(int left, ref int top)
|
||||
|
@ -13,7 +13,8 @@ namespace Client.ConsoleForms.Graphics
|
||||
public ConsoleColor SelectBackground { get; set; }
|
||||
public ConsoleColor SelectText { get; set; }
|
||||
|
||||
public override Region Occlusion => new Region(new Rectangle(0, 0, ContentWidth, ContentHeight));
|
||||
|
||||
public override Region Occlusion => new Region(new Rectangle(-padding.Left(), -padding.Top(), ContentWidth + padding.Right(), ContentHeight + padding.Bottom()));
|
||||
|
||||
public ListView(ViewData parameters, LangManager lang) : base(parameters, lang)
|
||||
{
|
||||
@ -30,6 +31,7 @@ namespace Client.ConsoleForms.Graphics
|
||||
if (limited && view.AttribueAsInt("width") > maxWidth) view.attributes["width"] = maxWidth.ToString();
|
||||
|
||||
Tuple<string, View> v = ConsoleController.LoadView(parameters.attributes["xmlns"], view, I18n); // Load the view in with standard namespace
|
||||
v.Item2.DrawBorder = false;
|
||||
innerViews.Add(v);
|
||||
|
||||
if (!limited) maxWidth = Math.Max(v.Item2.ContentWidth, maxWidth);
|
||||
|
@ -36,7 +36,7 @@ namespace Client.ConsoleForms.Graphics
|
||||
Dirty = true;
|
||||
}
|
||||
}
|
||||
public override Region Occlusion => new Region(new Rectangle(0, DrawBorder ? -1 : 0, ContentWidth + (DrawBorder ? 2 : 0), ContentHeight));
|
||||
public override Region Occlusion => new Region(new Rectangle(DrawBorder ? -1 : 0, DrawBorder ? -1 : 0, ContentWidth + (DrawBorder ? 3 : 0), ContentHeight));
|
||||
|
||||
//public char Border { get; set; }
|
||||
//public ConsoleColor BorderColor { get; set; }
|
||||
|
@ -39,7 +39,7 @@ namespace Client.ConsoleForms.Graphics
|
||||
BackgroundColor = (ConsoleColor)parameters.AttribueAsInt("color_background", (int)ConsoleColor.White);
|
||||
TextColor = (ConsoleColor)parameters.AttribueAsInt("color_text", (int)ConsoleColor.Black);
|
||||
Border = ' ';
|
||||
DrawBorder = parameters.attributes.ContainsKey("border");
|
||||
DrawBorder = true;// parameters.attributes.ContainsKey("border");
|
||||
I18n = lang;
|
||||
|
||||
back_data = parameters.GetAttribute("back");
|
||||
@ -68,17 +68,17 @@ namespace Client.ConsoleForms.Graphics
|
||||
public virtual void _DrawBorder(int left, int top)
|
||||
{
|
||||
Console.BackgroundColor = BorderColor;
|
||||
Console.SetCursorPosition(left, top - 1);
|
||||
Console.Write(Filler(Border, ContentWidth + 1));
|
||||
Console.SetCursorPosition(left - 1, top - 1);
|
||||
Console.Write(Filler(Border, ContentWidth + 2));
|
||||
for (int i = -1; i < ContentHeight; ++i)
|
||||
{
|
||||
Console.SetCursorPosition(left, top + i);
|
||||
Console.SetCursorPosition(left-1, top + i);
|
||||
Console.Write(Filler(Border, 2));
|
||||
Console.SetCursorPosition(left + ContentWidth, top + i);
|
||||
Console.SetCursorPosition(left + ContentWidth + 1, top + i);
|
||||
Console.Write(Filler(Border, 2));
|
||||
}
|
||||
Console.SetCursorPosition(left, top + ContentHeight);
|
||||
Console.Write(Filler(Border, ContentWidth + 2));
|
||||
Console.SetCursorPosition(left-1, top + ContentHeight);
|
||||
Console.Write(Filler(Border, ContentWidth + 4));
|
||||
Console.BackgroundColor = ConsoleColor.Black;
|
||||
}
|
||||
protected abstract void _Draw(int left, ref int top);
|
||||
|
@ -88,16 +88,18 @@ namespace Client.ConsoleForms
|
||||
{
|
||||
XmlDocument doc = new XmlDocument();
|
||||
doc.LoadXml(metaString);
|
||||
XmlNode lang = doc.GetElementsByTagName("Lang").Item(0);
|
||||
XmlNode lang = doc.GetElementsByTagName("Strings").Item(0);
|
||||
List<XmlElement> priorities = new List<XmlElement>();
|
||||
foreach(var node in lang.ChildNodes)
|
||||
if (node is XmlElement el)
|
||||
{
|
||||
if (el.Name.Equals("Default")) priorities.Insert(0, (XmlElement)node);
|
||||
else if (el.Name.Equals("Fallback"))
|
||||
//if (el.Name.Equals("Default")) priorities.Insert(0, (XmlElement)node);
|
||||
/*else*/ if (el.Name.Equals(/*"Fallback"*/"Lang"))
|
||||
{
|
||||
if (priorities.Count == 0) priorities.Add(el);
|
||||
else
|
||||
for (int i = 0; i < priorities.Count; ++i)
|
||||
if (!priorities[i].Name.Equals("Default") && ComparePriority(el, priorities[i]))
|
||||
if (/*!priorities[i].Name.Equals("Default") && */ComparePriority(el, priorities[i]))
|
||||
{
|
||||
priorities.Insert(i, el);
|
||||
break;
|
||||
@ -125,7 +127,7 @@ namespace Client.ConsoleForms
|
||||
break;
|
||||
}
|
||||
|
||||
// Use defults and fallbacks
|
||||
// Use defaults and fallbacks
|
||||
for (int i = 0; i<priorities.Count; ++i)
|
||||
{
|
||||
foreach (var prop in properties)
|
||||
|
@ -7,60 +7,86 @@ using System.Text;
|
||||
using System.Threading.Tasks;
|
||||
using Tofvesson.Collections;
|
||||
using Client.ConsoleForms.Graphics;
|
||||
using Tofvesson.Crypto;
|
||||
using Client.Properties;
|
||||
|
||||
namespace Client
|
||||
{
|
||||
public class NetContext : Context
|
||||
{
|
||||
private static readonly RandomProvider provider = new RegularRandomProvider();
|
||||
public NetContext(ContextManager manager) : base(manager, "Networking", "Common")
|
||||
{
|
||||
// Just close when anything is selected and "submitted"
|
||||
RegisterSelectListeners((s, i, v) => controller.CloseView(s), "EmptyFieldError", "IPError", "PortError", "ConnectionError");
|
||||
|
||||
((InputView)views.GetNamed("NetConnect")).SubmissionsListener = i =>
|
||||
bool connecting = false;
|
||||
|
||||
GetView<InputView>("NetConnect").SubmissionsListener = i =>
|
||||
{
|
||||
if (connecting)
|
||||
{
|
||||
controller.Popup("Already connecting!", 1000, ConsoleColor.DarkRed);
|
||||
return;
|
||||
}
|
||||
bool
|
||||
ip = ParseIP(i.Inputs[0].Text) != null,
|
||||
port = short.TryParse(i.Inputs[1].Text, out short prt) && prt > 0;
|
||||
|
||||
|
||||
if (ip && port)
|
||||
{
|
||||
connecting = true;
|
||||
// Connect to server here
|
||||
BankNetInteractor ita = new BankNetInteractor(i.Inputs[0].Text, prt, false); // Don't do identity check for now
|
||||
BankNetInteractor ita = new BankNetInteractor(i.Inputs[0].Text, prt);
|
||||
/*
|
||||
try
|
||||
{
|
||||
var t = ita.Connect();
|
||||
while (!t.IsCompleted)
|
||||
if (t.IsCanceled || t.IsFaulted)
|
||||
{
|
||||
controller.AddView(views.GetNamed("ConnectError"));
|
||||
return;
|
||||
}
|
||||
//var t = ita.Connect();
|
||||
//while (!t.IsCompleted)
|
||||
// if (t.IsCanceled || t.IsFaulted)
|
||||
// {
|
||||
// Show("ConnectError");
|
||||
// return;
|
||||
// }
|
||||
// else System.Threading.Thread.Sleep(125);
|
||||
}
|
||||
catch
|
||||
{
|
||||
controller.AddView(views.GetNamed("ConnectionError"));
|
||||
Show("ConnectionError");
|
||||
connecting = false;
|
||||
return;
|
||||
}
|
||||
manager.LoadContext(new WelcomeContext(manager, ita));
|
||||
*/
|
||||
|
||||
Promise verify = Promise.AwaitPromise(ita.CheckIdentity(new RSA(Resources.e_0x100, Resources.n_0x100), provider.NextUShort()));
|
||||
verify.Subscribe =
|
||||
p =>
|
||||
{
|
||||
void load() => manager.LoadContext(new WelcomeContext(manager, ita));
|
||||
|
||||
// Add condition check for remote peer verification
|
||||
if (bool.Parse(p.Value)) controller.Popup("Server identity verified!", 1000, ConsoleColor.Green, load);
|
||||
else controller.Popup("Remote server identity could not be verified!", 5000, ConsoleColor.Red, load);
|
||||
};
|
||||
DialogView identityNotify = GetView<DialogView>("IdentityVerify");
|
||||
identityNotify.RegisterSelectListener(
|
||||
(vw, ix, nm) => {
|
||||
verify.Subscribe = null; // Clear subscription
|
||||
#pragma warning disable CS4014 // Because this call is not awaited, execution of the current method continues before the call is completed
|
||||
ita.CancelAll();
|
||||
#pragma warning restore CS4014 // Because this call is not awaited, execution of the current method continues before the call is completed
|
||||
connecting = false;
|
||||
});
|
||||
Show(identityNotify);
|
||||
}
|
||||
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"));
|
||||
else if (!ip) Show("IPError");
|
||||
else Show("PortError");
|
||||
};
|
||||
}
|
||||
|
||||
public override void OnCreate()
|
||||
{
|
||||
controller.AddView(views.GetNamed("NetConnect"));
|
||||
}
|
||||
|
||||
public override void OnDestroy()
|
||||
{
|
||||
foreach (var view in views)
|
||||
controller.CloseView(view.Item2);
|
||||
}
|
||||
public override void OnCreate() => Show("NetConnect");
|
||||
public override void OnDestroy() => HideAll();
|
||||
|
||||
|
||||
//int gtrack = 0;
|
||||
|
@ -1,5 +1,6 @@
|
||||
using Client.ConsoleForms;
|
||||
using Client.ConsoleForms.Graphics;
|
||||
using Client.Properties;
|
||||
using ConsoleForms;
|
||||
using System;
|
||||
using System.Collections.Generic;
|
||||
@ -7,6 +8,8 @@ using System.Linq;
|
||||
using System.Text;
|
||||
using System.Threading.Tasks;
|
||||
using Tofvesson.Collections;
|
||||
using Tofvesson.Common;
|
||||
using Tofvesson.Crypto;
|
||||
|
||||
namespace Client
|
||||
{
|
||||
@ -20,24 +23,20 @@ namespace Client
|
||||
this.interactor = interactor;
|
||||
this.sessionID = sessionID;
|
||||
|
||||
((DialogView)views.GetNamed("Success")).RegisterSelectListener((v, i, s) =>
|
||||
GetView<DialogView>("Success").RegisterSelectListener((v, i, s) =>
|
||||
{
|
||||
interactor.Logout(sessionID);
|
||||
manager.LoadContext(new NetContext(manager));
|
||||
});
|
||||
}
|
||||
|
||||
public override void OnCreate()
|
||||
{
|
||||
//controller.AddView(views.GetNamed("Success"));
|
||||
controller.AddView(views.GetNamed("menu_options"));
|
||||
}
|
||||
public override void OnCreate() => Show("menu_options");
|
||||
|
||||
public override void OnDestroy()
|
||||
{
|
||||
controller.CloseView(views.GetNamed("Success"));
|
||||
#pragma warning disable CS4014 // Because this call is not awaited, execution of the current method continues before the call is completed
|
||||
interactor.Disconnect();
|
||||
interactor.CancelAll();
|
||||
#pragma warning restore CS4014 // Because this call is not awaited, execution of the current method continues before the call is completed
|
||||
}
|
||||
}
|
||||
|
@ -27,7 +27,7 @@ namespace Client
|
||||
RegisterSelectListeners((s, i, v) => controller.CloseView(s), "DuplicateAccountError", "EmptyFieldError", "IPError", "PortError", "AuthError", "PasswordMismatchError");
|
||||
|
||||
|
||||
((InputView)views.GetNamed("Login")).SubmissionsListener = i =>
|
||||
GetView<InputView>("Login").SubmissionsListener = i =>
|
||||
{
|
||||
bool success = true;
|
||||
|
||||
@ -43,14 +43,16 @@ namespace Client
|
||||
if (success)
|
||||
{
|
||||
// Authenticate against server here
|
||||
controller.AddView(views.GetNamed("AuthWait"));
|
||||
promise = interactor.Authenticate(i.Inputs[0].Text, i.Inputs[1].Text);
|
||||
Show("AuthWait");
|
||||
Task<Promise> prom = interactor.Authenticate(i.Inputs[0].Text, i.Inputs[1].Text);
|
||||
if(!prom.IsCompleted) prom.RunSynchronously();
|
||||
promise = prom.Result;
|
||||
promise.Subscribe =
|
||||
response =>
|
||||
{
|
||||
controller.CloseView(views.GetNamed("AuthWait"));
|
||||
Hide("AuthWait");
|
||||
if (response.Value.Equals("ERROR"))
|
||||
controller.AddView(views.GetNamed("AuthError"));
|
||||
Show("AuthError");
|
||||
else
|
||||
{
|
||||
forceDestroy = false;
|
||||
@ -58,18 +60,18 @@ namespace Client
|
||||
}
|
||||
};
|
||||
}
|
||||
else controller.AddView(views.GetNamed("EmptyFieldError"));
|
||||
else Show("EmptyFieldError");
|
||||
};
|
||||
|
||||
// For a smooth effect
|
||||
((InputView)views.GetNamed("Login")).InputListener = (v, c, i) =>
|
||||
GetView<InputView>("Login").InputListener = (v, c, i) =>
|
||||
{
|
||||
c.BackgroundColor = v.DefaultBackgroundColor;
|
||||
c.SelectBackgroundColor = v.DefaultSelectBackgroundColor;
|
||||
return true;
|
||||
};
|
||||
|
||||
((InputView)views.GetNamed("Register")).SubmissionsListener = i =>
|
||||
GetView<InputView>("Register").SubmissionsListener = i =>
|
||||
{
|
||||
bool success = true, mismatch = false;
|
||||
|
||||
@ -88,14 +90,14 @@ namespace Client
|
||||
{
|
||||
void a()
|
||||
{
|
||||
controller.AddView(views.GetNamed("RegWait"));
|
||||
promise = interactor.Register(i.Inputs[0].Text, i.Inputs[1].Text);
|
||||
Show("RegWait");
|
||||
promise = Promise.AwaitPromise(interactor.Register(i.Inputs[0].Text, i.Inputs[1].Text));
|
||||
promise.Subscribe =
|
||||
response =>
|
||||
{
|
||||
controller.CloseView(views.GetNamed("RegWait"));
|
||||
Hide("RegWait");
|
||||
if (response.Value.Equals("ERROR"))
|
||||
controller.AddView(views.GetNamed("DuplicateAccountError"));
|
||||
Show("DuplicateAccountError");
|
||||
else
|
||||
{
|
||||
forceDestroy = false;
|
||||
@ -106,18 +108,18 @@ namespace Client
|
||||
|
||||
if (i.Inputs[1].Text.Length < 5 || i.Inputs[1].Text.StartsWith("asdfasdf") || i.Inputs[1].Text.StartsWith("asdf1234"))
|
||||
{
|
||||
var warning = (DialogView)views.GetNamed("WeakPasswordWarning");
|
||||
var warning = GetView<DialogView>("WeakPasswordWarning");
|
||||
warning.RegisterSelectListener((wrn, idx, sel) =>
|
||||
{
|
||||
controller.CloseView(warning);
|
||||
Hide(warning);
|
||||
if (idx == 0) a();
|
||||
});
|
||||
controller.AddView(warning);
|
||||
Show(warning);
|
||||
}
|
||||
else a();
|
||||
}
|
||||
else if (mismatch) controller.AddView(views.GetNamed("PasswordMismatchError"));
|
||||
else controller.AddView(views.GetNamed("EmptyFieldError"));
|
||||
else if (mismatch) Show("PasswordMismatchError");
|
||||
else Show("EmptyFieldError");
|
||||
};
|
||||
|
||||
((InputView)views.GetNamed("Register")).InputListener = (v, c, i) =>
|
||||
@ -136,13 +138,13 @@ namespace Client
|
||||
});
|
||||
|
||||
// Add the initial view
|
||||
controller.AddView(views.GetNamed("WelcomeScreen"));
|
||||
Show("WelcomeScreen");
|
||||
}
|
||||
|
||||
public override void OnDestroy()
|
||||
{
|
||||
((InputView)views.GetNamed("Register")).SelectedField = 0;
|
||||
foreach (var v in ((InputView)views.GetNamed("Register")).Inputs)
|
||||
GetView<InputView>("Register").SelectedField = 0;
|
||||
foreach (var v in GetView<InputView>("Register").Inputs)
|
||||
{
|
||||
v.Text = "";
|
||||
v.SelectIndex = 0;
|
||||
@ -150,7 +152,7 @@ namespace Client
|
||||
}
|
||||
|
||||
((InputView)views.GetNamed("Login")).SelectedField = 0;
|
||||
foreach (var v in ((InputView)views.GetNamed("Login")).Inputs)
|
||||
foreach (var v in GetView<InputView>("Login").Inputs)
|
||||
{
|
||||
v.Text = "";
|
||||
v.SelectIndex = 0;
|
||||
@ -158,8 +160,7 @@ namespace Client
|
||||
}
|
||||
|
||||
// Close views
|
||||
foreach (var view in views)
|
||||
controller.CloseView(view.Item2);
|
||||
HideAll();
|
||||
|
||||
// Unsubscribe from events
|
||||
if (promise != null && !promise.HasValue) promise.Subscribe = null;
|
||||
@ -168,7 +169,7 @@ namespace Client
|
||||
interactor.UnregisterListener(token);
|
||||
|
||||
#pragma warning disable CS4014 // Because this call is not awaited, execution of the current method continues before the call is completed
|
||||
if (forceDestroy) interactor.Disconnect();
|
||||
if (forceDestroy) interactor.CancelAll();
|
||||
#pragma warning restore CS4014 // Because this call is not awaited, execution of the current method continues before the call is completed
|
||||
}
|
||||
}
|
||||
|
@ -3,9 +3,11 @@ using Common.Cryptography.KeyExchange;
|
||||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.Linq;
|
||||
using System.Net;
|
||||
using System.Numerics;
|
||||
using System.Text;
|
||||
using System.Threading.Tasks;
|
||||
using Tofvesson.Common;
|
||||
using Tofvesson.Crypto;
|
||||
|
||||
namespace Client
|
||||
@ -17,52 +19,57 @@ namespace Client
|
||||
|
||||
protected Dictionary<long, Promise> promises = new Dictionary<long, Promise>();
|
||||
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)
|
||||
protected readonly IPAddress addr;
|
||||
protected readonly short port;
|
||||
protected readonly EllipticDiffieHellman keyExchange;
|
||||
public bool IsAlive { get => client != null && client.IsAlive; }
|
||||
public bool IsLoggedIn
|
||||
{
|
||||
if(checkIdentity)
|
||||
new Task(() =>
|
||||
get
|
||||
{
|
||||
//AuthenticatedKeys = NetClient.CheckServerIdentity(address, port, provider);
|
||||
authenticating = false;
|
||||
authenticated = true;// AuthenticatedKeys != null;
|
||||
}).Start();
|
||||
else
|
||||
{
|
||||
authenticating = false;
|
||||
authenticated = false;
|
||||
if (loginTimeout >= DateTime.Now.Ticks) loginTimeout = -1;
|
||||
return loginTimeout != -1;
|
||||
}
|
||||
var addr = System.Net.IPAddress.Parse(address);
|
||||
}
|
||||
protected long loginTimeout = -1;
|
||||
protected string sessionID = null;
|
||||
|
||||
|
||||
public BankNetInteractor(string address, short port)
|
||||
{
|
||||
this.addr = IPAddress.Parse(address);
|
||||
this.port = port;
|
||||
this.keyExchange = EllipticDiffieHellman.Curve25519(EllipticDiffieHellman.Curve25519_GeneratePrivate(provider));
|
||||
}
|
||||
|
||||
protected virtual async Task Connect()
|
||||
{
|
||||
if (IsAlive) return;
|
||||
client = new NetClient(
|
||||
EllipticDiffieHellman.Curve25519(EllipticDiffieHellman.Curve25519_GeneratePrivate(provider)),
|
||||
keyExchange,
|
||||
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;
|
||||
await t;
|
||||
}
|
||||
public async virtual Task CancelAll()
|
||||
{
|
||||
if (client == null) return;
|
||||
await client.Disconnect();
|
||||
}
|
||||
public async virtual Task<object> Disconnect() => await client.Disconnect();
|
||||
|
||||
public long RegisterListener(OnClientConnectStateChanged stateListener)
|
||||
{
|
||||
long tkn;
|
||||
changeListeners[tkn = GetListenerToken()] = stateListener;
|
||||
long tkn = GetListenerToken();
|
||||
changeListeners[tkn] = stateListener;
|
||||
return tkn;
|
||||
}
|
||||
|
||||
@ -74,9 +81,8 @@ namespace Client
|
||||
if (err || !promises.ContainsKey(pID)) return null;
|
||||
Promise p = promises[pID];
|
||||
promises.Remove(pID);
|
||||
p.Value = response;
|
||||
p.HasValue = true;
|
||||
p.Subscribe?.Invoke(p);
|
||||
PostPromise(p, response);
|
||||
if (promises.Count == 0) keepAlive = false;
|
||||
return null;
|
||||
}
|
||||
|
||||
@ -86,7 +92,80 @@ namespace Client
|
||||
listener(client, connect);
|
||||
}
|
||||
|
||||
public virtual Promise CheckAccountAvailability(string username)
|
||||
public async virtual Task<Promise> CheckAccountAvailability(string username)
|
||||
{
|
||||
await Connect();
|
||||
if (username.Length > 60)
|
||||
return new Promise
|
||||
{
|
||||
HasValue = true,
|
||||
Value = "ERROR"
|
||||
};
|
||||
client.Send(CreateCommandMessage("Avail", username.ToBase64String(), out long pID));
|
||||
return RegisterPromise(pID);
|
||||
}
|
||||
|
||||
public async virtual Task<Promise> Authenticate(string username, string password)
|
||||
{
|
||||
await Connect();
|
||||
if (username.Length > 60)
|
||||
return new Promise
|
||||
{
|
||||
HasValue = true,
|
||||
Value = "ERROR"
|
||||
};
|
||||
client.Send(CreateCommandMessage("Auth", username.ToBase64String()+":"+password.ToBase64String(), out long pID));
|
||||
|
||||
return RegisterEventPromise(pID, p =>
|
||||
{
|
||||
bool b = !p.Value.StartsWith("ERROR");
|
||||
PostPromise(p.handler, b);
|
||||
if (b)
|
||||
{
|
||||
loginTimeout = 280 * TimeSpan.TicksPerSecond;
|
||||
sessionID = p.Value;
|
||||
}
|
||||
return false;
|
||||
});
|
||||
}
|
||||
|
||||
public async virtual Task<Promise> CreateAccount(string accountName)
|
||||
{
|
||||
if (!IsLoggedIn) throw new SystemException("Not logged in");
|
||||
await Connect();
|
||||
client.Send(CreateCommandMessage("Account_Create", $"{sessionID}:{accountName}", out long PID));
|
||||
return RegisterEventPromise(PID, RefreshSession);
|
||||
}
|
||||
|
||||
public async virtual Task<Promise> CheckIdentity(RSA check, ushort nonce)
|
||||
{
|
||||
long pID;
|
||||
Task connect = Connect();
|
||||
string ser;
|
||||
using(BitWriter writer = new BitWriter())
|
||||
{
|
||||
writer.WriteULong(nonce);
|
||||
ser = CreateCommandMessage("Verify", Convert.ToBase64String(writer.Finalize()), out pID);
|
||||
}
|
||||
await connect;
|
||||
client.Send(ser);
|
||||
return RegisterEventPromise(pID, manager =>
|
||||
{
|
||||
BitReader reader = new BitReader(Convert.FromBase64String(manager.Value));
|
||||
try
|
||||
{
|
||||
RSA remote = RSA.Deserialize(reader.ReadByteArray(), out int _);
|
||||
PostPromise(manager.handler, new BigInteger(remote.Decrypt(reader.ReadByteArray(), null, true)).Equals((BigInteger)nonce) && remote.Equals(check));
|
||||
}
|
||||
catch
|
||||
{
|
||||
PostPromise(manager.handler, false);
|
||||
}
|
||||
return false;
|
||||
});
|
||||
}
|
||||
|
||||
public async virtual Task<Promise> Register(string username, string password)
|
||||
{
|
||||
if (username.Length > 60)
|
||||
return new Promise
|
||||
@ -94,43 +173,43 @@ namespace Client
|
||||
HasValue = true,
|
||||
Value = "ERROR"
|
||||
};
|
||||
client.Send(CreateCommandMessage("Avail", username, out long pID));
|
||||
await Connect();
|
||||
client.Send(CreateCommandMessage("Reg", username.ToBase64String() + ":" + password.ToBase64String(), out long pID));
|
||||
return RegisterPromise(pID);
|
||||
}
|
||||
|
||||
public async virtual Task Logout(string sessionID)
|
||||
{
|
||||
if (!IsLoggedIn) return; // No need to unnecessarily trigger a logout that we know will fail
|
||||
await Connect();
|
||||
client.Send(CreateCommandMessage("Logout", sessionID, out long _));
|
||||
}
|
||||
|
||||
protected Promise RegisterPromise(long pID)
|
||||
{
|
||||
Promise p = new Promise();
|
||||
promises[pID] = p;
|
||||
return p;
|
||||
}
|
||||
|
||||
public virtual Promise Authenticate(string username, string password)
|
||||
protected Promise RegisterEventPromise(long pID, Func<Promise, bool> a)
|
||||
{
|
||||
if (username.Length > 60)
|
||||
return new Promise
|
||||
Promise p = RegisterPromise(pID);
|
||||
p.handler = new Promise();
|
||||
p.Subscribe = p1 =>
|
||||
{
|
||||
HasValue = true,
|
||||
Value = "ERROR"
|
||||
// If true, propogate result
|
||||
if (a(p1)) PostPromise(p1.handler, p1.Value);
|
||||
};
|
||||
client.Send(CreateCommandMessage("Auth", username+":"+password, out long pID));
|
||||
Promise p = new Promise();
|
||||
promises[pID] = p;
|
||||
return p;
|
||||
return p.handler;
|
||||
}
|
||||
|
||||
public virtual Promise Register(string username, string password)
|
||||
protected bool RefreshSession(Promise p)
|
||||
{
|
||||
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;
|
||||
if (!p.Value.StartsWith("ERROR")) loginTimeout = 280 * TimeSpan.TicksPerSecond;
|
||||
return true;
|
||||
}
|
||||
|
||||
public virtual void Logout(string sessionID)
|
||||
=> client.Send(CreateCommandMessage("Logout", sessionID, out long _));
|
||||
|
||||
protected long GetNewPromiseUID()
|
||||
{
|
||||
long l;
|
||||
@ -147,9 +226,9 @@ namespace Client
|
||||
return l;
|
||||
}
|
||||
|
||||
protected static void PostPromise(Promise p, string value)
|
||||
protected static void PostPromise(Promise p, dynamic value)
|
||||
{
|
||||
p.Value = value;
|
||||
p.Value = value?.ToString() ?? "null";
|
||||
p.HasValue = true;
|
||||
p.Subscribe?.Invoke(p);
|
||||
}
|
||||
@ -165,6 +244,7 @@ namespace Client
|
||||
public delegate void Event(Promise p);
|
||||
public class Promise
|
||||
{
|
||||
internal Promise handler = null; // For chained promise management
|
||||
private Event evt;
|
||||
public string Value { get; internal set; }
|
||||
public bool HasValue { get; internal set; }
|
||||
@ -173,10 +253,17 @@ namespace Client
|
||||
get => evt;
|
||||
set
|
||||
{
|
||||
evt = value;
|
||||
// Allows clearing subscriptions
|
||||
if (evt == null || value == null) evt = value;
|
||||
else evt += value;
|
||||
if (HasValue)
|
||||
evt(this);
|
||||
}
|
||||
}
|
||||
public static Promise AwaitPromise(Task<Promise> p)
|
||||
{
|
||||
if (!p.IsCompleted) p.RunSynchronously();
|
||||
return p.Result;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -1,44 +1,30 @@
|
||||
using System;
|
||||
using System.Collections;
|
||||
using System.Collections.Generic;
|
||||
using System.Diagnostics;
|
||||
using System.IO;
|
||||
using System.Numerics;
|
||||
using System.Runtime.InteropServices;
|
||||
using Client;
|
||||
using Client.ConsoleForms;
|
||||
using Client.ConsoleForms.Parameters;
|
||||
using Common;
|
||||
using Tofvesson.Common;
|
||||
using Tofvesson.Crypto;
|
||||
|
||||
namespace ConsoleForms
|
||||
{
|
||||
|
||||
class Program
|
||||
{
|
||||
private static readonly RandomProvider provider = new RegularRandomProvider(new Random(1337));
|
||||
public static TextWriter DebugStream = new DebugAdapterWriter();
|
||||
private static ConsoleController controller = ConsoleController.singleton;
|
||||
|
||||
static void Main(string[] args)
|
||||
public static void Main(string[] args)
|
||||
{
|
||||
// Set up timestamps in debug output
|
||||
DebugStream = new TimeStampWriter(DebugStream, "HH:mm:ss.fff");
|
||||
|
||||
|
||||
|
||||
byte[] serialized;
|
||||
|
||||
|
||||
using (BinaryCollector collector = new BinaryCollector(4))
|
||||
{
|
||||
collector.Push(true);
|
||||
collector.Push(new double[] { 6.0, 123 });
|
||||
collector.Push(new float[] { 512, 1.2f, 1.337f});
|
||||
collector.Push(5);
|
||||
serialized = collector.ToArray();
|
||||
}
|
||||
|
||||
BinaryDistributor bd = new BinaryDistributor(serialized);
|
||||
bool bit = bd.ReadBit();
|
||||
double[] result = bd.ReadDoubleArray();
|
||||
float[] f = bd.ReadFloatArray();
|
||||
int number = bd.ReadInt();
|
||||
|
||||
Padding p = new AbsolutePadding(2, 2, 1, 1);
|
||||
|
||||
Console.CursorVisible = false;
|
||||
|
75
Client/Properties/Resources.Designer.cs
generated
75
Client/Properties/Resources.Designer.cs
generated
@ -62,14 +62,14 @@ namespace Client.Properties {
|
||||
|
||||
/// <summary>
|
||||
/// Looks up a localized string similar to <?xml version="1.0" encoding="utf-8" ?>
|
||||
///<Resources>
|
||||
///<Resources xmlns="Client.ConsoleForms.Graphics">
|
||||
/// <DialogView id="EmptyFieldError"
|
||||
/// padding_left="2"
|
||||
/// padding_right="2"
|
||||
/// padding_top="1"
|
||||
/// padding_bottom="1">
|
||||
/// <Options>
|
||||
/// <Option>Ok</Option>
|
||||
/// <Option>@string/GENERIC_accept</Option>
|
||||
/// </Options>
|
||||
/// <Text>@string/ERR_empty</Text>
|
||||
/// </DialogView>
|
||||
@ -81,6 +81,16 @@ namespace Client.Properties {
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Looks up a localized resource of type System.Byte[].
|
||||
/// </summary>
|
||||
internal static byte[] e_0x100 {
|
||||
get {
|
||||
object obj = ResourceManager.GetObject("e_0x100", resourceCulture);
|
||||
return ((byte[])(obj));
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Looks up a localized resource of type System.Byte[].
|
||||
/// </summary>
|
||||
@ -91,6 +101,16 @@ namespace Client.Properties {
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Looks up a localized resource of type System.Byte[].
|
||||
/// </summary>
|
||||
internal static byte[] n_0x100 {
|
||||
get {
|
||||
object obj = ResourceManager.GetObject("n_0x100", resourceCulture);
|
||||
return ((byte[])(obj));
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Looks up a localized resource of type System.Byte[].
|
||||
/// </summary>
|
||||
@ -111,11 +131,10 @@ namespace Client.Properties {
|
||||
/// 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>
|
||||
/// <Field input_type="decimal" max_length="15">@string/NC_ip</Field>
|
||||
/// <Field default="80" input_type="integer" max_length="5">@string/NC_port</Field>
|
||||
/// </Fields>
|
||||
/// <Text>@string/NC_head</Text>
|
||||
/// </InputVie [rest of string was truncated]";.
|
||||
/// <Text>@string/NC_head</Text>
/// [rest of string was truncated]";.
|
||||
/// </summary>
|
||||
internal static string Networking {
|
||||
get {
|
||||
@ -142,10 +161,10 @@ namespace Client.Properties {
|
||||
/// padding_right="2"
|
||||
/// padding_top="1"
|
||||
/// padding_bottom="1">
|
||||
/// <Text>Balance: $balance</Text>
|
||||
/// <Text>@string/SE_bal</Text>
|
||||
/// </TextView>
|
||||
///
|
||||
/// <ListView id= [rest of string was truncated]";.
|
||||
/// <ListView id="me [rest of string was truncated]";.
|
||||
/// </summary>
|
||||
internal static string Session {
|
||||
get {
|
||||
@ -165,10 +184,10 @@ namespace Client.Properties {
|
||||
/// padding_bottom="1"
|
||||
/// width="42">
|
||||
/// <Options>
|
||||
/// <Option event="Setup:Login" close="true">Login</Option>
|
||||
/// <Option event="Setup:Register" close="true">Register</Option>
|
||||
/// <Option event="Setup:Login" close="true">@string/SU_login_label</Option>
|
||||
/// <Option event="Setup:Register" close="true">@string/SU_reg_label</Option>
|
||||
/// </Options>
|
||||
/// <Text>Welcome to the Tofvesson banking [rest of string was truncated]";.
|
||||
/// <Text>@st [rest of string was truncated]";.
|
||||
/// </summary>
|
||||
internal static string Setup {
|
||||
get {
|
||||
@ -182,9 +201,11 @@ namespace Client.Properties {
|
||||
/// <Entry name="NC_head">Server configuration</Entry>
|
||||
/// <Entry name="NC_sec">The selected server's identity could not be verified. This implies that it is not an official server. Continue?</Entry>
|
||||
/// <Entry name="NC_stall">Connecting to server...</Entry>
|
||||
///
|
||||
/// <Entry name="ERR_empty">One of more required field was empty!</Entry>
|
||||
///</Strings>.
|
||||
/// <Entry name="NC_next">Continue</Entry>
|
||||
/// <Entry name="NC_cancel">Cancel</Entry>
|
||||
/// <Entry name="NC_ip">Server IP:</Entry>
|
||||
/// <Entry name="NC_port">Port:</Entry>
|
||||
/// <Entry name="NC_iperr">The s [rest of string was truncated]";.
|
||||
/// </summary>
|
||||
internal static string strings_lang_en_GB {
|
||||
get {
|
||||
@ -198,9 +219,11 @@ namespace Client.Properties {
|
||||
/// <Entry name="NC_head">Server configuration</Entry>
|
||||
/// <Entry name="NC_sec">The selected server's identity could not be verified. This implies that it is not an official server. Continue?</Entry>
|
||||
/// <Entry name="NC_stall">Connecting to server...</Entry>
|
||||
///
|
||||
/// <Entry name="ERR_empty">One of more required field was empty!</Entry>
|
||||
///</Strings>.
|
||||
/// <Entry name="NC_next">Continue</Entry>
|
||||
/// <Entry name="NC_cancel">Cancel</Entry>
|
||||
/// <Entry name="NC_ip">Server IP:</Entry>
|
||||
/// <Entry name="NC_port">Port:</Entry>
|
||||
/// <Entry name="NC_iperr">The s [rest of string was truncated]";.
|
||||
/// </summary>
|
||||
internal static string strings_lang_en_US {
|
||||
get {
|
||||
@ -214,9 +237,11 @@ namespace Client.Properties {
|
||||
/// <Entry name="NC_head">Serverkonfiguration</Entry>
|
||||
/// <Entry name="NC_sec">Den valda serverns identitet kunde inte verifieras. Detta innebär att det inte är en officiell server. Fortsätt?</Entry>
|
||||
/// <Entry name="NC_stall">Kopplar upp mot servern...</Entry>
|
||||
///
|
||||
/// <Entry name="ERR_empty">Ett eller fler obligatoriska inputfält är tomma!</Entry>
|
||||
///</Strings>.
|
||||
/// <Entry name="NC_next">Fortsätt</Entry>
|
||||
/// <Entry name="NC_cancel">Avbryt</Entry>
|
||||
/// <Entry name="NC_ip">Server IP:</Entry>
|
||||
/// <Entry name="NC_port">Port:</Entry>
|
||||
/// <Entry name="NC_iperr">De [rest of string was truncated]";.
|
||||
/// </summary>
|
||||
internal static string strings_lang_sv_SE {
|
||||
get {
|
||||
@ -226,11 +251,11 @@ namespace Client.Properties {
|
||||
|
||||
/// <summary>
|
||||
/// Looks up a localized string similar to <?xml version="1.0" encoding="utf-8" ?>
|
||||
///<Lang>
|
||||
/// <Default>sv_SE</Default>
|
||||
/// <Fallback priority="0">en_US</Fallback>
|
||||
/// <Fallback priority="1">en_GB></Fallback>
|
||||
///</Lang>.
|
||||
///<Strings>
|
||||
/// <Lang priority="0">sv_SE</Lang>
|
||||
/// <Lang priority="1">en_US</Lang>
|
||||
/// <Lang priority="2">en_GB</Lang>
|
||||
///</Strings>.
|
||||
/// </summary>
|
||||
internal static string strings_meta {
|
||||
get {
|
||||
|
@ -148,4 +148,10 @@
|
||||
<data name="strings_lang_en_GB" type="System.Resources.ResXFileRef, System.Windows.Forms">
|
||||
<value>..\Resources\Strings\en_GB\strings.xml;System.String, mscorlib, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089;utf-8</value>
|
||||
</data>
|
||||
<data name="e_0x100" type="System.Resources.ResXFileRef, System.Windows.Forms">
|
||||
<value>..\Resources\0x100.e;System.Byte[], mscorlib, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089</value>
|
||||
</data>
|
||||
<data name="n_0x100" type="System.Resources.ResXFileRef, System.Windows.Forms">
|
||||
<value>..\Resources\0x100.n;System.Byte[], mscorlib, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089</value>
|
||||
</data>
|
||||
</root>
|
BIN
Client/Resources/0x100.e
Normal file
BIN
Client/Resources/0x100.e
Normal file
Binary file not shown.
BIN
Client/Resources/0x100.n
Normal file
BIN
Client/Resources/0x100.n
Normal file
Binary file not shown.
@ -37,6 +37,17 @@
|
||||
<Text>@string/NC_stall</Text>
|
||||
</DialogView>
|
||||
|
||||
<DialogView id="IdentityVerify"
|
||||
padding_left="2"
|
||||
padding_right="2"
|
||||
padding_top="1"
|
||||
padding_bottom="1">
|
||||
<Options>
|
||||
<Option>@string/NC_cancel</Option>
|
||||
</Options>
|
||||
<Text>@string/NC_identity</Text>
|
||||
</DialogView>
|
||||
|
||||
<DialogView id="IPError"
|
||||
padding_left="2"
|
||||
padding_right="2"
|
||||
|
@ -23,20 +23,39 @@
|
||||
padding_left="2"
|
||||
padding_right="2"
|
||||
padding_top="1"
|
||||
padding_bottom="1"
|
||||
border="2">
|
||||
padding_bottom="1">
|
||||
<Views>
|
||||
<ButtonView id="history">
|
||||
<Text>@string/SE_hist</Text>
|
||||
</ButtonView>
|
||||
|
||||
<ButtonView id="create">
|
||||
<Text>@string/SE_tx</Text>
|
||||
<ButtonView id="view">
|
||||
<Text>@string/SE_view</Text>
|
||||
</ButtonView>
|
||||
|
||||
<ButtonView id="update">
|
||||
<Text>@string/SE_pwdu</Text>
|
||||
</ButtonView>
|
||||
|
||||
<ButtonView id="exit">
|
||||
<Text>@string/SE_exit</Text>
|
||||
</ButtonView>
|
||||
</Views>
|
||||
</ListView>
|
||||
|
||||
<!-- Bank account list -->
|
||||
<ListView id="account_show">
|
||||
<Views>
|
||||
<ButtonView id="close">
|
||||
<Text>@string/GENERIC_dismiss</Text>
|
||||
</ButtonView>
|
||||
</Views>
|
||||
</ListView>
|
||||
|
||||
<DialogView id="account_info"
|
||||
padding_left="2"
|
||||
padding_right="2"
|
||||
padding_top="1"
|
||||
padding_bottom="1">
|
||||
<Options>
|
||||
<Option>Ok</Option>
|
||||
</Options>
|
||||
<Text>@string/SE_info</Text>
|
||||
</DialogView>
|
||||
</Elements>
|
@ -1,6 +1,6 @@
|
||||
<?xml version="1.0" encoding="utf-8" ?>
|
||||
<Lang>
|
||||
<Default>sv_SE</Default>
|
||||
<Fallback priority="0">en_US</Fallback>
|
||||
<Fallback priority="1">en_GB</Fallback>
|
||||
</Lang>
|
||||
<Strings>
|
||||
<Lang priority="0">sv_SE</Lang>
|
||||
<Lang priority="1">en_US</Lang>
|
||||
<Lang priority="2">en_GB</Lang>
|
||||
</Strings>
|
@ -10,6 +10,7 @@
|
||||
<Entry name="NC_iperr">The supplied IP-address is not valid</Entry>
|
||||
<Entry name="NC_porterr">The supplied port is not valid</Entry>
|
||||
<Entry name="NC_connerr">Could not connect to server</Entry>
|
||||
<Entry name="NC_identity">Verifying server identity...</Entry>
|
||||
|
||||
<Entry name="SU_welcome">Welcome to the Tofvesson banking system! To continue, press [ENTER] To go back, press [ESCAPE]</Entry>
|
||||
<Entry name="SU_reg">Register Account</Entry>
|
||||
@ -27,10 +28,23 @@
|
||||
<Entry name="SU_login_label">Login</Entry>
|
||||
|
||||
<Entry name="SE_bal">Balance: $1</Entry>
|
||||
<Entry name="SE_hist">View transaction history</Entry>
|
||||
<Entry name="SE_hist">Transaction history</Entry>
|
||||
<Entry name="SE_tx">Transfer funds</Entry>
|
||||
<Entry name="SE_who">Send to</Entry>
|
||||
<Entry name="SE_where">Account</Entry>
|
||||
<Entry name="SE_view">View accounts</Entry>
|
||||
<Entry name="SE_amount">Amount to transfer</Entry>
|
||||
<Entry name="SE_msg">Include a message</Entry>
|
||||
<Entry name="SE_pwdu">Update password</Entry>
|
||||
<Entry name="SE_exit">Log out</Entry>
|
||||
<Entry name="SE_open">Open an account</Entry>
|
||||
<Entry name="SE_close">Close an account</Entry>
|
||||
<Entry name="SE_accounts">Show accounts</Entry>
|
||||
<Entry name="SE_info">$0
|
||||
Balance: $1
|
||||
Date of creation: $2</Entry>
|
||||
|
||||
<Entry name="GENERIC_dismiss">Close</Entry>
|
||||
<Entry name="GENERIC_accept">Ok</Entry>
|
||||
<Entry name="GENERIC_positive">Yes</Entry>
|
||||
<Entry name="GENERIC_negative">No</Entry>
|
||||
|
@ -10,6 +10,7 @@
|
||||
<Entry name="NC_iperr">The supplied IP-address is not valid</Entry>
|
||||
<Entry name="NC_porterr">The supplied port is not valid</Entry>
|
||||
<Entry name="NC_connerr">Could not connect to server</Entry>
|
||||
<Entry name="NC_identity">Verifying server identity...</Entry>
|
||||
|
||||
<Entry name="SU_welcome">Welcome to the Tofvesson banking system! To continue, press [ENTER] To go back, press [ESCAPE]</Entry>
|
||||
<Entry name="SU_reg">Register Account</Entry>
|
||||
@ -27,10 +28,23 @@
|
||||
<Entry name="SU_login_label">Login</Entry>
|
||||
|
||||
<Entry name="SE_bal">Balance: $1</Entry>
|
||||
<Entry name="SE_hist">View transaction history</Entry>
|
||||
<Entry name="SE_hist">Transaction history</Entry>
|
||||
<Entry name="SE_tx">Transfer funds</Entry>
|
||||
<Entry name="SE_who">Send to</Entry>
|
||||
<Entry name="SE_where">Account</Entry>
|
||||
<Entry name="SE_view">View accounts</Entry>
|
||||
<Entry name="SE_amount">Amount to transfer</Entry>
|
||||
<Entry name="SE_msg">Include a message</Entry>
|
||||
<Entry name="SE_pwdu">Update password</Entry>
|
||||
<Entry name="SE_exit">Log out</Entry>
|
||||
<Entry name="SE_open">Open an account</Entry>
|
||||
<Entry name="SE_close">Close an account</Entry>
|
||||
<Entry name="SE_accounts">Show accounts</Entry>
|
||||
<Entry name="SE_info">$0
|
||||
Balance: $1
|
||||
Date of creation: $2</Entry>
|
||||
|
||||
<Entry name="GENERIC_dismiss">Close</Entry>
|
||||
<Entry name="GENERIC_accept">Ok</Entry>
|
||||
<Entry name="GENERIC_positive">Yes</Entry>
|
||||
<Entry name="GENERIC_negative">No</Entry>
|
||||
|
@ -3,6 +3,53 @@
|
||||
<Entry name="NC_head">Serverkonfiguration</Entry>
|
||||
<Entry name="NC_sec">Den valda serverns identitet kunde inte verifieras. Detta innebär att det inte är en officiell server. Fortsätt?</Entry>
|
||||
<Entry name="NC_stall">Kopplar upp mot servern...</Entry>
|
||||
<Entry name="NC_next">Fortsätt</Entry>
|
||||
<Entry name="NC_cancel">Avbryt</Entry>
|
||||
<Entry name="NC_ip">Server IP:</Entry>
|
||||
<Entry name="NC_port">Port:</Entry>
|
||||
<Entry name="NC_iperr">Den givna IP-addressen är inte giltig</Entry>
|
||||
<Entry name="NC_porterr">Den givna porten är inte giltig</Entry>
|
||||
<Entry name="NC_connerr">Kunde inte koppla till servern</Entry>
|
||||
<Entry name="NC_identity">Verifierar serverns identitet...</Entry>
|
||||
|
||||
<Entry name="ERR_empty">Ett eller fler obligatoriska inputfält är tomma!</Entry>
|
||||
<Entry name="SU_welcome">Välkommen till Tofvessons banksystem!
|
||||
För att fortsätta, tryck [ENTER]
|
||||
För att backa, tryck [ESCAPE]</Entry>
|
||||
<Entry name="SU_reg">Registrera konto</Entry>
|
||||
<Entry name="SU_regstall">Registrerar...</Entry>
|
||||
<Entry name="SU_dup">Ett konto med det givna användarnamnet finns redan!</Entry>
|
||||
<Entry name="SU_mismatch">De givna lösenorden matchar inte!</Entry>
|
||||
<Entry name="SU_weak">Det angivna lösenordet ases vara svagt. Är du säker på att du vill fortsätta?</Entry>
|
||||
<Entry name="SU_login">Logga in</Entry>
|
||||
<Entry name="SU_authstall">Autentiserar...</Entry>
|
||||
<Entry name="SU_usrerr">Det givna användarnamnet eller lösenordet var felaktigt</Entry>
|
||||
<Entry name="SU_usr">Användarnamn:</Entry>
|
||||
<Entry name="SU_pwd">Lösenord:</Entry>
|
||||
<Entry name="SU_pwdrep">Upprepa lösenord:</Entry>
|
||||
<Entry name="SU_reg_label">Registrera</Entry>
|
||||
<Entry name="SU_login_label">Logga in</Entry>
|
||||
|
||||
<Entry name="SE_bal">Kontobalans: $1</Entry>
|
||||
<Entry name="SE_hist">Transaktionshistorik</Entry>
|
||||
<Entry name="SE_tx">Överför pengar</Entry>
|
||||
<Entry name="SE_who">Skicka till</Entry>
|
||||
<Entry name="SE_where">Konto</Entry>
|
||||
<Entry name="SE_view">Visa konton</Entry>
|
||||
<Entry name="SE_amount">Värde att överföra</Entry>
|
||||
<Entry name="SE_msg">Inkludera ett meddelande</Entry>
|
||||
<Entry name="SE_pwdu">Uppdatera lösenord</Entry>
|
||||
<Entry name="SE_exit">Logga ut</Entry>
|
||||
<Entry name="SE_open">Öppna ett konto</Entry>
|
||||
<Entry name="SE_close">Stäng ett konto</Entry>
|
||||
<Entry name="SE_accounts">Visa konton</Entry>
|
||||
<Entry name="SE_info">"$0"
|
||||
Kontobalans: $1
|
||||
Begynnelsedatum: $2</Entry>
|
||||
|
||||
<Entry name="GENERIC_dismiss">Stäng</Entry>
|
||||
<Entry name="GENERIC_accept">Ok</Entry>
|
||||
<Entry name="GENERIC_positive">Ja</Entry>
|
||||
<Entry name="GENERIC_negative">Nej</Entry>
|
||||
|
||||
<Entry name="ERR_empty">Ett eller flera obligatoriska inputfält är tomma!</Entry>
|
||||
</Strings>
|
BIN
Client180412.vspx
Normal file
BIN
Client180412.vspx
Normal file
Binary file not shown.
@ -1,332 +0,0 @@
|
||||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.IO;
|
||||
using System.Linq;
|
||||
using System.Reflection;
|
||||
using System.Runtime.InteropServices;
|
||||
using System.Text;
|
||||
using System.Threading.Tasks;
|
||||
using Tofvesson.Crypto;
|
||||
|
||||
namespace Tofvesson.Common
|
||||
{
|
||||
public sealed class BinaryCollector : IDisposable
|
||||
{
|
||||
// Collects reusable
|
||||
private static readonly List<WeakReference<object[]>> expired = new List<WeakReference<object[]>>();
|
||||
|
||||
private static readonly byte[] holder = new byte[8];
|
||||
private static readonly float[] holder_f = new float[1];
|
||||
private static readonly double[] holder_d = new double[1];
|
||||
private static readonly ulong[] holder_u = new ulong[1];
|
||||
private static readonly uint[] holder_i = new uint[1];
|
||||
private static readonly List<Type> supportedTypes = new List<Type>()
|
||||
{
|
||||
typeof(bool),
|
||||
typeof(byte),
|
||||
typeof(sbyte),
|
||||
typeof(char),
|
||||
typeof(short),
|
||||
typeof(ushort),
|
||||
typeof(int),
|
||||
typeof(uint),
|
||||
typeof(long),
|
||||
typeof(ulong),
|
||||
typeof(float),
|
||||
typeof(double),
|
||||
typeof(decimal)
|
||||
};
|
||||
|
||||
private static readonly FieldInfo
|
||||
dec_lo,
|
||||
dec_mid,
|
||||
dec_hi,
|
||||
dec_flags;
|
||||
|
||||
static BinaryCollector()
|
||||
{
|
||||
dec_lo = typeof(decimal).GetField("lo", BindingFlags.NonPublic);
|
||||
dec_mid = typeof(decimal).GetField("mid", BindingFlags.NonPublic);
|
||||
dec_hi = typeof(decimal).GetField("hi", BindingFlags.NonPublic);
|
||||
dec_flags = typeof(decimal).GetField("flags", BindingFlags.NonPublic);
|
||||
}
|
||||
|
||||
private object[] collect;
|
||||
private readonly int bufferSize;
|
||||
private int collectCount = 0;
|
||||
|
||||
/// <summary>
|
||||
/// Allocates a new binary collector.
|
||||
/// </summary>
|
||||
public BinaryCollector(int bufferSize)
|
||||
{
|
||||
this.bufferSize = bufferSize;
|
||||
for (int i = expired.Count - 1; i >= 0; --i)
|
||||
if (expired[i].TryGetTarget(out collect))
|
||||
{
|
||||
if (collect.Length >= bufferSize)
|
||||
{
|
||||
expired.RemoveAt(i); // This entry he been un-expired for now
|
||||
break;
|
||||
}
|
||||
}
|
||||
else expired.RemoveAt(i); // Entry has been collected by GC
|
||||
if (collect == null || collect.Length < bufferSize)
|
||||
collect = new object[bufferSize];
|
||||
}
|
||||
|
||||
public void Push<T>(T b)
|
||||
{
|
||||
if (b is string || b.GetType().IsArray || IsSupportedType(b.GetType()))
|
||||
collect[collectCount++] = b is string ? Encoding.UTF8.GetBytes(b as string) : b as object;
|
||||
//else
|
||||
// Debug.LogWarning("MLAPI: The type \"" + b.GetType() + "\" is not supported by the Binary Serializer. It will be ignored");
|
||||
}
|
||||
|
||||
public byte[] ToArray()
|
||||
{
|
||||
long bitCount = 0;
|
||||
for (int i = 0; i < collectCount; ++i) bitCount += GetBitCount(collect[i]);
|
||||
|
||||
byte[] alloc = new byte[(bitCount / 8) + (bitCount % 8 == 0 ? 0 : 1)];
|
||||
long bitOffset = 0;
|
||||
foreach (var item in collect)
|
||||
Serialize(item, alloc, ref bitOffset);
|
||||
|
||||
return alloc;
|
||||
}
|
||||
|
||||
private static void Serialize<T>(T t, byte[] writeTo, ref long bitOffset)
|
||||
{
|
||||
Type type = t.GetType();
|
||||
bool size = false;
|
||||
if (type.IsArray)
|
||||
{
|
||||
var array = t as Array;
|
||||
Serialize((ushort)array.Length, writeTo, ref bitOffset);
|
||||
foreach (var element in array)
|
||||
Serialize(element, writeTo, ref bitOffset);
|
||||
}
|
||||
else if (IsSupportedType(type))
|
||||
{
|
||||
long offset = GetBitAllocation(type);
|
||||
if (type == typeof(bool))
|
||||
{
|
||||
WriteBit(writeTo, t as bool? ?? false, bitOffset);
|
||||
bitOffset += offset;
|
||||
}
|
||||
else if (type == typeof(decimal))
|
||||
{
|
||||
WriteDynamic(writeTo, dec_lo.GetValue(t), 4, bitOffset);
|
||||
WriteDynamic(writeTo, dec_mid.GetValue(t), 4, bitOffset + 32);
|
||||
WriteDynamic(writeTo, dec_hi.GetValue(t), 4, bitOffset + 64);
|
||||
WriteDynamic(writeTo, dec_flags.GetValue(t), 4, bitOffset + 96);
|
||||
bitOffset += offset;
|
||||
}
|
||||
else if ((size = type == typeof(float)) || type == typeof(double))
|
||||
{
|
||||
int bytes = size ? 4 : 8;
|
||||
Array type_holder = size ? holder_f as Array : holder_d as Array; // Fetch the preallocated array
|
||||
Array result_holder = size ? holder_i as Array : holder_u as Array;
|
||||
lock (result_holder)
|
||||
lock (type_holder)
|
||||
{
|
||||
// Clear artifacts
|
||||
if (size) result_holder.SetValue(0U, 0);
|
||||
else result_holder.SetValue(0UL, 0);
|
||||
type_holder.SetValue(t, 0); // Insert the value to convert into the preallocated holder array
|
||||
Buffer.BlockCopy(type_holder, 0, result_holder, 0, bytes); // Perform an internal copy to the byte-based holder
|
||||
dynamic d = result_holder.GetValue(0);
|
||||
|
||||
// Since floating point flag bits are seemingly the highest bytes of the floating point values
|
||||
// and even very small values have them, we swap the endianness in the hopes of reducing the size
|
||||
Serialize(Support.SwapEndian(d), writeTo, ref bitOffset);
|
||||
}
|
||||
//bitOffset += offset;
|
||||
}
|
||||
else
|
||||
{
|
||||
bool signed = IsSigned(t.GetType());
|
||||
dynamic value;
|
||||
if (signed)
|
||||
{
|
||||
Type t1 = t.GetType();
|
||||
if (t1 == typeof(sbyte)) value = (byte)ZigZagEncode(t);
|
||||
else if (t1 == typeof(short)) value = (ushort)ZigZagEncode(t);
|
||||
else if (t1 == typeof(int)) value = (uint)ZigZagEncode(t);
|
||||
else /*if (t1 == typeof(long))*/ value = (ulong)ZigZagEncode(t);
|
||||
}
|
||||
else value = t;
|
||||
|
||||
if (value <= 240) WriteByte(writeTo, value, bitOffset);
|
||||
else if (value <= 2287)
|
||||
{
|
||||
WriteByte(writeTo, (value - 240) / 256 + 241, bitOffset);
|
||||
WriteByte(writeTo, (value - 240) % 256, bitOffset + 8);
|
||||
}
|
||||
else if (value <= 67823)
|
||||
{
|
||||
WriteByte(writeTo, 249, bitOffset);
|
||||
WriteByte(writeTo, (value - 2288) / 256, bitOffset + 8);
|
||||
WriteByte(writeTo, (value - 2288) % 256, bitOffset + 16);
|
||||
}
|
||||
else
|
||||
{
|
||||
WriteByte(writeTo, value & 255, bitOffset + 8);
|
||||
WriteByte(writeTo, (value >> 8) & 255, bitOffset + 16);
|
||||
WriteByte(writeTo, (value >> 16) & 255, bitOffset + 24);
|
||||
if (value > 16777215)
|
||||
{
|
||||
WriteByte(writeTo, (value >> 24) & 255, bitOffset + 32);
|
||||
if (value > 4294967295)
|
||||
{
|
||||
WriteByte(writeTo, (value >> 32) & 255, bitOffset + 40);
|
||||
if (value > 1099511627775)
|
||||
{
|
||||
WriteByte(writeTo, (value >> 40) & 55, bitOffset + 48);
|
||||
if (value > 281474976710655)
|
||||
{
|
||||
WriteByte(writeTo, (value >> 48) & 255, bitOffset + 56);
|
||||
if (value > 72057594037927935)
|
||||
{
|
||||
WriteByte(writeTo, 255, bitOffset);
|
||||
WriteByte(writeTo, (value >> 56) & 255, bitOffset + 64);
|
||||
}
|
||||
else WriteByte(writeTo, 254, bitOffset);
|
||||
}
|
||||
else WriteByte(writeTo, 253, bitOffset);
|
||||
}
|
||||
else WriteByte(writeTo, 252, bitOffset);
|
||||
}
|
||||
else WriteByte(writeTo, 251, bitOffset);
|
||||
}
|
||||
else WriteByte(writeTo, 250, bitOffset);
|
||||
}
|
||||
bitOffset += BytesToRead(value) * 8;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private static byte Read7BitRange(byte higher, byte lower, int bottomBits) => (byte)((higher << bottomBits) & (lower & (0xFF << (8-bottomBits))));
|
||||
private static byte ReadNBits(byte from, int offset, int count) => (byte)(from & ((0xFF >> (8-count)) << offset));
|
||||
|
||||
private static bool IsSigned(Type t) => Convert.ToBoolean(t.GetField("MinValue").GetValue(null));
|
||||
|
||||
private static Type GetUnsignedType(Type t) =>
|
||||
t == typeof(sbyte) ? typeof(byte) :
|
||||
t == typeof(short) ? typeof(ushort) :
|
||||
t == typeof(int) ? typeof(uint) :
|
||||
t == typeof(long) ? typeof(ulong) :
|
||||
null;
|
||||
|
||||
private static dynamic ZigZagEncode(dynamic d) => (((d >> (int)(Marshal.SizeOf(d) * 8 - 1))&1) | (d << 1));
|
||||
|
||||
private static long GetBitCount<T>(T t)
|
||||
{
|
||||
Type type = t.GetType();
|
||||
long count = 0;
|
||||
if (type.IsArray)
|
||||
{
|
||||
Type elementType = type.GetElementType();
|
||||
|
||||
count += 16; // Int16 array size. Arrays shouldn't be syncing more than 65k elements
|
||||
foreach (var element in t as Array)
|
||||
count += GetBitCount(element);
|
||||
}
|
||||
else if (IsSupportedType(type))
|
||||
{
|
||||
long ba = GetBitAllocation(type);
|
||||
if (ba == 0) count += Encoding.UTF8.GetByteCount(t as string);
|
||||
else if (t is bool || t is decimal) count += ba;
|
||||
else count += BytesToRead(t) * 8;
|
||||
}
|
||||
//else
|
||||
// Debug.LogWarning("MLAPI: The type \"" + b.GetType() + "\" is not supported by the Binary Serializer. It will be ignored");
|
||||
return count;
|
||||
}
|
||||
|
||||
private static void WriteBit(byte[] b, bool bit, long index)
|
||||
=> b[index / 8] = (byte)((b[index / 8] & ~(1 << (int)(index % 8))) | (bit ? 1 << (int)(index % 8) : 0));
|
||||
private static void WriteByte(byte[] b, dynamic value, long index) => WriteByte(b, (byte)value, index);
|
||||
private static void WriteByte(byte[] b, byte value, long index)
|
||||
{
|
||||
int byteIndex = (int)(index / 8);
|
||||
int shift = (int)(index % 8);
|
||||
byte upper_mask = (byte)(0xFF << shift);
|
||||
byte lower_mask = (byte)~upper_mask;
|
||||
|
||||
b[byteIndex] = (byte)((b[byteIndex] & lower_mask) | (value << shift));
|
||||
if(shift != 0 && byteIndex + 1 < b.Length)
|
||||
b[byteIndex + 1] = (byte)((b[byteIndex + 1] & upper_mask) | (value >> (8 - shift)));
|
||||
}
|
||||
private static void WriteBits(byte[] b, byte value, int bits, int offset, long index)
|
||||
{
|
||||
for (int i = 0; i < bits; ++i)
|
||||
WriteBit(b, (value & (1 << (i + offset))) != 0, index + i);
|
||||
}
|
||||
private static void WriteDynamic(byte[] b, dynamic value, int byteCount, long index)
|
||||
{
|
||||
for (int i = 0; i < byteCount; ++i)
|
||||
WriteByte(b, (byte)((value >> (8 * i)) & 0xFF), index + (8 * i));
|
||||
}
|
||||
|
||||
private static int BytesToRead(dynamic integer)
|
||||
{
|
||||
bool size;
|
||||
if((size=integer is float) || integer is double)
|
||||
{
|
||||
int bytes = size ? 4 : 8;
|
||||
Array type_holder = size ? holder_f as Array : holder_d as Array; // Fetch the preallocated array
|
||||
Array result_holder = size ? holder_i as Array : holder_u as Array;
|
||||
lock (result_holder)
|
||||
lock (type_holder)
|
||||
{
|
||||
// Clear artifacts
|
||||
if (size) result_holder.SetValue(0U, 0);
|
||||
else result_holder.SetValue(0UL, 0);
|
||||
|
||||
type_holder.SetValue(integer, 0); // Insert the value to convert into the preallocated holder array
|
||||
Buffer.BlockCopy(type_holder, 0, result_holder, 0, bytes); // Perform an internal copy to the byte-based holder
|
||||
integer = Support.SwapEndian(integer = result_holder.GetValue(0));
|
||||
}
|
||||
}
|
||||
return
|
||||
integer <= 240 ? 1 :
|
||||
integer <= 2287 ? 2 :
|
||||
integer <= 67823 ? 3 :
|
||||
integer <= 16777215 ? 4 :
|
||||
integer <= 4294967295 ? 5 :
|
||||
integer <= 1099511627775 ? 6 :
|
||||
integer <= 281474976710655 ? 7 :
|
||||
integer <= 72057594037927935 ? 8 :
|
||||
9;
|
||||
}
|
||||
|
||||
// Supported datatypes for serialization
|
||||
private static bool IsSupportedType(Type t) => supportedTypes.Contains(t);
|
||||
|
||||
// Specifies how many bits will be written
|
||||
private static long GetBitAllocation(Type t) =>
|
||||
t == typeof(bool) ? 1 :
|
||||
t == typeof(byte) ? 8 :
|
||||
t == typeof(sbyte) ? 8 :
|
||||
t == typeof(short) ? 16 :
|
||||
t == typeof(char) ? 16 :
|
||||
t == typeof(ushort) ? 16 :
|
||||
t == typeof(int) ? 32 :
|
||||
t == typeof(uint) ? 32 :
|
||||
t == typeof(long) ? 64 :
|
||||
t == typeof(ulong) ? 64 :
|
||||
t == typeof(float) ? 32 :
|
||||
t == typeof(double) ? 64 :
|
||||
t == typeof(decimal) ? 128 :
|
||||
0; // Unknown type
|
||||
|
||||
// Creates a weak reference to the allocated collector so that reuse may be possible
|
||||
public void Dispose()
|
||||
{
|
||||
expired.Add(new WeakReference<object[]>(collect));
|
||||
collect = null;
|
||||
}
|
||||
}
|
||||
}
|
@ -1,108 +0,0 @@
|
||||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.Linq;
|
||||
using System.Runtime.InteropServices;
|
||||
using System.Text;
|
||||
using System.Threading.Tasks;
|
||||
using Tofvesson.Crypto;
|
||||
|
||||
namespace Tofvesson.Common
|
||||
{
|
||||
public class BinaryDistributor
|
||||
{
|
||||
private static readonly byte[] holder = new byte[8];
|
||||
private static readonly float[] holder_f = new float[1];
|
||||
private static readonly double[] holder_d = new double[1];
|
||||
private static readonly ulong[] holder_u = new ulong[1];
|
||||
private static readonly uint[] holder_i = new uint[1];
|
||||
|
||||
private readonly byte[] readFrom;
|
||||
private long bitCount = 0;
|
||||
public BinaryDistributor(byte[] readFrom) => this.readFrom = readFrom;
|
||||
|
||||
public bool ReadBit()
|
||||
{
|
||||
bool result = (readFrom[bitCount / 8] & (byte)(1 << (int)(bitCount % 8))) != 0;
|
||||
++bitCount;
|
||||
return result;
|
||||
}
|
||||
|
||||
public byte ReadByte()
|
||||
{
|
||||
int shift = (int)(bitCount % 8);
|
||||
int index = (int)(bitCount / 8);
|
||||
byte lower_mask = (byte)(0xFF << shift);
|
||||
byte upper_mask = (byte)~lower_mask;
|
||||
byte result = (byte)(((readFrom[index] & lower_mask) >> shift) | (shift == 0 ? 0 : (readFrom[index + 1] & upper_mask) << (8 - shift)));
|
||||
bitCount += 8;
|
||||
return result;
|
||||
}
|
||||
|
||||
public float ReadFloat() => ReadFloating<float>();
|
||||
public double ReadDouble() => ReadFloating<double>();
|
||||
public float[] ReadFloatArray() => ReadFloatingArray<float>();
|
||||
public double[] ReadDoubleArray() => ReadFloatingArray<double>();
|
||||
public ushort ReadUShort() => ReadUnsigned<ushort>();
|
||||
public uint ReadUInt() => ReadUnsigned<uint>();
|
||||
public ulong ReadULong() => ReadUnsigned<ulong>();
|
||||
public sbyte ReadSByte() => (sbyte)ZigZagDecode(ReadByte());
|
||||
public short ReadShort() => (short)ZigZagDecode(ReadUShort());
|
||||
public int ReadInt() => (int)ZigZagDecode(ReadUInt());
|
||||
public long ReadLong() => (long)ZigZagDecode(ReadULong());
|
||||
|
||||
private T ReadUnsigned<T>()
|
||||
{
|
||||
dynamic header = ReadByte();
|
||||
if (header <= 240) return (T) header;
|
||||
if (header <= 248) return (T) (240 + 256 * (header - 241) + ReadByte());
|
||||
if (header == 249) return (T) (header = 2288 + 256 * ReadByte() + ReadByte());
|
||||
dynamic res = ReadByte() | ((long)ReadByte() << 8) | ((long)ReadByte() << 16);
|
||||
if(header > 250)
|
||||
{
|
||||
res |= (long) ReadByte() << 24;
|
||||
if(header > 251)
|
||||
{
|
||||
res |= (long)ReadByte() << 32;
|
||||
if(header > 252)
|
||||
{
|
||||
res |= (long)ReadByte() << 40;
|
||||
if (header > 253)
|
||||
{
|
||||
res |= (long)ReadByte() << 48;
|
||||
if (header > 254) res |= (long)ReadByte() << 56;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
return (T) res;
|
||||
}
|
||||
private T[] ReadFloatingArray<T>()
|
||||
{
|
||||
ushort size = ReadUShort();
|
||||
T[] result = new T[size];
|
||||
for (short s = 0; s < size; ++s)
|
||||
result[s] = ReadFloating<T>();
|
||||
return result;
|
||||
}
|
||||
|
||||
private T ReadFloating<T>()
|
||||
{
|
||||
int size = Marshal.SizeOf(typeof(T));
|
||||
Array type_holder = size == 4 ? holder_f as Array : holder_d as Array;
|
||||
Array result_holder = size == 4 ? holder_i as Array : holder_u as Array;
|
||||
T result;
|
||||
lock(result_holder)
|
||||
lock (type_holder)
|
||||
{
|
||||
//for (int i = 0; i < size; ++i)
|
||||
// holder.SetValue(ReadByte(), i);
|
||||
if (size == 4) result_holder.SetValue(Support.SwapEndian(ReadUInt()), 0);
|
||||
else result_holder.SetValue(Support.SwapEndian(ReadULong()), 0);
|
||||
Buffer.BlockCopy(result_holder, 0, type_holder, 0, size);
|
||||
result = (T)type_holder.GetValue(0);
|
||||
}
|
||||
return result;
|
||||
}
|
||||
private static dynamic ZigZagDecode(dynamic d) => (((d << (int)(Marshal.SizeOf(d) * 8 - 1)) & 1) | (d >> 1));
|
||||
}
|
||||
}
|
137
Common/BinaryHelpers.cs
Normal file
137
Common/BinaryHelpers.cs
Normal file
@ -0,0 +1,137 @@
|
||||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.Linq;
|
||||
using System.Text;
|
||||
|
||||
namespace Tofvesson.Common
|
||||
{
|
||||
public static class BinaryHelpers
|
||||
{
|
||||
// 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 ulong SwapEndian(ulong value) =>
|
||||
((value >> 56) & 0xFF) |
|
||||
((value >> 40) & (0xFFUL << 8)) |
|
||||
((value >> 24) & (0xFFUL << 16)) |
|
||||
((value >> 8) & (0xFFUL << 24)) |
|
||||
((value << 56) & (0xFFUL << 56)) |
|
||||
((value << 40) & (0xFFUL << 48)) |
|
||||
((value << 24) & (0xFFUL << 40)) |
|
||||
((value << 8) & (0xFFUL << 32)) ;
|
||||
|
||||
|
||||
// How many bytes to write
|
||||
public static int VarIntSize(dynamic integer) =>
|
||||
integer is byte ||
|
||||
integer <= 240 ? 1 :
|
||||
integer <= 2287 ? 2 :
|
||||
integer <= 67823 ? 3 :
|
||||
integer <= 16777215 ? 4 :
|
||||
integer <= 4294967295 ? 5 :
|
||||
integer <= 1099511627775 ? 6 :
|
||||
integer <= 281474976710655 ? 7 :
|
||||
integer <= 72057594037927935 ? 8 :
|
||||
9;
|
||||
|
||||
public static ulong ReadVarInt(IEnumerable<byte> from, int offset)
|
||||
{
|
||||
ulong header = from.ElementAt(0);
|
||||
if (header <= 240) return header;
|
||||
if (header <= 248) return 240 + 256 * (header - 241) + from.ElementAt(1);
|
||||
if (header == 249) return 2288 + 256UL * from.ElementAt(1) + from.ElementAt(2);
|
||||
ulong res = from.ElementAt(1) | ((ulong)from.ElementAt(2) << 8) | ((ulong)from.ElementAt(3) << 16);
|
||||
if (header > 250)
|
||||
{
|
||||
res |= (ulong)from.ElementAt(4) << 24;
|
||||
if (header > 251)
|
||||
{
|
||||
res |= (ulong)from.ElementAt(5) << 32;
|
||||
if (header > 252)
|
||||
{
|
||||
res |= (ulong)from.ElementAt(6) << 40;
|
||||
if (header > 253)
|
||||
{
|
||||
res |= (ulong)from.ElementAt(7) << 48;
|
||||
if (header > 254) res |= (ulong)from.ElementAt(8) << 56;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
return res;
|
||||
}
|
||||
|
||||
public static bool TryReadVarInt(IEnumerable<byte> from, int offset, out int result)
|
||||
{
|
||||
bool b = TryReadVarInt(from, offset, out ulong res);
|
||||
result = (int)res;
|
||||
return b;
|
||||
}
|
||||
public static bool TryReadVarInt(IEnumerable<byte> from, int offset, out ulong result)
|
||||
{
|
||||
try
|
||||
{
|
||||
result = ReadVarInt(from, offset);
|
||||
return true;
|
||||
}
|
||||
catch
|
||||
{
|
||||
result = 0;
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
public static void WriteVarInt(byte[] to, int offset, dynamic t)
|
||||
{
|
||||
if (t is byte)
|
||||
{
|
||||
to[offset] = (byte)t;
|
||||
return;
|
||||
}
|
||||
|
||||
if (t <= 240) to[offset] = (byte)t;
|
||||
else if (t <= 2287)
|
||||
{
|
||||
to[offset] = (byte)((t - 240) / 256 + 241);
|
||||
to[offset + 1] = (byte)((t - 240) % 256);
|
||||
}
|
||||
else if (t <= 67823)
|
||||
{
|
||||
to[offset] = 249;
|
||||
to[offset + 1] = (byte)((t - 2288) / 256);
|
||||
to[offset + 2] = (byte)((t - 2288) % 256);
|
||||
}
|
||||
else
|
||||
{
|
||||
to[offset + 1] = (byte)(t & 0xFF);
|
||||
to[offset + 2] = (byte)((t >> 8) & 0xFF);
|
||||
to[offset + 3] = (byte)((t >> 16) & 0xFF);
|
||||
if (t > 16777215)
|
||||
{
|
||||
to[offset + 4] = (byte)((t >> 24) & 0xFF);
|
||||
if (t > 4294967295)
|
||||
{
|
||||
to[offset + 5] = (byte)((t >> 32) & 0xFF);
|
||||
if (t > 1099511627775)
|
||||
{
|
||||
to[offset + 6] = (byte)((t >> 40) & 0xFF);
|
||||
if (t > 281474976710655)
|
||||
{
|
||||
to[offset + 7] = (byte)((t >> 48) & 0xFF);
|
||||
if (t > 72057594037927935)
|
||||
{
|
||||
to[offset] = 255;
|
||||
to[offset + 8] = (byte)((t >> 56) & 0xFF);
|
||||
}
|
||||
else to[offset] = 254;
|
||||
}
|
||||
else to[offset] = 253;
|
||||
}
|
||||
else to[offset] = 252;
|
||||
}
|
||||
else to[offset] = 251;
|
||||
}
|
||||
else to[offset] = 250;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
111
Common/BitReader.cs
Normal file
111
Common/BitReader.cs
Normal file
@ -0,0 +1,111 @@
|
||||
using System;
|
||||
using System.Runtime.InteropServices;
|
||||
using System.Text;
|
||||
|
||||
namespace Tofvesson.Common
|
||||
{
|
||||
public class BitReader
|
||||
{
|
||||
private delegate T Getter<T>();
|
||||
private static readonly float[] holder_f = new float[1];
|
||||
private static readonly double[] holder_d = new double[1];
|
||||
private static readonly ulong[] holder_u = new ulong[1];
|
||||
private static readonly uint[] holder_i = new uint[1];
|
||||
|
||||
private readonly byte[] readFrom;
|
||||
private long bitCount = 0;
|
||||
public BitReader(byte[] readFrom) => this.readFrom = readFrom;
|
||||
|
||||
public bool ReadBool()
|
||||
{
|
||||
bool result = (readFrom[bitCount / 8] & (byte)(1 << (int)(bitCount % 8))) != 0;
|
||||
++bitCount;
|
||||
return result;
|
||||
}
|
||||
|
||||
public float ReadFloat() => ReadFloating<float>();
|
||||
public double ReadDouble() => ReadFloating<double>();
|
||||
public byte ReadByte()
|
||||
{
|
||||
int shift = (int)(bitCount % 8);
|
||||
int index = (int)(bitCount / 8);
|
||||
byte lower_mask = (byte)(0xFF << shift);
|
||||
byte upper_mask = (byte)~lower_mask;
|
||||
byte result = (byte)(((readFrom[index] & lower_mask) >> shift) | (shift == 0 ? 0 : (readFrom[index + 1] & upper_mask) << (8 - shift)));
|
||||
bitCount += 8;
|
||||
return result;
|
||||
}
|
||||
public void SkipPadded() => bitCount += (8 - (bitCount % 8)) % 8;
|
||||
public ushort ReadUShort() => (ushort)ReadULong();
|
||||
public uint ReadUInt() => (uint)ReadULong();
|
||||
public sbyte ReadSByte() => (sbyte)ZigZagDecode(ReadByte(), 1);
|
||||
public short ReadShort() => (short)ZigZagDecode(ReadUShort(), 2);
|
||||
public int ReadInt() => (int)ZigZagDecode(ReadUInt(), 4);
|
||||
public long ReadLong() => ZigZagDecode(ReadULong(), 8);
|
||||
public float[] ReadFloatArray(int known = -1) => ReadArray(ReadFloat, known);
|
||||
public double[] ReadDoubleArray(int known = -1) => ReadArray(ReadDouble, known);
|
||||
public byte[] ReadByteArray(int known = -1) => ReadArray(ReadByte, known);
|
||||
public ushort[] ReadUShortArray(int known = -1) => ReadArray(ReadUShort, known);
|
||||
public uint[] ReadUIntArray(int known = -1) => ReadArray(ReadUInt, known);
|
||||
public ulong[] ReadULongArray(int known = -1) => ReadArray(ReadULong, known);
|
||||
public sbyte[] ReadSByteArray(int known = -1) => ReadArray(ReadSByte, known);
|
||||
public short[] ReadShortArray(int known = -1) => ReadArray(ReadShort, known);
|
||||
public int[] ReadIntArray(int known = -1) => ReadArray(ReadInt, known);
|
||||
public long[] ReadLongArray(int known = -1) => ReadArray(ReadLong, known);
|
||||
public string ReadString() => Encoding.UTF8.GetString(ReadByteArray());
|
||||
|
||||
public ulong ReadULong()
|
||||
{
|
||||
ulong header = ReadByte();
|
||||
if (header <= 240) return header;
|
||||
if (header <= 248) return 240 + 256 * (header - 241) + ReadByte();
|
||||
if (header == 249) return 2288 + 256UL * ReadByte() + ReadByte();
|
||||
ulong res = ReadByte() | ((ulong)ReadByte() << 8) | ((ulong)ReadByte() << 16);
|
||||
if(header > 250)
|
||||
{
|
||||
res |= (ulong) ReadByte() << 24;
|
||||
if(header > 251)
|
||||
{
|
||||
res |= (ulong)ReadByte() << 32;
|
||||
if(header > 252)
|
||||
{
|
||||
res |= (ulong)ReadByte() << 40;
|
||||
if (header > 253)
|
||||
{
|
||||
res |= (ulong)ReadByte() << 48;
|
||||
if (header > 254) res |= (ulong)ReadByte() << 56;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
return res;
|
||||
}
|
||||
private T[] ReadArray<T>(Getter<T> g, int knownSize = -1)
|
||||
{
|
||||
T[] result = new T[knownSize > 0 ? (uint)knownSize : ReadUInt()];
|
||||
for (ushort s = 0; s < result.Length; ++s)
|
||||
result[s] = g();
|
||||
return result;
|
||||
}
|
||||
|
||||
private T ReadFloating<T>()
|
||||
{
|
||||
int size = Marshal.SizeOf(typeof(T));
|
||||
Array type_holder = size == 4 ? holder_f as Array : holder_d as Array;
|
||||
Array result_holder = size == 4 ? holder_i as Array : holder_u as Array;
|
||||
T result;
|
||||
lock(result_holder)
|
||||
lock (type_holder)
|
||||
{
|
||||
//for (int i = 0; i < size; ++i)
|
||||
// holder.SetValue(ReadByte(), i);
|
||||
if (size == 4) result_holder.SetValue(BinaryHelpers.SwapEndian(ReadUInt()), 0);
|
||||
else result_holder.SetValue(BinaryHelpers.SwapEndian(ReadULong()), 0);
|
||||
Buffer.BlockCopy(result_holder, 0, type_holder, 0, size);
|
||||
result = (T)type_holder.GetValue(0);
|
||||
}
|
||||
return result;
|
||||
}
|
||||
private static long ZigZagDecode(ulong d, int bytes) => (long)(((d << (bytes * 8 - 1)) & 1) | (d >> 1));
|
||||
}
|
||||
}
|
404
Common/BitWriter.cs
Normal file
404
Common/BitWriter.cs
Normal file
@ -0,0 +1,404 @@
|
||||
#define UNSAFE_PUSH
|
||||
|
||||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.Diagnostics;
|
||||
using System.Reflection;
|
||||
using System.Runtime.InteropServices;
|
||||
using System.Text;
|
||||
|
||||
namespace Tofvesson.Common
|
||||
{
|
||||
public sealed class BitWriter : IDisposable
|
||||
{
|
||||
private const int PREALLOC_COLLECT = 10;
|
||||
private static readonly Queue<List<object>> listPool = new Queue<List<object>>();
|
||||
|
||||
private static readonly float[] holder_f = new float[1];
|
||||
private static readonly double[] holder_d = new double[1];
|
||||
private static readonly ulong[] holder_u = new ulong[1];
|
||||
private static readonly uint[] holder_i = new uint[1];
|
||||
private static readonly List<Type> supportedTypes = new List<Type>()
|
||||
{
|
||||
typeof(bool),
|
||||
typeof(byte),
|
||||
typeof(sbyte),
|
||||
typeof(char),
|
||||
typeof(short),
|
||||
typeof(ushort),
|
||||
typeof(int),
|
||||
typeof(uint),
|
||||
typeof(long),
|
||||
typeof(ulong),
|
||||
typeof(float),
|
||||
typeof(double),
|
||||
typeof(decimal)
|
||||
};
|
||||
|
||||
private static readonly FieldInfo
|
||||
dec_lo,
|
||||
dec_mid,
|
||||
dec_hi,
|
||||
dec_flags;
|
||||
|
||||
static BitWriter()
|
||||
{
|
||||
dec_lo = typeof(decimal).GetField("lo", BindingFlags.NonPublic);
|
||||
dec_mid = typeof(decimal).GetField("mid", BindingFlags.NonPublic);
|
||||
dec_hi = typeof(decimal).GetField("hi", BindingFlags.NonPublic);
|
||||
dec_flags = typeof(decimal).GetField("flags", BindingFlags.NonPublic);
|
||||
|
||||
for (int i = 0; i < PREALLOC_COLLECT; i++)
|
||||
{
|
||||
listPool.Enqueue(new List<object>());
|
||||
}
|
||||
}
|
||||
|
||||
private List<object> collect = null;
|
||||
private bool tempAlloc = false;
|
||||
|
||||
/// <summary>
|
||||
/// Allocates a new binary collector.
|
||||
/// </summary>
|
||||
public BitWriter()
|
||||
{
|
||||
if (listPool.Count == 0)
|
||||
{
|
||||
Debug.WriteLine("BitWriter: Optimized for "+ PREALLOC_COLLECT + " BitWriters. Have you forgotten to dispose?");
|
||||
collect = new List<object>();
|
||||
tempAlloc = true;
|
||||
}
|
||||
else
|
||||
{
|
||||
collect = listPool.Dequeue();
|
||||
}
|
||||
}
|
||||
|
||||
#if UNSAFE_PUSH
|
||||
public
|
||||
#else
|
||||
private
|
||||
# endif
|
||||
void Push<T>(T b, bool known = false)
|
||||
{
|
||||
if (b == null) collect.Add(b);
|
||||
else if (b is string || b.GetType().IsArray || IsSupportedType(b.GetType()))
|
||||
collect.Add(b is string ? Encoding.UTF8.GetBytes(b as string) : b as object);
|
||||
else
|
||||
Debug.WriteLine("BitWriter: The type \"" + b.GetType() + "\" is not supported by the Binary Serializer. It will be ignored!");
|
||||
}
|
||||
|
||||
|
||||
public void WriteBool(bool b) => Push(b);
|
||||
public void WriteFloat(float f) => Push(f);
|
||||
public void WriteDouble(double d) => Push(d);
|
||||
public void WriteByte(byte b) => Push(b);
|
||||
public void WriteUShort(ushort s) => Push(s);
|
||||
public void WriteUInt(uint i) => Push(i);
|
||||
public void WriteULong(ulong l) => Push(l);
|
||||
public void WriteSByte(sbyte b) => Push(ZigZagEncode(b, 8));
|
||||
public void WriteShort(short s) => Push(ZigZagEncode(s, 8));
|
||||
public void WriteInt(int i) => Push(ZigZagEncode(i, 8));
|
||||
public void WriteLong(long l) => Push(ZigZagEncode(l, 8));
|
||||
public void WriteString(string s) => Push(s);
|
||||
public void WriteAlignBits() => Push<object>(null);
|
||||
public void WriteFloatArray(float[] f, bool known = false) => PushArray(f, known);
|
||||
public void WriteDoubleArray(double[] d, bool known = false) => PushArray(d, known);
|
||||
public void WriteByteArray(byte[] b, bool known = false) => PushArray(b, known);
|
||||
public void WriteUShortArray(ushort[] s, bool known = false) => PushArray(s, known);
|
||||
public void WriteUIntArray(uint[] i, bool known = false) => PushArray(i, known);
|
||||
public void WriteULongArray(ulong[] l, bool known = false) => PushArray(l, known);
|
||||
public void WriteSByteArray(sbyte[] b, bool known = false) => PushArray(b, known);
|
||||
public void WriteShortArray(short[] s, bool known = false) => PushArray(s, known);
|
||||
public void WriteIntArray(int[] i, bool known = false) => PushArray(i, known);
|
||||
public void WriteLongArray(long[] l, bool known = false) => PushArray(l, known);
|
||||
|
||||
#if UNSAFE_PUSH
|
||||
public
|
||||
#else
|
||||
private
|
||||
#endif
|
||||
void PushArray<T>(T[] t, bool knownSize = false)
|
||||
{
|
||||
if (!knownSize) Push((uint)t.Length);
|
||||
bool signed = IsSigned(t.GetType().GetElementType());
|
||||
int size = Marshal.SizeOf(t.GetType().GetElementType());
|
||||
foreach (T t1 in t) Push(signed ? (object)ZigZagEncode(t1 as long? ?? t1 as int? ?? t1 as short? ?? t1 as sbyte? ?? 0, size) : (object)t1);
|
||||
}
|
||||
|
||||
public byte[] Finalize()
|
||||
{
|
||||
byte[] b = new byte[GetFinalizeSize()];
|
||||
Finalize(ref b);
|
||||
return b;
|
||||
}
|
||||
public long Finalize(ref byte[] buffer)
|
||||
{
|
||||
if(buffer == null)
|
||||
{
|
||||
Debug.WriteLine("BitWriter: no buffer provided");
|
||||
return 0;
|
||||
}
|
||||
long bitCount = 0;
|
||||
for (int i = 0; i < collect.Count; ++i) bitCount += collect[i] == null ? (8 - (bitCount % 8)) % 8 : GetBitCount(collect[i]);
|
||||
|
||||
if (buffer.Length < ((bitCount / 8) + (bitCount % 8 == 0 ? 0 : 1)))
|
||||
{
|
||||
Debug.WriteLine("BitWriter: The buffer size is not large enough");
|
||||
return 0;
|
||||
}
|
||||
long bitOffset = 0;
|
||||
bool isAligned = true;
|
||||
foreach (var item in collect)
|
||||
if (item == null)
|
||||
{
|
||||
bitOffset += (8 - (bitOffset % 8)) % 8;
|
||||
isAligned = true;
|
||||
}
|
||||
else Serialize(item, buffer, ref bitOffset, ref isAligned);
|
||||
|
||||
return (bitCount / 8) + (bitCount % 8 == 0 ? 0 : 1);
|
||||
}
|
||||
|
||||
public long GetFinalizeSize()
|
||||
{
|
||||
long bitCount = 0;
|
||||
for (int i = 0; i < collect.Count; ++i) bitCount += collect[i]==null ? (8 - (bitCount % 8)) % 8 : GetBitCount(collect[i]);
|
||||
return ((bitCount / 8) + (bitCount % 8 == 0 ? 0 : 1));
|
||||
}
|
||||
|
||||
private static void Serialize<T>(T t, byte[] writeTo, ref long bitOffset, ref bool isAligned)
|
||||
{
|
||||
Type type = t.GetType();
|
||||
bool size = false;
|
||||
if (type.IsArray)
|
||||
{
|
||||
var array = t as Array;
|
||||
Serialize((uint)array.Length, writeTo, ref bitOffset, ref isAligned);
|
||||
foreach (var element in array)
|
||||
Serialize(element, writeTo, ref bitOffset, ref isAligned);
|
||||
}
|
||||
else if (IsSupportedType(type))
|
||||
{
|
||||
long offset = t is bool ? 1 : BytesToRead(t) * 8;
|
||||
if (type == typeof(bool))
|
||||
{
|
||||
WriteBit(writeTo, t as bool? ?? false, bitOffset);
|
||||
bitOffset += offset;
|
||||
isAligned = bitOffset % 8 == 0;
|
||||
}
|
||||
else if (type == typeof(decimal))
|
||||
{
|
||||
WriteDynamic(writeTo, (int)dec_lo.GetValue(t), 4, bitOffset, isAligned);
|
||||
WriteDynamic(writeTo, (int)dec_mid.GetValue(t), 4, bitOffset + 32, isAligned);
|
||||
WriteDynamic(writeTo, (int)dec_hi.GetValue(t), 4, bitOffset + 64, isAligned);
|
||||
WriteDynamic(writeTo, (int)dec_flags.GetValue(t), 4, bitOffset + 96, isAligned);
|
||||
bitOffset += offset;
|
||||
}
|
||||
else if ((size = type == typeof(float)) || type == typeof(double))
|
||||
{
|
||||
int bytes = size ? 4 : 8;
|
||||
Array type_holder = size ? holder_f as Array : holder_d as Array; // Fetch the preallocated array
|
||||
Array result_holder = size ? holder_i as Array : holder_u as Array;
|
||||
lock (result_holder)
|
||||
lock (type_holder)
|
||||
{
|
||||
// Clear artifacts
|
||||
if (size) result_holder.SetValue(0U, 0);
|
||||
else result_holder.SetValue(0UL, 0);
|
||||
type_holder.SetValue(t, 0); // Insert the value to convert into the preallocated holder array
|
||||
Buffer.BlockCopy(type_holder, 0, result_holder, 0, bytes); // Perform an internal copy to the byte-based holder
|
||||
|
||||
// Since floating point flag bits are seemingly the highest bytes of the floating point values
|
||||
// and even very small values have them, we swap the endianness in the hopes of reducing the size
|
||||
if(size) Serialize(BinaryHelpers.SwapEndian((uint)result_holder.GetValue(0)), writeTo, ref bitOffset, ref isAligned);
|
||||
else Serialize(BinaryHelpers.SwapEndian((ulong)result_holder.GetValue(0)), writeTo, ref bitOffset, ref isAligned);
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
//bool signed = IsSigned(t.GetType());
|
||||
ulong value;
|
||||
/*if (signed)
|
||||
{
|
||||
Type t1 = t.GetType();
|
||||
if (t1 == typeof(sbyte)) value = (byte)ZigZagEncode(t as sbyte? ?? 0, 1);
|
||||
else if (t1 == typeof(short)) value = (ushort)ZigZagEncode(t as short? ?? 0, 2);
|
||||
else if (t1 == typeof(int)) value = (uint)ZigZagEncode(t as int? ?? 0, 4);
|
||||
else /*if (t1 == typeof(long)) value = (ulong)ZigZagEncode(t as long? ?? 0, 8);
|
||||
}
|
||||
else*/
|
||||
if (t is byte)
|
||||
{
|
||||
WriteByte(writeTo, t as byte? ?? 0, bitOffset, isAligned);
|
||||
bitOffset += 8;
|
||||
return;
|
||||
}
|
||||
else if (t is ushort) value = t as ushort? ?? 0;
|
||||
else if (t is uint) value = t as uint? ?? 0;
|
||||
else /*if (t is ulong)*/ value = t as ulong? ?? 0;
|
||||
|
||||
if (value <= 240) WriteByte(writeTo, (byte)value, bitOffset, isAligned);
|
||||
else if (value <= 2287)
|
||||
{
|
||||
WriteByte(writeTo, (value - 240) / 256 + 241, bitOffset, isAligned);
|
||||
WriteByte(writeTo, (value - 240) % 256, bitOffset + 8, isAligned);
|
||||
}
|
||||
else if (value <= 67823)
|
||||
{
|
||||
WriteByte(writeTo, 249, bitOffset, isAligned);
|
||||
WriteByte(writeTo, (value - 2288) / 256, bitOffset + 8, isAligned);
|
||||
WriteByte(writeTo, (value - 2288) % 256, bitOffset + 16, isAligned);
|
||||
}
|
||||
else
|
||||
{
|
||||
WriteByte(writeTo, value & 255, bitOffset + 8, isAligned);
|
||||
WriteByte(writeTo, (value >> 8) & 255, bitOffset + 16, isAligned);
|
||||
WriteByte(writeTo, (value >> 16) & 255, bitOffset + 24, isAligned);
|
||||
if (value > 16777215)
|
||||
{
|
||||
WriteByte(writeTo, (value >> 24) & 255, bitOffset + 32, isAligned);
|
||||
if (value > 4294967295)
|
||||
{
|
||||
WriteByte(writeTo, (value >> 32) & 255, bitOffset + 40, isAligned);
|
||||
if (value > 1099511627775)
|
||||
{
|
||||
WriteByte(writeTo, (value >> 40) & 55, bitOffset + 48, isAligned);
|
||||
if (value > 281474976710655)
|
||||
{
|
||||
WriteByte(writeTo, (value >> 48) & 255, bitOffset + 56, isAligned);
|
||||
if (value > 72057594037927935)
|
||||
{
|
||||
WriteByte(writeTo, 255, bitOffset, isAligned);
|
||||
WriteByte(writeTo, (value >> 56) & 255, bitOffset + 64, isAligned);
|
||||
}
|
||||
else WriteByte(writeTo, 254, bitOffset, isAligned);
|
||||
}
|
||||
else WriteByte(writeTo, 253, bitOffset, isAligned);
|
||||
}
|
||||
else WriteByte(writeTo, 252, bitOffset, isAligned);
|
||||
}
|
||||
else WriteByte(writeTo, 251, bitOffset, isAligned);
|
||||
}
|
||||
else WriteByte(writeTo, 250, bitOffset, isAligned);
|
||||
}
|
||||
bitOffset += BytesToRead(value) * 8;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private static byte Read7BitRange(byte higher, byte lower, int bottomBits) => (byte)((higher << bottomBits) & (lower & (0xFF << (8-bottomBits))));
|
||||
private static byte ReadNBits(byte from, int offset, int count) => (byte)(from & ((0xFF >> (8-count)) << offset));
|
||||
|
||||
private static bool IsSigned(Type t) => t == typeof(sbyte) || t == typeof(short) || t == typeof(int) || t == typeof(long);
|
||||
|
||||
private static Type GetUnsignedType(Type t) =>
|
||||
t == typeof(sbyte) ? typeof(byte) :
|
||||
t == typeof(short) ? typeof(ushort) :
|
||||
t == typeof(int) ? typeof(uint) :
|
||||
t == typeof(long) ? typeof(ulong) :
|
||||
null;
|
||||
|
||||
private static ulong ZigZagEncode(long d, int bytes) => (ulong)(((d >> (bytes * 8 - 1))&1) | (d << 1));
|
||||
|
||||
private static long GetBitCount<T>(T t)
|
||||
{
|
||||
Type type = t.GetType();
|
||||
long count = 0;
|
||||
if (type.IsArray)
|
||||
{
|
||||
Type elementType = type.GetElementType();
|
||||
|
||||
count += BytesToRead((t as Array).Length) * 8; // Int16 array size. Arrays shouldn't be syncing more than 65k elements
|
||||
|
||||
if (elementType == typeof(bool)) count += (t as Array).Length;
|
||||
else
|
||||
foreach (var element in t as Array)
|
||||
count += GetBitCount(element);
|
||||
}
|
||||
else if (IsSupportedType(type))
|
||||
{
|
||||
long ba = t is bool ? 1 : BytesToRead(t)*8;
|
||||
if (ba == 0) count += Encoding.UTF8.GetByteCount(t as string);
|
||||
else if (t is bool || t is decimal) count += ba;
|
||||
else count += BytesToRead(t) * 8;
|
||||
}
|
||||
//else
|
||||
// Debug.LogWarning("MLAPI: The type \"" + b.GetType() + "\" is not supported by the Binary Serializer. It will be ignored");
|
||||
return count;
|
||||
}
|
||||
|
||||
private static void WriteBit(byte[] b, bool bit, long index)
|
||||
=> b[index / 8] = (byte)((b[index / 8] & ~(1 << (int)(index % 8))) | (bit ? 1 << (int)(index % 8) : 0));
|
||||
private static void WriteByte(byte[] b, ulong value, long index, bool isAligned) => WriteByte(b, (byte)value, index, isAligned);
|
||||
private static void WriteByte(byte[] b, byte value, long index, bool isAligned)
|
||||
{
|
||||
if (isAligned) b[index / 8] = value;
|
||||
else
|
||||
{
|
||||
int byteIndex = (int)(index / 8);
|
||||
int shift = (int)(index % 8);
|
||||
byte upper_mask = (byte)(0xFF << shift);
|
||||
|
||||
b[byteIndex] = (byte)((b[byteIndex] & (byte)~upper_mask) | (value << shift));
|
||||
b[byteIndex + 1] = (byte)((b[byteIndex + 1] & upper_mask) | (value >> (8 - shift)));
|
||||
}
|
||||
}
|
||||
private static void WriteDynamic(byte[] b, int value, int byteCount, long index, bool isAligned)
|
||||
{
|
||||
for (int i = 0; i < byteCount; ++i)
|
||||
WriteByte(b, (byte)((value >> (8 * i)) & 0xFF), index + (8 * i), isAligned);
|
||||
}
|
||||
|
||||
private static int BytesToRead(object i)
|
||||
{
|
||||
if (i is byte) return 1;
|
||||
bool size;
|
||||
ulong integer;
|
||||
if (i is decimal) return BytesToRead((int)dec_flags.GetValue(i)) + BytesToRead((int)dec_lo.GetValue(i)) + BytesToRead((int)dec_mid.GetValue(i)) + BytesToRead((int)dec_hi.GetValue(i));
|
||||
if ((size = i is float) || i is double)
|
||||
{
|
||||
int bytes = size ? 4 : 8;
|
||||
Array type_holder = size ? holder_f as Array : holder_d as Array; // Fetch the preallocated array
|
||||
Array result_holder = size ? holder_i as Array : holder_u as Array;
|
||||
lock (result_holder)
|
||||
lock (type_holder)
|
||||
{
|
||||
// Clear artifacts
|
||||
if (size) result_holder.SetValue(0U, 0);
|
||||
else result_holder.SetValue(0UL, 0);
|
||||
|
||||
type_holder.SetValue(i, 0); // Insert the value to convert into the preallocated holder array
|
||||
Buffer.BlockCopy(type_holder, 0, result_holder, 0, bytes); // Perform an internal copy to the byte-based holder
|
||||
if(size) integer = BinaryHelpers.SwapEndian((uint)result_holder.GetValue(0));
|
||||
else integer = BinaryHelpers.SwapEndian((ulong)result_holder.GetValue(0));
|
||||
}
|
||||
}
|
||||
else integer = i as ulong? ?? i as uint? ?? i as ushort? ?? i as byte? ?? 0;
|
||||
return
|
||||
integer <= 240 ? 1 :
|
||||
integer <= 2287 ? 2 :
|
||||
integer <= 67823 ? 3 :
|
||||
integer <= 16777215 ? 4 :
|
||||
integer <= 4294967295 ? 5 :
|
||||
integer <= 1099511627775 ? 6 :
|
||||
integer <= 281474976710655 ? 7 :
|
||||
integer <= 72057594037927935 ? 8 :
|
||||
9;
|
||||
}
|
||||
|
||||
// Supported datatypes for serialization
|
||||
private static bool IsSupportedType(Type t) => supportedTypes.Contains(t);
|
||||
|
||||
// Creates a weak reference to the allocated collector so that reuse may be possible
|
||||
public void Dispose()
|
||||
{
|
||||
if (!tempAlloc)
|
||||
{
|
||||
collect.Clear();
|
||||
listPool.Enqueue(collect);
|
||||
}
|
||||
collect = null; //GC picks this
|
||||
}
|
||||
}
|
||||
}
|
@ -63,8 +63,9 @@
|
||||
</ItemGroup>
|
||||
<ItemGroup>
|
||||
<Compile Include="AccountInfo.cs" />
|
||||
<Compile Include="BinaryCollector.cs" />
|
||||
<Compile Include="BinaryDistributor.cs" />
|
||||
<Compile Include="BinaryHelpers.cs" />
|
||||
<Compile Include="BitReader.cs" />
|
||||
<Compile Include="BitWriter.cs" />
|
||||
<Compile Include="Cryptography\AES.cs" />
|
||||
<Compile Include="Cryptography\CBC.cs" />
|
||||
<Compile Include="Collections.cs" />
|
||||
|
@ -9,6 +9,7 @@ using System.Numerics;
|
||||
using System.Text;
|
||||
using System.Threading;
|
||||
using System.Threading.Tasks;
|
||||
using Tofvesson.Common;
|
||||
using Tofvesson.Crypto;
|
||||
|
||||
namespace Common
|
||||
@ -155,7 +156,7 @@ namespace Common
|
||||
{
|
||||
lock (messageBuffer)
|
||||
{
|
||||
foreach (byte[] message in messageBuffer) Connection.Send(NetSupport.WithHeader(message));
|
||||
foreach (byte[] message in messageBuffer) Connection.Send(NetSupport.WithHeader(Crypto.Encrypt(message)));
|
||||
if (messageBuffer.Count > 0) lastComm = DateTime.Now.Ticks;
|
||||
messageBuffer.Clear();
|
||||
}
|
||||
@ -166,8 +167,8 @@ namespace Common
|
||||
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 && BinaryHelpers.TryReadVarInt(ibuf, 0, out mLen))
|
||||
ibuf.Dequeue(BinaryHelpers.VarIntSize(mLen));
|
||||
if (mLen != 0 && ibuf.Count >= mLen)
|
||||
{
|
||||
// Got a full message. Parse!
|
||||
@ -189,12 +190,13 @@ namespace Common
|
||||
byte[] read = Crypto.Decrypt(message);
|
||||
|
||||
// Read the decrypted message length
|
||||
int mlenInner = Support.ReadInt(read, 0);
|
||||
int mlenInner = (int) BinaryHelpers.ReadVarInt(read, 0);
|
||||
int size = BinaryHelpers.VarIntSize(mlenInner);
|
||||
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);
|
||||
string response = handler(read.SubArray(size, size + 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()))));
|
||||
@ -205,7 +207,7 @@ namespace Common
|
||||
{
|
||||
Connection.Close();
|
||||
}
|
||||
catch (Exception) { }
|
||||
catch { }
|
||||
return true;
|
||||
}
|
||||
}
|
||||
@ -220,13 +222,12 @@ namespace Common
|
||||
/// Disconnect from server
|
||||
/// </summary>
|
||||
/// <returns></returns>
|
||||
public virtual async Task<object> Disconnect()
|
||||
public virtual async Task Disconnect()
|
||||
{
|
||||
NetSupport.DoStateCheck(IsAlive, true);
|
||||
Running = false;
|
||||
|
||||
|
||||
return await new TaskFactory().StartNew<object>(() => { eventListener.Join(); return null; });
|
||||
await new TaskFactory().StartNew(eventListener.Join);
|
||||
}
|
||||
|
||||
// Methods for sending data to the server
|
||||
@ -244,7 +245,7 @@ namespace Common
|
||||
public virtual void Send(byte[] message)
|
||||
{
|
||||
NetSupport.DoStateCheck(IsAlive, true);
|
||||
lock (messageBuffer) messageBuffer.Enqueue(Crypto.Encrypt(NetSupport.WithHeader(message)));
|
||||
lock (messageBuffer) messageBuffer.Enqueue(NetSupport.WithHeader(message));
|
||||
}
|
||||
|
||||
private static bool Read(Socket sock, List<byte> read, byte[] buf, long timeout)
|
||||
|
@ -1,8 +1,11 @@
|
||||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.Diagnostics;
|
||||
using System.Linq;
|
||||
using System.Runtime.InteropServices;
|
||||
using System.Text;
|
||||
using System.Threading.Tasks;
|
||||
using Tofvesson.Common;
|
||||
using Tofvesson.Crypto;
|
||||
|
||||
namespace Common
|
||||
@ -10,20 +13,80 @@ namespace Common
|
||||
// Helper methods. WithHeader() should really just be in Support.cs
|
||||
public static class NetSupport
|
||||
{
|
||||
public enum Compression { int16, int32, int64 }
|
||||
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);
|
||||
int i = BinaryHelpers.VarIntSize(message.Length);
|
||||
byte[] nmsg = new byte[message.Length + i];
|
||||
//Support.WriteToArray(nmsg, message.Length, 0);
|
||||
BinaryHelpers.WriteVarInt(nmsg, 0, message.Length);
|
||||
Array.Copy(message, 0, nmsg, i, message.Length);
|
||||
Debug.WriteLine($"Compression: {nmsg.Length}/{Compress(nmsg, Compression.int16).Length}/{Compress(nmsg, Compression.int32).Length}/{Compress(nmsg, Compression.int64).Length}");
|
||||
Debug.WriteLine($"Matches: {Support.ArraysEqual(nmsg, Decompress(Compress(nmsg)))}");
|
||||
return nmsg;
|
||||
}
|
||||
|
||||
public static byte[] Decompress(byte[] cmpMessage, Compression method = Compression.int32)
|
||||
{
|
||||
BitReader reader = new BitReader(cmpMessage);
|
||||
byte[] decomp = new byte[reader.ReadUInt()];
|
||||
int size = method == Compression.int16 ? 2 : method == Compression.int32 ? 4 : 8;
|
||||
int count = (decomp.Length / size) + (decomp.Length % size == 0 ? 0 : 1);
|
||||
for(int i = 0; i<count; ++i)
|
||||
{
|
||||
dynamic value = size == 2 ? reader.ReadUShort() : size == 4 ? reader.ReadUInt() : reader.ReadULong();
|
||||
for (int j = Math.Min(size, decomp.Length - (i * size)) - 1; j >= 0; --j) decomp[(i * size) + j] = (byte)((int)(value >> (8 * j)) & 0xFF);
|
||||
}
|
||||
return decomp;
|
||||
}
|
||||
|
||||
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!");
|
||||
}
|
||||
|
||||
private delegate void WriteFunc(BitWriter writer, byte[] data, int index);
|
||||
private static WriteFunc
|
||||
func16 = (w, d, i) => w.WriteUShort(ReadUShort(d, i * 2)),
|
||||
func32 = (w, d, i) => w.WriteUInt(ReadUInt(d, i * 4)),
|
||||
func64 = (w, d, i) => w.WriteULong(ReadULong(d, i * 8));
|
||||
private static byte[] Compress(byte[] data, Compression method = Compression.int32)
|
||||
{
|
||||
int size = method == Compression.int16 ? 2 : method == Compression.int32 ? 4 : 8;
|
||||
int count = (data.Length / size) + (data.Length % size == 0 ? 0 : 1);
|
||||
WriteFunc func = size == 2 ? func16 : size == 4 ? func32 : func64;
|
||||
using (BitWriter writer = new BitWriter())
|
||||
{
|
||||
writer.WriteUInt((uint)data.Length);
|
||||
for (int i = 0; i < count; ++i)
|
||||
func(writer, data, i);
|
||||
return writer.Finalize();
|
||||
}
|
||||
}
|
||||
|
||||
private static ushort ReadUShort(byte[] b, int offset) =>
|
||||
(ushort)((ushort)TryReadByte(b, offset) |
|
||||
(ushort)((ushort)TryReadByte(b, offset + 1) << 8));
|
||||
|
||||
private static uint ReadUInt(byte[] b, int offset) =>
|
||||
(uint)TryReadByte(b, offset) |
|
||||
((uint)TryReadByte(b, offset + 1) << 8) |
|
||||
((uint)TryReadByte(b, offset + 2) << 16) |
|
||||
((uint)TryReadByte(b, offset + 3) << 24);
|
||||
|
||||
private static ulong ReadULong(byte[] b, int offset) =>
|
||||
(ulong)TryReadByte(b, offset) |
|
||||
((ulong)TryReadByte(b, offset + 1) << 8) |
|
||||
((ulong)TryReadByte(b, offset + 2) << 16) |
|
||||
((ulong)TryReadByte(b, offset + 3) << 24) |
|
||||
((ulong)TryReadByte(b, offset + 4) << 32) |
|
||||
((ulong)TryReadByte(b, offset + 5) << 40) |
|
||||
((ulong)TryReadByte(b, offset + 6) << 48) |
|
||||
((ulong)TryReadByte(b, offset + 7) << 56);
|
||||
|
||||
private static byte TryReadByte(byte[] b, int idx) => idx >= b.Length ? (byte) 0 : b[idx];
|
||||
}
|
||||
}
|
||||
|
@ -32,12 +32,13 @@ namespace Tofvesson.Crypto
|
||||
|
||||
int chunks = msg.Length / 64;
|
||||
|
||||
// Perform hashing for each 512-bit block
|
||||
for(int i = 0; i<chunks; ++i)
|
||||
{
|
||||
|
||||
// Split block into words
|
||||
// Split block into words (allocated out here to prevent massive garbage buildup)
|
||||
uint[] w = new uint[80];
|
||||
|
||||
// Perform hashing for each 512-bit block
|
||||
for (int i = 0; i<chunks; ++i)
|
||||
{
|
||||
// Compute initial source data from padded message
|
||||
for(int j = 0; j<16; ++j)
|
||||
w[j] |= (uint) ((msg[i * 64 + j * 4] << 24) | (msg[i * 64 + j * 4 + 1] << 16) | (msg[i * 64 + j * 4 + 2] << 8) | (msg[i * 64 + j * 4 + 3] << 0));
|
||||
|
||||
@ -63,6 +64,8 @@ namespace Tofvesson.Crypto
|
||||
b = a;
|
||||
a = tmp;
|
||||
}
|
||||
|
||||
// Add to result
|
||||
h0 += a;
|
||||
h1 += b;
|
||||
h2 += c;
|
||||
@ -73,6 +76,88 @@ namespace Tofvesson.Crypto
|
||||
return Support.WriteContiguous(new byte[20], 0, Support.SwapEndian(h0), Support.SwapEndian(h1), Support.SwapEndian(h2), Support.SwapEndian(h3), Support.SwapEndian(h4));
|
||||
}
|
||||
|
||||
public struct SHA1Result
|
||||
{
|
||||
public uint i0, i1, i2, i3, i4;
|
||||
public byte Get(int idx) => (byte)((idx < 4 ? i0 : idx < 8 ? i1 : idx < 12 ? i2 : idx < 16 ? i3 : i4)>>(8*(idx%4)));
|
||||
}
|
||||
public static SHA1Result SHA1_Opt(byte[] message)
|
||||
{
|
||||
SHA1Result result = new SHA1Result
|
||||
{
|
||||
// Initialize buffers
|
||||
i0 = 0x67452301,
|
||||
i1 = 0xEFCDAB89,
|
||||
i2 = 0x98BADCFE,
|
||||
i3 = 0x10325476,
|
||||
i4 = 0xC3D2E1F0
|
||||
};
|
||||
|
||||
// Pad message
|
||||
long len = message.Length * 8;
|
||||
int
|
||||
ml = message.Length + 1,
|
||||
max = ml + ((960 - (ml * 8 % 512)) % 512) / 8 + 8;
|
||||
|
||||
// Replaces the allocation of a lot of bytes
|
||||
byte GetMsg(int idx)
|
||||
{
|
||||
if (idx < message.Length)
|
||||
return message[idx];
|
||||
else if (idx == message.Length)
|
||||
return 0x80;
|
||||
else if (max - idx <= 8)
|
||||
return (byte)((len >> ((max - 1 - idx) * 8)) & 255);
|
||||
return 0;
|
||||
}
|
||||
|
||||
int chunks = max / 64;
|
||||
|
||||
// Replaces the recurring allocation of 80 uints
|
||||
uint ComputeIndex(int block, int idx)
|
||||
{
|
||||
if (idx < 16)
|
||||
return (uint)((GetMsg(block * 64 + idx * 4) << 24) | (GetMsg(block * 64 + idx * 4 + 1) << 16) | (GetMsg(block * 64 + idx * 4 + 2) << 8) | (GetMsg(block * 64 + idx * 4 + 3) << 0));
|
||||
else
|
||||
return Rot(ComputeIndex(block, idx - 3) ^ ComputeIndex(block, idx - 8) ^ ComputeIndex(block, idx - 14) ^ ComputeIndex(block, idx - 16), 1);
|
||||
}
|
||||
|
||||
// Perform hashing for each 512-bit block
|
||||
for (int i = 0; i < chunks; ++i)
|
||||
{
|
||||
|
||||
// Initialize chunk-hash
|
||||
uint
|
||||
a = result.i0,
|
||||
b = result.i1,
|
||||
c = result.i2,
|
||||
d = result.i3,
|
||||
e = result.i4;
|
||||
|
||||
// Do hash rounds
|
||||
for (int t = 0; t < 80; ++t)
|
||||
{
|
||||
uint tmp = Rot(a, 5) + func(t, b, c, d) + e + K(t) + ComputeIndex(i, t);
|
||||
e = d;
|
||||
d = c;
|
||||
c = Rot(b, 30);
|
||||
b = a;
|
||||
a = tmp;
|
||||
}
|
||||
result.i0 += a;
|
||||
result.i1 += b;
|
||||
result.i2 += c;
|
||||
result.i3 += d;
|
||||
result.i4 += e;
|
||||
}
|
||||
result.i0 = Support.SwapEndian(result.i0);
|
||||
result.i1 = Support.SwapEndian(result.i1);
|
||||
result.i2 = Support.SwapEndian(result.i2);
|
||||
result.i3 = Support.SwapEndian(result.i3);
|
||||
result.i4 = Support.SwapEndian(result.i4);
|
||||
return result;
|
||||
}
|
||||
|
||||
private static uint func(int t, uint b, uint c, uint d) =>
|
||||
t < 20 ? (b & c) | ((~b) & d) :
|
||||
t < 40 ? b ^ c ^ d :
|
||||
|
@ -422,13 +422,16 @@ namespace Tofvesson.Crypto
|
||||
|
||||
// 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 ulong SwapEndian(ulong value)
|
||||
{
|
||||
ulong res = 0;
|
||||
for(int i = 0; i<8; ++i)
|
||||
res = (res << 8) | ((value >> i * 8) & 0xFF);
|
||||
return res;
|
||||
}
|
||||
public static ulong SwapEndian(ulong value) =>
|
||||
((value >> 56) & 0xFF) |
|
||||
((value >> 40) & (0xFFUL << 8)) |
|
||||
((value >> 24) & (0xFFUL << 16)) |
|
||||
((value >> 8) & (0xFFUL << 24)) |
|
||||
((value << 56) & (0xFFUL << 56)) |
|
||||
((value << 40) & (0xFFUL << 48)) |
|
||||
((value << 24) & (0xFFUL << 40)) |
|
||||
((value << 8) & (0xFFUL << 32));
|
||||
|
||||
public static ulong RightShift(this ulong value, int shift) => shift < 0 ? value << -shift : value >> shift;
|
||||
public static string ToHexString(byte[] value)
|
||||
{
|
||||
@ -440,6 +443,8 @@ namespace Tofvesson.Crypto
|
||||
}
|
||||
return builder.ToString();
|
||||
}
|
||||
public static string ToBase64String(this string text) => Convert.ToBase64String(text.ToUTF8Bytes());
|
||||
public static string FromBase64String(this string text) => Convert.FromBase64String(text).ToUTF8String();
|
||||
|
||||
public static bool ReadYNBool(this TextReader reader, string nonDefault) => reader.ReadLine().ToLower().Equals(nonDefault);
|
||||
|
||||
|
@ -179,12 +179,19 @@ namespace Server
|
||||
writer.WriteStartElement("User");
|
||||
if (u.IsAdministrator) writer.WriteAttributeString("admin", "", "true");
|
||||
writer.WriteElementString("Name", u.Name);
|
||||
writer.WriteElementString("Balance", u.Balance.ToString());
|
||||
//writer.WriteElementString("Balance", u.Balance.ToString());
|
||||
writer.WriteElementString("Password", u.PasswordHash);
|
||||
writer.WriteElementString("Salt", u.Salt);
|
||||
foreach (var tx in u.History)
|
||||
foreach(var acc in u.accounts)
|
||||
{
|
||||
writer.WriteStartElement("Account");
|
||||
writer.WriteElementString("Name", acc.name);
|
||||
writer.WriteElementString("Balance", acc.balance.ToString());
|
||||
foreach (var tx in acc.History)
|
||||
{
|
||||
writer.WriteStartElement("Transaction");
|
||||
writer.WriteElementString("FromAccount", tx.fromAccount);
|
||||
writer.WriteElementString("ToAccount", tx.toAccount);
|
||||
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);
|
||||
@ -192,6 +199,8 @@ namespace Server
|
||||
}
|
||||
writer.WriteEndElement();
|
||||
}
|
||||
writer.WriteEndElement();
|
||||
}
|
||||
|
||||
private static string GenerateTempFileName(string prefix, string suffix)
|
||||
{
|
||||
@ -225,9 +234,10 @@ namespace Server
|
||||
return e;
|
||||
}
|
||||
|
||||
public User GetUser(string name) => FirstUser(u => u.Name.Equals(name));
|
||||
public User GetUser(string name) => name.Equals("System") ? null : FirstUser(u => u.Name.Equals(name));
|
||||
public User FirstUser(Predicate<User> p)
|
||||
{
|
||||
if (p == null) return null; // Done to conveniently handle system insertions
|
||||
User u;
|
||||
foreach (var entry in loadedUsers)
|
||||
if (p(u=FromEncoded(entry)))
|
||||
@ -248,7 +258,7 @@ namespace Server
|
||||
if (reader.Name.Equals("User"))
|
||||
{
|
||||
User n = User.Parse(ReadEntry(reader), this);
|
||||
if (n != null && p(n=FromEncoded(n)))
|
||||
if (n != null && p(FromEncoded(n)))
|
||||
{
|
||||
if (!loadedUsers.Contains(n)) loadedUsers.Add(n);
|
||||
return n;
|
||||
@ -259,19 +269,30 @@ namespace Server
|
||||
return null;
|
||||
}
|
||||
|
||||
public bool AddTransaction(string sender, string recipient, long amount, string message = null)
|
||||
public bool AddTransaction(string sender, string recipient, decimal amount, string fromAccount, string toAccount, string message = null)
|
||||
{
|
||||
User from = FirstUser(u => u.Name.Equals(sender));
|
||||
User to = FirstUser(u => u.Name.Equals(recipient));
|
||||
Account fromAcc = from?.GetAccount(fromAccount);
|
||||
Account toAcc = to.GetAccount(toAccount);
|
||||
|
||||
if (to == null || (from == null && !to.IsAdministrator)) return false;
|
||||
// Errant states
|
||||
if (
|
||||
to == null ||
|
||||
(from == null && !to.IsAdministrator) ||
|
||||
toAcc == null ||
|
||||
(from != null && fromAcc == null) ||
|
||||
(from != null && fromAcc.balance<amount)
|
||||
) return false;
|
||||
|
||||
Transaction tx = new Transaction(from == null ? "System" : from.Name, to.Name, amount, message);
|
||||
to.History.Add(tx);
|
||||
Transaction tx = new Transaction(from == null ? "System" : from.Name, to.Name, amount, message, fromAccount, toAccount);
|
||||
toAcc.History.Add(tx);
|
||||
toAcc.balance += amount;
|
||||
AddUser(to);
|
||||
if (from != null)
|
||||
{
|
||||
from.History.Add(tx);
|
||||
fromAcc.History.Add(tx);
|
||||
fromAcc.balance -= amount;
|
||||
AddUser(from);
|
||||
}
|
||||
return true;
|
||||
@ -299,8 +320,8 @@ namespace Server
|
||||
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;
|
||||
public bool ContainsUser(string user) => user.Equals("System") || FirstUser(u => u.Name.Equals(user)) != null;
|
||||
public bool ContainsUser(User user) => user.Name.Equals("System") || FirstUser(u => u.Name.Equals(user.Name)) != null;
|
||||
|
||||
private bool Traverse(XmlReader reader, params string[] downTo)
|
||||
{
|
||||
@ -324,11 +345,15 @@ namespace Server
|
||||
{
|
||||
User u = new User(entry);
|
||||
u.Name = Encode(u.Name);
|
||||
for (int i = 0; i < u.History.Count; ++i)
|
||||
foreach(var account in u.accounts)
|
||||
{
|
||||
u.History[i].to = Encode(u.History[i].to);
|
||||
u.History[i].from = Encode(u.History[i].from);
|
||||
u.History[i].meta = Encode(u.History[i].meta);
|
||||
account.name = Encode(account.name);
|
||||
foreach(var transaction in account.History)
|
||||
{
|
||||
transaction.to = Encode(transaction.to);
|
||||
transaction.from = Encode(transaction.from);
|
||||
transaction.meta = Encode(transaction.meta);
|
||||
}
|
||||
}
|
||||
return u;
|
||||
}
|
||||
@ -337,11 +362,15 @@ namespace Server
|
||||
{
|
||||
User u = new User(entry);
|
||||
u.Name = Decode(u.Name);
|
||||
for (int i = 0; i < u.History.Count; ++i)
|
||||
foreach (var account in u.accounts)
|
||||
{
|
||||
u.History[i].to = Decode(u.History[i].to);
|
||||
u.History[i].from = Decode(u.History[i].from);
|
||||
u.History[i].meta = Decode(u.History[i].meta);
|
||||
account.name = Decode(account.name);
|
||||
foreach (var transaction in account.History)
|
||||
{
|
||||
transaction.to = Decode(transaction.to);
|
||||
transaction.from = Decode(transaction.from);
|
||||
transaction.meta = Decode(transaction.meta);
|
||||
}
|
||||
}
|
||||
return u;
|
||||
}
|
||||
@ -380,6 +409,8 @@ namespace Server
|
||||
return this;
|
||||
}
|
||||
|
||||
public Entry AddNested(string name, string text) => AddNested(new Entry(name, text));
|
||||
|
||||
public Entry AddAttribute(string key, string value)
|
||||
{
|
||||
Attributes[key] = value;
|
||||
@ -415,40 +446,56 @@ namespace Server
|
||||
}
|
||||
}
|
||||
|
||||
public class Account
|
||||
{
|
||||
public User owner;
|
||||
public decimal balance;
|
||||
public string name;
|
||||
public List<Transaction> History { get; }
|
||||
public Account(User owner, decimal balance, string name)
|
||||
{
|
||||
History = new List<Transaction>();
|
||||
this.owner = owner;
|
||||
this.balance = balance;
|
||||
this.name = name;
|
||||
}
|
||||
public Account(Account copy) : this(copy.owner, copy.balance, copy.name)
|
||||
=> History.AddRange(copy.History);
|
||||
public Account AddTransaction(Transaction tx)
|
||||
{
|
||||
History.Add(tx);
|
||||
return this;
|
||||
}
|
||||
}
|
||||
|
||||
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<Transaction> History { get; }
|
||||
private User()
|
||||
{
|
||||
Name = "";
|
||||
History = new List<Transaction>();
|
||||
}
|
||||
public List<Account> accounts = new List<Account>();
|
||||
|
||||
public User(User copy) : this()
|
||||
private User()
|
||||
{ }
|
||||
|
||||
public User(User copy)
|
||||
{
|
||||
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);
|
||||
accounts.AddRange(copy.accounts);
|
||||
}
|
||||
|
||||
public User(string name, string passHash, string salt, long balance, bool generatePass = false, List<Transaction> transactionHistory = null, bool admin = false)
|
||||
: this(name, passHash, Encoding.UTF8.GetBytes(salt), balance, generatePass, transactionHistory, admin)
|
||||
public User(string name, string passHash, string salt, bool generatePass = false, bool admin = false)
|
||||
: this(name, passHash, Encoding.UTF8.GetBytes(salt), generatePass, admin)
|
||||
{ }
|
||||
|
||||
public User(string name, string passHash, byte[] salt, long balance, bool generatePass = false, List<Transaction> transactionHistory = null, bool admin = false)
|
||||
public User(string name, string passHash, byte[] salt, bool generatePass = false, bool admin = false)
|
||||
{
|
||||
History = transactionHistory ?? new List<Transaction>();
|
||||
Balance = balance;
|
||||
Name = name;
|
||||
IsAdministrator = admin;
|
||||
Salt = Convert.ToBase64String(salt);
|
||||
@ -458,26 +505,30 @@ namespace Server
|
||||
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 void AddAccount(Account a) => accounts.Add(a);
|
||||
public Account GetAccount(string name) => accounts.FirstOrDefault(a => a.name.Equals(name));
|
||||
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)
|
||||
.AddNested(new Entry("Name", Name));
|
||||
foreach (var account in accounts)
|
||||
{
|
||||
Entry acc = new Entry("Account")
|
||||
.AddNested("Name", account.name)
|
||||
.AddNested(new Entry("Balance", account.balance.ToString()).AddAttribute("omit", "true"));
|
||||
foreach (var transaction in account.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);
|
||||
.AddNested(transaction.to.Equals(Name) ? "From" : "To", transaction.to.Equals(Name) ? transaction.from : transaction.to)
|
||||
.AddNested("FromAccount", transaction.fromAccount)
|
||||
.AddNested("ToAccount", transaction.toAccount)
|
||||
.AddNested("Balance", transaction.amount.ToString());
|
||||
if (transaction.meta != null) tx.AddNested("Meta", transaction.meta);
|
||||
acc.AddNested(tx);
|
||||
}
|
||||
root.AddNested(acc);
|
||||
}
|
||||
return root;
|
||||
}
|
||||
@ -489,38 +540,68 @@ namespace Server
|
||||
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("Account"))
|
||||
{
|
||||
string name = null;
|
||||
decimal balance = 0;
|
||||
List<Transaction> history = new List<Transaction>();
|
||||
foreach (var accountData in entry.NestedEntries)
|
||||
{
|
||||
if (accountData.Name.Equals("Name")) name = accountData.Text;
|
||||
else if (entry.Name.Equals("Transaction"))
|
||||
{
|
||||
string fromAccount = null;
|
||||
string toAccount = null;
|
||||
string from = null;
|
||||
string to = null;
|
||||
long amount = -1;
|
||||
decimal 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("FromAccount")) fromAccount = e1.Text;
|
||||
else if (e1.Name.Equals("ToAccount")) toAccount = e1.Text;
|
||||
else if (e1.Name.Equals("Balance")) amount = decimal.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));
|
||||
if ( // Errant states for transaction data
|
||||
(from == null && to == null) ||
|
||||
(from != null && to != null) ||
|
||||
amount <= 0 ||
|
||||
fromAccount == null ||
|
||||
toAccount == null
|
||||
)
|
||||
user.ProblematicTransactions = true;
|
||||
else history.Add(new Transaction(from, to, amount, meta, fromAccount, toAccount));
|
||||
}
|
||||
else if (entry.Name.Equals("Balance")) balance = decimal.TryParse(entry.Text, out decimal l) ? l : 0;
|
||||
}
|
||||
if (name == null || balance < 0)
|
||||
{
|
||||
Output.Fatal($"Found errant account entry! Detected user name: {user.Name}");
|
||||
return null; // This is a hard error
|
||||
}
|
||||
Account a = new Account(user, balance, name);
|
||||
a.History.AddRange(history);
|
||||
user.AddAccount(a);
|
||||
}
|
||||
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)
|
||||
foreach (var account in user.accounts)
|
||||
foreach (var transaction in account.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 Transaction CreateTransaction(User recipient, long amount, Account fromAccount, Account toAccount, string message = null) =>
|
||||
new Transaction(this.Name, recipient.Name, amount, message, fromAccount.name, toAccount.name);
|
||||
|
||||
public override bool Equals(object obj) => obj is User && ((User)obj).Name.Equals(Name);
|
||||
|
||||
@ -532,13 +613,17 @@ namespace Server
|
||||
|
||||
public class Transaction
|
||||
{
|
||||
public string fromAccount;
|
||||
public string toAccount;
|
||||
public string from;
|
||||
public string to;
|
||||
public long amount;
|
||||
public decimal amount;
|
||||
public string meta;
|
||||
|
||||
public Transaction(string from, string to, long amount, string meta)
|
||||
public Transaction(string from, string to, decimal amount, string meta, string fromAccount, string toAccount)
|
||||
{
|
||||
this.fromAccount = fromAccount;
|
||||
this.toAccount = toAccount;
|
||||
this.from = from;
|
||||
this.to = to;
|
||||
this.amount = amount;
|
||||
|
38
Server/Output.cs
Normal file
38
Server/Output.cs
Normal file
@ -0,0 +1,38 @@
|
||||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.IO;
|
||||
using System.Linq;
|
||||
using System.Text;
|
||||
using System.Threading.Tasks;
|
||||
|
||||
namespace Server
|
||||
{
|
||||
public static class Output
|
||||
{
|
||||
public static void WriteLine(string message, bool error = false)
|
||||
{
|
||||
if (error) Error(message);
|
||||
else Info(message);
|
||||
}
|
||||
public static void Write(string message, bool error)
|
||||
{
|
||||
if (error) Error(message, false);
|
||||
else Info(message, false);
|
||||
}
|
||||
public static void Positive(string message, bool newline = true) => Write(message, ConsoleColor.DarkGreen, ConsoleColor.Black, newline, Console.Out);
|
||||
public static void Info(string message, bool newline = true) => Write(message, ConsoleColor.Gray, ConsoleColor.Black, newline, Console.Out);
|
||||
public static void Error(string message, bool newline = true) => Write(message, ConsoleColor.Gray, ConsoleColor.Black, newline, Console.Out);
|
||||
public static void Fatal(string message, bool newline = true) => Write(message, ConsoleColor.Gray, ConsoleColor.Black, newline, Console.Error);
|
||||
|
||||
private static void Write(string message, ConsoleColor f, ConsoleColor b, bool newline, TextWriter writer)
|
||||
{
|
||||
ConsoleColor f1 = Console.ForegroundColor, b1 = Console.BackgroundColor;
|
||||
Console.ForegroundColor = f;
|
||||
Console.BackgroundColor = b;
|
||||
writer.Write(message);
|
||||
if (newline) writer.WriteLine();
|
||||
Console.ForegroundColor = f1;
|
||||
Console.BackgroundColor = b1;
|
||||
}
|
||||
}
|
||||
}
|
@ -4,38 +4,124 @@ using Server.Properties;
|
||||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.Linq;
|
||||
using System.Numerics;
|
||||
using System.Text;
|
||||
using System.Threading.Tasks;
|
||||
using Tofvesson.Common;
|
||||
using Tofvesson.Crypto;
|
||||
|
||||
namespace Server
|
||||
{
|
||||
class Program
|
||||
{
|
||||
static void Main(string[] args)
|
||||
private const string VERBOSE_RESPONSE = "@string/REMOTE_";
|
||||
public static void Main(string[] args)
|
||||
{
|
||||
// Set up fancy output
|
||||
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);
|
||||
// Create a client session manager and allow sessions to remain valid for up to 5 minutes of inactivity (300 seconds)
|
||||
SessionManager manager = new SessionManager(300 * TimeSpan.TicksPerSecond, 20);
|
||||
|
||||
// Initialize the database
|
||||
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);
|
||||
|
||||
|
||||
// Create a secure random provider and start getting RSA stuff
|
||||
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!");
|
||||
//}
|
||||
Task<RSA> t = new Task<RSA>(() =>
|
||||
{
|
||||
RSA rsa = new RSA(Resources.e_0x100, Resources.n_0x100, Resources.d_0x100);
|
||||
if (rsa == null)
|
||||
{
|
||||
Output.Fatal("No RSA keys found! Server identity will not be verifiable!");
|
||||
Output.Info("Generating session-specific RSA-keys...");
|
||||
rsa = new RSA(128, 8, 7, 5);
|
||||
rsa.Save("0x100");
|
||||
Output.Info("Done!");
|
||||
}
|
||||
return rsa;
|
||||
});
|
||||
t.Start();
|
||||
|
||||
// Local methods to simplify common operations
|
||||
bool ParseDataPair(string cmd, out string user, out string pass)
|
||||
{
|
||||
int idx = cmd.IndexOf(':');
|
||||
user = "";
|
||||
pass = "";
|
||||
if (idx == -1) return false;
|
||||
user = cmd.Substring(0, idx);
|
||||
try
|
||||
{
|
||||
user = user.FromBase64String();
|
||||
pass = cmd.Substring(idx + 1).FromBase64String();
|
||||
}
|
||||
catch
|
||||
{
|
||||
Output.Error($"Recieved problematic username or password! (User: \"{user}\")");
|
||||
return false;
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
int ParseDataSet(string cmd, out string[] data)
|
||||
{
|
||||
List<string> gen = new List<string>();
|
||||
int idx;
|
||||
while ((idx = cmd.IndexOf(':')) != -1)
|
||||
{
|
||||
try
|
||||
{
|
||||
gen.Add(cmd.Substring(0, idx).FromBase64String());
|
||||
}
|
||||
catch
|
||||
{
|
||||
data = null;
|
||||
return -1; // Hard error
|
||||
}
|
||||
cmd = cmd.Substring(idx + 1);
|
||||
}
|
||||
try
|
||||
{
|
||||
gen.Add(cmd.FromBase64String());
|
||||
}
|
||||
catch
|
||||
{
|
||||
data = null;
|
||||
return -1; // Hard error
|
||||
}
|
||||
data = gen.ToArray();
|
||||
return gen.Count;
|
||||
}
|
||||
|
||||
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) };
|
||||
}
|
||||
|
||||
string GenerateResponse(long id, dynamic d) => id + ":" + d.ToString();
|
||||
|
||||
bool GetUser(string sid, out Database.User user)
|
||||
{
|
||||
user = manager.GetUser(sid);
|
||||
return user != null;
|
||||
}
|
||||
|
||||
bool GetAccount(string name, Database.User user, out Database.Account acc)
|
||||
{
|
||||
acc = user.accounts.FirstOrDefault(a => a.name.Equals(name));
|
||||
return acc != null;
|
||||
}
|
||||
|
||||
// Create server
|
||||
NetServer server = new NetServer(
|
||||
EllipticDiffieHellman.Curve25519(EllipticDiffieHellman.Curve25519_GeneratePrivate(random)),
|
||||
80,
|
||||
@ -48,51 +134,176 @@ namespace Server
|
||||
{
|
||||
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);
|
||||
if(!ParseDataPair(cmd[1], out string user, out string pass))
|
||||
{
|
||||
Output.Error($"Recieved problematic username or password! (User: \"{user}\")");
|
||||
return GenerateResponse(id, "ERROR");
|
||||
}
|
||||
Database.User usr = db.GetUser(user);
|
||||
if (usr == null || !usr.Authenticate(pass))
|
||||
{
|
||||
Console.WriteLine("Authentcation failure for user: "+user);
|
||||
Output.Error("Authentcation failure for user: "+user);
|
||||
return GenerateResponse(id, "ERROR");
|
||||
}
|
||||
|
||||
string sess = manager.GetSession(usr, "ERROR");
|
||||
Console.WriteLine("Authentication success for user: "+user+"\nSession: "+sess);
|
||||
Output.Positive("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]);
|
||||
if (manager.Expire(cmd[1])) Output.Info("Prematurely expired session: " + cmd[1]);
|
||||
else Output.Error("Attempted to expire a non-existent session!");
|
||||
break;
|
||||
case "Avail":
|
||||
{
|
||||
try
|
||||
{
|
||||
string name = cmd[1].FromBase64String();
|
||||
Output.Info($"Performing availability check on name \"{name}\"");
|
||||
return GenerateResponse(id, !db.ContainsUser(name));
|
||||
}
|
||||
catch
|
||||
{
|
||||
Output.Error($"Recieved improperly formatted base64 string: \"{cmd[1]}\"");
|
||||
return GenerateResponse(id, false);
|
||||
}
|
||||
}
|
||||
case "Account_Create":
|
||||
{
|
||||
if (!ParseDataPair(cmd[1], out string session, out string name) || // Get session id and account name
|
||||
!GetUser(session, out var user) || // Get user associated with session id
|
||||
!GetAccount(name, user, out var account))
|
||||
{
|
||||
// Don't print input data to output in case sensitive information was included
|
||||
Output.Error($"Recieved problematic session id or account name!");
|
||||
return GenerateResponse(id, "ERROR");
|
||||
}
|
||||
user.accounts.Add(new Database.Account(user, 0, name));
|
||||
db.AddUser(user); // Notify database of the update
|
||||
return GenerateResponse(id, true);
|
||||
}
|
||||
case "Account_Transaction_Create":
|
||||
{
|
||||
bool systemInsert = false;
|
||||
string error = VERBOSE_RESPONSE;
|
||||
|
||||
// Default values used here because compiler can't infer their valid parsing further down
|
||||
Database.User user = null;
|
||||
Database.Account account = null;
|
||||
Database.User tUser = null;
|
||||
Database.Account tAccount = null;
|
||||
decimal amount = 0;
|
||||
|
||||
// Expected data (in order): SessionID, AccountName, TargetUserName, TargetAccountName, Amount, [message]
|
||||
// Do checks to make sure the data we have been given isn't completely silly
|
||||
if (ParseDataSet(cmd[1], out string[] data) < 5 || data.Length > 6)
|
||||
error += "general"; // General error (parse failed)
|
||||
else if (!GetUser(data[0], out user))
|
||||
error += "badsession"; // Bad session id (could not get user from session manager)
|
||||
else if (!GetAccount(data[1], user, out account))
|
||||
error += "badacc"; // Bad source account name
|
||||
else if (!db.ContainsUser(data[2]))
|
||||
error += "notargetusr"; // Target user could not be found
|
||||
else if (!GetAccount(data[3], tUser = db.GetUser(data[2]), out tAccount))
|
||||
error += "notargetacc"; // Target account could not be found
|
||||
else if ((!user.IsAdministrator && (systemInsert = (data[2].Equals(user.Name) && account.name.Equals(tAccount.name)))))
|
||||
error += "unprivsysins"; // Unprivileged request for system-sourced transfer
|
||||
else if (!decimal.TryParse(data[4], out amount) || amount < 0)
|
||||
error += "badbalance"; // Given sum was not a valid amount
|
||||
else if ((!systemInsert && amount > account.balance))
|
||||
error += "insufficient"; // Insufficient funds in the source account
|
||||
|
||||
// Checks if an error ocurred and handles such a situation appropriately
|
||||
if(!error.Equals(VERBOSE_RESPONSE))
|
||||
{
|
||||
// Don't print input data to output in case sensitive information was included
|
||||
Output.Error($"Recieved problematic transaction data ({error}): {data?.ToList().ToString() ?? "Data could not be parsed"}");
|
||||
return GenerateResponse(id, $"ERROR:{error}");
|
||||
}
|
||||
// At this point, we know that all parsed variables above were successfully parsed and valid, therefore: no NREs
|
||||
// Parsed vars: 'user', 'account', 'tUser', 'tAccount', 'amount'
|
||||
// Perform and log the actual transaction
|
||||
return GenerateResponse(id,
|
||||
db.AddTransaction(
|
||||
systemInsert ? null : user.Name,
|
||||
tUser.Name,
|
||||
amount,
|
||||
account.name,
|
||||
tAccount.name,
|
||||
data.Length == 6 ? data[5] : null
|
||||
));
|
||||
}
|
||||
case "Account_Close":
|
||||
{
|
||||
Database.User user = null;
|
||||
Database.Account account = null;
|
||||
if (!ParseDataPair(cmd[1], out string session, out string name) || // Get session id and account name
|
||||
!GetUser(session, out user) || // Get user associated with session id
|
||||
!GetAccount(name, user, out account) ||
|
||||
account.balance != 0)
|
||||
{
|
||||
// Don't print input data to output in case sensitive information was included
|
||||
Output.Error($"Recieved problematic session id or account name!");
|
||||
|
||||
// Possible errors: bad session id, bad account name, balance in account isn't 0
|
||||
return GenerateResponse(id, $"ERROR:{VERBOSE_RESPONSE} {(user==null? "badsession" : account==null? "badacc" : "hasbal")}");
|
||||
}
|
||||
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);
|
||||
if (!ParseDataPair(cmd[1], out string user, out string pass))
|
||||
{
|
||||
// Don't print input data to output in case sensitive information was included
|
||||
Output.Error($"Recieved problematic username or password!");
|
||||
return GenerateResponse(id, $"ERROR:{VERBOSE_RESPONSE}userpass");
|
||||
}
|
||||
|
||||
// Cannot register an account with an existing username
|
||||
if (db.ContainsUser(user)) return GenerateResponse(id, $"ERROR:{VERBOSE_RESPONSE}exists");
|
||||
|
||||
// Create the database user entry and generate a personal password salt
|
||||
Database.User u = new Database.User(user, pass, random.GetBytes(Math.Abs(random.NextShort() % 60) + 20), true);
|
||||
db.AddUser(u);
|
||||
|
||||
// Generate a session token
|
||||
string sess = manager.GetSession(u, "ERROR");
|
||||
Console.WriteLine("Registered account: " + u.Name + "\nSession: "+sess);
|
||||
Output.Positive("Registered account: " + u.Name + "\nSession: "+sess);
|
||||
associations["session"] = sess;
|
||||
return GenerateResponse(id, sess);
|
||||
}
|
||||
case "Verify":
|
||||
{
|
||||
BitReader bd = new BitReader(Convert.FromBase64String(cmd[1]));
|
||||
try
|
||||
{
|
||||
while (!t.IsCompleted) System.Threading.Thread.Sleep(75);
|
||||
byte[] ser;
|
||||
using (BitWriter collector = new BitWriter())
|
||||
{
|
||||
collector.PushArray(t.Result.Serialize());
|
||||
collector.PushArray(t.Result.Encrypt(((BigInteger)bd.ReadUShort()).ToByteArray(), null, true));
|
||||
ser = collector.Finalize();
|
||||
}
|
||||
return GenerateResponse(id, Convert.ToBase64String(ser));
|
||||
}
|
||||
catch
|
||||
{
|
||||
return GenerateResponse(id, $"ERROR:{VERBOSE_RESPONSE}crypterr");
|
||||
}
|
||||
}
|
||||
default:
|
||||
return GenerateResponse(id, "ERROR");
|
||||
return GenerateResponse(id, $"ERROR:{VERBOSE_RESPONSE}unwn"); // Unknown request
|
||||
}
|
||||
|
||||
return null;
|
||||
},
|
||||
(c, b) =>
|
||||
(c, b) => // Called every time a client connects or disconnects (conn + dc with every command/request)
|
||||
{
|
||||
Console.WriteLine($"Client has {(b ? "C" : "Disc")}onnected");
|
||||
//if(!b && c.assignedValues.ContainsKey("session"))
|
||||
// manager.Expire(c.assignedValues["session"]);
|
||||
// Output.Info($"Client has {(b ? "C" : "Disc")}onnected");
|
||||
if(!b && c.assignedValues.ContainsKey("session"))
|
||||
manager.Expire(c.assignedValues["session"]);
|
||||
});
|
||||
server.StartListening();
|
||||
|
||||
@ -100,32 +311,5 @@ namespace Server
|
||||
|
||||
server.StopRunning();
|
||||
}
|
||||
|
||||
private static void HandleInput()
|
||||
{
|
||||
while (true)
|
||||
{
|
||||
|
||||
}
|
||||
}
|
||||
|
||||
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;
|
||||
}
|
||||
}
|
||||
|
30
Server/Properties/Resources.Designer.cs
generated
30
Server/Properties/Resources.Designer.cs
generated
@ -60,6 +60,16 @@ namespace Server.Properties {
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Looks up a localized resource of type System.Byte[].
|
||||
/// </summary>
|
||||
internal static byte[] d_0x100 {
|
||||
get {
|
||||
object obj = ResourceManager.GetObject("d_0x100", resourceCulture);
|
||||
return ((byte[])(obj));
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Looks up a localized resource of type System.Byte[].
|
||||
/// </summary>
|
||||
@ -70,6 +80,16 @@ namespace Server.Properties {
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Looks up a localized resource of type System.Byte[].
|
||||
/// </summary>
|
||||
internal static byte[] e_0x100 {
|
||||
get {
|
||||
object obj = ResourceManager.GetObject("e_0x100", resourceCulture);
|
||||
return ((byte[])(obj));
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Looks up a localized resource of type System.Byte[].
|
||||
/// </summary>
|
||||
@ -80,6 +100,16 @@ namespace Server.Properties {
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Looks up a localized resource of type System.Byte[].
|
||||
/// </summary>
|
||||
internal static byte[] n_0x100 {
|
||||
get {
|
||||
object obj = ResourceManager.GetObject("n_0x100", resourceCulture);
|
||||
return ((byte[])(obj));
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Looks up a localized resource of type System.Byte[].
|
||||
/// </summary>
|
||||
|
@ -118,12 +118,21 @@
|
||||
<value>System.Resources.ResXResourceWriter, System.Windows.Forms, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089</value>
|
||||
</resheader>
|
||||
<assembly alias="System.Windows.Forms" name="System.Windows.Forms, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089" />
|
||||
<data name="d_0x100" type="System.Resources.ResXFileRef, System.Windows.Forms">
|
||||
<value>..\Resources\0x100.d;System.Byte[], mscorlib, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089</value>
|
||||
</data>
|
||||
<data name="d_0x200" type="System.Resources.ResXFileRef, System.Windows.Forms">
|
||||
<value>..\Resources\0x200.d;System.Byte[], mscorlib, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089</value>
|
||||
</data>
|
||||
<data name="e_0x100" type="System.Resources.ResXFileRef, System.Windows.Forms">
|
||||
<value>..\Resources\0x100.e;System.Byte[], mscorlib, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089</value>
|
||||
</data>
|
||||
<data name="e_0x200" type="System.Resources.ResXFileRef, System.Windows.Forms">
|
||||
<value>..\Resources\0x200.e;System.Byte[], mscorlib, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089</value>
|
||||
</data>
|
||||
<data name="n_0x100" type="System.Resources.ResXFileRef, System.Windows.Forms">
|
||||
<value>..\Resources\0x100.n;System.Byte[], mscorlib, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089</value>
|
||||
</data>
|
||||
<data name="n_0x200" type="System.Resources.ResXFileRef, System.Windows.Forms">
|
||||
<value>..\Resources\0x200.n;System.Byte[], mscorlib, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089</value>
|
||||
</data>
|
||||
|
@ -44,6 +44,7 @@
|
||||
</ItemGroup>
|
||||
<ItemGroup>
|
||||
<Compile Include="Database.cs" />
|
||||
<Compile Include="Output.cs" />
|
||||
<Compile Include="Program.cs" />
|
||||
<Compile Include="Properties\AssemblyInfo.cs" />
|
||||
<Compile Include="Properties\Resources.Designer.cs">
|
||||
@ -55,6 +56,9 @@
|
||||
</ItemGroup>
|
||||
<ItemGroup>
|
||||
<None Include="App.config" />
|
||||
<None Include="Resources\0x100.d" />
|
||||
<None Include="Resources\0x100.e" />
|
||||
<None Include="Resources\0x100.n" />
|
||||
<None Include="Resources\0x200.d" />
|
||||
<None Include="Resources\0x200.e" />
|
||||
<None Include="Resources\0x200.n" />
|
||||
|
@ -34,6 +34,14 @@ namespace Server
|
||||
return s.sessionID;
|
||||
}
|
||||
|
||||
public Database.User GetUser(string SID)
|
||||
{
|
||||
foreach (var session in sessions)
|
||||
if (session.sessionID.Equals(SID))
|
||||
return session.user;
|
||||
return null;
|
||||
}
|
||||
|
||||
public bool Refresh(Database.User user)
|
||||
{
|
||||
Update();
|
||||
@ -74,16 +82,16 @@ namespace Server
|
||||
return false;
|
||||
}
|
||||
|
||||
public void Expire(string sid)
|
||||
public bool Expire(string sid)
|
||||
{
|
||||
Update();
|
||||
for (int i = sessions.Count - 1; i >= 0; --i)
|
||||
if (sessions[i].sessionID.Equals(sid))
|
||||
{
|
||||
sessions.RemoveAt(i);
|
||||
return;
|
||||
return true;
|
||||
}
|
||||
return;
|
||||
return false;
|
||||
}
|
||||
|
||||
public bool CheckSession(string sid, Database.User user)
|
||||
@ -106,7 +114,7 @@ namespace Server
|
||||
{
|
||||
string res;
|
||||
do res = random.NextString(sidLength);
|
||||
while (res.Equals(invalid));
|
||||
while (res.StartsWith(invalid));
|
||||
return res;
|
||||
}
|
||||
}
|
||||
|
Loading…
x
Reference in New Issue
Block a user