using System; using System.Collections.Generic; using System.Linq; using System.Threading.Tasks; using Dapper; using Wabbajack.Hashing.xxHash64; using Wabbajack.Server.DTOs; namespace Wabbajack.Server.DataLayer; public partial class SqlService { /// /// Adds a patch record /// /// /// public async Task AddPatch(Patch patch) { await using var conn = await Open(); await using var trans = conn.BeginTransaction(); if (await conn.QuerySingleOrDefaultAsync<(Guid, Guid)>( "Select SrcID, DestID FROM dbo.Patches where SrcID = @SrcId and DestID = @DestId", new {SrcId = patch.Src.Id, DestId = patch.Dest.Id}, trans) != default) return false; await conn.ExecuteAsync("INSERT INTO dbo.Patches (SrcId, DestId) VALUES (@SrcId, @DestId)", new {SrcId = patch.Src.Id, DestId = patch.Dest.Id}, trans); await trans.CommitAsync(); return true; } /// /// Adds a patch record /// /// /// public async Task FinializePatch(Patch patch) { await using var conn = await Open(); await conn.ExecuteAsync( "UPDATE dbo.Patches SET PatchSize = @PatchSize, Finished = @Finished, IsFailed = @IsFailed, FailMessage = @FailMessage WHERE SrcId = @SrcId AND DestID = @DestId", new { SrcId = patch.Src.Id, DestId = patch.Dest.Id, patch.PatchSize, patch.Finished, patch.IsFailed, patch.FailMessage }); } public async Task FindPatch(Guid src, Guid dest) { await using var conn = await Open(); var patch = await conn.QueryFirstOrDefaultAsync<(long, DateTime?, bool?, string)>( @"SELECT p.PatchSize, p.Finished, p.IsFailed, p.FailMessage FROM dbo.Patches p LEFT JOIN dbo.ArchiveDownloads src ON p.SrcId = src.Id LEFT JOIN dbo.ArchiveDownloads dest ON p.SrcId = dest.Id WHERE SrcId = @SrcId AND DestId = @DestId AND src.DownloadFinished IS NOT NULL AND dest.DownloadFinished IS NOT NULL", new { SrcId = src, DestId = dest }); if (patch == default) return default; return new Patch { Src = await GetArchiveDownload(src), Dest = await GetArchiveDownload(dest), PatchSize = patch.Item1, Finished = patch.Item2, IsFailed = patch.Item3, FailMessage = patch.Item4 }; } public async Task FindOrEnqueuePatch(Guid src, Guid dest) { await using var conn = await Open(); var trans = await conn.BeginTransactionAsync(); var patch = await conn.QueryFirstOrDefaultAsync<(Guid, Guid, long, DateTime?, bool?, string)>( "SELECT SrcId, DestId, PatchSize, Finished, IsFailed, FailMessage FROM dbo.Patches WHERE SrcId = @SrcId AND DestId = @DestId", new { SrcId = src, DestId = dest }, trans); if (patch == default) { await conn.ExecuteAsync("INSERT INTO dbo.Patches (SrcId, DestId) VALUES (@SrcId, @DestId)", new {SrcId = src, DestId = dest}, trans); await trans.CommitAsync(); return new Patch {Src = await GetArchiveDownload(src), Dest = await GetArchiveDownload(dest)}; } return new Patch { Src = await GetArchiveDownload(src), Dest = await GetArchiveDownload(dest), PatchSize = patch.Item3, Finished = patch.Item4, IsFailed = patch.Item5, FailMessage = patch.Item6 }; } public async Task GetPendingPatch() { await using var conn = await Open(); var patch = await conn.QueryFirstOrDefaultAsync<(Guid, Guid, long, DateTime?, bool?, string)>( @"SELECT p.SrcId, p.DestId, p.PatchSize, p.Finished, p.IsFailed, p.FailMessage FROM dbo.Patches p LEFT JOIN dbo.ArchiveDownloads src ON src.Id = p.SrcId LEFT JOIN dbo.ArchiveDownloads dest ON dest.Id = p.DestId WHERE p.Finished is NULL AND src.IsFailed = 0 AND dest.IsFailed = 0 "); if (patch == default) return default; return new Patch { Src = await GetArchiveDownload(patch.Item1), Dest = await GetArchiveDownload(patch.Item2), PatchSize = patch.Item3, Finished = patch.Item4, IsFailed = patch.Item5, FailMessage = patch.Item6 }; } public async Task> PatchesForSource(Guid sourceDownload) { await using var conn = await Open(); var patches = await conn.QueryAsync<(Guid, Guid, long, DateTime?, bool?, string)>( "SELECT SrcId, DestId, PatchSize, Finished, IsFailed, FailMessage FROM dbo.Patches WHERE SrcId = @SrcId", new {SrcId = sourceDownload}); return await AsPatches(patches); } public async Task> PatchesForSource(Hash sourceHash) { await using var conn = await Open(); var patches = await conn.QueryAsync<(Guid, Guid, long, DateTime?, bool?, string)>( @"SELECT p.SrcId, p.DestId, p.PatchSize, p.Finished, p.IsFailed, p.FailMessage FROM dbo.Patches p LEFT JOIN dbo.ArchiveDownloads a ON p.SrcId = a.Id WHERE a.Hash = @Hash AND p.Finished IS NOT NULL AND p.IsFailed = 0", new {Hash = sourceHash}); return await AsPatches(patches); } public async Task MarkPatchUsage(Guid srcId, Guid destId) { await using var conn = await Open(); await conn.ExecuteAsync( @"UPDATE dbo.Patches SET Downloads = Downloads + 1, LastUsed = GETUTCDATE() WHERE SrcId = @srcId AND DestID = @destId", new {SrcId = srcId, DestId = destId}); } public async Task> GetOldPatches() { await using var conn = await Open(); var patches = await conn.QueryAsync<(Guid, Guid, long, DateTime?, bool?, string)>( @"SELECT p.SrcId, p.DestId, p.PatchSize, p.Finished, p.IsFailed, p.FailMessage FROM dbo.Patches p LEFT JOIN dbo.ArchiveDownloads a ON p.SrcId = a.Id WHERE a.Hash not in (SELECT Hash FROM dbo.ModListArchives)"); return await AsPatches(patches); } private async Task> AsPatches(IEnumerable<(Guid, Guid, long, DateTime?, bool?, string)> patches) { var results = new List(); foreach (var (srcId, destId, patchSize, finished, isFailed, failMessage) in patches) results.Add(new Patch { Src = await GetArchiveDownload(srcId), Dest = await GetArchiveDownload(destId), PatchSize = patchSize, Finished = finished, IsFailed = isFailed, FailMessage = failMessage }); return results; } public async Task DeletePatch(Patch patch) { await using var conn = await Open(); await conn.ExecuteAsync(@"DELETE FROM dbo.Patches WHERE SrcId = @SrcId AND DestId = @DestID", new { SrcId = patch.Src.Id, DestId = patch.Dest.Id }); } public async Task> AllPatchHashes() { await using var conn = await Open(); return (await conn.QueryAsync<(Hash, Hash)>(@"SELECT a1.Hash, a2.Hash FROM dbo.Patches p LEFT JOIN dbo.ArchiveDownloads a1 ON a1.Id = p.SrcId LEFT JOIN dbo.ArchiveDownloads a2 on a2.Id = p.DestId WHERE p.Finished IS NOT NULL")).ToHashSet(); } public async Task DeletePatchesForHashPair((Hash, Hash) sqlFile) { await using var conn = await Open(); await conn.ExecuteAsync(@"DELETE p FROM dbo.Patches p LEFT JOIN dbo.ArchiveDownloads a1 ON a1.Id = p.SrcId LEFT JOIN dbo.ArchiveDownloads a2 on a2.Id = p.DestId WHERE a1.Hash = @SrcHash AND a2.Hash = @DestHash", new { SrcHash = sqlFile.Item1, DestHash = sqlFile.Item2 }); } public async Task PurgePatch(Hash hash, string rationale) { await using var conn = await Open(); await using var tx = await conn.BeginTransactionAsync(); await conn.ExecuteAsync( "DELETE p FROM dbo.Patches p LEFT JOIN dbo.ArchiveDownloads ad ON ad.Id = p.SrcId WHERE ad.Hash = @Hash ", new {Hash = hash}, tx); await conn.ExecuteAsync( "INSERT INTO dbo.NoPatch (Hash, Created, Rationale) VALUES (@Hash, GETUTCDATE(), @Rationale)", new { Hash = hash, Rationale = rationale }, tx); await tx.CommitAsync(); } public async Task IsNoPatch(Hash hash) { await using var conn = await Open(); return await conn.QueryFirstOrDefaultAsync("SELECT Hash FROM NoPatch WHERE Hash = @Hash", new {Hash = hash}) != default; } }