sockscape/server/Socks/MasterUdpServer.cs

133 lines
4.7 KiB
C#
Raw Normal View History

2017-06-16 21:00:01 +00:00
using System;
using System.Collections.Generic;
using System.Linq;
using System.Net;
using System.Net.Sockets;
using System.Text;
using System.Threading;
using System.Threading.Tasks;
2017-08-30 21:02:51 +00:00
using Glove;
using SockScape.Encryption;
2017-06-16 21:00:01 +00:00
2017-08-29 20:14:44 +00:00
namespace SockScape {
2017-06-16 21:00:01 +00:00
static class MasterUdpServer {
2017-08-30 21:02:51 +00:00
private static Dictionary<string, Client> Prospects;
private static Dictionary<string, Client> Clients;
private static ServerList Servers;
2017-08-30 21:02:51 +00:00
2017-06-16 21:00:01 +00:00
private static UdpClient Sock;
2017-08-21 21:03:32 +00:00
private static Thread ListeningThread;
private static bool IsOpen;
2017-06-16 21:00:01 +00:00
public static void Initialize() {
if(IsOpen || ListeningThread != null)
2017-06-16 21:00:01 +00:00
return;
Servers = new ServerList();
Prospects = new Dictionary<string, Client>();
2017-08-30 21:02:51 +00:00
Clients = new Dictionary<string, Client>();
ushort port = (ushort)Configuration.General["Master Port"];
2017-08-29 20:14:44 +00:00
Sock = new UdpClient(new IPEndPoint(IPAddress.Any, port));
2017-06-16 21:00:01 +00:00
IsOpen = true;
2017-08-21 21:03:32 +00:00
ListeningThread = new Thread(Listener);
2017-06-16 21:00:01 +00:00
ListeningThread.Start();
}
public static void Listener() {
while(IsOpen) {
2017-08-30 21:02:51 +00:00
IPEndPoint endPoint = new IPEndPoint(IPAddress.Any, 0);
2017-06-16 21:00:01 +00:00
while(Sock.Available > 0) {
2017-08-30 21:02:51 +00:00
var data = Sock.Receive(ref endPoint);
var client = endPoint.ToString();
var encryptor = IsClientConnected(client) ? Clients[client].Encryptor : null;
if(data.Take(Packet.MagicNumber.Length).SequenceEqual(Packet.MagicNumber))
encryptor = null;
2017-08-30 21:02:51 +00:00
Packet packet =
encryptor == null ? Packet.FromBytes(data)
: Packet.FromBytes(encryptor.Parse(data));
if(packet == null)
break;
byte[] sendBuffer;
2017-08-30 21:02:51 +00:00
switch((kIntraMasterId)packet.Id) {
case kIntraMasterId.InitiationAttempt:
if(packet.RegionCount != 1)
break;
if(packet[0] == Configuration.General["Master Secret"]) {
var key = new Key();
Prospects[client] = new Client {
LastReceive = DateTime.Now,
Address = endPoint,
Key = key
};
sendBuffer = key.GenerateRequestPacket().GetBytes();
Sock.Send(sendBuffer, sendBuffer.Length, endPoint);
2017-08-30 21:02:51 +00:00
}
break;
2017-08-30 21:02:51 +00:00
case kIntraMasterId.KeyExchange:
if(!IsProspectConnected(client))
break;
var privateKey = Prospects[client].Key.ParseResponsePacket(packet);
if(privateKey != -1) {
Prospects[client].Encryptor = new Cipher(privateKey);
Clients[client] = Prospects[client];
Prospects.Remove(client);
} else
Prospects.Remove(client);
break;
case kIntraMasterId.StatusUpdate:
if(!IsClientConnected(client) || packet.RegionCount < 1)
break;
2017-08-30 21:02:51 +00:00
break;
}
2017-06-16 21:00:01 +00:00
}
2017-06-16 21:00:01 +00:00
Thread.Sleep(1);
}
}
public static void Close() {
IsOpen = false;
ListeningThread.Join();
ListeningThread = null;
2017-08-29 20:14:44 +00:00
Sock.Dispose();
2017-06-16 21:00:01 +00:00
}
private static bool IsProspectConnected(string client) {
return Prospects.ContainsKey(client);
}
private static bool IsClientConnected(string client) {
return Clients.ContainsKey(client);
}
private static Packet PositiveAck(byte id) {
return new Packet((int)kIntraMasterAckId.PositiveAck, new { id });
}
private static Packet NegativeAck(byte id, string message) {
return new Packet((int)kIntraMasterAckId.NegativeAck, new { id, message });
}
2017-08-30 21:02:51 +00:00
class Client {
public IPEndPoint Address { get; set; }
2017-08-30 21:02:51 +00:00
public DateTime LastReceive { get; set; }
public Cipher Encryptor { get; set; }
public Key Key { get; set; }
}
}
2017-06-16 21:00:01 +00:00
}