using System; using System.Collections.Generic; using System.Linq; using System.Threading.Tasks; using Dapper; using Wabbajack.DTOs; using Wabbajack.DTOs.DownloadStates; using Wabbajack.Hashing.xxHash64; namespace Wabbajack.Server.DataLayer { public partial class SqlService { public async Task IngestModList(Hash hash, ModlistMetadata metadata, ModList modlist, bool brokenDownload) { await using var conn = await Open(); await using var tran = await conn.BeginTransactionAsync(); await conn.ExecuteAsync(@"DELETE FROM dbo.ModLists Where MachineUrl = @MachineUrl", new {MachineUrl = metadata.Links.MachineURL}, tran); var archives = modlist.Archives; var directives = modlist.Directives; modlist.Archives = Array.Empty(); modlist.Directives = Array.Empty(); await conn.ExecuteAsync( @"INSERT INTO dbo.ModLists (MachineUrl, Hash, Metadata, ModList, BrokenDownload) VALUES (@MachineUrl, @Hash, @Metadata, @ModList, @BrokenDownload)", new { MachineUrl = metadata.Links.MachineURL, Hash = hash, MetaData = _dtos.Serialize(metadata), ModList = _dtos.Serialize(modlist), BrokenDownload = brokenDownload }, tran); var entries = archives.Select(a => new { MachineUrl = metadata.Links.MachineURL, Name = a.Name, Hash = a.Hash, Size = a.Size, State = _dtos.Serialize(a.State), PrimaryKeyString = a.State.PrimaryKeyString }).ToArray(); await conn.ExecuteAsync(@"DELETE FROM dbo.ModListArchives WHERE MachineURL = @machineURL", new {MachineUrl = metadata.Links.MachineURL}, tran); foreach (var entry in entries) { await conn.ExecuteAsync( "INSERT INTO dbo.ModListArchives (MachineURL, Name, Hash, Size, PrimaryKeyString, State) VALUES (@MachineURL, @Name, @Hash, @Size, @PrimaryKeyString, @State)", entry, tran); } await tran.CommitAsync(); } public async Task HaveIndexedModlist(string machineUrl, Hash hash) { await using var conn = await Open(); var result = await conn.QueryFirstOrDefaultAsync( "SELECT MachineURL from dbo.Modlists WHERE MachineURL = @MachineUrl AND Hash = @Hash", new {MachineUrl = machineUrl, Hash = hash}); return result != null; } public async Task HashIsInAModlist(Hash hash) { await using var conn = await Open(); var result = await conn.QueryFirstOrDefaultAsync("SELECT Hash FROM dbo.ModListArchives Where Hash = @Hash", new {Hash = hash}); return result; } public async Task> ModListArchives(string machineURL) { await using var conn = await Open(); var archives = await conn.QueryAsync<(string, Hash, long, IDownloadState)>("SELECT Name, Hash, Size, State FROM dbo.ModListArchives WHERE MachineUrl = @MachineUrl", new {MachineUrl = machineURL}); return archives.Select(t => new Archive { State = t.Item4, Name = string.IsNullOrWhiteSpace(t.Item1) ? t.Item4.PrimaryKeyString : t.Item1, Size = t.Item3, Hash = t.Item2 }).ToList(); } public async Task> ModListArchives() { await using var conn = await Open(); var archives = await conn.QueryAsync<(string, Hash, long, IDownloadState)>("SELECT Name, Hash, Size, State FROM dbo.ModListArchives"); return archives.Select(t => new Archive { State = t.Item4, Name = string.IsNullOrWhiteSpace(t.Item1) ? t.Item4.PrimaryKeyString : t.Item1, Size = t.Item3, Hash = t.Item2 }).ToList(); } public async Task PurgeList(string machineURL) { await using var conn = await Open(); var ret1 = await conn.ExecuteAsync(@" delete from dbo.ModListArchives where MachineURL = @machineURL", new {machineURL}); var ret2 = await conn.ExecuteAsync(@" delete from dbo.ModLists where MachineURL = @machineURL", new {machineURL}); return ret1 + ret2; } } }