mirror of
https://github.com/wwiinnddyy/LanMountainDesktop.git
synced 2026-06-21 16:14:28 +08:00
* 激进的更新 * 试试 * fix.可爱的我一直在修CI( * fix.启动器一定要能够启动 * feat.尝试弄了AOT的启动器。 * fix.修CI,好像是因为Linux那边有个问题,反正修就对了。 * fix.ci难修,为什么liunx跑不起来呢? * Update build.yml * Update LanMountainDesktop.csproj * changed.调整了启动逻辑,优化了更新页面。 * changed.优化了更新体验 * feat.依旧试增量更新这一块,看看velopack * fix.我们试验性地修复了启动器无法正常启动的问题,原因可能是这个画面没有启动,就GUI没显示。然后还把编译问题修了一下。 * fix.继续修ci,ci怎么天天炸 * changed.velopack,试试rust * fix.修ci,修融合桌面,修启动器 * fix.GitHub Action工作流怎么天天出问题 * feat.引入velopack,不好,是rust(至少内存很安全了。 * chore: migrate release pipeline to signed filemap and wire rainyun s3 * fix: make optional s3 upload step workflow-parse safe * fix: make delta pack generation robust for empty diffs and linux paths * chore: rotate launcher update public key for pdc signing * fix: restore stable launcher update public key * fix: sync launcher public key with update signing secret * fix: normalize PEM line endings in signing key validation * fix: rotate launcher public key to match ci signing secret * fix: compare signing keys by SPKI instead of PEM text * refactor update backend to host-managed PDC pipeline * fix release workflow env key collisions * relax publish-pdc precheck to require S3 only * set GH_TOKEN for PDCC installer step * ci: add local pdc mock fallback for release publish * ci: fix pdc mock process log redirection * ci: fallback pdcc signing key to update private key * ci: ensure pdcc signing passphrase env is always set * ci: create pdcc publish root before invoking client * ci: set pdcc version variable from release version * ci: decouple pdcc installer version from publish config version * ci: package pdcc subchannels with generated filemap and changelog * ci: make local pdc mock diff return empty for fast fallback * ci: fix pdcc variable mapping and pdc signing prechecks * Update App.axaml.cs * ci: wire aws cli credentials for rainyun s3 * ci: pin pdcc client version separately from app version * ci: harden local pdc mock transport handling * ci: publish pdcc subchannels in one pass * ci: add pdcc publish heartbeat and timeout * ci: fix pdcc publish workdir bootstrap * feat.Penguin Logistics Online Network Distribution System * ci: fix plonds s3 probe and signing fallback * ci: validate signing key and quiet missing baselines * ci: relax aws checksum mode for rainyun s3 * ci: avoid multipart uploads to rainyun s3 * ci: handle empty plonds baselines safely * ci.plonds * Rebuild release pipeline around PLONDS and DDSS * Fix Windows installer script path in release workflow
139 lines
4.6 KiB
C#
139 lines
4.6 KiB
C#
using System.Text.Json;
|
|
using Plonds.Api.Configuration;
|
|
using Plonds.Shared;
|
|
using Plonds.Shared.Models;
|
|
|
|
namespace Plonds.Api.Services;
|
|
|
|
public sealed class FileSystemPlondsManifestStore : IPlondsManifestStore
|
|
{
|
|
private static readonly JsonSerializerOptions JsonOptions = new(JsonSerializerDefaults.Web)
|
|
{
|
|
WriteIndented = true
|
|
};
|
|
|
|
private readonly PlondsApiOptions _options;
|
|
private readonly string _storageRootFullPath;
|
|
private readonly string _metaRootFullPath;
|
|
|
|
public FileSystemPlondsManifestStore(PlondsApiOptions options)
|
|
{
|
|
_options = options;
|
|
_storageRootFullPath = ResolveRootPath(options.StorageRoot);
|
|
_metaRootFullPath = Path.Combine(_storageRootFullPath, options.MetaRoot);
|
|
}
|
|
|
|
public Task<PlondsMetadataCatalog> GetCatalogAsync(CancellationToken cancellationToken = default)
|
|
{
|
|
_ = cancellationToken;
|
|
|
|
var channelsRoot = Path.Combine(_metaRootFullPath, "channels");
|
|
var latest = new List<PlondsChannelPointer>();
|
|
if (Directory.Exists(channelsRoot))
|
|
{
|
|
foreach (var latestPath in Directory.EnumerateFiles(channelsRoot, "latest.json", SearchOption.AllDirectories))
|
|
{
|
|
var pointer = ReadLatestPointer(latestPath);
|
|
if (pointer is not null)
|
|
{
|
|
latest.Add(pointer);
|
|
}
|
|
}
|
|
}
|
|
|
|
var catalog = new PlondsMetadataCatalog(
|
|
ProtocolName: PlondsConstants.ProtocolName,
|
|
ProtocolVersion: PlondsConstants.ProtocolVersion,
|
|
StorageRoot: _storageRootFullPath,
|
|
MetaRoot: _metaRootFullPath,
|
|
Latest: latest.OrderBy(x => x.Channel, StringComparer.OrdinalIgnoreCase)
|
|
.ThenBy(x => x.Platform, StringComparer.OrdinalIgnoreCase)
|
|
.ToArray(),
|
|
Metadata: new Dictionary<string, string>
|
|
{
|
|
["apiBasePath"] = PlondsConstants.DefaultApiBasePath
|
|
});
|
|
|
|
return Task.FromResult(catalog);
|
|
}
|
|
|
|
public Task<PlondsChannelPointer?> GetLatestAsync(string channel, string platform, CancellationToken cancellationToken = default)
|
|
{
|
|
_ = cancellationToken;
|
|
return Task.FromResult(ReadLatestPointer(GetLatestPath(channel, platform)));
|
|
}
|
|
|
|
public Task<PlondsDistributionInfo?> GetDistributionAsync(string distributionId, CancellationToken cancellationToken = default)
|
|
{
|
|
_ = cancellationToken;
|
|
|
|
var path = GetDistributionPath(distributionId);
|
|
if (!File.Exists(path))
|
|
{
|
|
return Task.FromResult<PlondsDistributionInfo?>(null);
|
|
}
|
|
|
|
var json = File.ReadAllText(path);
|
|
var distribution = JsonSerializer.Deserialize<PlondsDistributionInfo>(json, JsonOptions);
|
|
return Task.FromResult(distribution);
|
|
}
|
|
|
|
private PlondsChannelPointer? ReadLatestPointer(string path)
|
|
{
|
|
if (!File.Exists(path))
|
|
{
|
|
return null;
|
|
}
|
|
|
|
var json = File.ReadAllText(path);
|
|
var pointer = JsonSerializer.Deserialize<PlondsChannelPointer>(json, JsonOptions);
|
|
return pointer;
|
|
}
|
|
|
|
private string GetLatestPath(string channel, string platform)
|
|
{
|
|
return Path.Combine(_metaRootFullPath, "channels", channel, platform, "latest.json");
|
|
}
|
|
|
|
private string GetDistributionPath(string distributionId)
|
|
{
|
|
return Path.Combine(_metaRootFullPath, "distributions", $"{distributionId}.json");
|
|
}
|
|
|
|
private static string ResolveRootPath(string root)
|
|
{
|
|
if (Path.IsPathRooted(root))
|
|
{
|
|
return Path.GetFullPath(root);
|
|
}
|
|
|
|
var candidates = new List<string>();
|
|
|
|
AddCandidateChain(candidates, Directory.GetCurrentDirectory(), root);
|
|
AddCandidateChain(candidates, AppContext.BaseDirectory, root);
|
|
|
|
foreach (var candidate in candidates.Distinct(StringComparer.OrdinalIgnoreCase))
|
|
{
|
|
if (Directory.Exists(candidate))
|
|
{
|
|
return candidate;
|
|
}
|
|
}
|
|
|
|
return candidates.FirstOrDefault() ?? Path.GetFullPath(Path.Combine(AppContext.BaseDirectory, root));
|
|
}
|
|
|
|
private static void AddCandidateChain(ICollection<string> candidates, string? startDirectory, string relativeRoot)
|
|
{
|
|
var current = string.IsNullOrWhiteSpace(startDirectory)
|
|
? null
|
|
: Path.GetFullPath(startDirectory);
|
|
|
|
while (!string.IsNullOrWhiteSpace(current))
|
|
{
|
|
candidates.Add(Path.GetFullPath(Path.Combine(current, relativeRoot)));
|
|
current = Directory.GetParent(current)?.FullName;
|
|
}
|
|
}
|
|
}
|