wabbajack/Wabbajack.Launcher/ViewModels/MainWindowViewModel.cs
JanuarySnow 58f9ea7afc
Protected folder subfolder handling (#2366)
* added more robust protected location checking, for exe and download/install paths

remove debug output and formatting

slight change to trigger rerun of CI

updated changelog

trigger CI re-run

* removed debug code

---------

Co-authored-by: Timothy Baldridge <tbaldridge@gmail.com>
2023-07-16 15:22:56 -06:00

401 lines
13 KiB
C#

using System;
using System.Collections.Generic;
using System.Diagnostics;
using System.IO;
using System.IO.Compression;
using System.Linq;
using System.Net;
using System.Text.Json;
using System.Text.Json.Serialization;
using System.Threading;
using System.Threading.Tasks;
using JetBrains.Annotations;
using MessageBox.Avalonia.DTO;
using ReactiveUI.Fody.Helpers;
using Wabbajack.Common;
using Wabbajack.Compression.Zip;
using Wabbajack.Downloaders.Http;
using Wabbajack.DTOs;
using Wabbajack.DTOs.DownloadStates;
using Wabbajack.DTOs.Logins;
using Wabbajack.Networking.Http.Interfaces;
using Wabbajack.Networking.NexusApi;
using Wabbajack.Paths;
using Wabbajack.Paths.IO;
#pragma warning disable SYSLIB0014
namespace Wabbajack.Launcher.ViewModels;
public class MainWindowViewModel : ViewModelBase
{
private readonly WebClient _client = new();
private readonly List<string> _errors = new();
private (Version Version, long Size, Func<Task<Uri>> Uri) _version;
public Uri GITHUB_REPO = new("https://api.github.com/repos/wabbajack-tools/wabbajack/releases");
private readonly NexusApi _nexusApi;
private readonly HttpDownloader _downloader;
private readonly ITokenProvider<NexusApiState> _tokenProvider;
public MainWindowViewModel(NexusApi nexusApi, HttpDownloader downloader, ITokenProvider<NexusApiState> tokenProvider)
{
_nexusApi = nexusApi;
Status = "Checking for new versions";
_downloader = downloader;
_tokenProvider = tokenProvider;
var tsk = CheckForUpdates();
}
[Reactive] public string Status { get; set; }
private async Task CheckForUpdates()
{
await VerifyCurrentLocation();
_client.Headers.Add("user-agent", "Wabbajack Launcher");
Status = "Selecting Release";
try
{
if (_tokenProvider.HaveToken())
{
try
{
_version = await GetNexusReleases(CancellationToken.None);
}
catch (Exception)
{
_errors.Add("Nexus error");
}
}
if (_version == default)
{
_version = await GetGithubRelease(CancellationToken.None);
}
}
catch (Exception ex)
{
_errors.Add(ex.Message);
await FinishAndExit();
}
if (_version == default)
{
_errors.Add("Unable to find releases");
await FinishAndExit();
}
Status = "Looking for Updates";
var baseFolder = KnownFolders.CurrentDirectory.Combine(_version.Version.ToString());
if (baseFolder.Combine("Wabbajack.exe").FileExists()) await FinishAndExit();
Status = $"Getting download Uri for {_version.Version}";
var uri = await _version.Uri();
/*
var archive = new Archive()
{
Name = $"{_version.Version}.zip",
Size = _version.Size,
State = new Http {Url = uri}
};
await using var stream = await _downloader.GetChunkedSeekableStream(archive, CancellationToken.None);
var rdr = new ZipReader(stream, true);
var entries = (await rdr.GetFiles()).OrderBy(d => d.FileOffset).ToArray();
foreach (var file in entries)
{
if (file.FileName.EndsWith("/") || file.FileName.EndsWith("\\")) continue;
var relPath = file.FileName.ToRelativePath();
Status = $"Extracting: {relPath.FileName}";
var outPath = baseFolder.Combine(relPath);
if (!outPath.Parent.DirectoryExists())
outPath.Parent.CreateDirectory();
await using var of = outPath.Open(FileMode.Create, FileAccess.Write, FileShare.None);
await rdr.Extract(file, of, CancellationToken.None);
}*/
var wc = new WebClient();
wc.DownloadProgressChanged += (sender, args) => UpdateProgress($"{_version.Version}", sender, args);
Status = $"Downloading {_version.Version} ...";
byte[] data;
try
{
data = await wc.DownloadDataTaskAsync(uri);
}
catch (Exception ex)
{
_errors.Add(ex.Message);
// Something went wrong so fallback to original URL
try
{
data = await wc.DownloadDataTaskAsync(uri);
}
catch (Exception ex2)
{
_errors.Add(ex2.Message);
await FinishAndExit();
throw; // avoid unsigned variable 'data'
}
}
try
{
using var zip = new ZipArchive(new MemoryStream(data), ZipArchiveMode.Read);
foreach (var entry in zip.Entries)
{
Status = $"Extracting: {entry.Name}";
var outPath = baseFolder.Combine(entry.FullName.ToRelativePath());
if (!outPath.Parent.DirectoryExists())
outPath.Parent.CreateDirectory();
if (entry.FullName.EndsWith("/") || entry.FullName.EndsWith("\\"))
continue;
await using var o = entry.Open();
await using var of = outPath.Open(FileMode.Create, FileAccess.Write, FileShare.None);
await o.CopyToAsync(of);
}
}
catch (Exception ex)
{
_errors.Add(ex.Message);
}
try
{
await InstallWebView();
}
catch (Exception e)
{
_errors.Add(e.Message);
}
await FinishAndExit();
}
[UriString]
private const string WebViewDownloadLink = "https://go.microsoft.com/fwlink/p/?LinkId=2124703";
private async Task InstallWebView(CancellationToken cancellationToken = default)
{
var setupPath = KnownFolders.WabbajackAppLocal.Combine("MicrosoftEdgeWebview2Setup.exe");
if (setupPath.FileExists()) return;
var wc = new WebClient();
wc.DownloadProgressChanged += (sender, args) => UpdateProgress("WebView2", sender, args);
Status = "Downloading WebView2 Runtime";
byte[] data;
try
{
data = await wc.DownloadDataTaskAsync(WebViewDownloadLink);
}
catch (Exception ex)
{
_errors.Add(ex.Message);
await FinishAndExit();
throw;
}
await setupPath.WriteAllBytesAsync(new Memory<byte>(data), cancellationToken);
var process = new ProcessHelper
{
Path = setupPath,
Arguments = new []{"/silent /install"}
};
await process.Start();
}
private async Task VerifyCurrentLocation()
{
try
{
var entryPoint = KnownFolders.EntryPoint;
if (KnownFolders.IsInSpecialFolder(entryPoint) || entryPoint.Depth <= 1)
{
var msg = MessageBox.Avalonia.MessageBoxManager
.GetMessageBoxStandardWindow(new MessageBoxStandardParams()
{
Topmost = true,
ShowInCenter = true,
ContentTitle = "Wabbajack Launcher: Bad startup path",
ContentMessage =
"Cannot start in the root of a drive, or protected folder locations such as Downloads, Desktop etc.\nPlease move Wabbajack to another folder."
});
var result = await msg.Show();
Environment.Exit(1);
}
}
catch (Exception ex)
{
Status = ex.Message;
var msg = MessageBox.Avalonia.MessageBoxManager
.GetMessageBoxStandardWindow(new MessageBoxStandardParams()
{
Topmost = true,
ShowInCenter = true,
ContentTitle = "Wabbajack Launcher: Error",
ContentMessage = ex.ToString()
});
var result = await msg.Show();
Environment.Exit(1);
}
}
[ContractAnnotation("=> halt")]
private async Task FinishAndExit()
{
try
{
Status = "Launching...";
var wjFolder = KnownFolders.CurrentDirectory.EnumerateDirectories(recursive:false)
.OrderByDescending(v =>
Version.TryParse(v.FileName.ToString(), out var ver) ? ver : new Version(0, 0, 0, 0))
.FirstOrDefault();
if (wjFolder == default)
{
_errors.Add("No WJ install found");
throw new Exception("No WJ install found");
}
var filename = wjFolder.Combine("Wabbajack.exe");
await CreateBatchFile(filename);
var info = new ProcessStartInfo
{
FileName = filename.ToString(),
Arguments = string.Join(" ",
Environment.GetCommandLineArgs().Skip(1).Select(s => s.Contains(' ') ? '\"' + s + '\"' : s)),
WorkingDirectory = wjFolder.ToString()
};
Process.Start(info);
}
catch (Exception ex)
{
_errors.Add(ex.Message);
if (_errors.Count == 0)
{
Status = "Failed: Unknown error";
await Task.Delay(10000);
}
foreach (var error in _errors)
{
Status = "Failed: " + error;
await Task.Delay(10000);
}
}
finally
{
Environment.Exit(0);
}
}
private async Task CreateBatchFile(AbsolutePath filename)
{
try
{
filename = filename.Parent.Combine("wabbajack-cli.exe");
var data = $"\"{filename}\" %*";
var file = Path.Combine(Directory.GetCurrentDirectory(), "wabbajack-cli.bat");
if (File.Exists(file) && await File.ReadAllTextAsync(file) == data) return;
await File.WriteAllTextAsync(file, data);
}
catch (Exception ex)
{
_errors.Add($"Creating Batch File : {ex}");
}
}
private void UpdateProgress(string what, object sender, DownloadProgressChangedEventArgs e)
{
Status = $"Downloading {what} ({e.ProgressPercentage}%)...";
}
private async Task<(Version Version, long Size, Func<Task<Uri>> Uri)> GetGithubRelease(CancellationToken token)
{
var releases = await GetGithubReleases();
var version = releases.Select(r =>
{
if (r.Tag.Split(".").Length == 4 && Version.TryParse(r.Tag, out var v))
return (v, r);
return (new Version(0, 0, 0, 0), r);
})
.OrderByDescending(r => r.Item1)
.FirstOrDefault();
var asset = version.r.Assets.FirstOrDefault(a => a.Name == version.Item1 + ".zip");
if (asset == null)
{
Status = $"Error, no asset found for Github Release {version.r}";
return default;
}
return (version.Item1, asset.Size, async () => asset!.BrowserDownloadUrl);
}
private async Task<Release[]> GetGithubReleases()
{
Status = "Checking GitHub Repository";
var data = await _client.DownloadStringTaskAsync(GITHUB_REPO);
Status = "Parsing Response";
return JsonSerializer.Deserialize<Release[]>(data)!;
}
private async Task<(Version Version, long Size, Func<Task<Uri>> uri)> GetNexusReleases(CancellationToken token)
{
Status = "Checking Nexus for updates";
if (!await _nexusApi.IsPremium(token))
return default;
var data = await _nexusApi.ModFiles("site", 403, token);
Status = "Parsing Response";
//return JsonSerializer.Deserialize<Release[]>(data)!;
var found = data.info.Files.Where(f => f.CategoryId == 5 || f.CategoryId == 3)
.Where(f => f.Name.EndsWith(".zip"))
.Select(f => Version.TryParse(f.Name[..^4], out var version) ? (version, f.SizeInBytes ?? f.Size, f.FileId) : default)
.FirstOrDefault(f => f != default);
if (found == default) return default;
return (found.version, found.Item2, async () =>
{
var link = await _nexusApi.DownloadLink("site", 403, found.FileId, token);
return link.info.First().URI;
});
}
private class Release
{
[JsonPropertyName("tag_name")] public string Tag { get; set; }
[JsonPropertyName("assets")] public Asset[] Assets { get; set; }
}
private class Asset
{
[JsonPropertyName("browser_download_url")]
public Uri BrowserDownloadUrl { get; set; }
[JsonPropertyName("name")] public string Name { get; set; }
[JsonPropertyName("size")] public long Size { get; set; }
}
}