* Added flush command to server to flush database

* Fixed how database manages user removal
* Added serverside user removal
* Added UI and netcode for user removal on clientside
* Fixed text rendering for TextView and InputView
* Removed redundant properties from TextView
* Simplified event system in Intro
This commit is contained in:
Gabriel Tofvesson 2018-05-15 23:21:58 +02:00
parent fdad3be98d
commit 9755f06120
15 changed files with 155 additions and 117 deletions

View File

@ -139,6 +139,17 @@ namespace Client
}); });
} }
public async virtual Task<Promise> DeleteUser()
{
await StatusCheck(true);
client.Send(CreateCommandMessage("RmUsr", sessionID, out var pID));
return RegisterEventPromise(pID, p =>
{
PostPromise(p.handler, !p.Value.StartsWith("ERROR"));
return false;
});
}
public async virtual Task<Promise> UpdatePassword(string newPass) public async virtual Task<Promise> UpdatePassword(string newPass)
{ {
await StatusCheck(true); await StatusCheck(true);

View File

@ -67,7 +67,7 @@ namespace Client.ConsoleForms.Graphics
int pl = padding.Left(), pr = padding.Right(); int pl = padding.Left(), pr = padding.Right();
Console.SetCursorPosition(left, top++); Console.SetCursorPosition(left, top++);
int pad = MaxWidth - options.CollectiveLength() - options.Length;// + pl + pr; int pad = ContentWidth - options.CollectiveLength() - options.Length;// + pl + pr;
int lpad = (int)(pad / 2f); int lpad = (int)(pad / 2f);
Console.BackgroundColor = BackgroundColor; Console.BackgroundColor = BackgroundColor;
Console.Write(Filler(' ', lpad)); Console.Write(Filler(' ', lpad));

View File

@ -32,6 +32,7 @@ namespace Client.ConsoleForms.Graphics
} }
} }
private string[][] splitInputs; private string[][] splitInputs;
protected ViewData data;
public InputView(ViewData parameters, LangManager lang) : base(parameters, lang) public InputView(ViewData parameters, LangManager lang) : base(parameters, lang)
@ -47,6 +48,8 @@ namespace Client.ConsoleForms.Graphics
DefaultSelectBackgroundColor = (ConsoleColor)sBC; DefaultSelectBackgroundColor = (ConsoleColor)sBC;
DefaultSelectTextColor = (ConsoleColor)sTC; DefaultSelectTextColor = (ConsoleColor)sTC;
this.data = parameters;
List<InputField> fields = new List<InputField>(); List<InputField> fields = new List<InputField>();
foreach (var data in parameters.nestedData.GetFirst(d => d.Name.Equals("Fields")).nestedData) foreach (var data in parameters.nestedData.GetFirst(d => d.Name.Equals("Fields")).nestedData)
if (!data.Name.Equals("Field")) continue; if (!data.Name.Equals("Field")) continue;
@ -63,14 +66,19 @@ namespace Client.ConsoleForms.Graphics
Inputs = fields.ToArray(); Inputs = fields.ToArray();
int max = ContentWidth;
int computedSize = 0; int computedSize = 0;
splitInputs = new string[Inputs.Length][]; splitInputs = new string[Inputs.Length][];
for (int i = 0; i < Inputs.Length; ++i) for (int i = 0; i < Inputs.Length; ++i)
{ {
splitInputs[i] = ComputeTextDimensions(Inputs[i].Label.Split(' ')); splitInputs[i] = ComputeTextDimensions(Inputs[i].Label.Split(' '));
foreach (var input in splitInputs[i])
if (input.Length > max)
max = input.Length;
computedSize += splitInputs[i].Length; computedSize += splitInputs[i].Length;
} }
ContentHeight += computedSize + Inputs.Length * 2; ContentHeight += computedSize + Inputs.Length * 2;
if (ContentWidth < max) ContentWidth = max;
} }
public int IndexOf(InputField field) public int IndexOf(InputField field)
@ -98,7 +106,7 @@ namespace Client.ConsoleForms.Graphics
{ {
Console.SetCursorPosition(left, top++); Console.SetCursorPosition(left, top++);
Console.BackgroundColor = BackgroundColor; Console.BackgroundColor = BackgroundColor;
Console.Write(splitInputs[j][i] + Filler(' ', MaxWidth - splitInputs[j][i].Length)); Console.Write(splitInputs[j][i] + Filler(' ', ContentWidth - splitInputs[j][i].Length));
} }
Console.SetCursorPosition(left, top++); Console.SetCursorPosition(left, top++);
@ -114,10 +122,10 @@ namespace Client.ConsoleForms.Graphics
if (Inputs[j].SelectIndex < Inputs[j].Text.Length) if (Inputs[j].SelectIndex < Inputs[j].Text.Length)
Console.Write( Console.Write(
Inputs[j].ShowText ? Inputs[j].ShowText ?
Inputs[j].Text.Substring(Inputs[j].SelectIndex + 1, drawn = Math.Min(maxWidth + Inputs[j].SelectIndex - Inputs[j].RenderStart - 1, Inputs[j].Text.Length - Inputs[j].SelectIndex - 1)) : Inputs[j].Text.Substring(Inputs[j].SelectIndex + 1, drawn = Math.Min(ContentWidth + Inputs[j].SelectIndex - Inputs[j].RenderStart - 1, Inputs[j].Text.Length - Inputs[j].SelectIndex - 1)) :
Filler('*', drawn = Math.Min(maxWidth + Inputs[j].SelectIndex - Inputs[j].RenderStart - 1, Inputs[j].Text.Length - Inputs[j].SelectIndex - 1)) Filler('*', drawn = Math.Min(ContentWidth + Inputs[j].SelectIndex - Inputs[j].RenderStart - 1, Inputs[j].Text.Length - Inputs[j].SelectIndex - 1))
); );
Console.Write(Filler(' ', maxWidth - 1 - drawn - Inputs[j].SelectIndex + Inputs[j].RenderStart)); Console.Write(Filler(' ', ContentWidth - 1 - drawn - Inputs[j].SelectIndex + Inputs[j].RenderStart));
Console.ForegroundColor = ConsoleColor.Black; Console.ForegroundColor = ConsoleColor.Black;
} }
} }
@ -140,7 +148,7 @@ namespace Client.ConsoleForms.Graphics
case ConsoleKey.RightArrow: case ConsoleKey.RightArrow:
if (Inputs[selectedField].SelectIndex < Inputs[selectedField].Text.Length) if (Inputs[selectedField].SelectIndex < Inputs[selectedField].Text.Length)
{ {
if (++Inputs[selectedField].SelectIndex - Inputs[selectedField].RenderStart == maxWidth) ++Inputs[selectedField].RenderStart; if (++Inputs[selectedField].SelectIndex - Inputs[selectedField].RenderStart == ContentWidth) ++Inputs[selectedField].RenderStart;
} }
else return changed; else return changed;
break; break;
@ -175,6 +183,7 @@ namespace Client.ConsoleForms.Graphics
else return changed; else return changed;
break; break;
case ConsoleKey.Enter: case ConsoleKey.Enter:
ParseAction(data)();
SubmissionsListener?.Invoke(this); SubmissionsListener?.Invoke(this);
return changed; return changed;
case ConsoleKey.Escape: case ConsoleKey.Escape:
@ -184,7 +193,7 @@ namespace Client.ConsoleForms.Graphics
{ {
if (InputListener?.Invoke(this, Inputs[selectedField], info, triggered) == false) break; if (InputListener?.Invoke(this, Inputs[selectedField], info, triggered) == false) break;
Inputs[selectedField].Text = Inputs[selectedField].Text.Substring(0, Inputs[selectedField].SelectIndex) + info.KeyChar + Inputs[selectedField].Text.Substring(Inputs[selectedField].SelectIndex); Inputs[selectedField].Text = Inputs[selectedField].Text.Substring(0, Inputs[selectedField].SelectIndex) + info.KeyChar + Inputs[selectedField].Text.Substring(Inputs[selectedField].SelectIndex);
if (++Inputs[selectedField].SelectIndex - Inputs[selectedField].RenderStart == maxWidth) ++Inputs[selectedField].RenderStart; if (++Inputs[selectedField].SelectIndex - Inputs[selectedField].RenderStart == ContentWidth) ++Inputs[selectedField].RenderStart;
} }
else return changed; else return changed;
break; break;

