2019-12-07 16:35:57 +00:00
|
|
|
|
using System;
|
|
|
|
|
using System.Diagnostics;
|
|
|
|
|
using System.IO;
|
2019-12-03 12:13:37 +00:00
|
|
|
|
using System.Linq;
|
2019-12-04 01:26:26 +00:00
|
|
|
|
using System.Threading.Tasks;
|
2019-12-03 21:56:18 +00:00
|
|
|
|
using System.Threading;
|
2019-12-03 12:13:37 +00:00
|
|
|
|
using System.Windows;
|
2019-11-09 20:40:25 +00:00
|
|
|
|
using Wabbajack.Common;
|
2020-01-03 17:22:50 +00:00
|
|
|
|
using Wabbajack.Common.StoreHandlers;
|
2019-11-09 21:45:10 +00:00
|
|
|
|
using Directory = Alphaleonis.Win32.Filesystem.Directory;
|
2019-12-07 16:35:57 +00:00
|
|
|
|
using DirectoryInfo = Alphaleonis.Win32.Filesystem.DirectoryInfo;
|
2019-11-09 21:45:10 +00:00
|
|
|
|
using File = Alphaleonis.Win32.Filesystem.File;
|
|
|
|
|
using Path = Alphaleonis.Win32.Filesystem.Path;
|
2020-04-09 17:17:24 +00:00
|
|
|
|
#nullable disable
|
2019-11-09 20:40:25 +00:00
|
|
|
|
|
|
|
|
|
namespace Wabbajack.Lib
|
|
|
|
|
{
|
2019-11-16 13:22:40 +00:00
|
|
|
|
public class VortexInstaller : AInstaller
|
2019-11-09 20:40:25 +00:00
|
|
|
|
{
|
2019-11-09 21:45:10 +00:00
|
|
|
|
public GameMetaData GameInfo { get; internal set; }
|
|
|
|
|
|
2019-12-01 20:22:33 +00:00
|
|
|
|
public override ModManager ModManager => ModManager.Vortex;
|
2019-11-09 21:45:10 +00:00
|
|
|
|
|
2020-03-26 21:15:44 +00:00
|
|
|
|
public VortexInstaller(AbsolutePath archive, ModList modList, AbsolutePath outputFolder, AbsolutePath downloadFolder, SystemParameters parameters)
|
2019-12-01 20:22:33 +00:00
|
|
|
|
: base(
|
2019-12-14 17:30:52 +00:00
|
|
|
|
archive: archive,
|
|
|
|
|
modList: modList,
|
|
|
|
|
outputFolder: outputFolder,
|
2020-01-07 13:50:11 +00:00
|
|
|
|
downloadFolder: downloadFolder,
|
2020-04-04 18:26:14 +00:00
|
|
|
|
parameters: parameters,
|
|
|
|
|
steps: 10)
|
2019-12-01 20:22:33 +00:00
|
|
|
|
{
|
|
|
|
|
#if DEBUG
|
2019-11-09 21:45:10 +00:00
|
|
|
|
// TODO: only for testing
|
|
|
|
|
IgnoreMissingFiles = true;
|
2019-12-01 20:22:33 +00:00
|
|
|
|
#endif
|
2019-11-09 21:45:10 +00:00
|
|
|
|
|
2019-12-09 22:46:03 +00:00
|
|
|
|
GameInfo = ModList.GameType.MetaData();
|
2019-11-09 20:40:25 +00:00
|
|
|
|
}
|
|
|
|
|
|
2019-12-04 01:26:26 +00:00
|
|
|
|
protected override async Task<bool> _Begin(CancellationToken cancel)
|
2019-11-09 21:45:10 +00:00
|
|
|
|
{
|
2019-12-03 21:56:18 +00:00
|
|
|
|
if (cancel.IsCancellationRequested) return false;
|
2020-01-18 19:28:24 +00:00
|
|
|
|
var metric = Metrics.Send(Metrics.BeginInstall, ModList.Name);
|
2020-01-07 13:03:46 +00:00
|
|
|
|
var result = await Utils.Log(new YesNoIntervention(
|
2019-12-03 12:20:16 +00:00
|
|
|
|
"Vortex Support is still experimental and may produce unexpected results. " +
|
2020-01-13 21:11:07 +00:00
|
|
|
|
"If anything fails please go to the special Vortex support channels on the Wabbajack Discord and contact @erri120#2285 " +
|
2020-01-07 13:03:46 +00:00
|
|
|
|
"for support.", "Continue with experimental feature?")).Task;
|
|
|
|
|
if (result == ConfirmationIntervention.Choice.Abort)
|
|
|
|
|
{
|
|
|
|
|
Utils.Log("Exiting at request of user");
|
|
|
|
|
return false;
|
|
|
|
|
}
|
2019-12-03 12:20:16 +00:00
|
|
|
|
|
2019-12-03 21:56:18 +00:00
|
|
|
|
if (cancel.IsCancellationRequested) return false;
|
2020-04-04 18:26:14 +00:00
|
|
|
|
Queue.SetActiveThreadsObservable(ConstructDynamicNumThreads(await RecommendQueueSize()));
|
2020-03-25 22:30:43 +00:00
|
|
|
|
DownloadFolder.CreateDirectory();
|
2019-11-09 21:45:10 +00:00
|
|
|
|
|
2019-12-03 21:56:18 +00:00
|
|
|
|
if (cancel.IsCancellationRequested) return false;
|
2019-12-17 16:25:15 +00:00
|
|
|
|
UpdateTracker.NextStep("Hashing Archives");
|
2019-12-04 01:26:26 +00:00
|
|
|
|
await HashArchives();
|
2019-12-03 21:56:18 +00:00
|
|
|
|
|
|
|
|
|
if (cancel.IsCancellationRequested) return false;
|
2019-12-17 16:25:15 +00:00
|
|
|
|
UpdateTracker.NextStep("Downloading Missing Archives");
|
2019-12-04 01:26:26 +00:00
|
|
|
|
await DownloadArchives();
|
2019-12-03 21:56:18 +00:00
|
|
|
|
|
|
|
|
|
if (cancel.IsCancellationRequested) return false;
|
2019-12-17 16:25:15 +00:00
|
|
|
|
UpdateTracker.NextStep("Hashing Remaining Archives");
|
2019-12-04 01:26:26 +00:00
|
|
|
|
await HashArchives();
|
2019-11-09 21:45:10 +00:00
|
|
|
|
|
2019-12-03 21:56:18 +00:00
|
|
|
|
if (cancel.IsCancellationRequested) return false;
|
2019-11-09 21:45:10 +00:00
|
|
|
|
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");
|
|
|
|
|
}
|
|
|
|
|
|
2019-12-17 16:25:15 +00:00
|
|
|
|
if (cancel.IsCancellationRequested) return false;
|
|
|
|
|
UpdateTracker.NextStep("Priming VFS");
|
2019-12-07 02:54:27 +00:00
|
|
|
|
await PrimeVFS();
|
2019-11-09 21:45:10 +00:00
|
|
|
|
|
2019-12-03 21:56:18 +00:00
|
|
|
|
if (cancel.IsCancellationRequested) return false;
|
2019-12-17 16:25:15 +00:00
|
|
|
|
UpdateTracker.NextStep("Building Folder Structure");
|
2019-11-09 21:45:10 +00:00
|
|
|
|
BuildFolderStructure();
|
2019-12-03 21:56:18 +00:00
|
|
|
|
|
|
|
|
|
if (cancel.IsCancellationRequested) return false;
|
2019-12-17 16:25:15 +00:00
|
|
|
|
UpdateTracker.NextStep("Installing Archives");
|
2019-12-04 01:26:26 +00:00
|
|
|
|
await InstallArchives();
|
2019-12-03 21:56:18 +00:00
|
|
|
|
|
|
|
|
|
if (cancel.IsCancellationRequested) return false;
|
2019-12-17 16:25:15 +00:00
|
|
|
|
UpdateTracker.NextStep("Installing Included files");
|
2019-12-04 01:26:26 +00:00
|
|
|
|
await InstallIncludedFiles();
|
2019-12-03 21:56:18 +00:00
|
|
|
|
|
2019-12-14 15:08:22 +00:00
|
|
|
|
if (cancel.IsCancellationRequested) return false;
|
2019-12-17 16:25:15 +00:00
|
|
|
|
UpdateTracker.NextStep("Installing Manual files");
|
2019-12-14 15:08:22 +00:00
|
|
|
|
await InstallManualGameFiles();
|
2019-12-08 06:55:27 +00:00
|
|
|
|
|
2019-12-03 21:56:18 +00:00
|
|
|
|
if (cancel.IsCancellationRequested) return false;
|
2019-12-17 16:25:15 +00:00
|
|
|
|
UpdateTracker.NextStep("Installing SteamWorkshopItems");
|
2019-12-04 01:26:26 +00:00
|
|
|
|
await InstallSteamWorkshopItems();
|
2019-11-09 21:45:10 +00:00
|
|
|
|
|
2019-12-15 04:33:48 +00:00
|
|
|
|
//InstallIncludedDownloadMetas();
|
2020-01-18 19:28:24 +00:00
|
|
|
|
var metric2 = Metrics.Send(Metrics.FinishInstall, ModList.Name);
|
2019-12-17 16:25:15 +00:00
|
|
|
|
UpdateTracker.NextStep("Installation complete! You may exit the program.");
|
2019-11-17 23:48:32 +00:00
|
|
|
|
return true;
|
2019-11-09 21:45:10 +00:00
|
|
|
|
}
|
|
|
|
|
|
2019-12-08 06:55:27 +00:00
|
|
|
|
private async Task InstallManualGameFiles()
|
2019-12-07 16:35:57 +00:00
|
|
|
|
{
|
|
|
|
|
if (!ModList.Directives.Any(d => d.To.StartsWith(Consts.ManualGameFilesDir)))
|
|
|
|
|
return;
|
|
|
|
|
|
2020-01-07 13:03:46 +00:00
|
|
|
|
var result = await Utils.Log(new YesNoIntervention("Some mods from this ModList must be installed directly into " +
|
2019-12-07 16:35:57 +00:00
|
|
|
|
"the game folder. Do you want to do this manually or do you want Wabbajack " +
|
2020-01-13 21:11:07 +00:00
|
|
|
|
"to do this for you?", "Move mods into game folder?")).Task;
|
2019-12-07 16:35:57 +00:00
|
|
|
|
|
2020-01-07 13:03:46 +00:00
|
|
|
|
if (result != ConfirmationIntervention.Choice.Continue)
|
2019-12-07 16:35:57 +00:00
|
|
|
|
return;
|
|
|
|
|
|
2020-03-25 22:30:43 +00:00
|
|
|
|
var manualFilesDir = OutputFolder.Combine(Consts.ManualGameFilesDir);
|
2019-12-07 16:35:57 +00:00
|
|
|
|
|
2020-04-12 19:35:17 +00:00
|
|
|
|
var gameFolder = GameInfo.TryGetGameLocation();
|
2019-12-07 16:35:57 +00:00
|
|
|
|
|
|
|
|
|
Info($"Copying files from {manualFilesDir} " +
|
|
|
|
|
$"to the game folder at {gameFolder}");
|
|
|
|
|
|
2020-03-25 22:30:43 +00:00
|
|
|
|
if (!manualFilesDir.Exists)
|
2019-12-07 16:35:57 +00:00
|
|
|
|
{
|
|
|
|
|
Info($"{manualFilesDir} does not exist!");
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2020-03-25 22:30:43 +00:00
|
|
|
|
/* TODO FIX THIS
|
|
|
|
|
await manualFilesDir.EnumerateFiles().PMap(Queue, dir =>
|
2019-12-07 16:35:57 +00:00
|
|
|
|
{
|
|
|
|
|
dirInfo.GetDirectories("*", SearchOption.AllDirectories).Do(d =>
|
|
|
|
|
{
|
2019-12-18 14:25:43 +00:00
|
|
|
|
var destPath = d.FullName.Replace(manualFilesDir, gameFolder);
|
2019-12-07 16:35:57 +00:00
|
|
|
|
Status($"Creating directory {destPath}");
|
|
|
|
|
Directory.CreateDirectory(destPath);
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
dirInfo.GetFiles("*", SearchOption.AllDirectories).Do(f =>
|
|
|
|
|
{
|
2019-12-18 14:25:43 +00:00
|
|
|
|
var destPath = f.FullName.Replace(manualFilesDir, gameFolder);
|
2019-12-07 16:35:57 +00:00
|
|
|
|
Status($"Copying file {f.FullName} to {destPath}");
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
File.Copy(f.FullName, destPath);
|
|
|
|
|
}
|
|
|
|
|
catch (Exception)
|
|
|
|
|
{
|
|
|
|
|
Info($"Could not copy file {f.FullName} to {destPath}. The file may already exist, skipping...");
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
});
|
2020-03-25 22:30:43 +00:00
|
|
|
|
|
|
|
|
|
*/
|
2019-12-07 16:35:57 +00:00
|
|
|
|
}
|
|
|
|
|
|
2019-12-04 01:26:26 +00:00
|
|
|
|
private async Task InstallSteamWorkshopItems()
|
2019-12-03 12:13:37 +00:00
|
|
|
|
{
|
|
|
|
|
//var currentLib = "";
|
|
|
|
|
SteamGame currentSteamGame = null;
|
2020-01-03 17:22:50 +00:00
|
|
|
|
StoreHandler.Instance.SteamHandler.Games.Where(g => g.Game == GameInfo.Game).Do(s => currentSteamGame = (SteamGame)s);
|
2019-12-03 12:13:37 +00:00
|
|
|
|
/*SteamHandler.Instance.InstallFolders.Where(f => f.Contains(currentSteamGame.InstallDir)).Do(s => currentLib = s);
|
|
|
|
|
|
|
|
|
|
var downloadFolder = Path.Combine(currentLib, "workshop", "downloads", currentSteamGame.AppId.ToString());
|
|
|
|
|
var contentFolder = Path.Combine(currentLib, "workshop", "content", currentSteamGame.AppId.ToString());
|
|
|
|
|
*/
|
|
|
|
|
if (!ModList.Directives.Any(s => s is SteamMeta))
|
|
|
|
|
return;
|
|
|
|
|
|
2020-01-07 13:03:46 +00:00
|
|
|
|
var result = await Utils.Log(new YesNoIntervention(
|
2020-01-13 21:11:07 +00:00
|
|
|
|
"The ModList you are installing requires Steam Workshop items to exist. " +
|
2020-01-07 13:03:46 +00:00
|
|
|
|
"You can check the Workshop Items in the manifest of this ModList. Wabbajack can start Steam for you " +
|
|
|
|
|
"and download the Items automatically. Do you want to proceed with this step?",
|
|
|
|
|
"Download Steam Workshop Items?")).Task;
|
2019-12-03 12:13:37 +00:00
|
|
|
|
|
2020-01-07 13:03:46 +00:00
|
|
|
|
if (result != ConfirmationIntervention.Choice.Continue)
|
2019-12-03 12:13:37 +00:00
|
|
|
|
return;
|
|
|
|
|
|
2019-12-04 01:26:26 +00:00
|
|
|
|
await ModList.Directives.OfType<SteamMeta>()
|
2020-03-25 22:30:43 +00:00
|
|
|
|
.PMap(Queue, async item =>
|
2019-12-03 12:13:37 +00:00
|
|
|
|
{
|
|
|
|
|
Status("Extracting Steam meta file to temp folder");
|
2020-03-25 22:30:43 +00:00
|
|
|
|
var path = DownloadFolder.Combine($"steamWorkshopItem_{item.ItemID}.meta");
|
|
|
|
|
if (!path.Exists)
|
|
|
|
|
await path.WriteAllBytesAsync(await LoadBytesFromPath(item.SourceDataID));
|
2019-12-03 12:13:37 +00:00
|
|
|
|
|
|
|
|
|
Status("Downloading Steam Workshop Item through steam cmd");
|
|
|
|
|
|
|
|
|
|
var p = new Process
|
|
|
|
|
{
|
|
|
|
|
StartInfo = new ProcessStartInfo
|
|
|
|
|
{
|
2020-01-03 17:22:50 +00:00
|
|
|
|
FileName = Path.Combine(StoreHandler.Instance.SteamHandler.SteamPath, "steam.exe"),
|
2019-12-03 12:13:37 +00:00
|
|
|
|
CreateNoWindow = true,
|
2020-01-03 17:22:50 +00:00
|
|
|
|
Arguments = $"console +workshop_download_item {currentSteamGame.ID} {currentSteamGame.ID}"
|
2019-12-03 12:13:37 +00:00
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
p.Start();
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
2019-12-04 01:26:26 +00:00
|
|
|
|
private async Task InstallIncludedFiles()
|
2019-11-09 21:45:10 +00:00
|
|
|
|
{
|
|
|
|
|
Info("Writing inline files");
|
2019-12-04 01:26:26 +00:00
|
|
|
|
await ModList.Directives.OfType<InlineFile>()
|
2020-03-25 22:30:43 +00:00
|
|
|
|
.PMap(Queue,async directive =>
|
2019-11-09 21:45:10 +00:00
|
|
|
|
{
|
2020-03-25 22:30:43 +00:00
|
|
|
|
if (directive.To.Extension == Consts.MetaFileExtension)
|
2019-12-04 10:42:58 +00:00
|
|
|
|
return;
|
|
|
|
|
|
2019-12-18 14:25:43 +00:00
|
|
|
|
Info($"Writing included file {directive.To}");
|
2020-03-25 22:30:43 +00:00
|
|
|
|
var outPath = OutputFolder.Combine(directive.To);
|
|
|
|
|
outPath.Delete();
|
|
|
|
|
await outPath.WriteAllBytesAsync(await LoadBytesFromPath(directive.SourceDataID));
|
2019-11-09 21:45:10 +00:00
|
|
|
|
});
|
|
|
|
|
}
|
2019-11-09 20:40:25 +00:00
|
|
|
|
}
|
|
|
|
|
}
|