wabbajack/Wabbajack.Compiler/CompilationSteps/DeconstructBSAs.cs

116 lines
4.5 KiB
C#
Raw Normal View History

using System;
using System.Collections.Generic;
using System.Linq;
using System.Threading.Tasks;
using Wabbajack.Common;
2021-09-27 12:42:46 +00:00
using Wabbajack.Compression.BSA;
using Wabbajack.DTOs;
using Wabbajack.DTOs.BSA.ArchiveStates;
using Wabbajack.DTOs.Directives;
using Wabbajack.Paths;
using Wabbajack.Paths.IO;
using Wabbajack.VFS;
namespace Wabbajack.Compiler.CompilationSteps
{
public class DeconstructBSAs : ACompilationStep
{
2020-04-27 21:32:19 +00:00
private readonly IEnumerable<RelativePath> _includeDirectly;
2020-02-05 05:17:12 +00:00
private readonly Func<VirtualFile, List<ICompilationStep>> _microstack;
private readonly Func<VirtualFile, List<ICompilationStep>> _microstackWithInclude;
private readonly MO2Compiler _mo2Compiler;
public DeconstructBSAs(ACompiler compiler) : base(compiler)
{
_mo2Compiler = (MO2Compiler)compiler;
2019-11-21 15:51:57 +00:00
_includeDirectly = _mo2Compiler.ModInis.Where(kv =>
{
2021-09-27 12:42:46 +00:00
var general = kv.Value["General"];
if (general["notes"] != null && (general["notes"].Contains(Consts.WABBAJACK_INCLUDE) || general["notes"].Contains(Consts.WABBAJACK_NOMATCH_INCLUDE))) return true;
if (general["comments"] != null && (general["comments"].Contains(Consts.WABBAJACK_INCLUDE) || general["comments"].Contains(Consts.WABBAJACK_NOMATCH_INCLUDE))) return true;
return false;
})
2021-09-27 12:42:46 +00:00
.Select(kv => kv.Key.RelativeTo(_mo2Compiler._settings.Source))
.ToList();
2020-02-05 05:17:12 +00:00
_microstack = bsa => new List<ICompilationStep>
{
new DirectMatch(_mo2Compiler),
new MatchSimilarTextures(_mo2Compiler),
2020-02-05 05:17:12 +00:00
new IncludePatches(_mo2Compiler, bsa),
new DropAll(_mo2Compiler)
};
2020-02-05 05:17:12 +00:00
_microstackWithInclude = bsa => new List<ICompilationStep>
{
new DirectMatch(_mo2Compiler),
new MatchSimilarTextures(_mo2Compiler),
2020-02-05 05:17:12 +00:00
new IncludePatches(_mo2Compiler, bsa),
new IncludeAll(_mo2Compiler)
};
}
public override async ValueTask<Directive?> Run(RawSourceFile source)
{
2020-03-25 22:30:43 +00:00
if (!Consts.SupportedBSAs.Contains(source.Path.Extension)) return null;
var defaultInclude = false;
2021-09-27 12:42:46 +00:00
if (source.Path.RelativeTo(_mo2Compiler._settings.Source).InFolder(_mo2Compiler._settings.Source.Combine(Consts.MO2ModFolderName)))
if (_includeDirectly.Any(path => source.Path.InFolder(path)))
defaultInclude = true;
2021-09-27 12:42:46 +00:00
if (source.AbsolutePath.Size() >= (long)2 << 31)
{
2021-09-27 12:42:46 +00:00
var bsaTest = await BSADispatch.Open(source.AbsolutePath);
if (bsaTest.State is BSAState)
{
2021-09-27 12:42:46 +00:00
throw new CompilerException(
$"BSA {source.AbsolutePath.FileName} is over 2GB in size, very few programs (Including Wabbajack) can create BSA files this large without causing CTD issues." +
$"Please re-compress this BSA into a more manageable size.");
}
}
2019-11-21 15:51:57 +00:00
var sourceFiles = source.File.Children;
2020-02-05 05:17:12 +00:00
var stack = defaultInclude ? _microstackWithInclude(source.File) : _microstack(source.File);
var id = Guid.NewGuid().ToString();
2020-04-27 21:32:19 +00:00
Func<Task>? _cleanup = null;
if (defaultInclude)
{
2020-09-05 19:36:44 +00:00
//_cleanup = await source.File.Context.Stage(source.File.Children);
2020-04-27 21:32:19 +00:00
}
2021-10-21 03:18:15 +00:00
var matches = await sourceFiles.PMapAll(_compiler.CompilerLimiter, e => _mo2Compiler.RunStack(stack, new RawSourceFile(e, Consts.BSACreationDir.Combine((RelativePath)id, (RelativePath)e.Name))))
2021-09-27 12:42:46 +00:00
.ToList();
foreach (var match in matches)
{
if (match is IgnoredDirectly)
2021-09-27 12:42:46 +00:00
{
throw new CompilerException($"File required for BSA {source.Path} creation doesn't exist: {match.To}");
}
_mo2Compiler.ExtraFiles.Add(match);
}
2021-09-27 12:42:46 +00:00
var bsa = await BSADispatch.Open(source.AbsolutePath);
var directive = new CreateBSA
{
2021-09-27 12:42:46 +00:00
State = bsa.State,
FileStates = bsa.Files.Select(f => f.State).ToArray(),
To = source.Path,
2021-09-27 12:42:46 +00:00
Hash = source.Hash,
TempID = (RelativePath)id,
};
2020-04-27 21:32:19 +00:00
if (_cleanup != null)
await _cleanup();
return directive;
}
}
2019-12-04 04:12:08 +00:00
}