View File

@ -32,29 +32,7 @@ namespace Client.ConsoleForms.Graphics
Dirty = true; Dirty = true;
} }
} }
public int MaxWidth
{
get => maxWidth;
set
{
maxWidth = value;
text_render = ComputeTextDimensions(text);
Dirty = true;
}
}
public int MaxHeight
{
get => maxHeight;
set
{
maxHeight = value;
text_render = ComputeTextDimensions(text);
Dirty = true;
}
}
public override Region Occlusion => new Region( public override Region Occlusion => new Region(
new Rectangle( new Rectangle(
-padding.Left() - (DrawBorder ? 2 : 0), // Left bound -padding.Left() - (DrawBorder ? 2 : 0), // Left bound
@ -231,7 +209,7 @@ namespace Client.ConsoleForms.Graphics
for (int i = 0; i < text_render.Length; ++i) for (int i = 0; i < text_render.Length; ++i)
{ {
Console.SetCursorPosition(left, top++); Console.SetCursorPosition(left, top++);
Console.Write(/*Filler(' ', pl) + */text_render[i] + Filler(' ', MaxWidth - text_render[i].Length)/* + Filler(' ', pr)*/); Console.Write(/*Filler(' ', pl) + */text_render[i] + Filler(' ', ContentWidth - text_render[i].Length)/* + Filler(' ', pr)*/);
} }
} }
@ -243,7 +221,7 @@ namespace Client.ConsoleForms.Graphics
{ {
Console.SetCursorPosition(left, top++); Console.SetCursorPosition(left, top++);
Console.BackgroundColor = BackgroundColor; Console.BackgroundColor = BackgroundColor;
Console.Write(Filler(' ', maxWidth/* + pl + pr*/)); Console.Write(Filler(' ', ContentWidth/* + pl + pr*/));
} }
} }
} }

