using System; using System.Collections.Generic; using System.Linq; using System.Threading; using System.Threading.Tasks; using Microsoft.Extensions.DependencyInjection; using Microsoft.Extensions.Logging; using Wabbajack.Common; using Wabbajack.Downloaders; using Wabbajack.DTOs; using Wabbajack.DTOs.JsonConverters; using Wabbajack.Installer; using Wabbajack.Paths; using Wabbajack.Paths.IO; using Xunit; namespace Wabbajack.Compiler.Test { public class ModListHarness { private readonly ILogger _logger; private readonly TemporaryFileManager _manager; public readonly AbsolutePath _source; private readonly AbsolutePath _downloadPath; private readonly AbsolutePath _installLocation; private readonly AbsolutePath _modsFolder; private readonly Dictionary _mods; public readonly FileExtractor.FileExtractor _fileExtractor; private readonly string _profileName; private readonly IServiceProvider _serviceProvider; private readonly TemporaryPath _outputFolder; private readonly DownloadDispatcher _downloadDispatcher; private readonly DTOSerializer _dtos; private readonly AbsolutePath _installDownloads; private readonly AbsolutePath _outputFile; private readonly AbsolutePath _gameFolder; public ModListHarness(ILogger logger, TemporaryFileManager manager, FileExtractor.FileExtractor fileExtractor, IServiceProvider serviceProvider, DownloadDispatcher downloadDispatcher, DTOSerializer dtos) { _logger = logger; _manager = manager; _source = _manager.CreateFolder(); _profileName = Guid.NewGuid().ToString(); _downloadPath = _manager.CreateFolder(); _installLocation = _manager.CreateFolder(); _outputFolder = _manager.CreateFolder(); _modsFolder = _source.Combine(Consts.MO2ModFolderName); _mods = new Dictionary(); _fileExtractor = fileExtractor; _serviceProvider = serviceProvider; _downloadDispatcher = downloadDispatcher; _gameFolder = _manager.CreateFolder(); _outputFile = _outputFolder.Path.Combine(_profileName + ".wabbajack"); _installDownloads = _installLocation.Combine("downloads"); _dtos = dtos; } public Mod AddMod(string? name = null) { name ??= Guid.NewGuid().ToString(); var mod = new Mod(name.ToRelativePath(), _modsFolder.Combine(name), this, new HashSet()); _mods[name!.ToRelativePath()] = mod; return mod; } public async Task CompileAndInstall(Action? configureSettings = null) { var modlist = await Compile(configureSettings); await Install(); return modlist; } public async Task Compile(Action? configureSettings = null) { configureSettings ??= x => {}; _source.Combine(Consts.MO2Profiles, _profileName).CreateDirectory(); using var scope = _serviceProvider.CreateScope(); var settings = scope.ServiceProvider.GetService()!; settings.Downloads = _downloadPath; settings.Game = Game.SkyrimSpecialEdition; settings.Source = _source; settings.ModListName = _profileName; settings.SelectedProfiles = new []{_profileName}; settings.Profile = _profileName; settings.OutputFile = _outputFile; configureSettings(settings); var modLines = _mods.Select( m => (m.Value.EnabledIn.Contains(_profileName) ? "+" : "-") + m.Key); await _source.Combine(Consts.MO2Profiles, _profileName, Consts.ModListTxt) .WriteAllLinesAsync(modLines, CancellationToken.None); var compiler = scope.ServiceProvider.GetService(); if (!await compiler.Begin(CancellationToken.None)) return null; var modlist = await StandardInstaller.LoadFromFile(_dtos, settings.OutputFile); return modlist; } public async Task Install() { using var scope = _serviceProvider.CreateScope(); var settings = scope.ServiceProvider.GetService()!; settings.Install = _installLocation; settings.Downloads = _installDownloads; settings.ModList = await StandardInstaller.LoadFromFile(_dtos, _outputFile); settings.ModlistArchive = _outputFile; settings.Game = Game.SkyrimSpecialEdition; settings.GameFolder = _gameFolder; settings.SystemParameters = new SystemParameters { ScreenWidth = 1920, ScreenHeight = 1080, SystemMemorySize = 8L * 1024 * 1024 * 1024, SystemPageSize = 8L * 1024 * 1024 * 1024, VideoMemorySize = 8L * 1024 * 1024 * 1024 }; var installer = scope.ServiceProvider.GetService()!; return await installer.Begin(CancellationToken.None); } public async Task AddManualDownload(AbsolutePath path) { var toPath = path.FileName.RelativeTo(_downloadPath); await path.CopyToAsync(toPath, true, CancellationToken.None); await toPath.WithExtension(Ext.Meta) .WriteAllLinesAsync(new[] { "[General]", $"manualURL={path.FileName}" }, CancellationToken.None); } public async Task InstallMod(Extension ext, Uri uri) { var state = _downloadDispatcher.Parse(uri); var file = (Guid.NewGuid() + ext.ToString()).ToRelativePath(); var archive = new Archive { State = state!, Name = file.ToString() }; _logger.LogInformation("Downloading: {uri}", uri); await _downloadDispatcher.Download(archive, file.RelativeTo(_downloadPath), CancellationToken.None); await file.WithExtension(Ext.Meta).RelativeTo(_downloadPath) .WriteAllTextAsync(_downloadDispatcher.MetaIniSection(archive), CancellationToken.None); var mod = AddMod(file.WithoutExtension().ToString()); mod.EnabledIn.Add(_profileName); _logger.LogInformation("Extracting: {file}", file); await mod.AddFromArchive(file.RelativeTo(_downloadPath)); return mod; } public void VerifyInstalledFile(AbsolutePath source) { var dest = source.RelativeTo(_source).RelativeTo(_installLocation); _logger.LogInformation("Verifying {file}", source.RelativeTo(_source)); Assert.Equal(source.Size(), dest.Size()); } } public record Mod(RelativePath Name, AbsolutePath FullPath, ModListHarness Harness, HashSet EnabledIn) { public async Task AddFile(AbsolutePath src) { var dest = FullPath.Combine(src.FileName); await src.CopyToAsync(dest, true, CancellationToken.None); return dest; } public async Task AddFromArchive(AbsolutePath src) { await Harness._fileExtractor.ExtractAll(src, FullPath, CancellationToken.None); } public async Task AddData(RelativePath path, string data) { var fullPath = FullPath.Combine(path); fullPath.Parent.CreateDirectory(); await fullPath.WriteAllTextAsync(data); return fullPath; } } }