mirror of
https://github.com/space-wizards/RobustToolbox.git
synced 2026-02-14 19:29:36 +01:00
Allows it to be queried from browser JS. No harm in not allowing this. Added helper function StatusExt.AddAllowOriginAny to make this easy to add.
180 lines
6.0 KiB
C#
180 lines
6.0 KiB
C#
using System;
|
|
using System.Linq;
|
|
using System.Threading.Tasks;
|
|
using System.Net;
|
|
using System.Text.Json.Nodes;
|
|
using System.Web;
|
|
using Robust.Shared;
|
|
using Robust.Shared.Utility;
|
|
|
|
namespace Robust.Server.ServerStatus
|
|
{
|
|
|
|
internal sealed partial class StatusHost
|
|
{
|
|
|
|
private void RegisterHandlers()
|
|
{
|
|
AddHandler(HandleTeapot);
|
|
AddHandler(HandleStatus);
|
|
AddHandler(HandleInfo);
|
|
AddAczHandlers();
|
|
}
|
|
|
|
private static async Task<bool> HandleTeapot(IStatusHandlerContext context)
|
|
{
|
|
if (!context.IsGetLike || context.Url!.AbsolutePath != "/teapot")
|
|
{
|
|
return false;
|
|
}
|
|
|
|
await context.RespondAsync("I am a teapot.", (HttpStatusCode) 418);
|
|
return true;
|
|
}
|
|
|
|
private async Task<bool> HandleStatus(IStatusHandlerContext context)
|
|
{
|
|
if (!context.IsGetLike || context.Url!.AbsolutePath != "/status")
|
|
{
|
|
return false;
|
|
}
|
|
|
|
var jObject = new JsonObject
|
|
{
|
|
// We need to send at LEAST name and player count to have the launcher work with us.
|
|
// Tags is optional technically but will be necessary practically for future organization.
|
|
// Content can override these if it wants (e.g. stealthmins).
|
|
["name"] = _serverNameCache,
|
|
["players"] = _playerManager.PlayerCount
|
|
};
|
|
|
|
var tagsCache = _serverTagsCache;
|
|
if (tagsCache != null)
|
|
{
|
|
var tags = new JsonArray();
|
|
foreach (var tag in tagsCache)
|
|
{
|
|
tags.Add(tag);
|
|
}
|
|
jObject["tags"] = tags;
|
|
}
|
|
|
|
OnStatusRequest?.Invoke(jObject);
|
|
|
|
context.AddAllowOriginAny();
|
|
await context.RespondJsonAsync(jObject);
|
|
|
|
return true;
|
|
}
|
|
|
|
private async Task<bool> HandleInfo(IStatusHandlerContext context)
|
|
{
|
|
if (!context.IsGetLike || context.Url!.AbsolutePath != "/info")
|
|
{
|
|
return false;
|
|
}
|
|
|
|
var downloadUrl = _cfg.GetCVar(CVars.BuildDownloadUrl);
|
|
|
|
JsonObject? buildInfo;
|
|
|
|
if (string.IsNullOrEmpty(downloadUrl))
|
|
{
|
|
var query = HttpUtility.ParseQueryString(context.Url.Query);
|
|
buildInfo = await PrepareACZBuildInfo(optional: query.Get("can_skip_build") == "1");
|
|
}
|
|
else
|
|
{
|
|
buildInfo = GetExternalBuildInfo();
|
|
}
|
|
|
|
var authInfo = new JsonObject
|
|
{
|
|
["mode"] = _netManager.Auth.ToString(),
|
|
["public_key"] = _netManager.CryptoPublicKey != null
|
|
? Convert.ToBase64String(_netManager.CryptoPublicKey)
|
|
: null
|
|
};
|
|
|
|
var jObject = new JsonObject
|
|
{
|
|
["connect_address"] = _cfg.GetCVar(CVars.StatusConnectAddress),
|
|
["auth"] = authInfo,
|
|
["build"] = buildInfo,
|
|
["desc"] = _serverDescCache,
|
|
};
|
|
|
|
var privacyPolicyLink = _cfg.GetCVar(CVars.StatusPrivacyPolicyLink);
|
|
var privacyPolicyIdentifier = _cfg.GetCVar(CVars.StatusPrivacyPolicyIdentifier);
|
|
var privacyPolicyVersion = _cfg.GetCVar(CVars.StatusPrivacyPolicyVersion);
|
|
|
|
if (!string.IsNullOrEmpty(privacyPolicyLink)
|
|
&& !string.IsNullOrEmpty(privacyPolicyIdentifier)
|
|
&& !string.IsNullOrEmpty(privacyPolicyVersion))
|
|
{
|
|
jObject["privacy_policy"] = new JsonObject
|
|
{
|
|
["identifier"] = privacyPolicyIdentifier,
|
|
["version"] = privacyPolicyVersion,
|
|
["link"] = privacyPolicyLink,
|
|
};
|
|
}
|
|
|
|
OnInfoRequest?.Invoke(jObject);
|
|
|
|
context.AddAllowOriginAny();
|
|
await context.RespondJsonAsync(jObject);
|
|
|
|
return true;
|
|
}
|
|
|
|
private JsonObject GetExternalBuildInfo()
|
|
{
|
|
var buildInfo = GameBuildInformation.GetBuildInfoFromConfig(_cfg);
|
|
|
|
return new JsonObject
|
|
{
|
|
["engine_version"] = buildInfo.EngineVersion,
|
|
["fork_id"] = buildInfo.ForkId,
|
|
["version"] = buildInfo.Version,
|
|
["download_url"] = buildInfo.ZipDownload,
|
|
["hash"] = buildInfo.ZipHash,
|
|
["acz"] = false,
|
|
["manifest_download_url"] = buildInfo.ManifestDownloadUrl,
|
|
["manifest_url"] = buildInfo.ManifestUrl,
|
|
["manifest_hash"] = buildInfo.ManifestHash
|
|
};
|
|
}
|
|
|
|
private async Task<JsonObject?> PrepareACZBuildInfo(bool optional)
|
|
{
|
|
var acm = await PrepareAcz(optional);
|
|
if (acm == null) return null;
|
|
|
|
// Fork ID is an interesting case, we don't want to cause too many redownloads but we also don't want to pollute disk.
|
|
// Call the fork "custom" if there's no explicit ID given.
|
|
var fork = _cfg.GetCVar(CVars.BuildForkId);
|
|
if (string.IsNullOrEmpty(fork))
|
|
{
|
|
fork = "custom";
|
|
}
|
|
return new JsonObject
|
|
{
|
|
["engine_version"] = _cfg.GetCVar(CVars.BuildEngineVersion),
|
|
["fork_id"] = fork,
|
|
["version"] = acm.ManifestHash,
|
|
// Don't supply a download URL - like supplying an empty self-address
|
|
["download_url"] = "",
|
|
["manifest_download_url"] = "",
|
|
["manifest_url"] = "",
|
|
// Pass acz so the launcher knows where to find the downloads.
|
|
["acz"] = true,
|
|
// Needs to be an empty 'hash' here or the launcher complains, this is from back when ACZ used zips
|
|
["hash"] = "",
|
|
["manifest_hash"] = acm.ManifestHash
|
|
};
|
|
}
|
|
}
|
|
|
|
}
|