using System; using System.Collections.Generic; using System.IO; using System.Linq; using System.Threading; using System.Threading.Tasks; using Microsoft.Extensions.Logging; using Wabbajack.Common; using Wabbajack.FileExtractor.ExtractedFiles; using Wabbajack.Hashing.xxHash64; using Wabbajack.Paths; using Wabbajack.Paths.IO; namespace Wabbajack.VFS { public class Context { public const ulong FileVersion = 0x03; public const string Magic = "WABBAJACK VFS FILE"; private readonly ParallelOptions _parallelOptions; private readonly TemporaryFileManager _manager; public readonly FileExtractor.FileExtractor Extractor; public readonly FileHashCache HashCache; public readonly ILogger Logger; public readonly VFSCache VfsCache; public Context(ILogger logger, ParallelOptions parallelOptions, TemporaryFileManager manager, VFSCache vfsCache, FileHashCache hashCache, FileExtractor.FileExtractor extractor) { Logger = logger; _manager = manager; Extractor = extractor; VfsCache = vfsCache; HashCache = hashCache; _parallelOptions = parallelOptions; } public IndexRoot Index { get; private set; } = IndexRoot.Empty; public async Task AddRoot(AbsolutePath root, CancellationToken token) { var filtered = Index.AllFiles.Where(file => file.IsNative && ((AbsolutePath)file.Name).FileExists()) .ToList(); var byPath = filtered.ToDictionary(f => f.Name); var filesToIndex = root.EnumerateFiles().Distinct().ToList(); var allFiles = await filesToIndex .PMap(_parallelOptions, async f => { if (byPath.TryGetValue(f, out var found)) if (found.LastModified == f.LastModifiedUtc().AsUnixTime() && found.Size == f.Size()) return found; return await VirtualFile.Analyze(this, null, new NativeFileStreamFactory(f), f, token); }).ToList(); var newIndex = await IndexRoot.Empty.Integrate(filtered.Concat(allFiles).ToList()); lock (this) { Index = newIndex; } return newIndex; } public async Task AddRoots(List roots, CancellationToken token) { var native = Index.AllFiles.Where(file => file.IsNative).ToDictionary(file => file.FullPath.Base); var filtered = Index.AllFiles.Where(file => ((AbsolutePath)file.Name).FileExists()).ToList(); var filesToIndex = roots.SelectMany(root => root.EnumerateFiles()).ToList(); var allFiles = await filesToIndex .PMap(_parallelOptions, async f => { if (native.TryGetValue(f, out var found)) if (found.LastModified == f.LastModifiedUtc().AsUnixTime() && found.Size == f.Size()) return found; return await VirtualFile.Analyze(this, null, new NativeFileStreamFactory(f), f, token); }).ToList(); var newIndex = await IndexRoot.Empty.Integrate(filtered.Concat(allFiles).ToList()); lock (this) { Index = newIndex; } return newIndex; } /// /// Extracts a file /// /// Work queue to use when required by some formats /// Predefined list of files to extract, all others will be skipped /// Func called for each file extracted /// Optional: folder to use for temporary storage /// Optional: Status update tracker /// /// public async Task Extract(HashSet files, Func callback, CancellationToken token, AbsolutePath? tempFolder = null) { var top = new VirtualFile(); var filesByParent = files.SelectMany(f => f.FilesInFullPath) .Distinct() .GroupBy(f => f.Parent ?? top) .ToDictionary(f => f.Key); async Task HandleFile(VirtualFile file, IExtractedFile sfn) { if (filesByParent.ContainsKey(file)) sfn.CanMove = false; if (files.Contains(file)) await callback(file, sfn); if (filesByParent.TryGetValue(file, out var children)) { var fileNames = children.ToDictionary(c => c.RelativeName); try { await Extractor.GatheringExtract(sfn, r => fileNames.ContainsKey(r), async (rel, csf) => { await HandleFile(fileNames[rel], csf); return 0; }, token, fileNames.Keys.ToHashSet()); } catch (Exception) { await using var stream = await sfn.GetStream(); var hash = await stream.HashingCopy(Stream.Null, token); if (hash != file.Hash) throw new Exception( $"File {file.FullPath} is corrupt, please delete it and retry the installation"); throw; } } } await filesByParent[top].PDo(_parallelOptions, async file => await HandleFile(file, new ExtractedNativeFile(file.AbsoluteName) { CanMove = false })); } private class Box { public T Value { get; set; } } #region KnownFiles private List _knownFiles = new(); private readonly Dictionary _knownArchives = new(); public void AddKnown(IEnumerable known, Dictionary archives) { _knownFiles.AddRange(known); foreach (var (key, value) in archives) _knownArchives.TryAdd(key, value); } public async Task BackfillMissing() { var newFiles = _knownArchives.ToDictionary(kv => kv.Key, kv => new VirtualFile { Name = kv.Value, Size = kv.Value.Size(), Hash = kv.Key }); foreach (var f in newFiles.Values) f.FillFullPath(0); var parentchild = new Dictionary<(VirtualFile, RelativePath), VirtualFile>(); void BackFillOne(HashRelativePath file) { var parent = newFiles[file.Hash]; foreach (var path in file.Parts) { if (parentchild.TryGetValue((parent, path), out var foundParent)) { parent = foundParent; continue; } var nf = new VirtualFile { Name = path, Parent = parent }; nf.FillFullPath(); parent.Children = parent.Children.Add(nf); parentchild.Add((parent, path), nf); parent = nf; } } _knownFiles.Where(f => f.Parts.Length > 0).Do(BackFillOne); var newIndex = await Index.Integrate(newFiles.Values.ToList()); lock (this) { Index = newIndex; } _knownFiles = new List(); } #endregion } }