diff --git a/.gitignore b/.gitignore
index f281e10..6395166 100644
--- a/.gitignore
+++ b/.gitignore
@@ -12,3 +12,6 @@
/Server/Resources/0x200.d
/Server/bin
/Server/obj
+/Server/Resources/0x100.d
+/Server/Resources/0x100.e
+/Server/Resources/0x100.n
diff --git a/Bank.sln b/Bank.sln
index 03b8605..854ffaa 100644
--- a/Bank.sln
+++ b/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
diff --git a/Client.psess b/Client.psess
new file mode 100644
index 0000000..23f31af
--- /dev/null
+++ b/Client.psess
@@ -0,0 +1,114 @@
+
+
+
+ Bank.sln
+ Sampling
+ None
+ true
+ true
+ Timestamp
+ Cycles
+ 10000000
+ 10
+ 10
+
+ false
+
+
+
+ false
+ 500
+
+ \Memory\Pages/sec
+ \PhysicalDisk(_Total)\Avg. Disk Queue Length
+ \Processor(_Total)\% Processor Time
+
+
+
+ true
+ false
+ false
+
+ false
+
+
+ false
+
+
+
+ Client\obj\Release\Client.exe
+ 01/01/0001 00:00:00
+ true
+ true
+ false
+ false
+ false
+ false
+ false
+ true
+ false
+ Executable
+ Client\bin\Release\Client.exe
+ Client\bin\Release\
+
+
+ IIS
+ InternetExplorer
+ true
+ false
+
+ false
+
+
+ false
+
+ {2236D5D4-7816-4630-8C86-0F0BDD46D7D8}|Client\Client.csproj
+ Client\Client.csproj
+ Client
+
+
+ Server\obj\Release\Server.exe
+ 01/01/0001 00:00:00
+ true
+ true
+ false
+ false
+ false
+ false
+ false
+ true
+ false
+ Executable
+ Server\bin\Release\Server.exe
+ Server\bin\Release\
+
+
+ IIS
+ InternetExplorer
+ true
+ false
+
+ false
+
+
+ false
+
+ {B458552A-5884-4B27-BA6B-826BC5590106}|Server\Server.csproj
+ Server\Server.csproj
+ Server
+
+
+
+
+ Client180412.vspx
+
+
+
+
+ :PB:{2236D5D4-7816-4630-8C86-0F0BDD46D7D8}|Client\Client.csproj
+
+
+ :PB:{B458552A-5884-4B27-BA6B-826BC5590106}|Server\Server.csproj
+
+
+
\ No newline at end of file
diff --git a/Client/Client.csproj b/Client/Client.csproj
index 46a1543..a8d8a4d 100644
--- a/Client/Client.csproj
+++ b/Client/Client.csproj
@@ -80,6 +80,8 @@
+
+
diff --git a/Client/ConsoleForms/ConsoleController.cs b/Client/ConsoleForms/ConsoleController.cs
index f018520..1d43bb7 100644
--- a/Client/ConsoleForms/ConsoleController.cs
+++ b/Client/ConsoleForms/ConsoleController.cs
@@ -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();
});
diff --git a/Client/ConsoleForms/Context.cs b/Client/ConsoleForms/Context.cs
index 2c3ec8b..c00f11f 100644
--- a/Client/ConsoleForms/Context.cs
+++ b/Client/ConsoleForms/Context.cs
@@ -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(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);
+ }
}
}
diff --git a/Client/ConsoleForms/Graphics/DialogView.cs b/Client/ConsoleForms/Graphics/DialogView.cs
index 30f08cc..28f6956 100644
--- a/Client/ConsoleForms/Graphics/DialogView.cs
+++ b/Client/ConsoleForms/Graphics/DialogView.cs
@@ -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[0])
)), lang)
{
ViewData optionsData = parameters.Get("Options");
diff --git a/Client/ConsoleForms/Graphics/InputView.cs b/Client/ConsoleForms/Graphics/InputView.cs
index d744298..dfc015a 100644
--- a/Client/ConsoleForms/Graphics/InputView.cs
+++ b/Client/ConsoleForms/Graphics/InputView.cs
@@ -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)
diff --git a/Client/ConsoleForms/Graphics/ListView.cs b/Client/ConsoleForms/Graphics/ListView.cs
index 7abf0a6..59ae739 100644
--- a/Client/ConsoleForms/Graphics/ListView.cs
+++ b/Client/ConsoleForms/Graphics/ListView.cs
@@ -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 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);
diff --git a/Client/ConsoleForms/Graphics/TextView.cs b/Client/ConsoleForms/Graphics/TextView.cs
index 3b749e4..a4974c4 100644
--- a/Client/ConsoleForms/Graphics/TextView.cs
+++ b/Client/ConsoleForms/Graphics/TextView.cs
@@ -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; }
diff --git a/Client/ConsoleForms/Graphics/View.cs b/Client/ConsoleForms/Graphics/View.cs
index c45a451..54108ff 100644
--- a/Client/ConsoleForms/Graphics/View.cs
+++ b/Client/ConsoleForms/Graphics/View.cs
@@ -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);
diff --git a/Client/ConsoleForms/LangManager.cs b/Client/ConsoleForms/LangManager.cs
index aa98b4e..9bfc4f3 100644
--- a/Client/ConsoleForms/LangManager.cs
+++ b/Client/ConsoleForms/LangManager.cs
@@ -88,25 +88,27 @@ 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 priorities = new List();
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"))
{
- for (int i = 0; i < priorities.Count; ++i)
- if (!priorities[i].Name.Equals("Default") && ComparePriority(el, priorities[i]))
- {
- priorities.Insert(i, el);
- break;
- }
- else if (i == priorities.Count - 1)
- {
- priorities.Add(el);
- break;
- }
+ 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]))
+ {
+ priorities.Insert(i, el);
+ break;
+ }
+ else if (i == priorities.Count - 1)
+ {
+ priorities.Add(el);
+ break;
+ }
}
}
@@ -125,7 +127,7 @@ namespace Client.ConsoleForms
break;
}
- // Use defults and fallbacks
+ // Use defaults and fallbacks
for (int i = 0; i controller.CloseView(s), "EmptyFieldError", "IPError", "PortError", "ConnectionError");
- ((InputView)views.GetNamed("NetConnect")).SubmissionsListener = i =>
+ bool connecting = false;
+
+ GetView("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("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;
diff --git a/Client/Context/SessionContext.cs b/Client/Context/SessionContext.cs
index 8394e8d..fed4302 100644
--- a/Client/Context/SessionContext.cs
+++ b/Client/Context/SessionContext.cs
@@ -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("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
}
}
diff --git a/Client/Context/WelcomeContext.cs b/Client/Context/WelcomeContext.cs
index e42543e..064dc9d 100644
--- a/Client/Context/WelcomeContext.cs
+++ b/Client/Context/WelcomeContext.cs
@@ -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("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 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("Login").InputListener = (v, c, i) =>
{
c.BackgroundColor = v.DefaultBackgroundColor;
c.SelectBackgroundColor = v.DefaultSelectBackgroundColor;
return true;
};
- ((InputView)views.GetNamed("Register")).SubmissionsListener = i =>
+ GetView("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("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("Register").SelectedField = 0;
+ foreach (var v in GetView("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("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
}
}
diff --git a/Client/Networking.cs b/Client/Networking.cs
index 9e60a83..4b3b2c0 100644
--- a/Client/Networking.cs
+++ b/Client/Networking.cs
@@ -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 promises = new Dictionary();
protected NetClient client;
- private bool authenticating = true, authenticated = false;
- public bool Authenticating { get => authenticating; }
- public bool PeerIsAuthenticated { get => authenticated; }
- public RSA AuthenticatedKeys { get; private set; }
- public bool IsAlive { get => client.IsAlive; }
-
- public BankNetInteractor(string address, short port, bool checkIdentity = true)
+ 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(() =>
- {
- //AuthenticatedKeys = NetClient.CheckServerIdentity(address, port, provider);
- authenticating = false;
- authenticated = true;// AuthenticatedKeys != null;
- }).Start();
- else
+ get
{
- 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
+
+
+
+
+ @string/NC_identity
+
+
+ padding_bottom="1">
-
- @string/SE_hist
-
-
-
- @string/SE_tx
+
+ @string/SE_view
@string/SE_pwdu
+
+
+ @string/SE_exit
+
+
+
+
+
+
+ @string/GENERIC_dismiss
+
+
+
+
+
+
+
+
+ @string/SE_info
+
\ No newline at end of file
diff --git a/Client/Resources/Strings/Meta.xml b/Client/Resources/Strings/Meta.xml
index ff86686..16a36fa 100644
--- a/Client/Resources/Strings/Meta.xml
+++ b/Client/Resources/Strings/Meta.xml
@@ -1,6 +1,6 @@
-
- sv_SE
- en_US
- en_GB
-
\ No newline at end of file
+
+ sv_SE
+ en_US
+ en_GB
+
\ No newline at end of file
diff --git a/Client/Resources/Strings/en_GB/strings.xml b/Client/Resources/Strings/en_GB/strings.xml
index 72ded50..a9e7c74 100644
--- a/Client/Resources/Strings/en_GB/strings.xml
+++ b/Client/Resources/Strings/en_GB/strings.xml
@@ -10,6 +10,7 @@
The supplied IP-address is not valid
The supplied port is not valid
Could not connect to server
+ Verifying server identity...
Welcome to the Tofvesson banking system! To continue, press [ENTER] To go back, press [ESCAPE]
Register Account
@@ -27,10 +28,23 @@
Login
Balance: $1
- View transaction history
+ Transaction history
Transfer funds
+ Send to
+ Account
+ View accounts
+ Amount to transfer
+ Include a message
Update password
+ Log out
+ Open an account
+ Close an account
+ Show accounts
+ $0
+Balance: $1
+Date of creation: $2
+ Close
Ok
Yes
No
diff --git a/Client/Resources/Strings/en_US/strings.xml b/Client/Resources/Strings/en_US/strings.xml
index 72ded50..a9e7c74 100644
--- a/Client/Resources/Strings/en_US/strings.xml
+++ b/Client/Resources/Strings/en_US/strings.xml
@@ -10,6 +10,7 @@
The supplied IP-address is not valid
The supplied port is not valid
Could not connect to server
+ Verifying server identity...
Welcome to the Tofvesson banking system! To continue, press [ENTER] To go back, press [ESCAPE]
Register Account
@@ -27,10 +28,23 @@
Login
Balance: $1
- View transaction history
+ Transaction history
Transfer funds
+ Send to
+ Account
+ View accounts
+ Amount to transfer
+ Include a message
Update password
+ Log out
+ Open an account
+ Close an account
+ Show accounts
+ $0
+Balance: $1
+Date of creation: $2
+ Close
Ok
Yes
No
diff --git a/Client/Resources/Strings/sv_SE/strings.xml b/Client/Resources/Strings/sv_SE/strings.xml
index 8157cf9..fba1d41 100644
--- a/Client/Resources/Strings/sv_SE/strings.xml
+++ b/Client/Resources/Strings/sv_SE/strings.xml
@@ -3,6 +3,53 @@
Serverkonfiguration
Den valda serverns identitet kunde inte verifieras. Detta innebär att det inte är en officiell server. Fortsätt?
Kopplar upp mot servern...
+ Fortsätt
+ Avbryt
+ Server IP:
+ Port:
+ Den givna IP-addressen är inte giltig
+ Den givna porten är inte giltig
+ Kunde inte koppla till servern
+ Verifierar serverns identitet...
- Ett eller fler obligatoriska inputfält är tomma!
+ Välkommen till Tofvessons banksystem!
+ För att fortsätta, tryck [ENTER]
+ För att backa, tryck [ESCAPE]
+ Registrera konto
+ Registrerar...
+ Ett konto med det givna användarnamnet finns redan!
+ De givna lösenorden matchar inte!
+ Det angivna lösenordet ases vara svagt. Är du säker på att du vill fortsätta?
+ Logga in
+ Autentiserar...
+ Det givna användarnamnet eller lösenordet var felaktigt
+ Användarnamn:
+ Lösenord:
+ Upprepa lösenord:
+ Registrera
+ Logga in
+
+ Kontobalans: $1
+ Transaktionshistorik
+ Överför pengar
+ Skicka till
+ Konto
+ Visa konton
+ Värde att överföra
+ Inkludera ett meddelande
+ Uppdatera lösenord
+ Logga ut
+ Öppna ett konto
+ Stäng ett konto
+ Visa konton
+ "$0"
+Kontobalans: $1
+Begynnelsedatum: $2
+
+ Stäng
+ Ok
+ Ja
+ Nej
+
+ Ett eller flera obligatoriska inputfält är tomma!
\ No newline at end of file
diff --git a/Client180412.vspx b/Client180412.vspx
new file mode 100644
index 0000000..9753c72
Binary files /dev/null and b/Client180412.vspx differ
diff --git a/Common/BinaryCollector.cs b/Common/BinaryCollector.cs
deleted file mode 100644
index a3d8b24..0000000
--- a/Common/BinaryCollector.cs
+++ /dev/null
@@ -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> expired = new List>();
-
- 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 supportedTypes = new List()
- {
- 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;
-
- ///
- /// Allocates a new binary collector.
- ///
- 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 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, 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)
- {
- 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