wabbajack/Wabbajack.Compiler/CompilerSettingsInferencer.cs

159 lines
7.2 KiB
C#
Raw Normal View History

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();
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));
}
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;
cs.Profile = selectedProfile;
2022-08-20 14:26:54 +00:00
cs.OutputFile = cs.Source.Parent;
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));
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));
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
}
_logger.LogInformation("Finding Always Enabled mods");
cs.AlwaysEnabled = Array.Empty<RelativePath>();
// Find mod tags
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"];
if ((generalModData["notes"]?.Contains(Consts.WABBAJACK_ALWAYS_ENABLE) ?? false) ||
(generalModData["comments"]?.Contains(Consts.WABBAJACK_ALWAYS_ENABLE) ?? false))
cs.AlwaysEnabled = cs.AlwaysEnabled.Append(modFolder.RelativeTo(mo2Folder)).ToArray();
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();
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();
}
_logger.LogInformation("Finding other profiles");
var otherProfilesFile = settingsFile.Parent.Combine("otherprofiles.txt");
if (otherProfilesFile.FileExists())
{
cs.AdditionalProfiles = await otherProfilesFile.ReadAllLinesAsync().ToArray();
}
2022-05-29 04:19:25 +00:00
cs.OutputFile = cs.Source.Parent.Combine(cs.Profile).WithExtension(Ext.Wabbajack);
}
return cs;
}
return null;
}
2022-09-20 02:56:03 +00:00
private static bool MatchesVariants(AbsolutePath file, string baseVariant, bool txtMode = false)
2022-09-20 02:56:03 +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
}
}