using System; using System.Collections.Generic; using System.IO; using System.Linq; using System.Net; using System.Threading.Tasks; using FluentFTP; using FluentFTP.Helpers; using Microsoft.Extensions.Logging; using Wabbajack.BuildServer; using Wabbajack.Common; using Wabbajack.Compiler.PatchCache; using Wabbajack.Hashing.xxHash64; using Wabbajack.Paths; using Wabbajack.Paths.IO; using Wabbajack.Server.DataLayer; using Wabbajack.Server.DTOs; using Wabbajack.Server.TokenProviders; using LogLevel = Microsoft.Extensions.Logging.LogLevel; namespace Wabbajack.Server.Services { public class PatchBuilder : AbstractService { private DiscordWebHook _discordWebHook; private SqlService _sql; private ArchiveMaintainer _maintainer; private readonly TemporaryFileManager _manager; private readonly IFtpSiteCredentials _ftpCreds; public PatchBuilder(ILogger logger, SqlService sql, AppSettings settings, ArchiveMaintainer maintainer, DiscordWebHook discordWebHook, QuickSync quickSync, TemporaryFileManager manager, IFtpSiteCredentials ftpCreds) : base(logger, settings, quickSync, TimeSpan.FromMinutes(1)) { _discordWebHook = discordWebHook; _sql = sql; _maintainer = maintainer; _manager = manager; _ftpCreds = ftpCreds; } public bool NoCleaning { get; set; } public override async Task Execute() { int count = 0; while (true) { count++; var patch = await _sql.GetPendingPatch(); if (patch == default) break; try { _logger.LogInformation( $"Building patch from {patch.Src.Archive.State.PrimaryKeyString} to {patch.Dest.Archive.State.PrimaryKeyString}"); await _discordWebHook.Send(Channel.Spam, new DiscordMessage { Content = $"Building patch from {patch.Src.Archive.State.PrimaryKeyString} to {patch.Dest.Archive.State.PrimaryKeyString}" }); if (patch.Src.Archive.Hash == patch.Dest.Archive.Hash && patch.Src.Archive.State.PrimaryKeyString == patch.Dest.Archive.State.PrimaryKeyString) { await patch.Fail(_sql, "Hashes match"); continue; } if (patch.Src.Archive.Size > 2_500_000_000 || patch.Dest.Archive.Size > 2_500_000_000) { await patch.Fail(_sql, "Too large to patch"); continue; } _maintainer.TryGetPath(patch.Src.Archive.Hash, out var srcPath); _maintainer.TryGetPath(patch.Dest.Archive.Hash, out var destPath); await using var sigFile = _manager.CreateFile(); await using var patchFile = _manager.CreateFile(); await using var srcStream = srcPath.Open(FileMode.Open, FileAccess.Read, FileShare.Read); await using var destStream = destPath.Open(FileMode.Open, FileAccess.Read, FileShare.Read); await using var sigStream = sigFile.Path.Open(FileMode.Create, FileAccess.ReadWrite); await using var patchOutput = patchFile.Path.Open(FileMode.Create, FileAccess.ReadWrite); OctoDiff.Create(destStream, srcStream, sigStream, patchOutput); await patchOutput.DisposeAsync(); var size = patchFile.Path.Size(); await UploadToCDN(patchFile.Path, PatchName(patch)); await patch.Finish(_sql, size); await _discordWebHook.Send(Channel.Spam, new DiscordMessage { Content = $"Built {size.ToFileSizeString()} patch from {patch.Src.Archive.State.PrimaryKeyString} to {patch.Dest.Archive.State.PrimaryKeyString}" }); } catch (Exception ex) { _logger.LogError(ex, "Error while building patch"); await patch.Fail(_sql, ex.ToString()); await _discordWebHook.Send(Channel.Spam, new DiscordMessage { Content = $"Failure building patch from {patch.Src.Archive.State.PrimaryKeyString} to {patch.Dest.Archive.State.PrimaryKeyString}" }); } } if (count > 0) { } if (!NoCleaning) await CleanupOldPatches(); return count; } private static string PatchName(Patch patch) { return PatchName(patch.Src.Archive.Hash, patch.Dest.Archive.Hash); } private static string PatchName(Hash oldHash, Hash newHash) { return $"{oldHash.ToHex()}_{newHash.ToHex()}"; } private async Task CleanupOldPatches() { var patches = await _sql.GetOldPatches(); using var client = await GetBunnyCdnFtpClient(); foreach (var patch in patches) { _logger.LogInformation($"Cleaning patch {patch.Src.Archive.Hash} -> {patch.Dest.Archive.Hash}"); await _discordWebHook.Send(Channel.Spam, new DiscordMessage { Content = $"Removing {patch.PatchSize.FileSizeToString()} patch from {patch.Src.Archive.State.PrimaryKeyString} to {patch.Dest.Archive.State.PrimaryKeyString} due it no longer being required by curated lists" }); if (!await DeleteFromCDN(client, PatchName(patch))) { _logger.LogWarning($"Patch file didn't exist {PatchName(patch)}"); } await _sql.DeletePatch(patch); var pendingPatch = await _sql.GetPendingPatch(); if (pendingPatch != default) break; } var files = await client.GetListingAsync($"\\"); _logger.LogInformation($"Found {files.Length} on the CDN"); var sqlFiles = await _sql.AllPatchHashes(); _logger.LogInformation($"Found {sqlFiles.Count} in SQL"); HashSet<(Hash, Hash)> NamesToPairs(IEnumerable ftpFiles) { return ftpFiles.Select(f => f.Name).Where(f => f.Contains("_")).Select(p => { try { var lst = p.Split("_", StringSplitOptions.RemoveEmptyEntries).Select(Hash.FromHex).ToArray(); return (lst[0], lst[1]); } catch (ArgumentException) { return default; } catch (FormatException) { return default; } }).Where(f => f != default).ToHashSet(); } var oldHashPairs = NamesToPairs(files.Where(f => DateTime.UtcNow - f.Modified > TimeSpan.FromDays(2))); foreach (var (oldHash, newHash) in oldHashPairs.Where(o => !sqlFiles.Contains(o))) { _logger.LogInformation($"Removing CDN File entry for {oldHash} -> {newHash} it's not SQL"); await client.DeleteFileAsync(PatchName(oldHash, newHash)); } var hashPairs = NamesToPairs(files); foreach (var sqlFile in sqlFiles.Where(s => !hashPairs.Contains(s))) { _logger.LogInformation("Removing SQL File entry for {from} -> {to} it's not on the CDN", sqlFile.Item1, sqlFile.Item2); await _sql.DeletePatchesForHashPair(sqlFile); } } private async Task UploadToCDN(AbsolutePath patchFile, string patchName) { for (var times = 0; times < 5; times ++) { try { _logger.Log(LogLevel.Information, $"Uploading {patchFile.Size().ToFileSizeString()} patch file to CDN {patchName}"); using var client = await GetBunnyCdnFtpClient(); await client.UploadFileAsync(patchFile.ToString(), patchName, FtpRemoteExists.Overwrite); return; } catch (Exception ex) { _logger.LogError(ex, $"Error uploading {patchFile} to CDN"); } } _logger.Log(LogLevel.Error, $"Couldn't upload {patchFile} to {patchName}"); } private async Task DeleteFromCDN(FtpClient client, string patchName) { if (!await client.FileExistsAsync(patchName)) return false; await client.DeleteFileAsync(patchName); return true; } private async Task GetBunnyCdnFtpClient() { var info = (await _ftpCreds.Get())[StorageSpace.Patches]; var client = new FtpClient(info.Hostname) {Credentials = new NetworkCredential(info.Username, info.Password)}; await client.ConnectAsync(); return client; } } }