View File

@ -139,7 +139,7 @@ namespace Client.ConsoleForms.Graphics
} }
protected EventAction ParseAction(ViewData data) protected EventAction ParseAction(ViewData data)
{ {
bool.TryParse(data.GetAttribute("close"), out bool close); bool.TryParse(data?.GetAttribute("close")??"", out bool close);
return ParseAction(data.GetAttribute("event"), close); return ParseAction(data.GetAttribute("event"), close);
} }
protected EventAction ParseAction(string action, bool close) protected EventAction ParseAction(string action, bool close)

View File

@ -13,56 +13,11 @@ namespace Client
public IntroContext(ContextManager manager, Action onComplete) : base(manager, "Intro", "Common") public IntroContext(ContextManager manager, Action onComplete) : base(manager, "Intro", "Common")
{ {
GetView<DialogView>("welcome").RegisterSelectListener((v, i, s) => GetView<DialogView>("welcome").RegisterSelectListener((v, i, s) =>
{
if (i == 1)
{
Hide(v);
onComplete();
}
else
{
Hide(v);
Show("describe1");
}
});
GetView<DialogView>("describe1").RegisterSelectListener((v, i, s) =>
{
if (i == 1) v.TriggerKeyEvent(new ConsoleKeyInfo('\0', ConsoleKey.Escape, false, false, false));
else
{
Hide(v);
Show("describe2");
}
});
GetView<DialogView>("describe2").RegisterSelectListener((v, i, s) =>
{
if (i == 1) v.TriggerKeyEvent(new ConsoleKeyInfo('\0', ConsoleKey.Escape, false, false, false));
else
{
Hide(v);
Show("describe3");
}
});
GetView<InputView>("describe3").SubmissionsListener = v =>
{ {
Hide(v); Hide(v);
Show("describe4"); if (i == 1) onComplete();
}; else Show("describe1");
});
GetView<InputView>("describe4").SubmissionsListener = v =>
{
Hide(v);
Show("describe4_1");
};
GetView<InputView>("describe4_1").SubmissionsListener = v =>
{
Hide(v);
Show("describe5");
};
GetView<DialogView>("describe5").RegisterSelectListener((v, i, s) => GetView<DialogView>("describe5").RegisterSelectListener((v, i, s) =>
{ {
@ -77,22 +32,7 @@ namespace Client
}; };
} }
public override void OnCreate() public override void OnCreate() => Show("welcome");
{ public override void OnDestroy() { }
Show("welcome");
}
public override void OnDestroy()
{
}
// Graphics update trigger
public override bool Update(ConsoleController.KeyEvent keypress, bool hasKeypress = true)
{
// Return: whether or not to redraw graphics
return base.Update(keypress, hasKeypress);
}
} }
} }

