Merge pull request #6 from cs8425/work-for-linux

fix some bug, upgrade to dotnet 10 and add docs for linux
This commit is contained in:
Kei-Luna
2026-04-30 05:43:01 +09:00
committed by GitHub
20 changed files with 271 additions and 64 deletions

View File

@@ -9,6 +9,9 @@ on:
permissions: permissions:
contents: write contents: write
env:
BRANCH_NAME: ${{ github.head_ref || github.ref_name }}
jobs: jobs:
build-release: build-release:
runs-on: windows-latest runs-on: windows-latest
@@ -18,7 +21,7 @@ jobs:
- uses: actions/setup-dotnet@v4 - uses: actions/setup-dotnet@v4
with: with:
dotnet-version: 9.0.x dotnet-version: 10.0.x
- name: Read version - name: Read version
id: version id: version
@@ -54,3 +57,55 @@ jobs:
files: | files: |
artifacts/MikuSB-win-x64.zip artifacts/MikuSB-win-x64.zip
artifacts/MikuSB-win-x64.zip.sha256 artifacts/MikuSB-win-x64.zip.sha256
build-release-linux:
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v4
- uses: actions/setup-dotnet@v4
with:
dotnet-version: '10.0.x'
- name: Publish server
run: |
dotnet publish ./MikuSB/MikuSB.csproj -c Release -r linux-x64 --self-contained true -p:PublishSingleFile=true --property:PublishDir=../artifacts/dist
- name: Read version
id: version
shell: bash
run: |
line=$(head -n 1 version.txt | xargs)
VERSION=${line#v=}
SHORT_HASH="$(git rev-parse --short=7 HEAD)"
if [ -z "$VERSION" ]; then
echo "version.txt is empty."
VERSION=${SHORT_HASH}
fi
if [[ "${{ env.BRANCH_NAME }}" == "main" ]]; then
echo "tag=v${VERSION}" >> $GITHUB_OUTPUT
else
SAFE_NAME=$(echo "${{ env.BRANCH_NAME }}" | tr '/' '-')
echo "tag=${SAFE_NAME}-v${VERSION}-${SHORT_HASH}" >> $GITHUB_OUTPUT
fi
- name: Assemble release package
run: |
packageDir="./artifacts/dist/"
zipFile="MikuSB-linux-x64.zip"
cp ./version.txt ${packageDir}
pushd ${packageDir}
zip -r ../${zipFile} .
popd
sha256sum ./artifacts/${zipFile} > ./artifacts/${zipFile}.sha256
- name: Create release
uses: softprops/action-gh-release@v2
with:
tag_name: ${{ steps.version.outputs.tag }}
name: ${{ steps.version.outputs.tag }}
generate_release_notes: true
files: |
./artifacts/MikuSB-linux-x64.zip
./artifacts/MikuSB-linux-x64.zip.sha256

View File

@@ -1,7 +1,7 @@
<Project Sdk="Microsoft.NET.Sdk"> <Project Sdk="Microsoft.NET.Sdk">
<PropertyGroup> <PropertyGroup>
<TargetFramework>net9.0</TargetFramework> <TargetFramework>net10.0</TargetFramework>
<ImplicitUsings>enable</ImplicitUsings> <ImplicitUsings>enable</ImplicitUsings>
<Nullable>enable</Nullable> <Nullable>enable</Nullable>
<CETCompat>false</CETCompat> <CETCompat>false</CETCompat>
@@ -21,7 +21,6 @@
<PackageReference Include="SQLitePCLRaw.provider.e_sqlite3" Version="2.1.10" /> <PackageReference Include="SQLitePCLRaw.provider.e_sqlite3" Version="2.1.10" />
<PackageReference Include="SqlSugarCore" Version="5.1.4.172" /> <PackageReference Include="SqlSugarCore" Version="5.1.4.172" />
<PackageReference Include="Swashbuckle.AspNetCore" Version="7.2.0" /> <PackageReference Include="Swashbuckle.AspNetCore" Version="7.2.0" />
<PackageReference Include="System.IO.Pipelines" Version="9.0.0" />
<PackageReference Include="System.Management" Version="9.0.0" /> <PackageReference Include="System.Management" Version="9.0.0" />
</ItemGroup> </ItemGroup>

View File

@@ -14,7 +14,10 @@ public class DatabaseHelper
public static readonly ConcurrentDictionary<int, List<BaseDatabaseDataHelper>> UidInstanceMap = []; public static readonly ConcurrentDictionary<int, List<BaseDatabaseDataHelper>> UidInstanceMap = [];
public static readonly List<int> ToSaveUidList = []; public static readonly List<int> ToSaveUidList = [];
public static long LastSaveTick = DateTime.UtcNow.Ticks; public static long LastSaveTick = DateTime.UtcNow.Ticks;
public static Thread? SaveThread;
private static int _saving = 0;
private static Task? _saveTask;
private static CancellationTokenSource? _cts;
public static bool LoadAccount; public static bool LoadAccount;
public static bool LoadAllData; public static bool LoadAllData;
@@ -77,15 +80,11 @@ public class DatabaseHelper
while (!res.IsCompleted) while (!res.IsCompleted)
{ {
Thread.Sleep(100);
} }
LastSaveTick = DateTime.UtcNow.Ticks; _cts = new CancellationTokenSource();
_saveTask = RunAutoSave(_cts.Token);
SaveThread = new Thread(() =>
{
while (true) CalcSaveDatabase();
});
SaveThread.Start();
LoadAllData = true; LoadAllData = true;
} }
@@ -248,6 +247,35 @@ public class DatabaseHelper
ToSaveUidList.RemoveAll(x => x == key); ToSaveUidList.RemoveAll(x => x == key);
} }
public static void Stop()
{
_cts?.Cancel();
}
public static async Task WaitAsync()
{
if (_saveTask != null)
await _saveTask;
}
private static async Task RunAutoSave(CancellationToken token)
{
LastSaveTick = DateTime.UtcNow.Ticks;
try
{
while (!token.IsCancellationRequested)
{
CalcSaveDatabase();
await Task.Delay(100, token);
}
}
catch (OperationCanceledException)
{
// exit normally
// Console.WriteLine($"RunAutoSave exit! - OperationCanceledException");
}
}
// Auto save per 5 min // Auto save per 5 min
public static void CalcSaveDatabase() public static void CalcSaveDatabase()
{ {
@@ -257,6 +285,10 @@ public class DatabaseHelper
public static void SaveDatabase() public static void SaveDatabase()
{ {
// ensure only one SaveDatabase() runnig
if (Interlocked.Exchange(ref _saving, 1) == 1)
return;
try try
{ {
var prev = DateTime.Now; var prev = DateTime.Now;
@@ -281,11 +313,18 @@ public class DatabaseHelper
Math.Round(t, 2).ToString(CultureInfo.InvariantCulture))); Math.Round(t, 2).ToString(CultureInfo.InvariantCulture)));
ToSaveUidList.Clear(); ToSaveUidList.Clear();
// Thread.Sleep(5000); // for test if saving process taking too long
} }
catch (Exception e) catch (Exception e)
{ {
logger.Error("An error occurred while saving the database", e); logger.Error("An error occurred while saving the database", e);
} }
finally
{
// release lock
Volatile.Write(ref _saving, 0);
}
LastSaveTick = DateTime.UtcNow.Ticks; LastSaveTick = DateTime.UtcNow.Ticks;
} }

