Compare commits

..

4 Commits

Author SHA1 Message Date
Kei-Luna
6c5d546026 Compress large packets 2026-05-13 19:09:41 +09:00
Kei-Luna
9e518edb8e README 2026-05-13 14:48:47 +09:00
Kei-Luna
79fad7df2e account list command 2026-05-13 13:38:35 +09:00
Kei-Luna
68a7d6cc61 We have tightened account management.
Accounts created using methods other than email address login will no longer be accessible.
(This is a necessary step when implementing multiplayer in the future.)
2026-05-13 13:35:56 +09:00
11 changed files with 289 additions and 60 deletions

View File

@@ -1,6 +1,8 @@
using MikuSB.Database;
using MikuSB.Database.Account; using MikuSB.Database.Account;
using MikuSB.Enums.Player; using MikuSB.Enums.Player;
using MikuSB.Internationalization; using MikuSB.Internationalization;
using System.Text;
namespace MikuSB.GameServer.Command.Commands; namespace MikuSB.GameServer.Command.Commands;
@@ -34,4 +36,25 @@ public class CommandAccount : ICommands
await arg.SendMsg(I18NManager.Translate("Game.Command.Account.CreateFailed", ex.Message)); await arg.SendMsg(I18NManager.Translate("Game.Command.Account.CreateFailed", ex.Message));
} }
} }
[CommandMethod("list")]
public async ValueTask List(CommandArg arg)
{
var accounts = DatabaseHelper.GetAllInstance<AccountData>()?
.OrderBy(account => account.Uid)
.ToList();
if (accounts == null || accounts.Count == 0)
{
await arg.SendMsg("No accounts found.");
return;
}
var builder = new StringBuilder();
builder.AppendLine("Accounts:");
foreach (var account in accounts)
builder.AppendLine($"{account.Username} -> UID {account.Uid}");
await arg.SendMsg(builder.ToString().TrimEnd());
}
} }

View File

@@ -226,9 +226,7 @@ public class PlayerInstance(PlayerGameData data)
continue; continue;
} }
//ToDo proto.Attrs[ToPackedAttrKey(gid, sid)] = val;
//Temporary fix for login issues(need to handle LoginRsp properly with zlib.)
//proto.Attrs[ToPackedAttrKey(gid, sid)] = val;
proto.Attrs[ToShiftedAttrKey(gid, sid)] = val; proto.Attrs[ToShiftedAttrKey(gid, sid)] = val;
} }

View File

