using System.Linq; using System.Net.Http; using System.Text.RegularExpressions; using System.Threading; using System.Threading.Tasks; using Newtonsoft.Json; using Wabbajack.Common; using Wabbajack.Common.Exceptions; using Wabbajack.Common.Serialization.Json; using Wabbajack.Lib.Http; using Wabbajack.Lib.Validation; namespace Wabbajack.Lib.Downloaders { public class GoogleDriveDownloader : IUrlDownloader { public async Task GetDownloaderState(dynamic archiveINI, bool quickMode) { var url = archiveINI.General?.directURL; return GetDownloaderState(url); } private static readonly Regex GDriveRegex = new("((?<=id=)[a-zA-Z0-9_-]*)|(?<=\\/file\\/d\\/)[a-zA-Z0-9_-]*", RegexOptions.Compiled); public AbstractDownloadState? GetDownloaderState(string? url) { if (url == null) return null; if (!url.StartsWith("https://drive.google.com")) return null; var match = GDriveRegex.Match(url); return new State(match.ToString()); } public async Task Prepare() { } [JsonName("GoogleDriveDownloader")] public class State : AbstractDownloadState { public string Id { get; } [JsonIgnore] public override object[] PrimaryKey => new object[] { Id }; public State(string id) { Id = id; } public override bool IsWhitelisted(ServerWhitelist whitelist) { return whitelist.GoogleIDs.Contains(Id); } public override async Task Download(Archive a, AbsolutePath destination) { var state = await ToHttpState(download: true); if (state == null) return false; return await state.Download(a, destination); } private async Task ToHttpState(bool download) { if (download) { var initialURL = $"https://drive.google.com/uc?id={Id}&export=download"; var client = new Wabbajack.Lib.Http.Client(); using var response = await client.GetAsync(initialURL); if (!response.IsSuccessStatusCode) throw new HttpException((int)response.StatusCode, response.ReasonPhrase ?? "Unknown"); var cookies = response.GetSetCookies(); var warning = cookies.FirstOrDefault(c => c.Key.StartsWith("download_warning_")); response.Dispose(); if (warning == default) { return new HTTPDownloader.State(initialURL) { Client = client }; } var url = $"https://drive.google.com/uc?export=download&confirm={warning.Value}&id={Id}"; var httpState = new HTTPDownloader.State(url) { Client = client }; return httpState; } else { var url = $"https://drive.google.com/file/d/{Id}/edit"; var client = new Wabbajack.Lib.Http.Client(); using var response = await client.GetAsync(url, errorsAsExceptions: false); if (!response.IsSuccessStatusCode) throw new HttpException((int)response.StatusCode, response.ReasonPhrase ?? "Unknown"); var httpState = new HTTPDownloader.State(url) { Client = client }; return httpState; } } public override async Task Verify(Archive a, CancellationToken? token) { var state = await ToHttpState(download: false); if (state == null) return false; return await state.Verify(a, token); } public override IDownloader GetDownloader() { return DownloadDispatcher.GetInstance(); } public override string GetManifestURL(Archive a) { return $"https://drive.google.com/uc?id={Id}&export=download"; } public override string[] GetMetaIni() { return new [] {"[General]",$"directURL=https://drive.google.com/uc?id={Id}&export=download"}; } } } }