View File

@@ -148,13 +148,29 @@ public class IConsole
#endregion #endregion
public static string ListenConsole() public static async Task ListenConsole(CancellationToken exitToken)
{ {
while (true) while (!exitToken.IsCancellationRequested)
{ {
ConsoleKeyInfo keyInfo; ConsoleKeyInfo keyInfo;
try { keyInfo = Console.ReadKey(true); } try
catch (InvalidOperationException) { continue; } {
if (!Console.KeyAvailable)
{
await Task.Delay(10, exitToken);
continue;
}
keyInfo = Console.ReadKey(true);
}
catch (OperationCanceledException)
{
break;
}
catch (InvalidOperationException)
{
await Task.Delay(50, exitToken);
continue;
}
switch (keyInfo.Key) switch (keyInfo.Key)
{ {

View File

@@ -2,7 +2,7 @@
<PropertyGroup> <PropertyGroup>
<OutputType>Library</OutputType> <OutputType>Library</OutputType>
<TargetFramework>net9.0</TargetFramework> <TargetFramework>net10.0</TargetFramework>
<ImplicitUsings>enable</ImplicitUsings> <ImplicitUsings>enable</ImplicitUsings>
<Nullable>enable</Nullable> <Nullable>enable</Nullable>
<CETCompat>false</CETCompat> <CETCompat>false</CETCompat>
@@ -22,7 +22,6 @@
<ProjectReference Include="..\Common\Common.csproj" /> <ProjectReference Include="..\Common\Common.csproj" />
<ProjectReference Include="..\TcpSharp\TcpSharp.csproj" /> <ProjectReference Include="..\TcpSharp\TcpSharp.csproj" />
<ProjectReference Include="..\Proto\Proto.csproj" /> <ProjectReference Include="..\Proto\Proto.csproj" />
<PackageReference Include="System.IO.Pipelines" Version="9.0.0" />
</ItemGroup> </ItemGroup>
</Project> </Project>

View File

@@ -2,7 +2,7 @@
<PropertyGroup> <PropertyGroup>
<OutputType>Exe</OutputType> <OutputType>Exe</OutputType>
<TargetFramework>net9.0</TargetFramework> <TargetFramework>net10.0</TargetFramework>
<RuntimeIdentifiers>win-x64</RuntimeIdentifiers> <RuntimeIdentifiers>win-x64</RuntimeIdentifiers>
<ImplicitUsings>enable</ImplicitUsings> <ImplicitUsings>enable</ImplicitUsings>
<Nullable>enable</Nullable> <Nullable>enable</Nullable>

View File

@@ -2,7 +2,7 @@
<PropertyGroup> <PropertyGroup>
<OutputType>Exe</OutputType> <OutputType>Exe</OutputType>
<TargetFramework>net9.0</TargetFramework> <TargetFramework>net10.0</TargetFramework>
<ImplicitUsings>enable</ImplicitUsings> <ImplicitUsings>enable</ImplicitUsings>
<Nullable>enable</Nullable> <Nullable>enable</Nullable>
<CETCompat>false</CETCompat> <CETCompat>false</CETCompat>

View File

@@ -160,7 +160,7 @@ public class LoaderManager : MikuSB
} }
} }
public static void InitCommand() public static async Task InitCommand(CancellationToken exitToken)
{ {
// Register the command handlers // Register the command handlers
try try
@@ -178,6 +178,6 @@ public class LoaderManager : MikuSB
IConsole.OnConsoleExcuteCommand += CommandExecutor.ConsoleExcuteCommand; IConsole.OnConsoleExcuteCommand += CommandExecutor.ConsoleExcuteCommand;
CommandExecutor.OnRunCommand += (sender, e) => { CommandManager.HandleCommand(e, sender); }; CommandExecutor.OnRunCommand += (sender, e) => { CommandManager.HandleCommand(e, sender); };
IConsole.ListenConsole(); await IConsole.ListenConsole(exitToken);
} }
} }