@@ -11,6 +11,8 @@ using MikuSB.GameServer.Server.Packet.Send.Misc;
using MikuSB.Proto; using MikuSB.Proto;
using MikuSB.TcpSharp; using MikuSB.TcpSharp;
using MikuSB.Util; using MikuSB.Util;
using System.Text;
using System.Text.Json;
using System.Text.Json.Nodes; using System.Text.Json.Nodes;
namespace MikuSB.GameServer.Server.Packet.Recv.Login; namespace MikuSB.GameServer.Server.Packet.Recv.Login;
@@ -18,21 +20,45 @@ namespace MikuSB.GameServer.Server.Packet.Recv.Login;
[Opcode(CmdIds.ReqLogin)] [Opcode(CmdIds.ReqLogin)]
public class HandlerReqLogin : Handler public class HandlerReqLogin : Handler
{ {
private static readonly Logger Logger = new("ReqLogin");
private static string? ExtractSdkAuthToken(string? token)
{
if (string.IsNullOrWhiteSpace(token))
return null;
try
{
var normalized = Uri.UnescapeDataString(token).Trim();
var padding = normalized.Length % 4;
if (padding > 0)
normalized = normalized.PadRight(normalized.Length + (4 - padding), '=');
var json = Encoding.UTF8.GetString(Convert.FromBase64String(normalized));
using var document = JsonDocument.Parse(json);
return document.RootElement.TryGetProperty("authToken", out var authToken)
? authToken.GetString()
: null;
}
catch
{
return null;
}
}
public override async Task OnHandle(Connection connection, byte[] data, ushort seqNo) public override async Task OnHandle(Connection connection, byte[] data, ushort seqNo)
{ {
var req = ReqLogin.Parser.ParseFrom(data); var req = ReqLogin.Parser.ParseFrom(data);
var sdkAuthToken = ExtractSdkAuthToken(req.Token);
var account = AccountData.GetAccountByComboToken(req.Token) var account = AccountData.GetAccountByComboToken(req.Token)
?? AccountData.GetAccountByDispatchToken(req.Token) ?? AccountData.GetAccountByDispatchToken(req.Token)
?? AccountData.GetAccountByUid(10001) ?? AccountData.GetAccountByComboToken(sdkAuthToken ?? "")
?? AccountData.GetAccountByUid(1); ?? AccountData.GetAccountByDispatchToken(sdkAuthToken ?? "");
if (account == null) if (account == null)
{ {
account = AccountData.CreateAccount("default@mikusb.local", 10001, ""); Logger.Warn($"Rejected login: provider={req.Provider}, token={req.Token}, authToken={sdkAuthToken}");
if (account == null) await connection.SendPacket(CmdIds.NtfLogout);
{ return;
await connection.SendPacket(CmdIds.NtfLogout);
return;
}
} }
if (!ResourceManager.IsLoaded) if (!ResourceManager.IsLoaded)
// resource manager not loaded, return // resource manager not loaded, return

View File

@@ -22,7 +22,7 @@ public class PacketRspLogin : BasePacket
}; };
var bytes = Google.Protobuf.MessageExtensions.ToByteArray(proto); var bytes = Google.Protobuf.MessageExtensions.ToByteArray(proto);
Logger.Info($"RspLogin proto size: {bytes.Length} bytes (limit: 65535)"); Logger.Info($"RspLogin proto size: {bytes.Length} bytes");
SetData(bytes); SetData(bytes);
} }

View File

@@ -42,7 +42,8 @@ dotnet build
``` ```
2. Set `GamePath` in `Config.json` to the path of your game executable. 2. Set `GamePath` in `Config.json` to the path of your game executable.
3. Start the server and run the `game` command. 3. Start the server and run the `game` command.
4. Enjoy. 4. Create an account in the server console.
5. Enjoy.
## Feature List ## Feature List

View File

@@ -42,7 +42,8 @@ dotnet build
``` ```
2. Config.json の`GamePath`にあなたのゲームの実行ファイルのパスを書き込みます 2. Config.json の`GamePath`にあなたのゲームの実行ファイルのパスを書き込みます
3. サーバーを起動し`game`コマンドを入力します 3. サーバーを起動し`game`コマンドを入力します
4. 楽しむ 4. サーバーコンソールでアカウントを作成する
5. 楽しむ
## 機能一覧 ## 機能一覧

View File

