BankProject/Server/Database.cs
GabrielTofvesson 100f5a32be 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
2018-04-26 00:24:58 +02:00

638 lines
27 KiB
C#

using System;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Xml;
using Tofvesson.Crypto;
using Tofvesson.Collections;
namespace Server
{
public sealed class Database
{
private static readonly RandomProvider random = new RegularRandomProvider();
public string[] MasterEntry { get; }
public string DatabaseName { get; }
public bool LoadFull { get; set; }
// Cached changes
private readonly List<User> changeList = new List<User>();
private readonly List<User> toRemove = new List<User>();
private readonly EvictionList<User> loadedUsers = new EvictionList<User>(40);
public Database(string dbName, string master, bool loadFullDB = false)
{
dbName += ".xml";
MasterEntry = master.Split('/');
if (!File.Exists(dbName))
{
FileStream strm = File.Create(dbName);
byte[] b;
strm.Write(b = Encoding.UTF8.GetBytes($"<?xml version=\"1.0\" encoding=\"utf-8\" ?>\n"), 0, b.Length);
// Generate root element for users
for (int i = 0; i < MasterEntry.Length; ++i) strm.Write(b = Encoding.UTF8.GetBytes($"<{MasterEntry[i]}>"), 0, b.Length);
for (int i = MasterEntry.Length - 1; i >= 0; --i) strm.Write(b = Encoding.UTF8.GetBytes($"</{MasterEntry[i]}>"), 0, b.Length);
strm.Close();
}
DatabaseName = dbName;
LoadFull = loadFullDB;
}
// Flush before deletion
~Database() { Flush(false); }
// UpdateUser is just another name for AddUser
public void UpdateUser(User entry) => AddUser(entry, true);
public void AddUser(User entry) => AddUser(entry, true);
private void AddUser(User entry, bool withFlush)
{
entry = ToEncoded(entry);
for (int i = 0; i < loadedUsers.Count; ++i)
if (entry.Equals(loadedUsers[i]))
loadedUsers[i] = entry;
for (int i = toRemove.Count - 1; i >= 0; --i)
if (toRemove[i].Equals(entry.Name))
toRemove.RemoveAt(i);
for (int i = 0; i < changeList.Count; ++i)
if (changeList[i].Equals(entry.Name))
return;
changeList.Add(entry);
if(withFlush) Flush(true);
}
public void RemoveUser(User entry) => RemoveUser(entry, true);
private void RemoveUser(User entry, bool withFlush)
{
entry = ToEncoded(entry);
for (int i = 0; i < loadedUsers.Count; ++i)
if (entry.Equals(loadedUsers[i]))
loadedUsers.RemoveAt(i);
for (int i = changeList.Count - 1; i >= 0; --i)
if (changeList[i].Equals(entry.Name))
changeList.RemoveAt(i);
for (int i = toRemove.Count - 1; i >= 0; --i)
if (toRemove[i].Equals(entry.Name))
return;
toRemove.Add(entry);
if(withFlush) Flush(true);
}
// Triggers a forceful flush
public void Flush() => Flush(false);
// Permissive (cache-dependent) flush
private void Flush(bool optional)
{
if(!(optional || changeList.Count > 30 || toRemove.Count > 30)) return; // No need to flush
string temp = GenerateTempFileName("tmp_", ".xml");
using(var writer = XmlWriter.Create(temp))
{
using(var reader = XmlReader.Create(DatabaseName))
{
int masterDepth = 0;
bool trigger = false, wn = false, recent = false;
while (wn || reader.Read())
{
wn = false;
if (trigger)
{
foreach (var user in changeList)
WriteUser(writer, user);
bool wroteNode = false;
while ((wroteNode || reader.Name.Equals("User") || reader.Read()) && reader.NodeType != XmlNodeType.EndElement)
{
wroteNode = false;
if (reader.Name.Equals("User"))
{
User u = User.Parse(ReadEntry(reader), this);
if (u != null)
{
bool shouldWrite = true;
foreach (var toChange in changeList)
if (toChange.Name.Equals(u.Name))
{
shouldWrite = false;
break;
}
if (shouldWrite)
foreach (var remove in toRemove)
if (remove.Name.Equals(u.Name))
{
shouldWrite = false;
break;
}
if (shouldWrite) WriteUser(writer, u);
}
}
else
{
wroteNode = true;
writer.WriteNode(reader, true);
}
}
trigger = false;
recent = true;
writer.WriteEndElement();
toRemove.Clear();
changeList.Clear();
}
if (masterDepth != MasterEntry.Length && reader.Name.Equals(MasterEntry[masterDepth]))
{
trigger = reader.NodeType == XmlNodeType.Element && ++masterDepth == MasterEntry.Length;
reader.MoveToContent();
writer.WriteStartElement(MasterEntry[masterDepth - 1]);
}
else if (masterDepth == MasterEntry.Length && recent)
{
if(masterDepth!=1) writer.WriteEndElement();
recent = false;
}
else
{
wn = true;
writer.WriteNode(reader, true);
}
}
}
writer.Flush();
}
File.Delete(DatabaseName);
File.Move(temp, DatabaseName);
}
private static void WriteUser(XmlWriter writer, User u)
{
writer.WriteStartElement("User");
if (u.IsAdministrator) writer.WriteAttributeString("admin", "", "true");
writer.WriteElementString("Name", u.Name);
//writer.WriteElementString("Balance", u.Balance.ToString());
writer.WriteElementString("Password", u.PasswordHash);
writer.WriteElementString("Salt", u.Salt);
foreach(var 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);
writer.WriteEndElement();
}
writer.WriteEndElement();
}
writer.WriteEndElement();
}
private static string GenerateTempFileName(string prefix, string suffix)
{
string s;
do s = prefix + random.NextString((Math.Abs(random.NextInt())%16)+8) + suffix;
while (File.Exists(s));
return s;
}
private Entry ReadEntry(XmlReader reader)
{
Entry e = new Entry(reader.Name);
if (reader.HasAttributes)
{
reader.MoveToAttribute(0);
do e.Attributes.Add(reader.Name, reader.Value);
while (reader.MoveToNextAttribute());
}
reader.MoveToContent();
while (reader.Read() && reader.NodeType != XmlNodeType.EndElement)
{
if (reader.NodeType == XmlNodeType.Element)
using (var subRead = reader.ReadSubtree())
{
SkipSpaces(subRead);
e.NestedEntries.Add(ReadEntry(subRead));
}
else if (reader.NodeType == XmlNodeType.Text) e.Text = reader.Value;
}
reader.Read();
return e;
}
public User GetUser(string name) => 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)))
return u;
foreach (var entry in changeList)
if (p(u=FromEncoded(entry)))
{
if (!loadedUsers.Contains(entry)) loadedUsers.Add(u);
return u;
}
using (var reader = XmlReader.Create(DatabaseName))
{
if (!Traverse(reader, MasterEntry)) return null;
while (reader.Read() && reader.NodeType != XmlNodeType.EndElement)
{
if (reader.Name.Equals("User"))
{
User n = User.Parse(ReadEntry(reader), this);
if (n != null && p(FromEncoded(n)))
{
if (!loadedUsers.Contains(n)) loadedUsers.Add(n);
return n;
}
}
}
}
return 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);
// 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, fromAccount, toAccount);
toAcc.History.Add(tx);
toAcc.balance += amount;
AddUser(to);
if (from != null)
{
fromAcc.History.Add(tx);
fromAcc.balance -= amount;
AddUser(from);
}
return true;
}
public User[] Users(Predicate<User> p)
{
List<User> l = new List<User>();
User u;
foreach (var entry in changeList)
if (p(u=FromEncoded(entry)))
l.Add(entry);
using (var reader = XmlReader.Create(DatabaseName))
{
if (!Traverse(reader, MasterEntry)) return null;
while (SkipSpaces(reader) && reader.NodeType != XmlNodeType.EndElement)
{
if (reader.NodeType == XmlNodeType.EndElement) break;
User e = User.Parse(ReadEntry(reader), this);
if (e!=null && p(e=FromEncoded(e))) l.Add(e);
}
}
return l.ToArray();
}
public bool ContainsUser(string user) => 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)
{
for(int i = 0; i<downTo.Length; ++i)
{
while (reader.Read() && !downTo[i].Equals(reader.Name)) ;
if (!downTo[i].Equals(reader.Name)) return false;
reader.MoveToContent();
}
return true;
}
private bool SkipSpaces(XmlReader reader)
{
bool b;
while ((b = reader.Read()) && reader.NodeType == XmlNodeType.Whitespace) ;
return b;
}
private static User ToEncoded(User entry)
{
User u = new User(entry);
u.Name = Encode(u.Name);
foreach(var account in u.accounts)
{
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;
}
private static User FromEncoded(User entry)
{
User u = new User(entry);
u.Name = Decode(u.Name);
foreach (var account in u.accounts)
{
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;
}
private static string Encode(string s) => Convert.ToBase64String(Encoding.UTF8.GetBytes(s));
private static string Decode(string s) => Convert.FromBase64String(s).ToUTF8String();
internal class Entry
{
public string Text { get; set; }
public string Name { get; set; }
public Dictionary<string, string> Attributes { get; } // Transient properties in comparison
public List<Entry> NestedEntries { get; } // Semi-transient comparison properties
public Entry(string name, string text = "")
{
Name = name;
Text = text;
Attributes = new Dictionary<string, string>();
NestedEntries = new List<Entry>();
}
public Entry GetNestedEntry(Predicate<Entry> p)
{
foreach (var entry in NestedEntries)
if (p(entry))
return entry;
return null;
}
public bool BoolAttribute(string attrName, bool def = false, bool ignoreCase = true)
=> Attributes.ContainsKey(attrName) && bool.TryParse(ignoreCase ? Attributes[attrName].ToLower() :Attributes[attrName], out bool b) ? b : def;
public Entry AddNested(Entry e)
{
NestedEntries.Add(e);
return this;
}
public Entry AddNested(string name, string text) => AddNested(new Entry(name, text));
public Entry AddAttribute(string key, string value)
{
Attributes[key] = value;
return this;
}
public override bool Equals(object obj)
{
if (!(obj is Entry)) return false;
Entry cmp = (Entry)obj;
if (cmp.Attributes.Count != Attributes.Count || cmp.NestedEntries.Count != NestedEntries.Count || !Text.Equals(cmp.Text) || !Name.Equals(cmp.Name)) return false;
foreach(var entry in NestedEntries)
{
if (entry.BoolAttribute("omit")) goto Next;
foreach (var cmpEntry in cmp.NestedEntries)
if (cmpEntry.BoolAttribute("omit")) continue;
else if (cmpEntry.Equals(entry)) goto Next;
return false;
Next: { }
}
return true;
}
public override int GetHashCode()
{
var hashCode = 495068346;
hashCode = hashCode * -1521134295 + EqualityComparer<string>.Default.GetHashCode(Text);
hashCode = hashCode * -1521134295 + EqualityComparer<string>.Default.GetHashCode(Name);
hashCode = hashCode * -1521134295 + EqualityComparer<Dictionary<string, string>>.Default.GetHashCode(Attributes);
hashCode = hashCode * -1521134295 + EqualityComparer<List<Entry>>.Default.GetHashCode(NestedEntries);
return hashCode;
}
}
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 bool IsAdministrator { get; set; }
public string PasswordHash { get; internal set; }
public string Salt { get; internal set; }
public List<Account> accounts = new List<Account>();
private User()
{ }
public User(User copy)
{
this.ProblematicTransactions = copy.ProblematicTransactions;
this.Name = copy.Name;
this.IsAdministrator = copy.IsAdministrator;
this.PasswordHash = copy.PasswordHash;
this.Salt = copy.Salt;
accounts.AddRange(copy.accounts);
}
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, bool generatePass = false, bool admin = false)
{
Name = name;
IsAdministrator = admin;
Salt = Convert.ToBase64String(salt);
PasswordHash = generatePass ? Convert.ToBase64String(KDF.PBKDF2(KDF.HMAC_SHA1, Encoding.UTF8.GetBytes(passHash), Encoding.UTF8.GetBytes(Salt), 8192, 320)) : passHash;
}
public bool Authenticate(string password)
=> Convert.ToBase64String(KDF.PBKDF2(KDF.HMAC_SHA1, Encoding.UTF8.GetBytes(password), Encoding.UTF8.GetBytes(Salt), 8192, 320)).Equals(PasswordHash);
public 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));
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(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;
}
internal static User Parse(Entry e, Database db)
{
if (!e.Name.Equals("User")) return null;
User user = new User();
foreach (var entry in e.NestedEntries)
{
if (entry.Name.Equals("Name")) user.Name = entry.Text;
else if (entry.Name.Equals("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;
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("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 ( // 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;
// Populate transaction names
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, 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);
public override int GetHashCode()
{
return 539060726 + EqualityComparer<string>.Default.GetHashCode(Name);
}
}
public class Transaction
{
public string fromAccount;
public string toAccount;
public string from;
public string to;
public decimal amount;
public 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;
this.meta = meta;
}
public User GetTxToUser(Database db) => db.FirstUser(u => u.Name.Equals(to));
public User GetTxFromUser(Database db) => db.FirstUser(u => u.Name.Equals(from));
}
}
}