View File

@@ -18,6 +18,10 @@ public class MikuSB
public static readonly Listener Listener = new(); public static readonly Listener Listener = new();
public static readonly CommandManager CommandManager = new(); public static readonly CommandManager CommandManager = new();
// for exit signal
private static readonly CancellationTokenSource _cts = new();
private static int _exitCode = 0;
public static async Task Main() public static async Task Main()
{ {
var time = DateTime.Now; var time = DateTime.Now;
@@ -50,11 +54,15 @@ public class MikuSB
ResourceManager.IsLoaded = true; ResourceManager.IsLoaded = true;
HandbookGenerator.GenerateAll(); HandbookGenerator.GenerateAll();
LoaderManager.InitCommand(); var consoleTask = Task.Run(() => LoaderManager.InitCommand(_cts.Token), _cts.Token);
var elapsed = DateTime.Now - time; var elapsed = DateTime.Now - time;
Logger.Info(I18NManager.Translate("Server.ServerInfo.ServerStarted", Logger.Info(I18NManager.Translate("Server.ServerInfo.ServerStarted",
Math.Round(elapsed.TotalSeconds, 2).ToString(CultureInfo.InvariantCulture))); Math.Round(elapsed.TotalSeconds, 2).ToString(CultureInfo.InvariantCulture)));
await consoleTask;
await ProcessExit(Volatile.Read(ref _exitCode));
} }
#region Exit #region Exit
@@ -64,30 +72,40 @@ public class MikuSB
AppDomain.CurrentDomain.ProcessExit += (_, _) => AppDomain.CurrentDomain.ProcessExit += (_, _) =>
{ {
Logger.Info(I18NManager.Translate("Server.ServerInfo.Shutdown")); Logger.Info(I18NManager.Translate("Server.ServerInfo.Shutdown"));
ProcessExit(); RequestShutdown(0);
}; };
AppDomain.CurrentDomain.UnhandledException += (obj, arg) => AppDomain.CurrentDomain.UnhandledException += (obj, arg) =>
{ {
Logger.Error(I18NManager.Translate("Server.ServerInfo.UnhandledException", obj.GetType().Name), Logger.Error(I18NManager.Translate("Server.ServerInfo.UnhandledException", obj.GetType().Name),
(Exception)arg.ExceptionObject); (Exception)arg.ExceptionObject);
Logger.Info(I18NManager.Translate("Server.ServerInfo.Shutdown")); Logger.Info(I18NManager.Translate("Server.ServerInfo.Shutdown"));
ProcessExit(); RequestShutdown(1);
Environment.Exit(1);
}; };
Console.CancelKeyPress += (_, eventArgs) => Console.CancelKeyPress += (_, eventArgs) =>
{ {
Logger.Info(I18NManager.Translate("Server.ServerInfo.CancelKeyPressed")); Logger.Info(I18NManager.Translate("Server.ServerInfo.CancelKeyPressed"));
eventArgs.Cancel = true; eventArgs.Cancel = true;
Environment.Exit(0); RequestShutdown(0);
}; };
} }
private static void ProcessExit() private static void RequestShutdown(int exitCode)
{
Interlocked.Exchange(ref _exitCode, exitCode);
_cts.Cancel();
}
private static async Task ProcessExit(int exitCode)
{ {
SocketListener.Connections.Values.ToList().ForEach(x => x.Stop(true)); SocketListener.Connections.Values.ToList().ForEach(x => x.Stop(true));
DatabaseHelper.SaveThread?.Interrupt();
DatabaseHelper.SaveDatabase(); DatabaseHelper.Stop(); // notify stop
await DatabaseHelper.WaitAsync(); // wait AutoSave thread exit
DatabaseHelper.SaveDatabase(); // final flush
Environment.Exit(exitCode);
} }
# endregion # endregion

