Compare commits

...

5 Commits
v2.5 ... v2.6

Author SHA1 Message Date
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
Kei-Luna
548c77850e Merge pull request #34 from frearb/fix_server_time
fix server time
2026-05-13 09:31:33 +09:00
Kei-Luna
d8c356a01f Update PatchDownloadService.cs 2026-05-13 09:06:57 +09:00
frearb
4ee11618be fix server time 2026-05-12 00:06:23 +08:00
7 changed files with 267 additions and 57 deletions

View File

@@ -6,8 +6,7 @@ public static class PatchDownloadService
{ {
private static readonly Logger Logger = new("PatchDownloader"); private static readonly Logger Logger = new("PatchDownloader");
private const string PatchRelativePath = @"Patch\MikuSB-Patch.dll"; private const string PatchRelativePath = @"Patch\MikuSB-Patch.dll";
private const string PatchDownloadUrl = private const string PatchDownloadUrl = "https://github.com/Kei-Luna/MikuSB-Patch/releases/download/MikuSB-Patch/MikuSB-Patch.dll";
"https://github.com/Kei-Luna/MikuSB-Patch/releases/download/MikuSB-Patch/MikuSB-Patch.dll";
private const int DownloadTimeoutSeconds = 60; private const int DownloadTimeoutSeconds = 60;
public static void EnsurePatchPresent() public static void EnsurePatchPresent()

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

@@ -1,16 +1,14 @@
using MikuSB.Util.Extensions;
namespace MikuSB.GameServer.Server.CallGS.Handlers.Misc; namespace MikuSB.GameServer.Server.CallGS.Handlers.Misc;
// Client requests server time to calculate timezone offset. // Client requests server time to calculate timezone offset.
// nTime1/nTime2 are DST transition reference timestamps; returning the same value means no offset. // In the client, ZoneTime.lua hardcodes sTime1/sTime2; if nTime1/nTime2 are false, the client ignores this update.
// Otherwise, offset = nTimeX - ParseTimeNative(sTimeX).
[CallGSApi("ZoneTime_ReqTime")] [CallGSApi("ZoneTime_ReqTime")]
public class ZoneTime_ReqTime : ICallGSHandler public class ZoneTime_ReqTime : ICallGSHandler
{ {
public async Task Handle(Connection connection, string param, ushort seqNo) public async Task Handle(Connection connection, string param, ushort seqNo)
{ {
var now = Extensions.GetUnixSec(); var arg = $"{{\"nTime1\":false,\"nTime2\":false}}";
var arg = $"{{\"nTime1\":{now},\"nTime2\":{now}}}";
await CallGSRouter.SendScript(connection, "ZoneTime_ChangeTime", arg); await CallGSRouter.SendScript(connection, "ZoneTime_ChangeTime", arg);
} }
} }

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

@@ -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

@@ -1 +1 @@
v=2.5 v=2.6