2022-05-25 02:59:15 +00:00
|
|
|
using System;
|
|
|
|
using System.Linq;
|
|
|
|
using System.Threading.Tasks;
|
|
|
|
using Microsoft.Extensions.Logging;
|
|
|
|
using Wabbajack.Common;
|
|
|
|
using Wabbajack.DTOs;
|
|
|
|
using Wabbajack.Installer;
|
|
|
|
using Wabbajack.Paths;
|
|
|
|
using Wabbajack.Paths.IO;
|
|
|
|
|
|
|
|
namespace Wabbajack.Compiler;
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// Given a modlist.txt file, infer as much of CompilerSettings as possible
|
|
|
|
/// </summary>
|
|
|
|
public class CompilerSettingsInferencer
|
|
|
|
{
|
|
|
|
private readonly ILogger<CompilerSettingsInferencer> _logger;
|
|
|
|
|
|
|
|
public CompilerSettingsInferencer(ILogger<CompilerSettingsInferencer> logger)
|
|
|
|
{
|
|
|
|
_logger = logger;
|
|
|
|
|
|
|
|
}
|
2022-05-27 05:41:11 +00:00
|
|
|
|
|
|
|
public async Task<CompilerSettings?> InferFromRootPath(AbsolutePath rootPath)
|
|
|
|
{
|
|
|
|
var mo2File = rootPath.Combine(Consts.MO2IniName).LoadIniFile();
|
2022-05-31 23:14:27 +00:00
|
|
|
var profile = mo2File["General"]["selected_profile"].FromMO2Ini();
|
2022-05-27 05:41:11 +00:00
|
|
|
|
|
|
|
return await InferModListFromLocation(rootPath.Combine(Consts.MO2Profiles, profile, Consts.ModListTxt));
|
|
|
|
}
|
2022-05-25 02:59:15 +00:00
|
|
|
|
|
|
|
public async Task<CompilerSettings?> InferModListFromLocation(AbsolutePath settingsFile)
|
|
|
|
{
|
|
|
|
var cs = new CompilerSettings();
|
|
|
|
|
|
|
|
if (settingsFile.FileName == "modlist.txt".ToRelativePath() && settingsFile.Depth > 3)
|
|
|
|
{
|
|
|
|
_logger.LogInformation("Inferencing basic settings");
|
|
|
|
var mo2Folder = settingsFile.Parent.Parent.Parent;
|
|
|
|
var mo2Ini = mo2Folder.Combine(Consts.MO2IniName);
|
|
|
|
if (mo2Ini.FileExists())
|
|
|
|
{
|
|
|
|
var iniData = mo2Ini.LoadIniFile();
|
|
|
|
|
|
|
|
var general = iniData["General"];
|
|
|
|
|
|
|
|
cs.Game = GameRegistry.GetByFuzzyName(general["gameName"].FromMO2Ini()).Game;
|
|
|
|
cs.Source = mo2Folder;
|
|
|
|
|
|
|
|
var selectedProfile = general["selected_profile"].FromMO2Ini();
|
|
|
|
//cs.GamePath = general["gamePath"].FromMO2Ini().ToAbsolutePath();
|
|
|
|
cs.ModListName = selectedProfile;
|
2022-05-30 20:50:24 +00:00
|
|
|
cs.Profile = selectedProfile;
|
|
|
|
|
2022-08-20 14:26:54 +00:00
|
|
|
cs.OutputFile = cs.Source.Parent;
|
2022-05-25 02:59:15 +00:00
|
|
|
|
|
|
|
var settings = iniData["Settings"];
|
|
|
|
cs.Downloads = settings["download_directory"].FromMO2Ini().ToAbsolutePath();
|
|
|
|
|
|
|
|
if (cs.Downloads == default)
|
|
|
|
cs.Downloads = cs.Source.Combine("downloads");
|
|
|
|
|
2022-05-26 05:21:12 +00:00
|
|
|
cs.NoMatchInclude = Array.Empty<RelativePath>();
|
2022-08-09 11:54:21 +00:00
|
|
|
cs.Include = Array.Empty<RelativePath>();
|
2022-05-26 05:21:12 +00:00
|
|
|
foreach (var file in mo2Folder.EnumerateFiles())
|
|
|
|
{
|
2022-09-20 02:56:03 +00:00
|
|
|
if (MatchesVariants(file, Consts.WABBAJACK_INCLUDE))
|
2022-08-09 11:54:21 +00:00
|
|
|
cs.Include = cs.Include.Add(file.Parent.RelativeTo(mo2Folder));
|
2022-08-18 23:02:19 +00:00
|
|
|
|
2022-09-20 02:56:03 +00:00
|
|
|
if (MatchesVariants(file, Consts.WABBAJACK_NOMATCH_INCLUDE))
|
2022-08-18 23:02:19 +00:00
|
|
|
cs.NoMatchInclude = cs.NoMatchInclude.Add(file.Parent.RelativeTo(mo2Folder));
|
2022-11-05 22:36:26 +00:00
|
|
|
if (MatchesVariants(file, Consts.WABBAJACK_NOMATCH_INCLUDE, true))
|
|
|
|
{
|
|
|
|
cs.NoMatchInclude = cs.NoMatchInclude.Add(file.RelativeTo(mo2Folder));
|
|
|
|
var taggedFiles = await file.ReadAllLinesAsync().ToArray();
|
|
|
|
foreach (var taggedFile in taggedFiles)
|
|
|
|
{
|
|
|
|
var noMatchIncludeAsset =
|
|
|
|
(AbsolutePath) file.ToString().Replace(file.FileName.ToString(), taggedFile);
|
|
|
|
if (noMatchIncludeAsset.DirectoryExists())
|
|
|
|
cs.NoMatchInclude = cs.NoMatchInclude.Add(noMatchIncludeAsset
|
|
|
|
.RelativeTo(mo2Folder));
|
|
|
|
if (noMatchIncludeAsset.FileExists())
|
|
|
|
cs.Include = cs.Include.Add(noMatchIncludeAsset
|
|
|
|
.RelativeTo(mo2Folder));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-09-20 02:56:03 +00:00
|
|
|
if (MatchesVariants(file, Consts.WABBAJACK_IGNORE))
|
2022-08-23 22:38:47 +00:00
|
|
|
cs.Ignore = cs.Ignore.Add(file.Parent.RelativeTo(mo2Folder));
|
2022-11-05 22:36:26 +00:00
|
|
|
if (MatchesVariants(file, Consts.WABBAJACK_IGNORE, true))
|
|
|
|
{
|
|
|
|
cs.Ignore = cs.Ignore.Add(file.RelativeTo(mo2Folder));
|
|
|
|
var taggedFiles = await file.ReadAllLinesAsync().ToArray();
|
|
|
|
foreach (var taggedFile in taggedFiles)
|
|
|
|
{
|
|
|
|
cs.Ignore = cs.Ignore.Add(
|
|
|
|
((AbsolutePath) file.ToString().Replace(file.FileName.ToString(), taggedFile))
|
|
|
|
.RelativeTo(mo2Folder));
|
|
|
|
}
|
|
|
|
}
|
2022-05-26 05:21:12 +00:00
|
|
|
}
|
|
|
|
|
2022-05-25 02:59:15 +00:00
|
|
|
_logger.LogInformation("Finding Always Enabled mods");
|
|
|
|
cs.AlwaysEnabled = Array.Empty<RelativePath>();
|
2022-05-30 21:02:54 +00:00
|
|
|
|
|
|
|
// Find mod tags
|
2022-05-25 02:59:15 +00:00
|
|
|
foreach (var modFolder in mo2Folder.Combine("mods").EnumerateDirectories())
|
|
|
|
{
|
|
|
|
var iniFile = modFolder.Combine("meta.ini");
|
|
|
|
if (!iniFile.FileExists()) continue;
|
|
|
|
|
|
|
|
var data = iniFile.LoadIniFile();
|
|
|
|
var generalModData = data["General"];
|
2022-05-30 21:02:54 +00:00
|
|
|
if ((generalModData["notes"]?.Contains(Consts.WABBAJACK_ALWAYS_ENABLE) ?? false) ||
|
|
|
|
(generalModData["comments"]?.Contains(Consts.WABBAJACK_ALWAYS_ENABLE) ?? false))
|
2022-05-25 02:59:15 +00:00
|
|
|
cs.AlwaysEnabled = cs.AlwaysEnabled.Append(modFolder.RelativeTo(mo2Folder)).ToArray();
|
2022-05-30 21:02:54 +00:00
|
|
|
|
|
|
|
if ((generalModData["notes"]?.Contains(Consts.WABBAJACK_NOMATCH_INCLUDE) ?? false) ||
|
|
|
|
(generalModData["comments"]?.Contains(Consts.WABBAJACK_NOMATCH_INCLUDE) ?? false))
|
|
|
|
cs.NoMatchInclude = cs.NoMatchInclude.Append(modFolder.RelativeTo(mo2Folder)).ToArray();
|
2022-05-31 23:14:27 +00:00
|
|
|
|
|
|
|
if ((generalModData["notes"]?.Contains(Consts.WABBAJACK_INCLUDE) ?? false) ||
|
|
|
|
(generalModData["comments"]?.Contains(Consts.WABBAJACK_INCLUDE) ?? false))
|
|
|
|
cs.Include = cs.Include.Append(modFolder.RelativeTo(mo2Folder)).ToArray();
|
2022-08-23 22:38:47 +00:00
|
|
|
|
|
|
|
if ((generalModData["notes"]?.Contains(Consts.WABBAJACK_IGNORE) ?? false) ||
|
|
|
|
(generalModData["comments"]?.Contains(Consts.WABBAJACK_IGNORE) ?? false))
|
|
|
|
cs.Ignore = cs.Ignore.Append(modFolder.RelativeTo(mo2Folder)).ToArray();
|
2022-05-25 02:59:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
_logger.LogInformation("Finding other profiles");
|
|
|
|
var otherProfilesFile = settingsFile.Parent.Combine("otherprofiles.txt");
|
|
|
|
if (otherProfilesFile.FileExists())
|
|
|
|
{
|
2022-05-30 20:50:24 +00:00
|
|
|
cs.AdditionalProfiles = await otherProfilesFile.ReadAllLinesAsync().ToArray();
|
2022-05-25 02:59:15 +00:00
|
|
|
}
|
2022-05-29 04:19:25 +00:00
|
|
|
|
|
|
|
cs.OutputFile = cs.Source.Parent.Combine(cs.Profile).WithExtension(Ext.Wabbajack);
|
2022-05-25 02:59:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return cs;
|
|
|
|
}
|
|
|
|
|
|
|
|
return null;
|
|
|
|
}
|
2022-09-20 02:56:03 +00:00
|
|
|
|
2022-11-05 22:36:26 +00:00
|
|
|
private static bool MatchesVariants(AbsolutePath file, string baseVariant, bool txtMode = false)
|
2022-09-20 02:56:03 +00:00
|
|
|
{
|
2022-11-05 22:36:26 +00:00
|
|
|
var fileNameString = file.FileName.ToString();
|
|
|
|
if (!txtMode)
|
|
|
|
{
|
|
|
|
return fileNameString == baseVariant;
|
|
|
|
}
|
|
|
|
return fileNameString == baseVariant + "_FILES.txt" || fileNameString == baseVariant + "_FILES.TXT";
|
2022-09-20 02:56:03 +00:00
|
|
|
}
|
2022-05-25 02:59:15 +00:00
|
|
|
}
|