View File

@@ -9,7 +9,7 @@ https://go.microsoft.com/fwlink/?LinkID=208121.
<PublishDir>bin\MikuSB-Win64-Debug</PublishDir> <PublishDir>bin\MikuSB-Win64-Debug</PublishDir>
<PublishProtocol>FileSystem</PublishProtocol> <PublishProtocol>FileSystem</PublishProtocol>
<_TargetId>Folder</_TargetId> <_TargetId>Folder</_TargetId>
<TargetFramework>net9.0</TargetFramework> <TargetFramework>net10.0</TargetFramework>
<RuntimeIdentifier>win-x64</RuntimeIdentifier> <RuntimeIdentifier>win-x64</RuntimeIdentifier>
<SelfContained>false</SelfContained> <SelfContained>false</SelfContained>
<PublishSingleFile>false</PublishSingleFile> <PublishSingleFile>false</PublishSingleFile>

View File

@@ -9,7 +9,7 @@ https://go.microsoft.com/fwlink/?LinkID=208121.
<PublishDir>bin\MikuSB-MultiFile\</PublishDir> <PublishDir>bin\MikuSB-MultiFile\</PublishDir>
<PublishProtocol>FileSystem</PublishProtocol> <PublishProtocol>FileSystem</PublishProtocol>
<_TargetId>Folder</_TargetId> <_TargetId>Folder</_TargetId>
<TargetFramework>net9.0</TargetFramework> <TargetFramework>net10.0</TargetFramework>
<RuntimeIdentifier>win-x64</RuntimeIdentifier> <RuntimeIdentifier>win-x64</RuntimeIdentifier>
<SelfContained>false</SelfContained> <SelfContained>false</SelfContained>
<PublishSingleFile>false</PublishSingleFile> <PublishSingleFile>false</PublishSingleFile>

