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
129 lines
3.6 KiB
C#
129 lines
3.6 KiB
C#
using System;
|
|
using System.Collections.Generic;
|
|
using Tofvesson.Crypto;
|
|
|
|
namespace Server
|
|
{
|
|
public sealed class SessionManager
|
|
{
|
|
private static readonly RandomProvider random = new RegularRandomProvider();
|
|
private readonly List<Session> sessions = new List<Session>();
|
|
private readonly long timeout;
|
|
private readonly int sidLength;
|
|
|
|
public SessionManager(long timeout, int sidLength = 10)
|
|
{
|
|
this.timeout = timeout;
|
|
this.sidLength = sidLength < 10 ? 10 : sidLength;
|
|
}
|
|
|
|
public string GetSession(Database.User user, string invalidSID)
|
|
{
|
|
Update();
|
|
for (int i = 0; i < sessions.Count; ++i)
|
|
if (sessions[i].user.Equals(user))
|
|
return sessions[i].sessionID;
|
|
|
|
Session s = new Session
|
|
{
|
|
sessionID = GenerateRandomSID(invalidSID),
|
|
user = user,
|
|
expiry = DateTime.Now.Ticks + timeout
|
|
};
|
|
sessions.Add(s);
|
|
return s.sessionID;
|
|
}
|
|
|
|
public 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();
|
|
for (int i = sessions.Count - 1; i >= 0; --i)
|
|
if (sessions[i].user.Equals(user))
|
|
{
|
|
Session s = sessions[i];
|
|
s.expiry = DateTime.Now.Ticks + timeout;
|
|
sessions[i] = s;
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
public void Expire(Database.User user)
|
|
{
|
|
Update();
|
|
for (int i = sessions.Count - 1; i >= 0; --i)
|
|
if (sessions[i].user.Equals(user))
|
|
{
|
|
sessions.RemoveAt(i);
|
|
return;
|
|
}
|
|
return;
|
|
}
|
|
|
|
public bool Refresh(string sid)
|
|
{
|
|
Update();
|
|
for (int i = sessions.Count - 1; i >= 0; --i)
|
|
if (sessions[i].sessionID.Equals(sid))
|
|
{
|
|
Session s = sessions[i];
|
|
s.expiry = DateTime.Now.Ticks + timeout;
|
|
sessions[i] = s;
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
public bool Expire(string sid)
|
|
{
|
|
Update();
|
|
for (int i = sessions.Count - 1; i >= 0; --i)
|
|
if (sessions[i].sessionID.Equals(sid))
|
|
{
|
|
sessions.RemoveAt(i);
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
public bool CheckSession(string sid, Database.User user)
|
|
{
|
|
foreach (var session in sessions)
|
|
if (session.sessionID.Equals(sid) && session.user.Equals(user))
|
|
return true;
|
|
|
|
return false;
|
|
}
|
|
|
|
public void Update()
|
|
{
|
|
for(int i = sessions.Count - 1; i>=0; --i)
|
|
if (sessions[i].expiry < DateTime.Now.Ticks)
|
|
sessions.RemoveAt(i);
|
|
}
|
|
|
|
private string GenerateRandomSID(string invalid)
|
|
{
|
|
string res;
|
|
do res = random.NextString(sidLength);
|
|
while (res.StartsWith(invalid));
|
|
return res;
|
|
}
|
|
}
|
|
|
|
public struct Session
|
|
{
|
|
public string sessionID;
|
|
public Database.User user;
|
|
public long expiry; // Measured in ticks
|
|
}
|
|
}
|