using System; using System.Collections.Generic; using System.IO; using System.IO.Compression; using System.Linq; using System.Net; using System.Threading.Tasks; using FluentFTP; using Microsoft.Extensions.Logging; using Org.BouncyCastle.Utilities.Collections; using Wabbajack.BuildServer; using Wabbajack.BuildServer.Controllers; using Wabbajack.Common; using Wabbajack.Lib; using Wabbajack.Lib.AuthorApi; using Wabbajack.Lib.FileUploader; using Wabbajack.Server.DataLayer; using Wabbajack.Server.DTOs; namespace Wabbajack.Server.Services { public class MirrorUploader : AbstractService { private SqlService _sql; private ArchiveMaintainer _archives; private DiscordWebHook _discord; public bool ActiveFileSyncEnabled { get; set; } = true; public MirrorUploader(ILogger logger, AppSettings settings, SqlService sql, QuickSync quickSync, ArchiveMaintainer archives, DiscordWebHook discord) : base(logger, settings, quickSync, TimeSpan.FromHours(1)) { _sql = sql; _archives = archives; _discord = discord; } public override async Task Execute() { int uploaded = 0; if (ActiveFileSyncEnabled) await _sql.SyncActiveMirroredFiles(); TOP: var toUpload = await _sql.GetNextMirroredFile(); if (toUpload == default) { await DeleteOldMirrorFiles(); return uploaded; } uploaded += 1; try { var creds = await BunnyCdnFtpInfo.GetCreds(StorageSpace.Mirrors); using var queue = new WorkQueue(); if (_archives.TryGetPath(toUpload.Hash, out var path)) { _logger.LogInformation($"Uploading mirror file {toUpload.Hash} {path.Size.FileSizeToString()}"); bool exists = false; using (var client = await GetClient(creds)) { exists = await client.FileExistsAsync($"{toUpload.Hash.ToHex()}/definition.json.gz"); } if (exists) { _logger.LogInformation($"Skipping {toUpload.Hash} it's already on the server"); await toUpload.Finish(_sql); goto TOP; } await _discord.Send(Channel.Spam, new DiscordMessage { Content = $"Uploading {toUpload.Hash} - {toUpload.Created} because {toUpload.Rationale}" }); var definition = await Client.GenerateFileDefinition(queue, path, (s, percent) => { }); using (var client = await GetClient(creds)) { await client.CreateDirectoryAsync($"{definition.Hash.ToHex()}"); await client.CreateDirectoryAsync($"{definition.Hash.ToHex()}/parts"); } string MakePath(long idx) { return $"{definition.Hash.ToHex()}/parts/{idx}"; } await definition.Parts.PMap(queue, async part => { _logger.LogInformation($"Uploading mirror part ({part.Index}/{definition.Parts.Length})"); var buffer = new byte[part.Size]; await using (var fs = await path.OpenShared()) { fs.Position = part.Offset; await fs.ReadAsync(buffer); } await CircuitBreaker.WithAutoRetryAllAsync(async () =>{ using var client = await GetClient(creds); var name = MakePath(part.Index); await client.UploadAsync(new MemoryStream(buffer), name); }); }); await CircuitBreaker.WithAutoRetryAllAsync(async () => { using var client = await GetClient(creds); _logger.LogInformation($"Finishing mirror upload"); await using var ms = new MemoryStream(); await using (var gz = new GZipStream(ms, CompressionLevel.Optimal, true)) { definition.ToJson(gz); } ms.Position = 0; var remoteName = $"{definition.Hash.ToHex()}/definition.json.gz"; await client.UploadAsync(ms, remoteName); }); await toUpload.Finish(_sql); } else { await toUpload.Fail(_sql, "Archive not found"); } } catch (Exception ex) { _logger.LogInformation($"{toUpload.Created} {toUpload.Uploaded}"); _logger.LogError(ex, "Error uploading"); await toUpload.Fail(_sql, ex.ToString()); } goto TOP; } private static async Task GetClient(BunnyCdnFtpInfo creds = null) { return await CircuitBreaker.WithAutoRetryAllAsync(async () => { creds ??= await BunnyCdnFtpInfo.GetCreds(StorageSpace.Mirrors); var ftpClient = new FtpClient(creds.Hostname, new NetworkCredential(creds.Username, creds.Password)); ftpClient.DataConnectionType = FtpDataConnectionType.EPSV; await ftpClient.ConnectAsync(); return ftpClient; }); } /// /// Gets a list of all the Mirrored file hashes that physically exist on the CDN (via FTP lookup) /// /// public async Task> GetHashesOnCDN() { using var ftpClient = await GetClient(); var serverFiles = (await ftpClient.GetNameListingAsync("\\")); return serverFiles .Select(f => ((RelativePath)f).FileName) .Select(l => { try { return Hash.FromHex((string)l); } catch (Exception) { return default; } }) .Where(h => h != default) .ToHashSet(); } public async Task DeleteOldMirrorFiles() { var existingHashes = await GetHashesOnCDN(); var fromSql = await _sql.GetAllMirroredHashes(); foreach (var (hash, _) in fromSql.Where(s => s.Value)) { Utils.Log($"Removing {hash} from SQL it's no longer in the CDN"); if (!existingHashes.Contains(hash)) await _sql.DeleteMirroredFile(hash); } var toDelete = existingHashes.Where(h => !fromSql.ContainsKey(h)).ToArray(); using var client = await GetClient(); foreach (var hash in toDelete) { await _discord.Send(Channel.Spam, new DiscordMessage {Content = $"Removing mirrored file {hash}, as it's no longer in sql"}); Utils.Log($"Removing {hash} from the CDN it's no longer in SQL"); await client.DeleteDirectoryAsync(hash.ToHex()); } } } }