View File

@@ -9,7 +9,7 @@ https://go.microsoft.com/fwlink/?LinkID=208121.
<PublishDir>bin\MikuSB-OneFile\</PublishDir> <PublishDir>bin\MikuSB-OneFile\</PublishDir>
<PublishProtocol>FileSystem</PublishProtocol> <PublishProtocol>FileSystem</PublishProtocol>
<_TargetId>Folder</_TargetId> <_TargetId>Folder</_TargetId>
<TargetFramework>net9.0</TargetFramework> <TargetFramework>net10.0</TargetFramework>
<RuntimeIdentifier>win-x64</RuntimeIdentifier> <RuntimeIdentifier>win-x64</RuntimeIdentifier>
<SelfContained>true</SelfContained> <SelfContained>true</SelfContained>
<PublishSingleFile>true</PublishSingleFile> <PublishSingleFile>true</PublishSingleFile>

View File

@@ -1,7 +1,7 @@
<Project Sdk="Microsoft.NET.Sdk"> <Project Sdk="Microsoft.NET.Sdk">
<PropertyGroup> <PropertyGroup>
<TargetFramework>net9.0</TargetFramework> <TargetFramework>net10.0</TargetFramework>
<ImplicitUsings>enable</ImplicitUsings> <ImplicitUsings>enable</ImplicitUsings>
<Nullable>enable</Nullable> <Nullable>enable</Nullable>
<CETCompat>false</CETCompat> <CETCompat>false</CETCompat>

View File

@@ -1,6 +1,6 @@
<Project Sdk="Microsoft.NET.Sdk"> <Project Sdk="Microsoft.NET.Sdk">
<PropertyGroup> <PropertyGroup>
<TargetFramework>net9.0</TargetFramework> <TargetFramework>net10.0</TargetFramework>
<ImplicitUsings>enable</ImplicitUsings> <ImplicitUsings>enable</ImplicitUsings>
<Nullable>enable</Nullable> <Nullable>enable</Nullable>
<RootNamespace>MikuSB.Proxy</RootNamespace> <RootNamespace>MikuSB.Proxy</RootNamespace>

View File

