using System; using System.Collections.Generic; using System.Diagnostics; using System.Linq; using System.Net.Http; using System.Runtime.InteropServices; using System.Threading; using System.Threading.Tasks; using Microsoft.Extensions.DependencyInjection; using Microsoft.Extensions.Logging; using Wabbajack.Compiler; using Wabbajack.Configuration; using Wabbajack.Downloaders; using Wabbajack.Downloaders.GameFile; using Wabbajack.Downloaders.VerificationCache; using Wabbajack.DTOs; using Wabbajack.DTOs.Interventions; using Wabbajack.DTOs.JsonConverters; using Wabbajack.DTOs.Logins; using Wabbajack.Hashing.PHash; using Wabbajack.Installer; using Wabbajack.Networking.BethesdaNet; using Wabbajack.Networking.Discord; using Wabbajack.Networking.Http; using Wabbajack.Networking.Http.Interfaces; using Wabbajack.Networking.NexusApi; using Wabbajack.Networking.Steam; using Wabbajack.Networking.WabbajackClientApi; using Wabbajack.Paths; using Wabbajack.Paths.IO; using Wabbajack.RateLimiter; using Wabbajack.Services.OSIntegrated.Services; using Wabbajack.Services.OSIntegrated.TokenProviders; using Wabbajack.VFS; using Wabbajack.VFS.Interfaces; using Client = Wabbajack.Networking.WabbajackClientApi.Client; namespace Wabbajack.Services.OSIntegrated; public static class ServiceExtensions { /// /// Adds variants of services that integrate into global OS services. These are not testing /// variants or services that require Environment variables. These are the "full fat" services. /// /// public static IServiceCollection AddOSIntegrated(this IServiceCollection service, Action? cfn = null) { // Register app-wide cancellation token source to allow clean termination service.AddSingleton(new CancellationTokenSource()); service.AddTransient(typeof(CancellationToken), s => s.GetRequiredService().Token); var options = new OSIntegratedOptions(); cfn?.Invoke(options); var tempBase = KnownFolders.EntryPoint.Combine("temp"); service.AddTransient(s => new TemporaryFileManager(tempBase.Combine(Environment.ProcessId + "_" + Guid.NewGuid()))); Task.Run(() => CleanAllTempData(tempBase)); service.AddSingleton(s => options.UseLocalCache ? new FileHashCache(s.GetService()!.CreateFile().Path, s.GetService>()!) : new FileHashCache(KnownFolders.AppDataLocal.Combine("Wabbajack", "GlobalHashCache.sqlite"), s.GetService>()!)); service.AddSingleton(s => { var diskCache = options.UseLocalCache ? new VFSDiskCache(s.GetService()!.CreateFile().Path) : new VFSDiskCache(KnownFolders.WabbajackAppLocal.Combine("GlobalVFSCache4.sqlite")); var cesiCache = new CesiVFSCache(s.GetRequiredService>(), s.GetRequiredService()); return new FallthroughVFSCache(new IVfsCache[] {diskCache}); }); service.AddSingleton(s => options.UseLocalCache ? new BinaryPatchCache(s.GetRequiredService>(), s.GetService()!.CreateFolder().Path) : new BinaryPatchCache(s.GetRequiredService>(),KnownFolders.WabbajackAppLocal.Combine("PatchCache"))); service.AddSingleton(s => { var dtos = s.GetRequiredService(); return options.UseLocalCache ? new VerificationCache(s.GetRequiredService>(), s.GetService()!.CreateFile().Path, TimeSpan.FromDays(1), dtos) : new VerificationCache(s.GetRequiredService>(), KnownFolders.WabbajackAppLocal.Combine("VerificationCacheV2.sqlite"), TimeSpan.FromDays(1), dtos); }); service.AddSingleton(new ParallelOptions {MaxDegreeOfParallelism = Environment.ProcessorCount}); MainSettings GetAppSettings(IServiceProvider provider, string name) { var settingsManager = provider.GetService(); var settings = settingsManager!.Load(name).Result; if (settings.Upgrade()) { settingsManager.Save("app_settings", settings).Wait(); } return settings; } Func> GetResourceSettings(IServiceProvider provider, string name) { return async () => { var s = await provider.GetService()!.GetSettings(name); return ((int) s.MaxTasks, s.MaxThroughput); }; } // Settings service.AddSingleton(s => new Configuration { EncryptedDataLocation = KnownFolders.WabbajackAppLocal.Combine("encrypted"), ModListsDownloadLocation = KnownFolders.EntryPoint.Combine("downloaded_mod_lists"), SavedSettingsLocation = KnownFolders.WabbajackAppLocal.Combine("saved_settings"), LogLocation = KnownFolders.LauncherAwarePath.Combine("logs"), ImageCacheLocation = KnownFolders.WabbajackAppLocal.Combine("image_cache") }); service.AddSingleton(); service.AddSingleton(); service.AddSingleton(s => GetAppSettings(s, "app_settings")); // Resources service.AddAllSingleton>(s => new Resource("Downloads", GetResourceSettings(s, "Downloads"), s.GetRequiredService())); service.AddAllSingleton>(s => new Resource("Web Requests", GetResourceSettings(s, "Web Requests"), s.GetRequiredService())); service.AddAllSingleton>(s => new Resource("VFS", GetResourceSettings(s, "VFS"), s.GetRequiredService())); service.AddAllSingleton>(s => new Resource("File Hashing", GetResourceSettings(s, "File Hashing"), s.GetRequiredService())); service.AddAllSingleton>(s => new Resource("Wabbajack Client", GetResourceSettings(s, "Wabbajack Client"), s.GetRequiredService())); service.AddAllSingleton>(s => new Resource("File Extractor", GetResourceSettings(s, "File Extractor"), s.GetRequiredService())); service.AddAllSingleton>(s => new Resource("Compiler", GetResourceSettings(s, "Compiler"), s.GetRequiredService())); service.AddAllSingleton>(s => new Resource("Installer", GetResourceSettings(s, "Installer"), s.GetRequiredService())); service.AddAllSingleton>(s => new Resource("User Intervention", 1, token: s.GetRequiredService())); service.AddSingleton(); service.AddScoped(); service.AddSingleton(); service.AddSingleton(); // Networking service.AddSingleton(); service.AddAllSingleton(); service.AddSteam(); service.AddSingleton(); service.AddSingleton(); service.AddBethesdaNet(); // Token Providers service.AddAllSingleton, EncryptedJsonTokenProvider, NexusApiTokenProvider>(); service.AddAllSingleton, EncryptedJsonTokenProvider, BethesdaNetTokenProvider>(); service .AddAllSingleton, EncryptedJsonTokenProvider, LoversLabTokenProvider>(); service .AddAllSingleton, EncryptedJsonTokenProvider, VectorPlexusTokenProvider>(); service .AddAllSingleton, EncryptedJsonTokenProvider, SteamTokenProvider>(); service.AddAllSingleton, WabbajackApiTokenProvider>(); service .AddAllSingleton>, EncryptedJsonTokenProvider>, DiscordTokenProvider>(); service.AddAllSingleton(); service.AddDownloadDispatcher(); if (options.UseStubbedGameFolders) service.AddAllSingleton(); else service.AddAllSingleton(); // ImageLoader if (RuntimeInformation.IsOSPlatform(OSPlatform.Windows)) service.AddSingleton(); else service.AddSingleton(); // Installer/Compiler Configuration service.AddScoped(); service.AddScoped(); service.AddScoped(); service.AddScoped(); service.AddSingleton(); // Application Info var version = $"{ThisAssembly.Git.SemVer.Major}.{ThisAssembly.Git.SemVer.Major}.{ThisAssembly.Git.SemVer.Patch}{ThisAssembly.Git.SemVer.DashLabel}"; service.AddSingleton(s => new ApplicationInfo { ApplicationSlug = "Wabbajack", ApplicationName = Environment.ProcessPath?.ToAbsolutePath().FileName.ToString() ?? "Wabbajack", ApplicationSha = ThisAssembly.Git.Sha, Platform = RuntimeInformation.ProcessArchitecture.ToString(), OperatingSystemDescription = RuntimeInformation.OSDescription, RuntimeIdentifier = RuntimeInformation.RuntimeIdentifier, OSVersion = Environment.OSVersion.VersionString, Version = version }); return service; } private static void CleanAllTempData(AbsolutePath path) { // Get directories first and cache them, this freezes the directories were looking at // so any new ones don't show up in the middle of our deletes. var dirs = path.EnumerateDirectories().ToList(); var processIds = Process.GetProcesses().Select(p => p.Id).ToHashSet(); foreach (var dir in dirs) { var name = dir.FileName.ToString().Split("_"); if (!int.TryParse(name[0], out var processId)) continue; if (processIds.Contains(processId)) continue; try { dir.DeleteDirectory(); } catch (Exception) { // ignored } } } public class OSIntegratedOptions { public bool UseLocalCache { get; set; } = false; public bool UseStubbedGameFolders { get; set; } = false; } }