2025-05-01 19:01:59 +03:00
|
|
|
|
using System;
|
|
|
|
|
|
using System.Collections.Generic;
|
2025-05-01 21:03:55 +03:00
|
|
|
|
using System.Diagnostics;
|
|
|
|
|
|
using System.IO;
|
2025-05-01 19:01:59 +03:00
|
|
|
|
using System.Net.Http;
|
2025-05-01 21:03:55 +03:00
|
|
|
|
using System.Text;
|
2025-05-01 19:01:59 +03:00
|
|
|
|
using System.Threading;
|
|
|
|
|
|
using System.Threading.Tasks;
|
|
|
|
|
|
using Avalonia.Controls;
|
2025-05-02 10:56:19 +03:00
|
|
|
|
using Nebula.UpdateResolver.Configuration;
|
|
|
|
|
|
using Nebula.UpdateResolver.Rest;
|
2025-05-01 19:01:59 +03:00
|
|
|
|
|
|
|
|
|
|
namespace Nebula.UpdateResolver;
|
|
|
|
|
|
|
|
|
|
|
|
public partial class MainWindow : Window
|
|
|
|
|
|
{
|
2025-05-02 10:56:19 +03:00
|
|
|
|
public static readonly string RootPath = Path.Join(Environment.GetFolderPath(
|
|
|
|
|
|
Environment.SpecialFolder.ApplicationData), "Datum");
|
|
|
|
|
|
|
2025-05-01 19:01:59 +03:00
|
|
|
|
private readonly HttpClient _httpClient = new HttpClient();
|
2025-05-02 10:56:19 +03:00
|
|
|
|
public readonly FileApi FileApi = new FileApi(Path.Join(RootPath,"app"));
|
2025-05-06 11:19:19 +03:00
|
|
|
|
private string LogStr = "";
|
2025-05-02 10:56:19 +03:00
|
|
|
|
public MainWindow()
|
2025-05-01 19:01:59 +03:00
|
|
|
|
{
|
|
|
|
|
|
InitializeComponent();
|
2025-05-01 21:03:55 +03:00
|
|
|
|
Start();
|
2025-05-01 19:01:59 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
2025-05-01 21:03:55 +03:00
|
|
|
|
private async Task Start()
|
2025-05-01 19:01:59 +03:00
|
|
|
|
{
|
2025-05-02 10:56:19 +03:00
|
|
|
|
try
|
2025-05-01 19:01:59 +03:00
|
|
|
|
{
|
2025-05-02 10:56:19 +03:00
|
|
|
|
var info = await EnsureFiles();
|
|
|
|
|
|
Log("Downloading files...");
|
2025-05-01 21:03:55 +03:00
|
|
|
|
|
2025-05-02 10:56:19 +03:00
|
|
|
|
foreach (var file in info.ToDelete)
|
|
|
|
|
|
{
|
|
|
|
|
|
Log("Deleting " + file.Path);
|
|
|
|
|
|
FileApi.Remove(file.Path);
|
|
|
|
|
|
}
|
2025-05-01 21:03:55 +03:00
|
|
|
|
|
2025-05-02 10:56:19 +03:00
|
|
|
|
var loadedManifest = info.FilesExist;
|
2025-05-01 21:03:55 +03:00
|
|
|
|
Save(loadedManifest);
|
2025-05-02 10:56:19 +03:00
|
|
|
|
|
|
|
|
|
|
var count = info.ToDownload.Count;
|
|
|
|
|
|
var resolved = 0;
|
|
|
|
|
|
|
|
|
|
|
|
foreach (var file in info.ToDownload)
|
|
|
|
|
|
{
|
|
|
|
|
|
using var response = await _httpClient.GetAsync(
|
|
|
|
|
|
ConfigurationStandalone.GetConfigValue(UpdateConVars.UpdateCacheUrl)
|
|
|
|
|
|
+ "/" + file.Hash);
|
|
|
|
|
|
|
|
|
|
|
|
response.EnsureSuccessStatusCode();
|
|
|
|
|
|
await using var stream = await response.Content.ReadAsStreamAsync();
|
|
|
|
|
|
FileApi.Save(file.Path, stream);
|
|
|
|
|
|
resolved++;
|
|
|
|
|
|
Log("Saving " + file.Path, (int)(resolved / (float)count * 100f));
|
|
|
|
|
|
|
|
|
|
|
|
loadedManifest.Add(file);
|
|
|
|
|
|
Save(loadedManifest);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Log("Download finished. Running launcher...");
|
|
|
|
|
|
|
|
|
|
|
|
var process = Process.Start(new ProcessStartInfo
|
|
|
|
|
|
{
|
|
|
|
|
|
FileName = "dotnet.exe",
|
|
|
|
|
|
Arguments = Path.Join(FileApi.RootPath, "Nebula.Launcher.dll"),
|
|
|
|
|
|
CreateNoWindow = true,
|
|
|
|
|
|
UseShellExecute = false,
|
|
|
|
|
|
RedirectStandardOutput = true,
|
|
|
|
|
|
RedirectStandardError = true,
|
|
|
|
|
|
StandardOutputEncoding = Encoding.UTF8
|
|
|
|
|
|
});
|
2025-05-01 19:01:59 +03:00
|
|
|
|
}
|
2025-05-06 11:19:19 +03:00
|
|
|
|
catch(HttpRequestException e){
|
|
|
|
|
|
LogError(e);
|
|
|
|
|
|
Log("Проблемы с интернет-соединением...");
|
|
|
|
|
|
var logPath = Path.Join(RootPath,"updateResloverError.txt");
|
|
|
|
|
|
File.WriteAllText(logPath, LogStr);
|
|
|
|
|
|
Process.Start(new ProcessStartInfo(){
|
|
|
|
|
|
FileName = "notepad",
|
|
|
|
|
|
Arguments = logPath
|
|
|
|
|
|
});
|
|
|
|
|
|
}
|
2025-05-02 10:56:19 +03:00
|
|
|
|
catch (Exception e)
|
2025-05-01 21:03:55 +03:00
|
|
|
|
{
|
2025-05-06 11:19:19 +03:00
|
|
|
|
LogError(e);
|
|
|
|
|
|
var logPath = Path.Join(RootPath,"updateResloverError.txt");
|
|
|
|
|
|
File.WriteAllText(logPath, LogStr);
|
|
|
|
|
|
Process.Start(new ProcessStartInfo(){
|
|
|
|
|
|
FileName = "notepad",
|
|
|
|
|
|
Arguments = logPath
|
|
|
|
|
|
});
|
2025-05-02 10:56:19 +03:00
|
|
|
|
}
|
2025-05-01 21:03:55 +03:00
|
|
|
|
|
2025-05-06 11:19:19 +03:00
|
|
|
|
Thread.Sleep(4000);
|
2025-05-01 21:03:55 +03:00
|
|
|
|
|
|
|
|
|
|
Environment.Exit(0);
|
2025-05-01 19:01:59 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
private async Task<ManifestEnsureInfo> EnsureFiles()
|
|
|
|
|
|
{
|
2025-05-01 21:03:55 +03:00
|
|
|
|
Log("Ensuring launcher manifest...");
|
2025-05-02 10:56:19 +03:00
|
|
|
|
var manifest = await RestStandalone.GetAsync<LauncherManifest>(
|
|
|
|
|
|
new Uri(ConfigurationStandalone.GetConfigValue(UpdateConVars.UpdateCacheUrl)! + "/manifest.json"), CancellationToken.None);
|
2025-05-01 21:03:55 +03:00
|
|
|
|
|
2025-05-01 19:01:59 +03:00
|
|
|
|
var toDownload = new HashSet<LauncherManifestEntry>();
|
|
|
|
|
|
var toDelete = new HashSet<LauncherManifestEntry>();
|
2025-05-01 21:03:55 +03:00
|
|
|
|
var filesExist = new HashSet<LauncherManifestEntry>();
|
2025-05-01 19:01:59 +03:00
|
|
|
|
|
2025-05-01 21:03:55 +03:00
|
|
|
|
Log("Manifest loaded!");
|
2025-05-02 10:56:19 +03:00
|
|
|
|
if (ConfigurationStandalone.TryGetConfigValue(UpdateConVars.CurrentLauncherManifest, out var currentManifest))
|
2025-05-01 19:01:59 +03:00
|
|
|
|
{
|
2025-05-01 21:03:55 +03:00
|
|
|
|
Log("Delta manifest loaded!");
|
2025-05-01 19:01:59 +03:00
|
|
|
|
foreach (var file in currentManifest.Entries)
|
|
|
|
|
|
{
|
2025-05-01 21:03:55 +03:00
|
|
|
|
if (!manifest.Entries.Contains(file))
|
2025-05-02 10:56:19 +03:00
|
|
|
|
toDelete.Add(EnsurePath(file));
|
2025-05-01 21:03:55 +03:00
|
|
|
|
else
|
2025-05-02 10:56:19 +03:00
|
|
|
|
filesExist.Add(EnsurePath(file));
|
2025-05-01 19:01:59 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
foreach (var file in manifest.Entries)
|
|
|
|
|
|
{
|
|
|
|
|
|
if(!currentManifest.Entries.Contains(file))
|
2025-05-02 10:56:19 +03:00
|
|
|
|
toDownload.Add(EnsurePath(file));
|
2025-05-01 19:01:59 +03:00
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
else
|
|
|
|
|
|
{
|
|
|
|
|
|
toDownload = manifest.Entries;
|
|
|
|
|
|
}
|
2025-05-01 21:03:55 +03:00
|
|
|
|
|
|
|
|
|
|
Log("Saving launcher manifest...");
|
|
|
|
|
|
|
|
|
|
|
|
return new ManifestEnsureInfo(toDownload, toDelete, filesExist);
|
|
|
|
|
|
}
|
2025-05-06 11:19:19 +03:00
|
|
|
|
private void LogError(Exception e){
|
|
|
|
|
|
Log($"{e.GetType().Name}: "+ e.Message);
|
|
|
|
|
|
Log(e.StackTrace);
|
|
|
|
|
|
if(e.InnerException != null)
|
|
|
|
|
|
LogError(e.InnerException);
|
|
|
|
|
|
}
|
|
|
|
|
|
private void Log(string? message, int percentage = 0)
|
2025-05-01 21:03:55 +03:00
|
|
|
|
{
|
2025-05-06 11:19:19 +03:00
|
|
|
|
if(message is null) return;
|
|
|
|
|
|
|
2025-05-01 21:03:55 +03:00
|
|
|
|
ProgressLabel.Content = message;
|
|
|
|
|
|
if (percentage == 0)
|
|
|
|
|
|
PercentLabel.Content = "";
|
|
|
|
|
|
else
|
|
|
|
|
|
PercentLabel.Content = percentage + "%";
|
|
|
|
|
|
|
2025-05-06 11:19:19 +03:00
|
|
|
|
var messageOut = $"[{DateTime.Now.ToUniversalTime():yyyy-MM-dd HH:mm:ss}]: {message} {PercentLabel.Content}";
|
|
|
|
|
|
Console.WriteLine(messageOut);
|
|
|
|
|
|
LogStr += messageOut + "\n";
|
|
|
|
|
|
|
2025-05-01 21:03:55 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
private void Save(HashSet<LauncherManifestEntry> entries)
|
|
|
|
|
|
{
|
2025-05-02 10:56:19 +03:00
|
|
|
|
ConfigurationStandalone.SetConfigValue(UpdateConVars.CurrentLauncherManifest, new LauncherManifest(entries));
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
private LauncherManifestEntry EnsurePath(LauncherManifestEntry entry)
|
|
|
|
|
|
{
|
|
|
|
|
|
if(!PathValidator.IsSafePath(FileApi.RootPath, entry.Path))
|
|
|
|
|
|
throw new ArgumentException("Path contains invalid characters. Manifest hash: " + entry.Hash);
|
|
|
|
|
|
|
|
|
|
|
|
return entry;
|
2025-05-01 19:01:59 +03:00
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-05-02 10:56:19 +03:00
|
|
|
|
|
|
|
|
|
|
public static class PathValidator
|
|
|
|
|
|
{
|
|
|
|
|
|
public static bool IsSafePath(string baseDirectory, string relativePath)
|
|
|
|
|
|
{
|
|
|
|
|
|
if (Path.IsPathRooted(relativePath))
|
|
|
|
|
|
return false;
|
|
|
|
|
|
|
|
|
|
|
|
var fullBase = Path.GetFullPath(baseDirectory);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
var combinedPath = Path.Combine(fullBase, relativePath);
|
|
|
|
|
|
var fullPath = Path.GetFullPath(combinedPath);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (!fullPath.StartsWith(fullBase, StringComparison.Ordinal))
|
|
|
|
|
|
return false;
|
|
|
|
|
|
|
|
|
|
|
|
if (File.Exists(fullPath) || Directory.Exists(fullPath))
|
|
|
|
|
|
{
|
|
|
|
|
|
FileInfo fileInfo = new FileInfo(fullPath);
|
|
|
|
|
|
if (fileInfo.Attributes.HasFlag(FileAttributes.ReparsePoint))
|
|
|
|
|
|
return false;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
return true;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|