PSO2SERVER/Server/Program.cs

299 lines
12 KiB
C#
Raw Normal View History

2024-09-10 00:31:40 +08:00
using System;
using System.Collections.Generic;
using System.IO;
using System.Net;
using System.Threading;
using System.Security.Cryptography;
2024-09-10 01:13:20 +08:00
using PSO2SERVER.Database;
using PSO2SERVER.Protocol.Handlers;
2024-09-11 16:33:10 +08:00
using System.Threading.Tasks;
2024-09-11 17:51:59 +08:00
using Google.Protobuf.WellKnownTypes;
using static Org.BouncyCastle.Math.EC.ECCurve;
using System.Reflection;
using System.Linq;
using Newtonsoft.Json.Linq;
2024-12-08 10:56:05 +08:00
using Google.Protobuf.Compiler;
using Newtonsoft.Json;
using PSO2SERVER.Zone;
2024-12-08 11:33:06 +08:00
using PSO2SERVER.Json;
2024-12-09 04:09:09 +08:00
using PSO2SERVER.Models;
2024-09-10 00:31:40 +08:00
2024-09-10 01:13:20 +08:00
namespace PSO2SERVER
2024-09-10 00:31:40 +08:00
{
2024-09-10 01:13:20 +08:00
internal class ServerApp
2024-09-10 00:31:40 +08:00
{
2024-09-10 01:13:20 +08:00
public static ServerApp Instance { get; private set; }
2024-09-10 00:31:40 +08:00
// Will be using these around the app later [KeyPhact]
2024-09-10 01:13:20 +08:00
public const string ServerName = "Phantasy Star Online 2 Server";
public const string ServerShortName = "PSO2";
public const string ServerAuthor = "Sancaros (https://github.com/Sancaros/PSO2SERVER)";
public const string ServerCopyright = "(C) 2024 Sancaros.";
public const string ServerLicense = "All licenced under AGPL.";
2024-09-11 17:51:59 +08:00
public const string ServerVersion = "v0.1.2";
2024-09-10 01:13:20 +08:00
public const string ServerVersionName = "Sancaros";
2024-09-10 00:31:40 +08:00
2024-09-12 10:22:58 +08:00
public const int ServerShipProtNums = 10;
public const int ServerShipProt = 12000;
2024-12-02 11:27:14 +08:00
public const int ServerShipListProtNums_JP = 10;
public const int ServerShipListProt_JP = 12099;
public const int ServerShipListProtNums_NA = 6;
public const int ServerShipListProt_NA = 13001;
2024-09-12 10:22:58 +08:00
2024-09-11 00:44:21 +08:00
public const string ServerSettingsKey = "Resources\\settings.txt";
2024-12-07 23:01:27 +08:00
public const string ServerMemoryPacket = "Resources\\setMemoryPacket.bin"; //TODO
2024-09-11 00:44:21 +08:00
// 密钥BLOB格式
2024-12-02 11:27:14 +08:00
public const string ServerPrivateKeyBlob_JP = "key\\privateKey_jp.blob";
public const string ServerPublicKeyBlob_JP = "key\\publicKey_jp.blob";
public const string ServerPrivateKeyBlob_VITA = "key\\privateKey_vita.blob";
public const string ServerPublicKeyBlob_VITA = "key\\publicKey_vita.blob";
2024-09-12 02:14:42 +08:00
public const string ServerSEGAKeyBlob = "key\\SEGAKey.blob";
2024-09-11 00:44:21 +08:00
2024-09-12 10:22:58 +08:00
// 密钥PEM格式 来自Schthack
2024-09-11 00:44:21 +08:00
public const string ServerPrivatePem = "key\\privateKey.pem";
public const string ServerSEGAPem = "key\\SEGAKey.pem";
2024-09-10 00:31:40 +08:00
public static IPAddress BindAddress = IPAddress.Parse("127.0.0.1");
public static Config Config;
public static ConsoleSystem ConsoleSystem;
public List<QueryServer> QueryServers = new List<QueryServer>();
public Server Server;
2024-09-13 02:23:34 +08:00
private static Assembly ResolveAssembly(object sender, ResolveEventArgs args)
{
string libDir = Path.Combine(AppDomain.CurrentDomain.BaseDirectory, "lib");
string assemblyPath = Path.Combine(libDir, new AssemblyName(args.Name).Name + ".dll");
if (File.Exists(assemblyPath))
{
return Assembly.LoadFrom(assemblyPath);
}
return null;
}
2024-09-10 00:31:40 +08:00
public static void Main(string[] args)
{
2024-09-13 02:23:34 +08:00
// 设置 AssemblyResolve 事件处理程序
AppDomain.CurrentDomain.AssemblyResolve += new ResolveEventHandler(ResolveAssembly);
2024-09-10 00:31:40 +08:00
Config = new Config();
ConsoleSystem = new ConsoleSystem { Thread = new Thread(ConsoleSystem.StartThread) };
ConsoleSystem.Thread.Start();
// Setup function exit handlers to guarentee Exit() is run before closing
Console.CancelKeyPress += Exit;
AppDomain.CurrentDomain.ProcessExit += Exit;
2024-12-09 12:14:17 +08:00
JsonTest.JsonReadTest();
2024-12-09 04:09:09 +08:00
2024-09-10 00:31:40 +08:00
try
{
for (var i = 0; i < args.Length; i++)
{
switch (args[i].ToLower())
{
case "-b":
case "--bind-address":
if (++i < args.Length)
2024-09-11 17:51:59 +08:00
{
var value = args[i];
try
{
if (IPAddress.TryParse(value, out IPAddress ipAddress))
{
// IP address is valid
BindAddress = ipAddress;
}
else
{
// Not an IP address, try resolving as a domain name
var addresses = Dns.GetHostAddresses(value);
if (addresses.Length > 0)
{
// Prefer IPv4 addresses over IPv6
ipAddress = addresses.FirstOrDefault(addr => addr.AddressFamily == System.Net.Sockets.AddressFamily.InterNetwork) ?? addresses[0];
BindAddress = ipAddress; // Use the first resolved IP address
}
else
{
Logger.WriteError($"No IP addresses found for domain: {value}");
}
}
}
catch (Exception ex)
{
Logger.WriteError($"Error resolving domain {value}: {ex.Message}");
}
}
2024-09-10 00:31:40 +08:00
break;
case "-s":
case "--size":
var splitArgs = args[++i].Split(',');
var width = int.Parse(splitArgs[0]);
var height = int.Parse(splitArgs[1]);
if (width < ConsoleSystem.Width)
{
Logger.WriteWarning("[ARG] Capping console width to {0} columns", ConsoleSystem.Width);
width = ConsoleSystem.Width;
}
if (height < ConsoleSystem.Height)
{
Logger.WriteWarning("[ARG] Capping console height to {0} rows", ConsoleSystem.Height);
height = ConsoleSystem.Height;
}
ConsoleSystem.SetSize(width, height);
break;
}
}
}
catch (Exception ex)
{
Logger.WriteException("An error has occurred while parsing command line parameters", ex);
}
// Check for settings.txt [AIDA]
2024-09-11 00:44:21 +08:00
if (!File.Exists(ServerSettingsKey))
2024-09-10 00:31:40 +08:00
{
// If it doesn't exist, throw an error and quit [AIDA]
2024-09-11 00:44:21 +08:00
Logger.WriteError("[ERR] 载入 {0} 文件错误. 按任意键退出.", ServerSettingsKey);
2024-09-10 00:31:40 +08:00
Console.ReadKey();
Environment.Exit(0);
}
2024-09-12 12:32:38 +08:00
Instance = new ServerApp();
2024-09-12 02:14:42 +08:00
2024-09-12 12:32:38 +08:00
Instance.GenerateKeys();
2024-09-10 00:31:40 +08:00
// Fix up startup message [KeyPhact]
Logger.WriteHeader();
2024-09-10 01:13:20 +08:00
Logger.Write(ServerName + " - " + ServerVersion + " (" + ServerVersionName + ")");
Logger.Write("作者 " + ServerAuthor);
//Logger.Write(ServerLicense);
2024-09-10 00:31:40 +08:00
Thread.Sleep(1000);
2024-09-10 01:13:20 +08:00
//System.Data.Entity.Database.SetInitializer(new System.Data.Entity.DropCreateDatabaseIfModelChanges<ServerEf>());
2024-09-11 16:33:10 +08:00
_ = Instance.StartAsync();
2024-09-10 00:31:40 +08:00
}
2024-09-12 12:32:38 +08:00
public void GenerateKeys()
{
// Process private key files
2024-12-02 11:27:14 +08:00
KeyLoader.ProcessKeyFiles(ServerPrivatePem, ServerPrivateKeyBlob_JP, true, File.Exists(ServerPrivatePem));
2024-09-12 12:32:38 +08:00
// Process SEGA public key files
KeyLoader.ProcessKeyFiles(ServerSEGAPem, ServerSEGAKeyBlob, false, File.Exists(ServerSEGAPem));
// Process general RSA keys
using (RSACryptoServiceProvider rsa = new RSACryptoServiceProvider())
{
// Process private and public RSA keys
2024-12-02 11:27:14 +08:00
KeyLoader.GenerateAndSaveKeyIfNotExists(rsa, ServerPrivateKeyBlob_JP, true);
KeyLoader.GenerateAndSaveKeyIfNotExists(rsa, ServerPublicKeyBlob_JP, false);
KeyLoader.GenerateAndSaveKeyIfNotExists(rsa, ServerPrivateKeyBlob_VITA, true);
KeyLoader.GenerateAndSaveKeyIfNotExists(rsa, ServerPublicKeyBlob_VITA, false);
2024-09-12 12:32:38 +08:00
}
}
2024-09-11 16:33:10 +08:00
public async Task StartAsync()
2024-09-10 00:31:40 +08:00
{
2024-09-11 19:17:42 +08:00
var startTime = DateTime.Now; // 记录启动开始时间
2024-09-10 00:31:40 +08:00
Server = new Server();
2024-09-11 16:33:10 +08:00
await InitializeConfigurationAsync();
2024-09-12 15:06:07 +08:00
2024-09-11 16:33:10 +08:00
await InitializeDatabaseAsync();
2024-09-12 10:22:58 +08:00
2024-12-02 11:27:14 +08:00
await InitializeQueryServers(QueryMode.AuthList, "认证", 11000, 3, 1000);
await InitializeQueryServers(QueryMode.AuthList, "认证", 13099, 1, 0);
await InitializeQueryServers(QueryMode.AuthList, "认证", ServerShipProt, ServerShipProtNums, 100);
await InitializeQueryServers(QueryMode.ShipList, "舰船_JP", 12100, 10, 100);
await InitializeQueryServers(QueryMode.ShipList, "舰船_JP", 12193, 10, 100);
await InitializeQueryServers(QueryMode.ShipList, "舰船_JP", 12194, 10, 100);
await InitializeQueryServers(QueryMode.ShipList, "舰船_JP", ServerShipListProt_JP, ServerShipListProtNums_JP, 100);
2024-09-12 10:22:58 +08:00
2024-12-02 11:27:14 +08:00
await InitializeQueryServers(QueryMode.ShipList, "舰船_NA", ServerShipListProt_NA, ServerShipListProtNums_NA, 1);
2024-09-10 00:31:40 +08:00
2024-09-11 16:33:10 +08:00
Logger.WriteInternal("服务器启动完成 " + DateTime.Now);
2024-09-11 19:17:42 +08:00
var endTime = DateTime.Now; // 记录启动结束时间
var duration = endTime - startTime; // 计算启动耗时
Logger.WriteInternal($"服务器启动耗时: {duration.TotalSeconds} 秒"); // 记录启动耗时
2024-09-11 16:33:10 +08:00
Server.Run();
}
2024-09-10 00:31:40 +08:00
2024-09-11 16:33:10 +08:00
private async Task InitializeConfigurationAsync()
{
await Task.Run(() =>
2024-09-10 00:31:40 +08:00
{
2024-09-11 16:33:10 +08:00
Config.Load();
PacketHandlers.LoadPacketHandlers();
});
}
2024-09-10 00:31:40 +08:00
2024-09-11 16:33:10 +08:00
private async Task InitializeDatabaseAsync()
{
2024-09-15 17:17:05 +08:00
try
2024-09-11 16:33:10 +08:00
{
2024-09-15 17:17:05 +08:00
Logger.WriteInternal("[DBC] 载入数据库...");
2024-09-11 16:33:10 +08:00
using (var db = new ServerEf())
{
2024-09-15 17:17:05 +08:00
await Task.Run(() =>
{
db.TestDatabaseConnection();
2024-12-03 18:17:43 +08:00
//db.SetupDB();
2024-09-15 17:17:05 +08:00
});
Logger.WriteInternal("[DBC] 数据库初始化完成。");
2024-09-11 16:33:10 +08:00
}
2024-09-15 17:17:05 +08:00
}
catch (Exception ex)
{
Logger.WriteException("[DBC] 数据库初始化异常", ex);
throw; // 重新抛出异常,或者根据需要处理它
}
2024-09-11 16:33:10 +08:00
}
2024-09-10 00:31:40 +08:00
2024-12-02 11:27:14 +08:00
public async Task InitializeQueryServers(QueryMode queryMode, string portname, int port, int portnums, int add)
2024-09-11 16:33:10 +08:00
{
2024-09-12 12:15:17 +08:00
await Task.Run(() =>
2024-09-11 16:33:10 +08:00
{
2024-09-12 12:15:17 +08:00
if (portnums <= 0)
portnums = 1;
if (portnums > 0)
2024-09-12 10:22:58 +08:00
{
2024-09-12 12:15:17 +08:00
for (var i = 0; i < portnums; i++)
{
2024-12-02 11:27:14 +08:00
QueryServers.Add(new QueryServer(queryMode, portname, port + (add * i)));
2024-09-12 12:15:17 +08:00
}
2024-09-12 10:22:58 +08:00
}
2024-09-12 12:15:17 +08:00
});
2024-09-10 00:31:40 +08:00
}
private static void Exit(object sender, EventArgs e)
{
// Save the configuration
Config.Save();
}
2024-12-09 12:14:17 +08:00
2024-09-10 00:31:40 +08:00
}
}