View File

@ -148,6 +148,26 @@ namespace Client
}; };
}; };
options.GetView<ButtonView>("delete").SetEvent(v => Show("account_delete"));
GetView<DialogView>("account_delete").RegisterSelectListener((v, i, s) =>
{
Hide(v);
if (i == 1)
{
Show("delete_stall");
Promise deletion = Promise.AwaitPromise(interactor.DeleteUser());
deletion.Subscribe = p =>
{
Hide("delete_stall");
if (bool.Parse(p.Value))
controller.Popup(GetIntlString("SE_delete_success"), 2500, ConsoleColor.Green, () => manager.LoadContext(new NetContext(manager)));
else
controller.Popup(GetIntlString("SE_delete_failure"), 1500, ConsoleColor.Red);
};
}
});
// Actual "create account" input box thingy // Actual "create account" input box thingy
var input = GetView<InputView>("account_create"); var input = GetView<InputView>("account_create");
input.SubmissionsListener = __ => input.SubmissionsListener = __ =>

View File

@ -25,6 +25,8 @@ namespace Client
// Just close when anything is selected and "submitted" // Just close when anything is selected and "submitted"
RegisterSelectListeners((s, i, v) => controller.CloseView(s), "DuplicateAccountError", "EmptyFieldError", "IPError", "PortError", "AuthError", "PasswordMismatchError"); RegisterSelectListeners((s, i, v) => controller.CloseView(s), "DuplicateAccountError", "EmptyFieldError", "IPError", "PortError", "AuthError", "PasswordMismatchError");
// If Escape key is pressed, suggest to controller to terminate
GetView("WelcomeScreen").OnBackEvent = v => controller.ShouldExit = true;
GetView<InputView>("Login").SubmissionsListener = i => GetView<InputView>("Login").SubmissionsListener = i =>
{ {

View File

@ -22,7 +22,7 @@
padding_bottom="1" padding_bottom="1"
back="Intro:welcome"> back="Intro:welcome">
<Options> <Options>
<Option>@string/WS_continue</Option> <Option event="Intro:describe2" close="true">@string/WS_continue</Option>
</Options> </Options>
<Text>@string/WS_describe1</Text> <Text>@string/WS_describe1</Text>
</DialogView> </DialogView>
@ -35,8 +35,8 @@
padding_bottom="1" padding_bottom="1"
back="Intro:describe1"> back="Intro:describe1">
<Options> <Options>
<Option>@string/WS_continue</Option> <Option event="Intro:describe3" close="true">@string/WS_continue</Option>
<Option>@string/WS_back</Option> <Option event="Intro:describe3" close="true">@string/WS_back</Option>
</Options> </Options>
<Text>@string/WS_describe2</Text> <Text>@string/WS_describe2</Text>
</DialogView> </DialogView>
@ -47,7 +47,9 @@
padding_right="2" padding_right="2"
padding_top="1" padding_top="1"
padding_bottom="1" padding_bottom="1"
back="Intro:describe2"> back="Intro:describe2"
event="Intro:describe4"
close="true">
<Fields> <Fields>
<Field>@string/WS_input</Field> <Field>@string/WS_input</Field>
<Field>@string/WS_input</Field> <Field>@string/WS_input</Field>
@ -62,7 +64,9 @@
padding_right="2" padding_right="2"
padding_top="1" padding_top="1"
padding_bottom="1" padding_bottom="1"
back="Intro:describe3"> back="Intro:describe3"
event="Intro:describe4_1"
close="true">
<Fields> <Fields>
<Field>@string/WS_input</Field> <Field>@string/WS_input</Field>
<Field input_type="integer">@string/WS_input_integer</Field> <Field input_type="integer">@string/WS_input_integer</Field>
@ -77,7 +81,9 @@
padding_right="2" padding_right="2"
padding_top="1" padding_top="1"
padding_bottom="1" padding_bottom="1"
back="Intro:describe4"> back="Intro:describe4"
event="Intro:describe5"
close="true">
<Fields> <Fields>
<Field input_type="alphanumeric">@string/WS_input_alphanum</Field> <Field input_type="alphanumeric">@string/WS_input_alphanum</Field>
<Field hide="true">@string/WS_input_password</Field> <Field hide="true">@string/WS_input_password</Field>

View File

@ -90,12 +90,37 @@
<Text>@string/SE_pwdu</Text> <Text>@string/SE_pwdu</Text>
</ButtonView> </ButtonView>
<ButtonView id="delete">
<Text>@string/SE_delete</Text>
</ButtonView>
<ButtonView id="exit"> <ButtonView id="exit">
<Text>@string/SE_exit</Text> <Text>@string/SE_exit</Text>
</ButtonView> </ButtonView>
</Views> </Views>
</ListView> </ListView>
<TextView id="delete_stall"
padding_left="2"
padding_right="2"
padding_top="1"
padding_bottom="1">
<Text>@string/SE_delete_stall</Text>
</TextView>
<DialogView id="account_delete"
padding_left="2"
padding_right="2"
padding_top="1"
padding_bottom="1"
border="4">
<Options>
<Option>@string/GENERIC_negative</Option>
<Option>@string/GENERIC_positive</Option>
</Options>
<Text>@string/SE_delete_warn</Text>
</DialogView>
<InputView id="account_create" <InputView id="account_create"
padding_left="2" padding_left="2"
padding_right="2" padding_right="2"

View File

@ -79,7 +79,7 @@ To go back, press [ESCAPE]</Entry>
<Entry name="SE_hist">Transaction history</Entry> <Entry name="SE_hist">Transaction history</Entry>
<Entry name="SE_tx">Transfer funds</Entry> <Entry name="SE_tx">Transfer funds</Entry>
<Entry name="SE_tx_success">Funds transferred!</Entry> <Entry name="SE_tx_success">Funds transferred!</Entry>
<Entry name="SE_who">Send to</Entry> <Entry name="SE_who">Send to:</Entry>
<Entry name="SE_where">Account</Entry> <Entry name="SE_where">Account</Entry>
<Entry name="SE_where_f">From Account:</Entry> <Entry name="SE_where_f">From Account:</Entry>
<Entry name="SE_where_t">To Account:</Entry> <Entry name="SE_where_t">To Account:</Entry>
@ -98,6 +98,12 @@ Is this correct?</Entry>
<Entry name="SE_exit">Log out</Entry> <Entry name="SE_exit">Log out</Entry>
<Entry name="SE_open">Open an account</Entry> <Entry name="SE_open">Open an account</Entry>
<Entry name="SE_close">Close account</Entry> <Entry name="SE_close">Close account</Entry>
<Entry name="SE_delete">Delete user account</Entry>
<Entry name="SE_delete_warn">WARNING: This will delete the current user and all connected accounts!
Are you sure you would like to continue?</Entry>
<Entry name="SE_delete_stall">Deleting...</Entry>
<Entry name="SE_delete_success">User deleted</Entry>
<Entry name="SE_delete_failure">User could not be deleted</Entry>
<Entry name="SE_accounts">Show accounts</Entry> <Entry name="SE_accounts">Show accounts</Entry>
<Entry name="SE_balance_toohigh">Supplied balance is higher than available amount in source account! <Entry name="SE_balance_toohigh">Supplied balance is higher than available amount in source account!
Available balance: $0 SEK</Entry> Available balance: $0 SEK</Entry>

View File

@ -79,7 +79,7 @@ To go back, press [ESCAPE]</Entry>
<Entry name="SE_hist">Transaction history</Entry> <Entry name="SE_hist">Transaction history</Entry>
<Entry name="SE_tx">Transfer funds</Entry> <Entry name="SE_tx">Transfer funds</Entry>
<Entry name="SE_tx_success">Funds transferred!</Entry> <Entry name="SE_tx_success">Funds transferred!</Entry>
<Entry name="SE_who">Send to</Entry> <Entry name="SE_who">Send to:</Entry>
<Entry name="SE_where">Account</Entry> <Entry name="SE_where">Account</Entry>
<Entry name="SE_where_f">From Account:</Entry> <Entry name="SE_where_f">From Account:</Entry>
<Entry name="SE_where_t">To Account:</Entry> <Entry name="SE_where_t">To Account:</Entry>
@ -98,6 +98,12 @@ Is this correct?</Entry>
<Entry name="SE_exit">Log out</Entry> <Entry name="SE_exit">Log out</Entry>
<Entry name="SE_open">Open an account</Entry> <Entry name="SE_open">Open an account</Entry>
<Entry name="SE_close">Close account</Entry> <Entry name="SE_close">Close account</Entry>
<Entry name="SE_delete">Delete user account</Entry>
<Entry name="SE_delete_warn">WARNING: This will delete the current user and all connected accounts!
Are you sure you would like to continue?</Entry>
<Entry name="SE_delete_stall">Deleting...</Entry>
<Entry name="SE_delete_success">User deleted</Entry>
<Entry name="SE_delete_failure">User could not be deleted</Entry>
<Entry name="SE_accounts">Show accounts</Entry> <Entry name="SE_accounts">Show accounts</Entry>
<Entry name="SE_balance_toohigh">Supplied balance is higher than available amount in source account! <Entry name="SE_balance_toohigh">Supplied balance is higher than available amount in source account!
Available balance: $0 SEK</Entry> Available balance: $0 SEK</Entry>

View File

@ -84,7 +84,7 @@ För att backa, tryck [ESCAPE]</Entry>
<Entry name="SE_hist">Transaktionshistorik</Entry> <Entry name="SE_hist">Transaktionshistorik</Entry>
<Entry name="SE_tx">Överför pengar</Entry> <Entry name="SE_tx">Överför pengar</Entry>
<Entry name="SE_tx_success">Belopp överfört!</Entry> <Entry name="SE_tx_success">Belopp överfört!</Entry>
<Entry name="SE_who">Skicka till</Entry> <Entry name="SE_who">Skicka till:</Entry>
<Entry name="SE_where">Konto</Entry> <Entry name="SE_where">Konto</Entry>
<Entry name="SE_where_f">Från konto:</Entry> <Entry name="SE_where_f">Från konto:</Entry>
<Entry name="SE_where_t">Till konto:</Entry> <Entry name="SE_where_t">Till konto:</Entry>
@ -103,6 +103,13 @@ Till kontot: $2
<Entry name="SE_exit">Logga ut</Entry> <Entry name="SE_exit">Logga ut</Entry>
<Entry name="SE_open">Öppna ett konto</Entry> <Entry name="SE_open">Öppna ett konto</Entry>
<Entry name="SE_close">Stäng konto</Entry> <Entry name="SE_close">Stäng konto</Entry>
<Entry name="SE_delete">Radera användare</Entry>
<Entry name="SE_delete_warn">VARNING: Detta kommer att radera den nuvarande användaren
och alla kopplade konton!
Vill du fortsätta?</Entry>
<Entry name="SE_delete_stall">Raderar...</Entry>
<Entry name="SE_delete_success">Användare raderad</Entry>
<Entry name="SE_delete_failure">Användare kunde inte raderas</Entry>
<Entry name="SE_accounts">Visa konton</Entry> <Entry name="SE_accounts">Visa konton</Entry>
<Entry name="SE_balance_toohigh">Angivet belopp är högre än det tillgängliga beloppet i ursprungskontot! <Entry name="SE_balance_toohigh">Angivet belopp är högre än det tillgängliga beloppet i ursprungskontot!
Tillgängligt saldo: $0 SEK</Entry> Tillgängligt saldo: $0 SEK</Entry>

View File

@ -73,17 +73,19 @@ namespace Server
public void RemoveUser(User entry) => RemoveUser(entry, true); public void RemoveUser(User entry) => RemoveUser(entry, true);
private void RemoveUser(User entry, bool withFlush) private void RemoveUser(User entry, bool withFlush)
{ {
entry = ToEncoded(entry); // Remove from loaded users collection
for (int i = 0; i < loadedUsers.Count; ++i) for (int i = 0; i < loadedUsers.Count; ++i)
if (entry.Equals(loadedUsers[i])) if (entry.Name.Equals(loadedUsers[i].Name))
loadedUsers.RemoveAt(i); loadedUsers.RemoveAt(i);
// Changes are retracted from change collectino
for (int i = changeList.Count - 1; i >= 0; --i) for (int i = changeList.Count - 1; i >= 0; --i)
if (changeList[i].Equals(entry.Name)) if (changeList[i].Name.Equals(entry.Name))
changeList.RemoveAt(i); changeList.RemoveAt(i);
// Check if user already is scheduled for deletion
for (int i = toRemove.Count - 1; i >= 0; --i) for (int i = toRemove.Count - 1; i >= 0; --i)
if (toRemove[i].Equals(entry.Name)) if (toRemove[i].Name.Equals(entry.Name))
return; return;
toRemove.Add(entry); toRemove.Add(entry);
@ -104,9 +106,10 @@ namespace Server
using(var reader = XmlReader.Create(DatabaseName)) using(var reader = XmlReader.Create(DatabaseName))
{ {
int masterDepth = 0; int masterDepth = 0;
bool trigger = false, wn = false, recent = false; bool trigger = false, wn = false, recent = false, justTriggered = false;
while (wn || reader.Read()) while (wn || reader.Read() || (reader.NodeType==XmlNodeType.None && justTriggered))
{ {
justTriggered = false;
wn = false; wn = false;
if (trigger) if (trigger)
{ {
@ -114,7 +117,7 @@ namespace Server
WriteUser(writer, user); WriteUser(writer, user);
bool wroteNode = false; bool wroteNode = false;
while ((wroteNode || reader.Name.Equals("User") || reader.Read()) && reader.NodeType != XmlNodeType.EndElement) while ((wroteNode || reader.Name.Equals("User") || reader.Read()) && reader.NodeType != XmlNodeType.EndElement && reader.NodeType != XmlNodeType.None)
{ {
wroteNode = false; wroteNode = false;
if (reader.Name.Equals("User")) if (reader.Name.Equals("User"))
@ -154,6 +157,7 @@ namespace Server
if (masterDepth != MasterEntry.Length && reader.Name.Equals(MasterEntry[masterDepth])) if (masterDepth != MasterEntry.Length && reader.Name.Equals(MasterEntry[masterDepth]))
{ {
trigger = reader.NodeType == XmlNodeType.Element && ++masterDepth == MasterEntry.Length; trigger = reader.NodeType == XmlNodeType.Element && ++masterDepth == MasterEntry.Length;
justTriggered = true;
reader.MoveToContent(); reader.MoveToContent();
writer.WriteStartElement(MasterEntry[masterDepth - 1]); writer.WriteStartElement(MasterEntry[masterDepth - 1]);
} }
@ -241,10 +245,18 @@ namespace Server
public User FirstUser(Predicate<User> p) public User FirstUser(Predicate<User> p)
{ {
if (p == null) return null; // Done to conveniently handle system insertions if (p == null) return null; // Done to conveniently handle system insertions
// Check if user is scheduled for removal
foreach (var entry in toRemove)
if (p(entry))
return null;
// Check loaded users
foreach (var entry in loadedUsers) foreach (var entry in loadedUsers)
if (p(entry)) if (p(entry))
return entry; return entry;
// Check modified users
foreach (var entry in changeList) foreach (var entry in changeList)
if (p(entry)) if (p(entry))
{ {
@ -252,6 +264,7 @@ namespace Server
return entry; return entry;
} }
// Read from database
using (var reader = XmlReader.Create(DatabaseName)) using (var reader = XmlReader.Create(DatabaseName))
{ {
if (!Traverse(reader, MasterEntry)) return null; if (!Traverse(reader, MasterEntry)) return null;

View File

@ -147,11 +147,23 @@ Use command 'help' to get a list of available commands";
// Server endpoints // Server endpoints
switch (cmd[0]) switch (cmd[0])
{ {
case "RmUsr":
{
if (!GetUser(cmd[1], out var user))
{
if(verbosity > 0) Output.Error($"Could not delete user from session as session isn't valid. (SessionID=\"{cmd[1]}\")");
return ErrorResponse(id, "badsession");
}
manager.Expire(user);
db.RemoveUser(user);
if (verbosity > 0) Output.Info($"Removed user \"{user.Name}\" (SessionID={cmd[1]})");
return GenerateResponse(id, true);
}
case "Auth": // Log in to a user account (get a session id) case "Auth": // Log in to a user account (get a session id)
{ {
if(!ParseDataPair(cmd[1], out string user, out string pass)) if(!ParseDataPair(cmd[1], out string user, out string pass))
{ {
Output.Error($"Recieved problematic username or password! (User: \"{user}\")"); if(verbosity > 0) Output.Error($"Recieved problematic username or password! (User: \"{user}\")");
return ErrorResponse(id); return ErrorResponse(id);
} }
Database.User usr = db.GetUser(user); Database.User usr = db.GetUser(user);
@ -447,7 +459,9 @@ Use command 'help' to get a list of available commands";
} }
Output.Raw($"Current verbosity level: {(verbosity<1?"FATAL":verbosity==1?"INFO":"DEBUG")}"); Output.Raw($"Current verbosity level: {(verbosity<1?"FATAL":verbosity==1?"INFO":"DEBUG")}");
}), "Get or set verbosity level: DEBUG, INFO, FATAL (alternatively enter 0, 1 or 2 respectively)") }), "Get or set verbosity level: DEBUG, INFO, FATAL (alternatively enter 0, 1 or 2 respectively)")
.Append(new Command("sess").WithParameter("sessionID", 'r', Parameter.ParamType.STRING, true).SetAction( // Display active sessions .Append(new Command("sess") // Display active sessions
.WithParameter("sessionID", 'r', Parameter.ParamType.STRING, true)
.SetAction(
(c, l) => { (c, l) => {
StringBuilder builder = new StringBuilder(); StringBuilder builder = new StringBuilder();
manager.Update(); // Ensure that we don't show expired sessions (artifacts exist until it is necessary to remove them) manager.Update(); // Ensure that we don't show expired sessions (artifacts exist until it is necessary to remove them)
@ -498,7 +512,8 @@ Use command 'help' to get a list of available commands";
db.AddUser(user); db.AddUser(user);
} }
else Output.Raw(user.IsAdministrator); else Output.Raw(user.IsAdministrator);
}), "Show or set admin status for a user"); }), "Show or set admin status for a user")
.Append(new Command("flush").SetAction(() => { db.Flush(); Output.Raw("Database flushed"); }), "Flush database");// Flush database to database file
// Set up a persistent terminal-esque input design // Set up a persistent terminal-esque input design
Output.OnNewLine = () => Output.WriteOverwritable(">> "); Output.OnNewLine = () => Output.WriteOverwritable(">> ");