@@ -30,7 +30,8 @@ public sealed class ProxyCertificateAuthority
RootCerPath); RootCerPath);
} }
public string RootCerPath => Path.Combine(CertificateDirectory, "MikuSB.Proxy.Root.cer"); public string RootCerPath => Path.Join(CertificateDirectory, "MikuSB.Proxy.Root.cer");
public string RootCerPemPath => Path.Join(CertificateDirectory, "MikuSB.Proxy.Root.pem");
private static string CertificateDirectory => Path.Combine(AppContext.BaseDirectory, "proxy-certs"); private static string CertificateDirectory => Path.Combine(AppContext.BaseDirectory, "proxy-certs");
@@ -55,6 +56,12 @@ public sealed class ProxyCertificateAuthority
if (!File.Exists(RootCerPath)) if (!File.Exists(RootCerPath))
File.WriteAllBytes(RootCerPath, existing.Export(X509ContentType.Cert)); File.WriteAllBytes(RootCerPath, existing.Export(X509ContentType.Cert));
if (!File.Exists(RootCerPemPath))
{
string pemString = existing.ExportCertificatePem();
File.WriteAllText(RootCerPemPath, pemString);
}
return existing; return existing;
} }
@@ -78,6 +85,9 @@ public sealed class ProxyCertificateAuthority
File.WriteAllBytes(pfxPath, exportable.Export(X509ContentType.Pfx, Password)); File.WriteAllBytes(pfxPath, exportable.Export(X509ContentType.Pfx, Password));
File.WriteAllBytes(RootCerPath, exportable.Export(X509ContentType.Cert)); File.WriteAllBytes(RootCerPath, exportable.Export(X509ContentType.Cert));
_logger.LogInformation("Created MikuSB proxy root certificate at {CertificatePath}", RootCerPath); _logger.LogInformation("Created MikuSB proxy root certificate at {CertificatePath}", RootCerPath);
File.WriteAllText(RootCerPemPath, exportable.ExportCertificatePem());
_logger.LogInformation("Created MikuSB proxy root certificate (PEM) at {CertificatePath}", RootCerPemPath);
return exportable; return exportable;
} }

View File

