continuity/Server/AesSession.cs
2023-08-09 16:23:41 +02:00

216 lines
No EOL
7.1 KiB
C#

using System.Buffers.Binary;
using System.Runtime.InteropServices;
using System.Runtime.InteropServices.Marshalling;
using System.Runtime.Serialization.Formatters.Binary;
using System.Security.Cryptography;
using System.Text;
using System.Text.Json.Serialization;
using MassTransit.Mediator;
using Microsoft.Extensions.Hosting;
using Microsoft.Extensions.Logging;
using NetCoreServer;
using Buffer = NetCoreServer.Buffer;
namespace Server;
public abstract class AesSession : TcpSession
{
private readonly ILogger _logger;
private readonly IMediator _mediator;
private static readonly byte[] Key = new byte[] { 1, 2, 3, 4, 5, 6, 7, 8, 9, 1, 2, 3, 4, 5, 6, 7 }
.Reverse().ToArray();
private static readonly byte[] IV = new byte[]
{ 0xfe, 220, 0xba, 0x98, 0x76, 0x54, 50, 0x10, 15, 30, 0x2d, 60, 0x4b, 90, 0x69, 120 }.Reverse().ToArray();
private readonly ICryptoTransform _encryptor;
private readonly ICryptoTransform _decryptor;
protected AesSession(TcpServer
server, ILogger logger, IMediator mediator) : base(server)
{
_logger = logger;
_mediator = mediator;
var aes = Aes.Create();
aes.Key = Key;
aes.IV = IV;
aes.Padding = PaddingMode.None;
aes.Mode = CipherMode.ECB;
_decryptor = aes.CreateDecryptor(aes.Key, aes.IV);
_encryptor = aes.CreateEncryptor(aes.Key, aes.IV);
}
public override long Send(byte[] buffer)
{
byte[] encryptedBuffer;
using (var ms = new MemoryStream())
using (var cs = new CryptoStream(ms, _encryptor, CryptoStreamMode.Write))
{
cs.Write(buffer);
cs.FlushFinalBlock();
encryptedBuffer = ms.ToArray();
}
return base.Send(encryptedBuffer);
}
public override long Send(byte[] buffer, long offset, long size)
{
byte[] encryptedBuffer;
using (var ms = new MemoryStream())
using (var cs = new CryptoStream(ms, _encryptor, CryptoStreamMode.Write))
{
cs.Write(buffer, (int)offset, (int)size);
cs.FlushFinalBlock();
encryptedBuffer = ms.ToArray();
}
return base.Send(encryptedBuffer, 0, encryptedBuffer.Length);
}
public override long Send(ReadOnlySpan<byte> buffer)
{
Span<byte> encryptedBuffer;
using (var ms = new MemoryStream())
using (var cs = new CryptoStream(ms, _encryptor, CryptoStreamMode.Write))
{
cs.Write(buffer);
cs.FlushFinalBlock();
encryptedBuffer = ms.ToArray();
}
return base.Send(encryptedBuffer);
}
public override bool SendAsync(byte[] buffer)
{
byte[] encryptedBuffer;
using (var ms = new MemoryStream())
using (var cs = new CryptoStream(ms, _encryptor, CryptoStreamMode.Write))
{
cs.Write(buffer);
cs.FlushFinalBlock();
encryptedBuffer = ms.ToArray();
}
return base.SendAsync(encryptedBuffer);
}
public override bool SendAsync(byte[] buffer, long offset, long size)
{
byte[] encryptedBuffer;
using (var ms = new MemoryStream())
using (var cs = new CryptoStream(ms, _encryptor, CryptoStreamMode.Write))
{
cs.Write(buffer, (int)offset, (int)size);
cs.FlushFinalBlock();
encryptedBuffer = ms.ToArray();
}
return base.SendAsync(encryptedBuffer, 0, encryptedBuffer.Length);
}
public override bool SendAsync(ReadOnlySpan<byte> buffer)
{
Span<byte> encryptedBuffer;
using (var ms = new MemoryStream())
using (var cs = new CryptoStream(ms, _encryptor, CryptoStreamMode.Write))
{
cs.Write(buffer);
cs.FlushFinalBlock();
encryptedBuffer = ms.ToArray();
}
return base.SendAsync(encryptedBuffer);
}
public override long Receive(byte[] buffer)
{
Span<byte> decryptedBuffer;
using (var ms = new MemoryStream())
using (var cs = new CryptoStream(ms, _decryptor, CryptoStreamMode.Write))
{
cs.Write(buffer);
cs.FlushFinalBlock();
decryptedBuffer = ms.ToArray();
}
return base.Receive(decryptedBuffer.ToArray());
}
public override long Receive(byte[] buffer, long offset, long size)
{
byte[] decryptedBuffer;
using (var ms = new MemoryStream())
using (var cs = new CryptoStream(ms, _decryptor, CryptoStreamMode.Write))
{
cs.Write(buffer, (int)offset, (int)size);
cs.FlushFinalBlock();
decryptedBuffer = ms.ToArray();
}
return base.Receive(decryptedBuffer, 0, decryptedBuffer.Length);
}
protected override void OnReceived(byte[] buffer, long offset, long size)
{
try
{
Console.WriteLine($"Length: {size} & offset: {offset}");
//Console.WriteLine(BitConverter.ToString(buffer.ToArray()).Replace("-", string.Empty));
Span<byte> decryptedBuffer = new byte[size];
// xor every value after the first 8 bytes
// then apply AES decryption
var dataBuffer = Decrypt(new ArraySegment<byte>(buffer, 8, (int)size - 8).ToArray());
/*using (var ms = new MemoryStream(dataBuffer))
using (var cs = new CryptoStream(ms, _decryptor, CryptoStreamMode.Read))
{
cs.Read(decryptedBuffer);
}*/
Console.WriteLine("Length " + BitConverter.ToUInt16(buffer, 0));
var opCode = BitConverter.ToUInt16(buffer.ToArray(), 2);
Console.WriteLine("Packet Op Code: " + opCode);
Console.WriteLine("Some Value: " + buffer[4]);
/*
* 20s = 5
* 15s = 4
* 10s = 3
*
* client alive time * 5s => uptime
*/
var clientAliveTime = BitConverter.ToUInt16(buffer.ToArray(), 5);
Console.WriteLine("Client Alive time: " + clientAliveTime);
Console.WriteLine("Might be a flag:" + buffer[7]);
Console.WriteLine("username: " + Encoding.ASCII.GetString(dataBuffer.ToArray(), 0, 20));
Console.WriteLine("password: " + Encoding.ASCII.GetString(dataBuffer.ToArray(), 20, 32));
Console.WriteLine("Full buffer: " + Encoding.ASCII.GetString(dataBuffer.ToArray()));
Packet packet = new Packet((OperationCode)opCode, dataBuffer, clientAliveTime, buffer[0], buffer[3],
Id);
Parallel.Invoke(() => _mediator.Send(packet));
base.OnReceived(decryptedBuffer.ToArray(), offset, decryptedBuffer.Length);
}
catch (CryptographicException ex)
{
_logger.LogError("An error has occured while decrypting: {ErrorMessage}", ex.Message);
_logger.LogError("Default buffer message: {Message}", Encoding.ASCII.GetString(buffer));
}
}
private byte[] Decrypt(byte[] buffer)
{
for (int i = 0; i < buffer.Length; ++i)
{
buffer[i] = (byte)(buffer[i] ^ i ^ (3 * (0xFE - i)));
}
return buffer;
}
}