using System; using System.Collections.Generic; using System.Collections.ObjectModel; using System.IO; using System.Linq; using System.Net.Http; using ReactiveUI; using System.Reactive.Disposables; using System.Windows.Media.Imaging; using ReactiveUI.Fody.Helpers; using DynamicData; using System.Reactive; using System.Reactive.Linq; using System.Text.Json; using System.Threading; using System.Threading.Tasks; using System.Windows.Shell; using System.Windows.Threading; using Microsoft.Extensions.Logging; using Microsoft.WindowsAPICodePack.Dialogs; using Wabbajack.Common; using Wabbajack.Downloaders; using Wabbajack.Downloaders.GameFile; using Wabbajack.DTOs; using Wabbajack.DTOs.DownloadStates; using Wabbajack.DTOs.JsonConverters; using Wabbajack.Hashing.xxHash64; using Wabbajack.Installer; using Wabbajack.LoginManagers; using Wabbajack.Messages; using Wabbajack.Models; using Wabbajack.Paths; using Wabbajack.RateLimiter; using Wabbajack.Paths.IO; using Wabbajack.Services.OSIntegrated; using Wabbajack.Util; namespace Wabbajack; public enum ModManager { Standard } public enum InstallState { Configuration, Installing, Success, Failure } public class InstallerVM : BackNavigatingVM, IBackNavigatingVM, ICpuStatusVM { private const string LastLoadedModlist = "last-loaded-modlist"; private const string InstallSettingsPrefix = "install-settings-"; private Random _random = new(); [Reactive] public Percent StatusProgress { get; set; } [Reactive] public string StatusText { get; set; } [Reactive] public ModList ModList { get; set; } [Reactive] public ModlistMetadata ModlistMetadata { get; set; } [Reactive] public ErrorResponse? Completed { get; set; } [Reactive] public FilePickerVM ModListLocation { get; set; } [Reactive] public MO2InstallerVM Installer { get; set; } [Reactive] public BitmapFrame ModListImage { get; set; } [Reactive] public BitmapFrame SlideShowImage { get; set; } [Reactive] public InstallState InstallState { get; set; } [Reactive] protected ErrorResponse[] Errors { get; private set; } [Reactive] public ErrorResponse Error { get; private set; } /// /// Slideshow Data /// [Reactive] public string SlideShowTitle { get; set; } [Reactive] public string SlideShowAuthor { get; set; } [Reactive] public string SlideShowDescription { get; set; } private readonly DTOSerializer _dtos; private readonly ILogger _logger; private readonly SettingsManager _settingsManager; private readonly IServiceProvider _serviceProvider; private readonly SystemParametersConstructor _parametersConstructor; private readonly IGameLocator _gameLocator; private readonly ResourceMonitor _resourceMonitor; private readonly Services.OSIntegrated.Configuration _configuration; private readonly HttpClient _client; private readonly DownloadDispatcher _downloadDispatcher; private readonly IEnumerable _logins; private readonly CancellationToken _cancellationToken; public ReadOnlyObservableCollection StatusList => _resourceMonitor.Tasks; [Reactive] public bool Installing { get; set; } [Reactive] public ErrorResponse ErrorState { get; set; } [Reactive] public bool ShowNSFWSlides { get; set; } public LogStream LoggerProvider { get; } // Command properties public ReactiveCommand ShowManifestCommand { get; } public ReactiveCommand OpenReadmeCommand { get; } public ReactiveCommand OpenDiscordButton { get; } public ReactiveCommand VisitModListWebsiteCommand { get; } public ReactiveCommand CloseWhenCompleteCommand { get; } public ReactiveCommand OpenLogsCommand { get; } public ReactiveCommand GoToInstallCommand { get; } public ReactiveCommand BeginCommand { get; } public InstallerVM(ILogger logger, DTOSerializer dtos, SettingsManager settingsManager, IServiceProvider serviceProvider, SystemParametersConstructor parametersConstructor, IGameLocator gameLocator, LogStream loggerProvider, ResourceMonitor resourceMonitor, Wabbajack.Services.OSIntegrated.Configuration configuration, HttpClient client, DownloadDispatcher dispatcher, IEnumerable logins, CancellationToken cancellationToken) : base(logger) { _logger = logger; _configuration = configuration; LoggerProvider = loggerProvider; _settingsManager = settingsManager; _dtos = dtos; _serviceProvider = serviceProvider; _parametersConstructor = parametersConstructor; _gameLocator = gameLocator; _resourceMonitor = resourceMonitor; _client = client; _downloadDispatcher = dispatcher; _logins = logins; _cancellationToken = cancellationToken; Installer = new MO2InstallerVM(this); BackCommand = ReactiveCommand.Create(() => NavigateToGlobal.Send(NavigateToGlobal.ScreenType.ModeSelectionView)); BeginCommand = ReactiveCommand.Create(() => BeginInstall().FireAndForget()); OpenReadmeCommand = ReactiveCommand.Create(() => { UIUtils.OpenWebsite(new Uri(ModList!.Readme)); }, this.WhenAnyValue(vm => vm.LoadingLock.IsNotLoading, vm => vm.ModList.Readme, (isNotLoading, readme) => isNotLoading && !string.IsNullOrWhiteSpace(readme))); VisitModListWebsiteCommand = ReactiveCommand.Create(() => { UIUtils.OpenWebsite(ModList!.Website); }, LoadingLock.IsNotLoadingObservable); ModListLocation = new FilePickerVM { ExistCheckOption = FilePickerVM.CheckOptions.On, PathType = FilePickerVM.PathTypeOptions.File, PromptTitle = "Select a ModList to install" }; ModListLocation.Filters.Add(new CommonFileDialogFilter("Wabbajack Modlist", "*.wabbajack")); OpenLogsCommand = ReactiveCommand.Create(() => { UIUtils.OpenFolder(_configuration.LogLocation); }); OpenDiscordButton = ReactiveCommand.Create(() => { UIUtils.OpenWebsite(new Uri(ModlistMetadata.Links.DiscordURL)); }, this.WhenAnyValue(x => x.ModlistMetadata) .WhereNotNull() .Select(md => !string.IsNullOrWhiteSpace(md.Links.DiscordURL))); ShowManifestCommand = ReactiveCommand.Create(() => { UIUtils.OpenWebsite(new Uri("https://www.wabbajack.org/search/" + ModlistMetadata.NamespacedName)); }, this.WhenAnyValue(x => x.ModlistMetadata) .WhereNotNull() .Select(md => !string.IsNullOrWhiteSpace(md.Links.MachineURL))); CloseWhenCompleteCommand = ReactiveCommand.Create(() => { Environment.Exit(0); }); GoToInstallCommand = ReactiveCommand.Create(() => { UIUtils.OpenFolder(Installer.Location.TargetPath); }); MessageBus.Current.Listen() .Subscribe(msg => LoadModlist(msg.Path, msg.Metadata).FireAndForget()) .DisposeWith(CompositeDisposable); MessageBus.Current.Listen() .Subscribe(msg => { LoadLastModlist().FireAndForget(); }); this.WhenActivated(disposables => { ModListLocation.WhenAnyValue(l => l.TargetPath) .Subscribe(p => LoadModlist(p, null).FireAndForget()) .DisposeWith(disposables); var token = new CancellationTokenSource(); BeginSlideShow(token.Token).FireAndForget(); Disposable.Create(() => token.Cancel()) .DisposeWith(disposables); this.WhenAny(vm => vm.ModListLocation.ErrorState) .CombineLatest(this.WhenAny(vm => vm.Installer.DownloadLocation.ErrorState), this.WhenAny(vm => vm.Installer.Location.ErrorState), this.WhenAny(vm => vm.ModListLocation.TargetPath), this.WhenAny(vm => vm.Installer.Location.TargetPath), this.WhenAny(vm => vm.Installer.DownloadLocation.TargetPath)) .Select(t => { var errors = new[] {t.First, t.Second, t.Third} .Where(t => t.Failed) .Concat(Validate()) .ToArray(); if (!errors.Any()) return ErrorResponse.Success; return ErrorResponse.Fail(string.Join("\n", errors.Select(e => e.Reason))); }) .BindTo(this, vm => vm.ErrorState) .DisposeWith(disposables); }); } private IEnumerable Validate() { if (!ModListLocation.TargetPath.FileExists()) yield return ErrorResponse.Fail("Mod list source does not exist"); var downloadPath = Installer.DownloadLocation.TargetPath; if (downloadPath.Depth <= 1) yield return ErrorResponse.Fail("Download path isn't set to a folder"); var installPath = Installer.Location.TargetPath; if (installPath.Depth <= 1) yield return ErrorResponse.Fail("Install path isn't set to a folder"); if (installPath.InFolder(KnownFolders.Windows)) yield return ErrorResponse.Fail("Don't install modlists into your Windows folder"); foreach (var game in GameRegistry.Games) { if (!_gameLocator.TryFindLocation(game.Key, out var location)) continue; if (installPath.InFolder(location)) yield return ErrorResponse.Fail("Can't install a modlist into a game folder"); if (location.ThisAndAllParents().Any(path => installPath == path)) { yield return ErrorResponse.Fail( "Can't install in this path, installed files may overwrite important game files"); } } if (installPath.InFolder(KnownFolders.EntryPoint)) yield return ErrorResponse.Fail("Can't install a modlist into the Wabbajack.exe path"); if (KnownFolders.EntryPoint.ThisAndAllParents().Any(path => installPath == path)) { yield return ErrorResponse.Fail("Installing in this folder may overwrite Wabbajack"); } } private async Task BeginSlideShow(CancellationToken token) { while (!token.IsCancellationRequested) { await Task.Delay(5000, token); if (InstallState == InstallState.Installing) { await PopulateNextModSlide(ModList); } } } private async Task LoadLastModlist() { var lst = await _settingsManager.Load(LastLoadedModlist); if (lst.FileExists()) { ModListLocation.TargetPath = lst; } } private async Task LoadModlist(AbsolutePath path, ModlistMetadata? metadata) { using var ll = LoadingLock.WithLoading(); InstallState = InstallState.Configuration; ModListLocation.TargetPath = path; try { ModList = await StandardInstaller.LoadFromFile(_dtos, path); ModListImage = BitmapFrame.Create(await StandardInstaller.ModListImageStream(path)); if (!string.IsNullOrWhiteSpace(ModList.Readme)) UIUtils.OpenWebsite(new Uri(ModList.Readme)); StatusText = $"Install configuration for {ModList.Name}"; TaskBarUpdate.Send($"Loaded {ModList.Name}", TaskbarItemProgressState.Normal); var hex = (await ModListLocation.TargetPath.ToString().Hash()).ToHex(); var prevSettings = await _settingsManager.Load(InstallSettingsPrefix + hex); if (path.WithExtension(Ext.MetaData).FileExists()) { try { metadata = JsonSerializer.Deserialize(await path.WithExtension(Ext.MetaData) .ReadAllTextAsync()); ModlistMetadata = metadata; } catch (Exception ex) { _logger.LogInformation(ex, "Can't load metadata cached next to file"); } } if (prevSettings.ModListLocation == path) { ModListLocation.TargetPath = prevSettings.ModListLocation; Installer.Location.TargetPath = prevSettings.InstallLocation; Installer.DownloadLocation.TargetPath = prevSettings.DownloadLoadction; ModlistMetadata = metadata ?? prevSettings.Metadata; } PopulateSlideShow(ModList); ll.Succeed(); await _settingsManager.Save(LastLoadedModlist, path); } catch (Exception ex) { _logger.LogError(ex, "While loading modlist"); ll.Fail(); } } private async Task BeginInstall() { await Task.Run(async () => { InstallState = InstallState.Installing; foreach (var downloader in await _downloadDispatcher.AllDownloaders(ModList.Archives.Select(a => a.State))) { _logger.LogInformation("Preparing {Name}", downloader.GetType().Name); if (await downloader.Prepare()) continue; var manager = _logins .FirstOrDefault(l => l.LoginFor() == downloader.GetType()); if (manager == null) { _logger.LogError("Cannot install, could not prepare {Name} for downloading", downloader.GetType().Name); throw new Exception($"No way to prepare {downloader}"); } RxApp.MainThreadScheduler.Schedule(manager, (_, _) => { manager.TriggerLogin.Execute(null); return Disposable.Empty; }); while (true) { if (await downloader.Prepare()) break; await Task.Delay(1000); } } var postfix = (await ModListLocation.TargetPath.ToString().Hash()).ToHex(); await _settingsManager.Save(InstallSettingsPrefix + postfix, new SavedInstallSettings { ModListLocation = ModListLocation.TargetPath, InstallLocation = Installer.Location.TargetPath, DownloadLoadction = Installer.DownloadLocation.TargetPath, Metadata = ModlistMetadata }); await _settingsManager.Save(LastLoadedModlist, ModListLocation.TargetPath); try { var installer = StandardInstaller.Create(_serviceProvider, new InstallerConfiguration { Game = ModList.GameType, Downloads = Installer.DownloadLocation.TargetPath, Install = Installer.Location.TargetPath, ModList = ModList, ModlistArchive = ModListLocation.TargetPath, SystemParameters = _parametersConstructor.Create(), GameFolder = _gameLocator.GameLocation(ModList.GameType) }); installer.OnStatusUpdate = update => { StatusText = update.StatusText; StatusProgress = update.StepsProgress; TaskBarUpdate.Send(update.StatusText, TaskbarItemProgressState.Indeterminate, update.StepsProgress.Value); }; if (!await installer.Begin(_cancellationToken)) { TaskBarUpdate.Send($"Error during install of {ModList.Name}", TaskbarItemProgressState.Error); InstallState = InstallState.Failure; StatusText = $"Error during install of {ModList.Name}"; StatusProgress = Percent.Zero; } else { TaskBarUpdate.Send($"Finished install of {ModList.Name}", TaskbarItemProgressState.Normal); InstallState = InstallState.Success; if (!string.IsNullOrWhiteSpace(ModList.Readme)) UIUtils.OpenWebsite(new Uri(ModList.Readme)); } } catch (Exception ex) { TaskBarUpdate.Send($"Error during install of {ModList.Name}", TaskbarItemProgressState.Error); _logger.LogError(ex, ex.Message); InstallState = InstallState.Failure; StatusText = $"Error during install of {ModList.Name}"; StatusProgress = Percent.Zero; } }); } class SavedInstallSettings { public AbsolutePath ModListLocation { get; set; } public AbsolutePath InstallLocation { get; set; } public AbsolutePath DownloadLoadction { get; set; } public ModlistMetadata Metadata { get; set; } } private void PopulateSlideShow(ModList modList) { SlideShowTitle = modList.Name; SlideShowAuthor = modList.Author; SlideShowDescription = modList.Description; SlideShowImage = ModListImage; } private async Task PopulateNextModSlide(ModList modList) { try { var mods = modList.Archives.Select(a => a.State) .OfType() .Where(t => ShowNSFWSlides || !t.IsNSFW) .Where(t => t.ImageURL != null) .ToArray(); var thisMod = mods[_random.Next(0, mods.Length)]; var data = await _client.GetByteArrayAsync(thisMod.ImageURL!); var image = BitmapFrame.Create(new MemoryStream(data)); SlideShowTitle = thisMod.Name; SlideShowAuthor = thisMod.Author; SlideShowDescription = thisMod.Description; SlideShowImage = image; } catch (Exception ex) { _logger.LogTrace(ex, "While loading slide"); } } }