@@ -5,6 +5,7 @@ using System.Net.Sockets;
using System.Security.Authentication; using System.Security.Authentication;
using System.Text; using System.Text;
using MikuSB.Configuration; using MikuSB.Configuration;
using MikuSB.Util;
using Microsoft.Extensions.Hosting; using Microsoft.Extensions.Hosting;
using Microsoft.Extensions.Logging; using Microsoft.Extensions.Logging;
using Microsoft.Extensions.Options; using Microsoft.Extensions.Options;
@@ -15,7 +16,7 @@ public sealed class ProxyServer(
IOptions<ProxyOptions> options, IOptions<ProxyOptions> options,
ProxyCertificateAuthority certificateAuthority, ProxyCertificateAuthority certificateAuthority,
HttpClient httpClient, HttpClient httpClient,
ILogger<ProxyServer> logger) : BackgroundService Logger logger) : BackgroundService
{ {
private const string ListenAddress = "127.0.0.1"; private const string ListenAddress = "127.0.0.1";
private const string ServerHost = "127.0.0.1"; private const string ServerHost = "127.0.0.1";
@@ -53,14 +54,14 @@ public sealed class ProxyServer(
{ {
if (!_options.Enabled) if (!_options.Enabled)
{ {
logger.LogInformation("MikuSB proxy is disabled"); logger.Info("MikuSB proxy is disabled");
return; return;
} }
var address = IPAddress.Parse(ListenAddress); var address = IPAddress.Parse(ListenAddress);
_listener = new TcpListener(address, _options.Port); _listener = new TcpListener(address, _options.Port);
_listener.Start(); _listener.Start();
logger.LogInformation("MikuSB proxy listening on {Address}:{Port}", ListenAddress, _options.Port); logger.Info($"MikuSB proxy listening on {ListenAddress}:{_options.Port}");
try try
{ {
@@ -85,6 +86,7 @@ public sealed class ProxyServer(
{ {
using (client) using (client)
{ {
logger.Info($"Proxy New client: {client.Client.RemoteEndPoint}");
try try
{ {
await HandleClientCoreAsync(client, cancellationToken); await HandleClientCoreAsync(client, cancellationToken);
@@ -100,12 +102,13 @@ public sealed class ProxyServer(
} }
catch (AuthenticationException ex) catch (AuthenticationException ex)
{ {
logger.LogWarning(ex, "Proxy TLS authentication failed"); logger.Warn($"Proxy TLS authentication failed: {ex}");
} }
catch (Exception ex) catch (Exception ex)
{ {
logger.LogWarning(ex, "Proxy client failed"); logger.Warn($"Proxy client failed {ex}");
} }
logger.Info($"Proxy client close: {client.Client.RemoteEndPoint}");
} }
} }
@@ -187,7 +190,7 @@ public sealed class ProxyServer(
{ {
var pathAndQuery = request.GetPathAndQuery(); var pathAndQuery = request.GetPathAndQuery();
var uri = new Uri($"http://{ServerHost}:{_options.ServerHttpPort}{pathAndQuery}"); var uri = new Uri($"http://{ServerHost}:{_options.ServerHttpPort}{pathAndQuery}");
logger.LogInformation("[Proxy] Redirect: {Method} {Host}{Path} -> {Uri}", request.Method, request.HostOverride ?? request.Host, pathAndQuery, uri); logger.Info($"Redirect: {request.Method} {request.HostOverride ?? request.Host}{pathAndQuery} -> {uri}");
await SendHttpRequestAsync(clientStream, request, uri, true, cancellationToken); await SendHttpRequestAsync(clientStream, request, uri, true, cancellationToken);
} }
@@ -202,7 +205,7 @@ public sealed class ProxyServer(
if (IsSelfReference(uri)) if (IsSelfReference(uri))
{ {
logger.LogWarning("[Proxy] Self-reference blocked: {Method} {Uri}", request.Method, uri); logger.Info($"Self-reference blocked: {request.Method} {uri}");
await WriteSimpleResponseAsync(clientStream, HttpStatusCode.LoopDetected, "Proxy self-reference detected", cancellationToken); await WriteSimpleResponseAsync(clientStream, HttpStatusCode.LoopDetected, "Proxy self-reference detected", cancellationToken);
return; return;
} }
@@ -351,7 +354,10 @@ public sealed class ProxyServer(
public string GetPathAndQuery() public string GetPathAndQuery()
{ {
if (Uri.TryCreate(Target, UriKind.Absolute, out var uri)) // "/query?version=a.b.c&platform=PC"
// => Uri.TryCreate() return true && uri.Scheme == "file"
// => will return "/query%3Fversion=a.b.c&platform=PC" cause 404
if (Uri.TryCreate(Target, UriKind.Absolute, out var uri) && uri.IsAbsoluteUri && (uri.Scheme == Uri.UriSchemeHttp || uri.Scheme == Uri.UriSchemeHttps))
return uri.PathAndQuery; return uri.PathAndQuery;
if (string.IsNullOrEmpty(Target)) if (string.IsNullOrEmpty(Target))

66
README_linux.md Normal file
View File

@@ -0,0 +1,66 @@
# MikuSB on Linux
## Config
### setup steam launch options as following
`HTTP_PROXY="http://127.0.0.1:8888" HTTPS_PROXY="http://127.0.0.1:8888" ALL_PROXY="http://127.0.0.1:8888" %command%`
### start local server and keep it running
```
./MikuSB
```
### find root CA cert, and create ca bundle
root CA cert, should in the path: `proxy-certs/MikuSB.Proxy.Root.pem`
### setup root CA for proton/wine
not sure, even I remove Proton PFX (Wine prefix) folder, without redo this step, still no cert issue.
`Proton Hotfix` is the proton version which selected in steam `Force the use of a specific Steam Play compatibility tool`
```bash
APPID=<THE-APP-ID-OF-THE-GAME>
STEAM_COMPAT_DATA_PATH=~/.steam/steam/steamapps/compatdata/$APPID/pfx
STEAM_WINE_PATH="$HOME/.steam/steam/steamapps/common/Proton Hotfix/files/bin/wine"
WINEPREFIX=$STEAM_COMPAT_DATA_PATH $STEAM_WINE_PATH certutil -addstore -f Root proxy-certs/MikuSB.Proxy.Root.pem
```
### start the game and enjoy
## development
1. Restore dependencies and build.
```bash
dotnet build
```
2. run it
```bash
dotnet run --project ./MikuSB
```
## release build
```bash
DOTNET_CLI_UI_LANGUAGE=en time dotnet publish ./MikuSB/MikuSB.csproj -c Release -r linux-x64 --self-contained true -p:PublishSingleFile=true --property:PublishDir=../publish
# output will in ./publish/*
cd ./publish
# start server
./MikuSB
```
## TODO:
* [ ] tool/script for CA cert create and install to proton/wine
* [ ] automatic done in main program

View File

@@ -17,17 +17,17 @@ public static class SdkServer
{ {
public static void Start(string[] args) public static void Start(string[] args)
{ {
BuildWebHost(args).RunAsync(); var builder = Host.CreateDefaultBuilder(args)
} .ConfigureWebHostDefaults(webBuilder =>
private static IWebHost BuildWebHost(string[] args)
{ {
var builder = WebHost.CreateDefaultBuilder(args) webBuilder
.UseStartup<Startup>() .UseStartup<Startup>()
.ConfigureLogging((_, logging) => { logging.ClearProviders(); }) .ConfigureLogging((_, logging) => { logging.ClearProviders(); })
.UseUrls(ConfigManager.Config.HttpServer.GetDisplayAddress()); .UseUrls(ConfigManager.Config.HttpServer.GetDisplayAddress());
});
return builder.Build(); var host = builder.Build();
host.RunAsync();
} }
} }
@@ -91,7 +91,7 @@ public class Startup
{ {
options.JsonSerializerOptions.PropertyNamingPolicy = JsonNamingPolicy.SnakeCaseLower; options.JsonSerializerOptions.PropertyNamingPolicy = JsonNamingPolicy.SnakeCaseLower;
}); });
services.AddSingleton<Logger>(_ => new Logger("HttpServer")); services.AddSingleton<Logger>(_ => new Logger("Proxy"));
services.AddMikuSbProxy(ConfigManager.Config.Proxy); services.AddMikuSbProxy(ConfigManager.Config.Proxy);
} }
} }

View File

@@ -1,7 +1,7 @@
<Project Sdk="Microsoft.NET.Sdk"> <Project Sdk="Microsoft.NET.Sdk">
<PropertyGroup> <PropertyGroup>
<TargetFramework>net9.0</TargetFramework> <TargetFramework>net10.0</TargetFramework>
<ImplicitUsings>enable</ImplicitUsings> <ImplicitUsings>enable</ImplicitUsings>
<Nullable>enable</Nullable> <Nullable>enable</Nullable>
<RootNamespace>MikuSB.SdkServer</RootNamespace> <RootNamespace>MikuSB.SdkServer</RootNamespace>

View File

@@ -1,7 +1,7 @@
<Project Sdk="Microsoft.NET.Sdk"> <Project Sdk="Microsoft.NET.Sdk">
<PropertyGroup> <PropertyGroup>
<TargetFramework>net9.0</TargetFramework> <TargetFramework>net10.0</TargetFramework>
<ImplicitUsings>enable</ImplicitUsings> <ImplicitUsings>enable</ImplicitUsings>
<Nullable>enable</Nullable> <Nullable>enable</Nullable>
<CETCompat>false</CETCompat> <CETCompat>false</CETCompat>
@@ -11,7 +11,6 @@
<ItemGroup> <ItemGroup>
<PackageReference Include="Google.Protobuf" Version="3.29.2" /> <PackageReference Include="Google.Protobuf" Version="3.29.2" />
<PackageReference Include="System.IO.Pipelines" Version="9.0.0" />
</ItemGroup> </ItemGroup>
<ItemGroup> <ItemGroup>