2022-05-27 05:41:11 +00:00
|
|
|
using System;
|
|
|
|
using System.Collections;
|
|
|
|
using System.Collections.Generic;
|
|
|
|
using System.CommandLine;
|
|
|
|
using System.CommandLine.Invocation;
|
2022-09-29 05:10:49 +00:00
|
|
|
using System.CommandLine.NamingConventionBinder;
|
2022-05-27 05:41:11 +00:00
|
|
|
using System.Linq;
|
|
|
|
using System.Threading;
|
|
|
|
using System.Threading.Tasks;
|
|
|
|
using Microsoft.Extensions.Logging;
|
2022-05-28 22:53:52 +00:00
|
|
|
using Wabbajack.Common;
|
2022-05-27 05:41:11 +00:00
|
|
|
using Wabbajack.Compiler;
|
|
|
|
using Wabbajack.Downloaders;
|
|
|
|
using Wabbajack.Downloaders.GameFile;
|
|
|
|
using Wabbajack.DTOs;
|
|
|
|
using Wabbajack.DTOs.JsonConverters;
|
|
|
|
using Wabbajack.Installer;
|
|
|
|
using Wabbajack.Networking.WabbajackClientApi;
|
|
|
|
using Wabbajack.Paths;
|
|
|
|
using Wabbajack.Paths.IO;
|
|
|
|
using Wabbajack.VFS;
|
|
|
|
|
|
|
|
namespace Wabbajack.CLI.Verbs;
|
|
|
|
|
|
|
|
public class InstallCompileInstallVerify : IVerb
|
|
|
|
{
|
|
|
|
private readonly ILogger<InstallCompileInstallVerify> _logger;
|
|
|
|
private readonly Client _wjClient;
|
|
|
|
private readonly DownloadDispatcher _dispatcher;
|
|
|
|
|
|
|
|
private readonly DTOSerializer _dtos;
|
|
|
|
private readonly IServiceProvider _serviceProvider;
|
|
|
|
private readonly FileHashCache _cache;
|
|
|
|
private readonly GameLocator _gameLocator;
|
|
|
|
private readonly CompilerSettingsInferencer _inferencer;
|
|
|
|
|
|
|
|
public InstallCompileInstallVerify(ILogger<InstallCompileInstallVerify> logger, Client wjClient, DownloadDispatcher dispatcher, DTOSerializer dtos,
|
|
|
|
FileHashCache cache, GameLocator gameLocator, IServiceProvider serviceProvider, CompilerSettingsInferencer inferencer)
|
|
|
|
{
|
|
|
|
_logger = logger;
|
|
|
|
_wjClient = wjClient;
|
|
|
|
_dispatcher = dispatcher;
|
|
|
|
_dtos = dtos;
|
|
|
|
_serviceProvider = serviceProvider;
|
|
|
|
_cache = cache;
|
|
|
|
_gameLocator = gameLocator;
|
|
|
|
_inferencer = inferencer;
|
|
|
|
}
|
|
|
|
|
|
|
|
public Command MakeCommand()
|
|
|
|
{
|
|
|
|
var command = new Command("install-compile-install-verify");
|
|
|
|
command.Add(new Option<AbsolutePath>(new[] {"-m", "-machineUrls"}, "Machine url(s) to download"));
|
|
|
|
command.Add(new Option<AbsolutePath>(new[] {"-d", "-downloads"}, "Downloads path"));
|
|
|
|
command.Add(new Option<AbsolutePath>(new[] {"-o", "-outputs"}, "Outputs path"));
|
|
|
|
command.Description = "Installs a modlist, compiles it, installs it again, verifies it";
|
|
|
|
command.Handler = CommandHandler.Create(Run);
|
|
|
|
return command;
|
|
|
|
}
|
|
|
|
|
|
|
|
public async Task<int> Run(AbsolutePath outputs, AbsolutePath downloads, IEnumerable<string> machineUrls, CancellationToken token)
|
|
|
|
{
|
|
|
|
foreach (var machineUrl in machineUrls)
|
|
|
|
{
|
|
|
|
_logger.LogInformation("Installing {MachineUrl}", machineUrl);
|
2022-05-28 22:53:52 +00:00
|
|
|
var wabbajackPath = downloads.Combine(machineUrl.Replace("/", "_@@_")).WithExtension(Ext.Wabbajack);
|
2022-05-27 05:41:11 +00:00
|
|
|
if (!await DownloadMachineUrl(machineUrl, wabbajackPath, token))
|
|
|
|
throw new Exception("Can't download modlist");
|
|
|
|
|
|
|
|
var installPath = outputs.Combine(machineUrl);
|
|
|
|
|
|
|
|
var modlist = await StandardInstaller.LoadFromFile(_dtos, wabbajackPath);
|
|
|
|
|
|
|
|
var installer = StandardInstaller.Create(_serviceProvider, new InstallerConfiguration
|
|
|
|
{
|
|
|
|
Downloads = downloads,
|
|
|
|
Install = installPath,
|
|
|
|
ModList = modlist,
|
|
|
|
Game = modlist.GameType,
|
|
|
|
ModlistArchive = wabbajackPath,
|
|
|
|
GameFolder = _gameLocator.GameLocation(modlist.GameType)
|
|
|
|
});
|
|
|
|
|
|
|
|
var result = await installer.Begin(token);
|
|
|
|
if (!result)
|
|
|
|
{
|
|
|
|
_logger.LogInformation("Error installing {MachineUrl}", machineUrl);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2022-05-28 22:53:52 +00:00
|
|
|
|
2022-05-27 05:41:11 +00:00
|
|
|
_logger.LogInformation("Inferring settings");
|
2022-05-30 21:02:54 +00:00
|
|
|
var inferredSettings = await _inferencer.InferFromRootPath(installPath);
|
|
|
|
if (inferredSettings == null)
|
2022-05-27 05:41:11 +00:00
|
|
|
{
|
|
|
|
_logger.LogInformation("Error inferencing settings for {MachineUrl}", machineUrl);
|
|
|
|
return 2;
|
|
|
|
}
|
|
|
|
|
2022-05-30 21:02:54 +00:00
|
|
|
inferredSettings.UseGamePaths = true;
|
2022-05-27 05:41:11 +00:00
|
|
|
|
|
|
|
|
2022-05-30 21:02:54 +00:00
|
|
|
var compiler = MO2Compiler.Create(_serviceProvider, inferredSettings);
|
2022-05-27 05:41:11 +00:00
|
|
|
result = await compiler.Begin(token);
|
2022-05-30 20:50:24 +00:00
|
|
|
if (!result)
|
|
|
|
return result ? 0 : 3;
|
|
|
|
|
|
|
|
|
|
|
|
var installPath2 = outputs.Combine("verify_list");
|
|
|
|
|
|
|
|
var comparison = await StandardInstaller.LoadFromFile(_dtos, wabbajackPath);
|
|
|
|
|
2022-05-30 21:02:54 +00:00
|
|
|
var modlist2 = await StandardInstaller.LoadFromFile(_dtos, inferredSettings.OutputFile);
|
2022-05-30 20:50:24 +00:00
|
|
|
if (CompareModlists(comparison, modlist2))
|
|
|
|
return 3;
|
|
|
|
|
|
|
|
var installer2 = StandardInstaller.Create(_serviceProvider, new InstallerConfiguration
|
|
|
|
{
|
|
|
|
Downloads = downloads,
|
|
|
|
Install = installPath2,
|
|
|
|
ModList = modlist2,
|
|
|
|
Game = modlist2.GameType,
|
2022-05-30 21:02:54 +00:00
|
|
|
ModlistArchive = inferredSettings.OutputFile,
|
2022-05-30 20:50:24 +00:00
|
|
|
GameFolder = _gameLocator.GameLocation(modlist2.GameType)
|
|
|
|
});
|
2022-05-27 05:41:11 +00:00
|
|
|
|
2022-05-30 20:50:24 +00:00
|
|
|
result = await installer2.Begin(token);
|
|
|
|
if (!result)
|
|
|
|
{
|
|
|
|
_logger.LogInformation("Error installing recompiled {MachineUrl}", machineUrl);
|
|
|
|
return 1;
|
|
|
|
}
|
2022-05-27 05:41:11 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
2022-05-30 20:50:24 +00:00
|
|
|
|
|
|
|
private bool CompareModlists(ModList a, ModList b)
|
|
|
|
{
|
|
|
|
var aDirectives = a.Directives.ToDictionary(d => d.To);
|
|
|
|
var bDirectives = b.Directives.ToDictionary(d => d.To);
|
|
|
|
|
|
|
|
var found = false;
|
|
|
|
foreach (var missing in aDirectives.Where(ad => !bDirectives.ContainsKey(ad.Key)))
|
|
|
|
{
|
2022-05-30 21:02:54 +00:00
|
|
|
if (missing.Key.Extension == Ext.Meta)
|
|
|
|
continue;
|
2022-05-30 20:50:24 +00:00
|
|
|
_logger.LogWarning("File {To} is missing in recompiled list", missing.Key);
|
|
|
|
found = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
foreach (var missing in bDirectives.Where(bd => !aDirectives.ContainsKey(bd.Key)))
|
|
|
|
{
|
|
|
|
_logger.LogWarning("File {To} is missing in original list", missing.Key);
|
|
|
|
found = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return found;
|
|
|
|
}
|
|
|
|
|
2022-05-27 05:41:11 +00:00
|
|
|
private async Task<bool> DownloadMachineUrl(string machineUrl, AbsolutePath wabbajack, CancellationToken token)
|
|
|
|
{
|
|
|
|
_logger.LogInformation("Downloading {MachineUrl}", machineUrl);
|
|
|
|
|
|
|
|
var lists = await _wjClient.LoadLists();
|
|
|
|
var list = lists.FirstOrDefault(l => l.NamespacedName == machineUrl);
|
|
|
|
if (list == null)
|
|
|
|
{
|
|
|
|
_logger.LogInformation("Couldn't find list {MachineUrl}", machineUrl);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (wabbajack.FileExists() && await _cache.FileHashCachedAsync(wabbajack, token) == list.DownloadMetadata!.Hash)
|
|
|
|
{
|
|
|
|
_logger.LogInformation("File already exists, using cached file");
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
var state = _dispatcher.Parse(new Uri(list.Links.Download));
|
|
|
|
|
|
|
|
await _dispatcher.Download(new Archive
|
|
|
|
{
|
|
|
|
Name = wabbajack.FileName.ToString(),
|
|
|
|
Hash = list.DownloadMetadata!.Hash,
|
|
|
|
Size = list.DownloadMetadata.Size,
|
|
|
|
State = state!
|
|
|
|
}, wabbajack, token);
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|