mirror of
https://github.com/wabbajack-tools/wabbajack.git
synced 2024-08-30 18:42:17 +00:00
280 lines
11 KiB
C#
280 lines
11 KiB
C#
using System;
|
|
using System.IO;
|
|
using System.Linq;
|
|
using System.Text;
|
|
using System.Windows;
|
|
using Wabbajack.Common;
|
|
using Wabbajack.Lib.Downloaders;
|
|
using Wabbajack.Lib.NexusApi;
|
|
using Wabbajack.Lib.Validation;
|
|
using Directory = Alphaleonis.Win32.Filesystem.Directory;
|
|
using File = Alphaleonis.Win32.Filesystem.File;
|
|
using Path = Alphaleonis.Win32.Filesystem.Path;
|
|
|
|
namespace Wabbajack.Lib
|
|
{
|
|
public class MO2Installer : AInstaller
|
|
{
|
|
public bool WarnOnOverwrite { get; set; } = true;
|
|
|
|
public MO2Installer(string archive, ModList modList, string outputFolder)
|
|
{
|
|
ModManager = ModManager.MO2;
|
|
ModListArchive = archive;
|
|
OutputFolder = outputFolder;
|
|
DownloadFolder = Path.Combine(OutputFolder, "downloads");
|
|
ModList = modList;
|
|
}
|
|
|
|
public string GameFolder { get; set; }
|
|
|
|
protected override bool _Begin()
|
|
{
|
|
ConfigureProcessor(17, RecommendQueueSize());
|
|
var game = GameRegistry.Games[ModList.GameType];
|
|
|
|
if (GameFolder == null)
|
|
GameFolder = game.GameLocation;
|
|
|
|
if (GameFolder == null)
|
|
{
|
|
MessageBox.Show(
|
|
$"In order to do a proper install Wabbajack needs to know where your {game.MO2Name} folder resides. We tried looking the" +
|
|
"game location up in the windows registry but were unable to find it, please make sure you launch the game once before running this installer. ",
|
|
"Could not find game location", MessageBoxButton.OK);
|
|
Utils.Log("Exiting because we couldn't find the game folder.");
|
|
return false;
|
|
}
|
|
|
|
UpdateTracker.NextStep("Validating Game ESMs");
|
|
ValidateGameESMs();
|
|
|
|
UpdateTracker.NextStep("Validating Modlist");
|
|
ValidateModlist.RunValidation(ModList);
|
|
|
|
Directory.CreateDirectory(OutputFolder);
|
|
Directory.CreateDirectory(DownloadFolder);
|
|
|
|
if (Directory.Exists(Path.Combine(OutputFolder, "mods")) && WarnOnOverwrite)
|
|
{
|
|
if (MessageBox.Show(
|
|
"There already appears to be a Mod Organizer 2 install in this folder, are you sure you wish to continue" +
|
|
" with installation? If you do, you may render both your existing install and the new modlist inoperable.",
|
|
"Existing MO2 installation in install folder",
|
|
MessageBoxButton.YesNo,
|
|
MessageBoxImage.Exclamation) == MessageBoxResult.No)
|
|
{
|
|
Utils.Log("Existing installation at the request of the user, existing mods folder found.");
|
|
return false;
|
|
}
|
|
}
|
|
|
|
UpdateTracker.NextStep("Optimizing Modlist");
|
|
OptimizeModlist();
|
|
|
|
UpdateTracker.NextStep("Hashing Archives");
|
|
HashArchives();
|
|
|
|
UpdateTracker.NextStep("Downloading Missing Archives");
|
|
DownloadArchives();
|
|
|
|
UpdateTracker.NextStep("Hashing Remaining Archives");
|
|
HashArchives();
|
|
|
|
var missing = ModList.Archives.Where(a => !HashedArchives.ContainsKey(a.Hash)).ToList();
|
|
if (missing.Count > 0)
|
|
{
|
|
foreach (var a in missing)
|
|
Info($"Unable to download {a.Name}");
|
|
if (IgnoreMissingFiles)
|
|
Info("Missing some archives, but continuing anyways at the request of the user");
|
|
else
|
|
Error("Cannot continue, was unable to download one or more archives");
|
|
}
|
|
|
|
UpdateTracker.NextStep("Priming VFS");
|
|
PrimeVFS();
|
|
|
|
UpdateTracker.NextStep("Building Folder Structure");
|
|
BuildFolderStructure();
|
|
|
|
UpdateTracker.NextStep("Installing Archives");
|
|
InstallArchives();
|
|
|
|
UpdateTracker.NextStep("Installing Included files");
|
|
InstallIncludedFiles();
|
|
|
|
UpdateTracker.NextStep("Installing Archive Metas");
|
|
InstallIncludedDownloadMetas();
|
|
|
|
UpdateTracker.NextStep("Building BSAs");
|
|
BuildBSAs();
|
|
|
|
UpdateTracker.NextStep("Generating Merges");
|
|
zEditIntegration.GenerateMerges(this);
|
|
|
|
UpdateTracker.NextStep("Installation complete! You may exit the program.");
|
|
return true;
|
|
}
|
|
|
|
|
|
private void InstallIncludedDownloadMetas()
|
|
{
|
|
ModList.Directives
|
|
.OfType<ArchiveMeta>()
|
|
.PMap(Queue, directive =>
|
|
{
|
|
Status($"Writing included .meta file {directive.To}");
|
|
var outPath = Path.Combine(DownloadFolder, directive.To);
|
|
if (File.Exists(outPath)) File.Delete(outPath);
|
|
File.WriteAllBytes(outPath, LoadBytesFromPath(directive.SourceDataID));
|
|
});
|
|
}
|
|
|
|
private void ValidateGameESMs()
|
|
{
|
|
foreach (var esm in ModList.Directives.OfType<CleanedESM>().ToList())
|
|
{
|
|
var filename = Path.GetFileName(esm.To);
|
|
var gameFile = Path.Combine(GameFolder, "Data", filename);
|
|
Utils.Log($"Validating {filename}");
|
|
var hash = gameFile.FileHash();
|
|
if (hash != esm.SourceESMHash)
|
|
{
|
|
Utils.Error("Game ESM hash doesn't match, is the ESM already cleaned? Please verify your local game files.");
|
|
}
|
|
}
|
|
}
|
|
|
|
private void AskToEndorse()
|
|
{
|
|
var mods = ModList.Archives
|
|
.Select(m => m.State)
|
|
.OfType<NexusDownloader.State>()
|
|
.GroupBy(f => (f.GameName, f.ModID))
|
|
.Select(mod => mod.First())
|
|
.ToArray();
|
|
|
|
var result = MessageBox.Show(
|
|
$"Installation has completed, but you have installed {mods.Length} from the Nexus, would you like to" +
|
|
" endorse these mods to show support to the authors? It will only take a few moments.", "Endorse Mods?",
|
|
MessageBoxButton.YesNo, MessageBoxImage.Question);
|
|
|
|
if (result != MessageBoxResult.Yes) return;
|
|
|
|
// Shuffle mods so that if we hit a API limit we don't always miss the same mods
|
|
var r = new Random();
|
|
for (var i = 0; i < mods.Length; i++)
|
|
{
|
|
var a = r.Next(mods.Length);
|
|
var b = r.Next(mods.Length);
|
|
var tmp = mods[a];
|
|
mods[a] = mods[b];
|
|
mods[b] = tmp;
|
|
}
|
|
|
|
mods.PMap(Queue, mod =>
|
|
{
|
|
var er = new NexusApiClient().EndorseMod(mod);
|
|
Utils.Log($"Endorsed {mod.GameName} - {mod.ModID} - Result: {er.message}");
|
|
});
|
|
Info("Done! You may now exit the application!");
|
|
}
|
|
|
|
private void BuildBSAs()
|
|
{
|
|
var bsas = ModList.Directives.OfType<CreateBSA>().ToList();
|
|
Info($"Building {bsas.Count} bsa files");
|
|
|
|
bsas.Do(bsa =>
|
|
{
|
|
Status($"Building {bsa.To}");
|
|
var sourceDir = Path.Combine(OutputFolder, Consts.BSACreationDir, bsa.TempID);
|
|
|
|
using (var a = bsa.State.MakeBuilder())
|
|
{
|
|
bsa.FileStates.PMap(Queue, state =>
|
|
{
|
|
Status($"Adding {state.Path} to BSA");
|
|
using (var fs = File.OpenRead(Path.Combine(sourceDir, state.Path)))
|
|
{
|
|
a.AddFile(state, fs);
|
|
}
|
|
});
|
|
|
|
Info($"Writing {bsa.To}");
|
|
a.Build(Path.Combine(OutputFolder, bsa.To));
|
|
}
|
|
});
|
|
|
|
|
|
var bsaDir = Path.Combine(OutputFolder, Consts.BSACreationDir);
|
|
if (Directory.Exists(bsaDir))
|
|
{
|
|
Info($"Removing temp folder {Consts.BSACreationDir}");
|
|
Utils.DeleteDirectory(bsaDir);
|
|
}
|
|
}
|
|
|
|
private void InstallIncludedFiles()
|
|
{
|
|
Info("Writing inline files");
|
|
ModList.Directives
|
|
.OfType<InlineFile>()
|
|
.PMap(Queue, directive =>
|
|
{
|
|
Status($"Writing included file {directive.To}");
|
|
var outPath = Path.Combine(OutputFolder, directive.To);
|
|
if (File.Exists(outPath)) File.Delete(outPath);
|
|
if (directive is RemappedInlineFile)
|
|
WriteRemappedFile((RemappedInlineFile)directive);
|
|
else if (directive is CleanedESM)
|
|
GenerateCleanedESM((CleanedESM)directive);
|
|
else
|
|
File.WriteAllBytes(outPath, LoadBytesFromPath(directive.SourceDataID));
|
|
});
|
|
}
|
|
|
|
private void GenerateCleanedESM(CleanedESM directive)
|
|
{
|
|
var filename = Path.GetFileName(directive.To);
|
|
var gameFile = Path.Combine(GameFolder, "Data", filename);
|
|
Info($"Generating cleaned ESM for {filename}");
|
|
if (!File.Exists(gameFile)) throw new InvalidDataException($"Missing {filename} at {gameFile}");
|
|
Status($"Hashing game version of {filename}");
|
|
var sha = gameFile.FileHash();
|
|
if (sha != directive.SourceESMHash)
|
|
throw new InvalidDataException(
|
|
$"Cannot patch {filename} from the game folder hashes don't match have you already cleaned the file?");
|
|
|
|
var patchData = LoadBytesFromPath(directive.SourceDataID);
|
|
var toFile = Path.Combine(OutputFolder, directive.To);
|
|
Status($"Patching {filename}");
|
|
using (var output = File.OpenWrite(toFile))
|
|
using (var input = File.OpenRead(gameFile))
|
|
{
|
|
BSDiff.Apply(input, () => new MemoryStream(patchData), output);
|
|
}
|
|
}
|
|
|
|
private void WriteRemappedFile(RemappedInlineFile directive)
|
|
{
|
|
var data = Encoding.UTF8.GetString(LoadBytesFromPath(directive.SourceDataID));
|
|
|
|
data = data.Replace(Consts.GAME_PATH_MAGIC_BACK, GameFolder);
|
|
data = data.Replace(Consts.GAME_PATH_MAGIC_DOUBLE_BACK, GameFolder.Replace("\\", "\\\\"));
|
|
data = data.Replace(Consts.GAME_PATH_MAGIC_FORWARD, GameFolder.Replace("\\", "/"));
|
|
|
|
data = data.Replace(Consts.MO2_PATH_MAGIC_BACK, OutputFolder);
|
|
data = data.Replace(Consts.MO2_PATH_MAGIC_DOUBLE_BACK, OutputFolder.Replace("\\", "\\\\"));
|
|
data = data.Replace(Consts.MO2_PATH_MAGIC_FORWARD, OutputFolder.Replace("\\", "/"));
|
|
|
|
data = data.Replace(Consts.DOWNLOAD_PATH_MAGIC_BACK, DownloadFolder);
|
|
data = data.Replace(Consts.DOWNLOAD_PATH_MAGIC_DOUBLE_BACK, DownloadFolder.Replace("\\", "\\\\"));
|
|
data = data.Replace(Consts.DOWNLOAD_PATH_MAGIC_FORWARD, DownloadFolder.Replace("\\", "/"));
|
|
|
|
File.WriteAllText(Path.Combine(OutputFolder, directive.To), data);
|
|
}
|
|
}
|
|
}
|