@@ -1,3 +1,4 @@
using Microsoft.AspNetCore.Http;
using Microsoft.AspNetCore.Mvc; using Microsoft.AspNetCore.Mvc;
using MikuSB.Configuration; using MikuSB.Configuration;
using MikuSB.Database.Account; using MikuSB.Database.Account;
@@ -13,8 +14,6 @@ public class RouteController : ControllerBase
{ {
public static ConfigContainer Config = ConfigManager.Config; public static ConfigContainer Config = ConfigManager.Config;
private const int DefaultAccountUid = 10001;
public static object BuildServerList(string version = "") public static object BuildServerList(string version = "")
{ {
return new return new
@@ -129,29 +128,15 @@ public class RouteController : ControllerBase
return Ok(rsp); return Ok(rsp);
} }
private static AccountData EnsureDefaultAccount() private static AccountData? ResolveAccountByUid(string? uid)
{
var account = AccountData.GetAccountByUid(DefaultAccountUid)
?? AccountData.GetAccountByEmail("default@mikusb.local");
if (account != null)
return account;
return AccountData.CreateAccount("default@mikusb.local", DefaultAccountUid, "");
}
private static AccountData ResolveAccountByUid(string? uid)
{ {
if (int.TryParse(uid, out var parsedUid)) if (int.TryParse(uid, out var parsedUid))
{ return AccountData.GetAccountByUid(parsedUid);
var accountByUid = AccountData.GetAccountByUid(parsedUid);
if (accountByUid != null)
return accountByUid;
}
return EnsureDefaultAccount(); return null;
} }
private static AccountData ResolveAccountForSdkLogin(string? email, string? uid, string? token) private static AccountData? ResolveAccountForSdkLogin(string? email, string? uid, string? token)
{ {
if (!string.IsNullOrWhiteSpace(token)) if (!string.IsNullOrWhiteSpace(token))
{ {
@@ -174,18 +159,78 @@ public class RouteController : ControllerBase
return ResolveAccountByUid(uid); return ResolveAccountByUid(uid);
} }
private async Task<string?> GetJsonBodyValue(string propertyName)
{
if (!Request.HasJsonContentType())
return null;
Request.EnableBuffering();
Request.Body.Position = 0;
using var reader = new StreamReader(Request.Body, Encoding.UTF8, leaveOpen: true);
var body = await reader.ReadToEndAsync();
Request.Body.Position = 0;
if (string.IsNullOrWhiteSpace(body))
return null;
try
{
using var document = JsonDocument.Parse(body);
if (document.RootElement.ValueKind != JsonValueKind.Object)
return null;
return document.RootElement.TryGetProperty(propertyName, out var value)
? value.GetString()
: null;
}
catch
{
return null;
}
}
private IActionResult BuildLoginFailedResponse(string message)
{
object rsp = new
{
code = 1001,
data = (object?)null,
msg = message
};
return Ok(rsp);
}
private IActionResult BuildNotFoundResponse(string message)
{
object rsp = new
{
code = 1001,
data = (object?)null,
msg = message
};
return Ok(rsp);
}
[HttpGet("/seasun/loginByToken")] [HttpGet("/seasun/loginByToken")]
[HttpPost("/seasun/loginByToken")] [HttpPost("/seasun/loginByToken")]
public IActionResult LoginByToken( public async Task<IActionResult> LoginByToken(
[FromQuery] string? uid, [FromQuery] string? uid,
[FromQuery] string? token, [FromQuery] string? token,
[FromForm] string? form_uid, [FromForm] string? form_uid,
[FromForm] string? form_token [FromForm] string? form_token
) )
{ {
var account = ResolveAccountForSdkLogin(null, uid ?? form_uid, token ?? form_token); var finalUid = uid ?? form_uid ?? await GetJsonBodyValue("uid");
var finalUid = account.Uid.ToString(); var finalToken = token ?? form_token ?? await GetJsonBodyValue("token");
var finalToken = account.GenerateComboToken(); var account = ResolveAccountForSdkLogin(null, finalUid, finalToken);
if (account == null)
return BuildLoginFailedResponse("Account not found.");
var responseUid = account.Uid.ToString();
var responseToken = account.GenerateComboToken();
object rsp = new object rsp = new
{ {
@@ -195,13 +240,13 @@ public class RouteController : ControllerBase
associatedAccounts = Array.Empty<string>(), associatedAccounts = Array.Empty<string>(),
isFirstLogin = false, isFirstLogin = false,
isNeedKoreaSciAuth = false, isNeedKoreaSciAuth = false,
ksOpenId = $"ks_{finalUid}", ksOpenId = $"ks_{responseUid}",
nickname = account.Username, nickname = account.Username,
passportId = finalUid, passportId = responseUid,
playerFillAgeUrl = "", playerFillAgeUrl = "",
status = 0, status = 0,
thirdPartyUid = "", thirdPartyUid = "",
token = finalToken, token = responseToken,
type = "guest", type = "guest",
uid = account.Uid uid = account.Uid
}, },
@@ -213,7 +258,7 @@ public class RouteController : ControllerBase
[HttpGet("/seasun/login")] [HttpGet("/seasun/login")]
[HttpPost("/seasun/login")] [HttpPost("/seasun/login")]
public IActionResult Login( public async Task<IActionResult> Login(
[FromQuery] string? uid, [FromQuery] string? uid,
[FromQuery] string? token, [FromQuery] string? token,
[FromQuery] string? email, [FromQuery] string? email,
@@ -222,10 +267,48 @@ public class RouteController : ControllerBase
[FromForm] string? form_email [FromForm] string? form_email
) )
{ {
var finalEmail = email ?? form_email; var finalEmail = email ?? form_email ?? await GetJsonBodyValue("email");
var account = ResolveAccountForSdkLogin(finalEmail, uid ?? form_uid, token ?? form_token); if (!string.IsNullOrWhiteSpace(finalEmail))
var finalUid = account.Uid.ToString(); {
var finalToken = account.GenerateComboToken(); var accountByEmail = AccountData.GetAccountByEmail(finalEmail);
if (accountByEmail == null)
return BuildLoginFailedResponse("Account not found.");
var finalUidValue = accountByEmail.Uid.ToString();
var finalTokenValue = accountByEmail.GenerateComboToken();
object emailLoginRsp = new
{
code = 0,
data = new
{
associatedAccounts = Array.Empty<string>(),
isFirstLogin = false,
isNeedKoreaSciAuth = false,
ksOpenId = $"ks_{finalUidValue}",
nickname = accountByEmail.Username,
passportId = finalUidValue,
playerFillAgeUrl = "",
status = 0,
thirdPartyUid = "",
token = finalTokenValue,
type = "guest",
uid = accountByEmail.Uid
},
msg = "操作成功"
};
return Ok(emailLoginRsp);
}
var finalUid = uid ?? form_uid ?? await GetJsonBodyValue("uid");
var finalToken = token ?? form_token ?? await GetJsonBodyValue("token");
var account = ResolveAccountForSdkLogin(finalEmail, finalUid, finalToken);
if (account == null)
return BuildLoginFailedResponse("Account not found.");
var responseUid = account.Uid.ToString();
var responseToken = account.GenerateComboToken();
object rsp = new object rsp = new
{ {
@@ -235,13 +318,13 @@ public class RouteController : ControllerBase
associatedAccounts = Array.Empty<string>(), associatedAccounts = Array.Empty<string>(),
isFirstLogin = false, isFirstLogin = false,
isNeedKoreaSciAuth = false, isNeedKoreaSciAuth = false,
ksOpenId = $"ks_{finalUid}", ksOpenId = $"ks_{responseUid}",
nickname = account.Username, nickname = account.Username,
passportId = finalUid, passportId = responseUid,
playerFillAgeUrl = "", playerFillAgeUrl = "",
status = 0, status = 0,
thirdPartyUid = "", thirdPartyUid = "",
token = finalToken, token = responseToken,
type = "guest", type = "guest",
uid = account.Uid uid = account.Uid
}, },
@@ -259,6 +342,9 @@ public class RouteController : ControllerBase
) )
{ {
var account = ResolveAccountByUid(uid ?? form_uid); var account = ResolveAccountByUid(uid ?? form_uid);
if (account == null)
return BuildNotFoundResponse("Account not found.");
var uidString = account.Uid.ToString(); var uidString = account.Uid.ToString();
object rsp = new object rsp = new
@@ -338,7 +424,11 @@ public class RouteController : ControllerBase
[HttpGet("/account/query-uid/{appId}")] [HttpGet("/account/query-uid/{appId}")]
public IActionResult QueryUid(string appId, [FromQuery] string authInfo) public IActionResult QueryUid(string appId, [FromQuery] string authInfo)
{ {
var uid = ResolveAccountByUid(ExtractUid(authInfo)).Uid.ToString(); var account = ResolveAccountByUid(ExtractUid(authInfo));
if (account == null)
return BuildNotFoundResponse("Account not found.");
var uid = account.Uid.ToString();
object rsp = new object rsp = new
{ {

View File

@@ -1,34 +1,108 @@
using Microsoft.AspNetCore.Http; using Microsoft.AspNetCore.Http;
using MikuSB.Util; using MikuSB.Util;
using System.Text;
namespace MikuSB.SdkServer.Utils; namespace MikuSB.SdkServer.Utils;
public class RequestLoggingMiddleware(RequestDelegate next) public class RequestLoggingMiddleware(RequestDelegate next)
{ {
private const long MaxLoggedBodyBytes = 1024 * 1024;
private static bool ShouldSkip(string path)
=> path.StartsWith("/report") || path.Contains("/log/") || path == "/alive";
private static bool ShouldLogBody(HttpRequest request)
{
if (request.ContentLength is null or 0)
return false;
if (request.ContentLength > MaxLoggedBodyBytes)
return false;
if (string.IsNullOrWhiteSpace(request.ContentType))
return false;
return request.ContentType.Contains("json", StringComparison.OrdinalIgnoreCase)
|| request.ContentType.Contains("text", StringComparison.OrdinalIgnoreCase)
|| request.ContentType.Contains("xml", StringComparison.OrdinalIgnoreCase)
|| request.ContentType.Contains("x-www-form-urlencoded", StringComparison.OrdinalIgnoreCase);
}
private static string SanitizeForLog(string value)
{
var builder = new StringBuilder(value.Length);
foreach (var ch in value)
{
if (ch == '\r')
{
builder.Append(@"\r");
continue;
}
if (ch == '\n')
{
builder.Append(@"\n");
continue;
}
if (char.IsControl(ch))
{
builder.Append(@"\u");
builder.Append(((int)ch).ToString("x4"));
continue;
}
builder.Append(ch);
}
return builder.ToString();
}
private static async Task<string> ReadBodyAsString(HttpRequest request)
{
request.EnableBuffering();
request.Body.Position = 0;
using var reader = new StreamReader(request.Body, Encoding.UTF8, leaveOpen: true);
var body = await reader.ReadToEndAsync();
request.Body.Position = 0;
return body;
}
public async Task InvokeAsync(HttpContext context, Logger logger) public async Task InvokeAsync(HttpContext context, Logger logger)
{ {
var request = context.Request; var request = context.Request;
var method = request.Method; var method = request.Method;
var path = request.Path + request.QueryString; var path = request.Path.ToString();
var pathWithQuery = path + request.QueryString;
if (ConfigManager.Config.HttpServer.EnableLog && !ShouldSkip(path))
{
var body = ShouldLogBody(request)
? SanitizeForLog(await ReadBodyAsString(request))
: "<omitted>";
logger.Info($"REQ {method} {pathWithQuery} body={body}");
}
await next(context); await next(context);
var statusCode = context.Response.StatusCode; var statusCode = context.Response.StatusCode;
if (path.StartsWith("/report") || path.Contains("/log/") || path == "/alive") if (ShouldSkip(path))
return; return;
if (!ConfigManager.Config.HttpServer.EnableLog) return; if (!ConfigManager.Config.HttpServer.EnableLog) return;
if (statusCode == 200) if (statusCode == 200)
{ {
logger.Info($"{method} {path} => {statusCode}"); logger.Info($"{method} {pathWithQuery} => {statusCode}");
} }
else if (statusCode == 404) else if (statusCode == 404)
{ {
logger.Warn($"{method} {path} => {statusCode}"); logger.Warn($"{method} {pathWithQuery} => {statusCode}");
} }
else else
{ {
logger.Error($"{method} {path} => {statusCode}"); logger.Error($"{method} {pathWithQuery} => {statusCode}");
} }
} }
} }

View File

@@ -12,6 +12,7 @@ public class BasePacket
public long Timestamp { get; set; } public long Timestamp { get; set; }
public IMessage? Message { get; set; } public IMessage? Message { get; set; }
public PacketFraming Framing { get; set; } public PacketFraming Framing { get; set; }
public int UncompressedBodySize { get; set; }
public BasePacket(ushort cmdId) public BasePacket(ushort cmdId)
{ {

View File

@@ -2,6 +2,7 @@
using MikuSB.Enums.Packet; using MikuSB.Enums.Packet;
using MikuSB.Util; using MikuSB.Util;
using System.Buffers.Binary; using System.Buffers.Binary;
using System.IO.Compression;
using System.Net.Sockets; using System.Net.Sockets;
namespace MikuSB.TcpSharp namespace MikuSB.TcpSharp
@@ -63,11 +64,11 @@ namespace MikuSB.TcpSharp
} }
} }
public byte[] Encode(ushort packetId, byte[] payload, PacketFraming framing = PacketFraming.FourByteLittleEndianLength) public byte[] Encode(ushort packetId, byte[] payload, PacketFraming framing = PacketFraming.FourByteLittleEndianLength, int uncompressedSize = 0)
{ {
return framing switch return framing switch
{ {
PacketFraming.TwoByteBigEndianLength => EncodeTwoByteFrame(packetId, payload), PacketFraming.TwoByteBigEndianLength => EncodeTwoByteFrame(packetId, payload, uncompressedSize),
PacketFraming.FourByteLittleEndianLength => EncodeFourByteFrame(packetId, payload), PacketFraming.FourByteLittleEndianLength => EncodeFourByteFrame(packetId, payload),
_ => EncodeFourByteFrame(packetId, payload) _ => EncodeFourByteFrame(packetId, payload)
}; };
@@ -220,8 +221,20 @@ namespace MikuSB.TcpSharp
return payload; return payload;
} }
private byte[] EncodeTwoByteFrame(ushort packetId, byte[] payload) private const int CompressionThreshold = 60000;
private static byte[] ZlibCompress(byte[] data)
{ {
using var ms = new MemoryStream();
using (var zlib = new ZLibStream(ms, CompressionLevel.Optimal, leaveOpen: true))
zlib.Write(data, 0, data.Length);
return ms.ToArray();
}
private byte[] EncodeTwoByteFrame(ushort packetId, byte[] payload, int uncompressedSize = 0)
{
if (payload.Length > CompressionThreshold)
payload = ZlibCompress(payload);
var wrappedPayload = WrapPayload(payload); var wrappedPayload = WrapPayload(payload);
var buffer = new byte[HeaderSize4Byte + wrappedPayload.Length]; var buffer = new byte[HeaderSize4Byte + wrappedPayload.Length];
@@ -246,7 +259,9 @@ namespace MikuSB.TcpSharp
const int wrapperHeaderSize = 35; const int wrapperHeaderSize = 35;
var wrapped = new byte[wrapperHeaderSize + payload.Length]; var wrapped = new byte[wrapperHeaderSize + payload.Length];
BinaryPrimitives.WriteUInt16LittleEndian(wrapped.AsSpan(6, 2), (ushort)payload.Length); BinaryPrimitives.WriteUInt16LittleEndian(wrapped.AsSpan(6, 2), (ushort)payload.Length);
wrapped[11] = 1; if (payload.Length >= 2 && payload[0] == 0x78 &&
(payload[1] == 0x01 || payload[1] == 0x5E || payload[1] == 0x9C || payload[1] == 0xDA))
wrapped[10] = 2;
payload.CopyTo(wrapped.AsSpan(wrapperHeaderSize)); payload.CopyTo(wrapped.AsSpan(wrapperHeaderSize));
return wrapped; return wrapped;

View File

@@ -1 +1 @@
v=2.5 v=2.7