using System; using System.Collections.Generic; using System.Diagnostics; using System.IO.Compression; using System.Linq; using System.Reactive.Linq; using System.Threading; using System.Threading.Tasks; using System.Web; using Newtonsoft.Json; using Wabbajack.Common; using Wabbajack.Common.Serialization.Json; using Wabbajack.Lib.Validation; using YoutubeExplode; using YoutubeExplode.Exceptions; using YoutubeExplode.Videos.Streams; using File = Alphaleonis.Win32.Filesystem.File; using Path = Alphaleonis.Win32.Filesystem.Path; namespace Wabbajack.Lib.Downloaders { public class YouTubeDownloader : IDownloader { public async Task GetDownloaderState(dynamic archiveINI, bool quickMode) { var directURL = (Uri)DownloaderUtils.GetDirectURL(archiveINI); var state = UriToState(directURL) as State; if (state == null) return state; var idx = 0; while (true) { var section = archiveINI[$"track/{idx}"]; if (section.name == null) break; var track = new State.Track(); track.Name = section.name; track.Start = TimeSpan.Parse(section.start); track.End = TimeSpan.Parse(section.end); track.Format = Enum.Parse(section.format); state.Tracks.Add(track); idx += 1; } return state; } internal static AbstractDownloadState? UriToState(Uri directURL) { if (directURL == null || !directURL.Host.EndsWith("youtube.com")) { return null; } var key = HttpUtility.ParseQueryString(directURL.Query)["v"]; return key != null ? new State(key) : null; } public async Task Prepare() { } [JsonName("YouTubeDownloader")] public class State : AbstractDownloadState { public string Key { get; } public List Tracks { get; set; } = new List(); [JsonIgnore] public override object[] PrimaryKey => new object[] {Key}; public State(string key) { Key = key; } [JsonName("YouTubeTrack")] public class Track { public enum FormatEnum { XWM, WAV } public FormatEnum Format { get; set; } public string Name { get; set; } = string.Empty; public TimeSpan Start { get; set; } public TimeSpan End { get; set; } } public override bool IsWhitelisted(ServerWhitelist whitelist) { return true; } public override async Task Download(Archive a, AbsolutePath destination) { try { using var queue = new WorkQueue(); await using var folder = await TempFolder.Create(); folder.Dir.Combine("tracks").CreateDirectory(); var client = new YoutubeClient(Common.Http.ClientFactory.Client); var meta = await client.Videos.GetAsync(Key); var video = await client.Videos.Streams.GetManifestAsync(Key); var stream = video.Streams.OfType().Where(f => f.AudioCodec.StartsWith("mp4a")).OrderByDescending(a => a.Bitrate) .ToArray().First(); var initialDownload = folder.Dir.Combine("initial_download"); var trackFolder = folder.Dir.Combine("tracks"); await using (var fs = initialDownload.Create()) { await client.Videos.Streams.CopyToAsync(stream, fs, new Progress($"Downloading {a.Name}"), CancellationToken.None); } await initialDownload.CopyToAsync(destination.WithExtension(new Extension(".dest_stream"))); await Tracks.PMap(queue, async track => { Utils.Status($"Extracting track {track.Name}"); await ExtractTrack(initialDownload, trackFolder, track); }); await using var dest = destination.Create(); using var ar = new ZipArchive(dest, ZipArchiveMode.Create); foreach (var track in trackFolder.EnumerateFiles().OrderBy(e => e)) { Utils.Status($"Adding {track.FileName} to archive"); var entry = ar.CreateEntry(Path.Combine("Data", "tracks", (string)track.RelativeTo(trackFolder)), CompressionLevel.NoCompression); entry.LastWriteTime = meta.UploadDate; await using var es = entry.Open(); await using var ins = track.OpenRead(); await ins.CopyToAsync(es); } return true; } catch (VideoUnavailableException) { return false; } } private AbsolutePath FFMpegPath => "Downloaders/Converters/ffmpeg.exe".RelativeTo(AbsolutePath.EntryPoint); private AbsolutePath xWMAEncodePath = "Downloaders/Converters/xWMAEncode.exe".RelativeTo(AbsolutePath.EntryPoint); private Extension WAVExtension = new Extension(".wav"); private Extension XWMExtension = new Extension(".xwm"); private async Task ExtractTrack(AbsolutePath source, AbsolutePath destFolder, Track track) { var process = new ProcessHelper { Path = FFMpegPath, Arguments = new object[] {"-threads", 1, "-i", source, "-ss", track.Start, "-t", track.End - track.Start, track.Name.RelativeTo(destFolder).WithExtension(WAVExtension)}, ThrowOnNonZeroExitCode = true }; var ffmpegLogs = process.Output.Where(arg => arg.Type == ProcessHelper.StreamType.Output) .ForEachAsync(val => { Utils.Status($"Extracting {track.Name} - {val.Line}"); }); await process.Start(); if (track.Format == Track.FormatEnum.WAV) return; process = new ProcessHelper() { Path = xWMAEncodePath, Arguments = new object[] {"-b", 192000, track.Name.RelativeTo(destFolder).WithExtension(WAVExtension), track.Name.RelativeTo(destFolder).WithExtension(XWMExtension)}, ThrowOnNonZeroExitCode = true }; var xwmLogs = process.Output.Where(arg => arg.Type == ProcessHelper.StreamType.Output) .ForEachAsync(val => { Utils.Status($"Encoding {track.Name} - {val.Line}"); }); await process.Start(); if (File.Exists($"{destFolder}\\{track.Name}.wav")) File.Delete($"{destFolder}\\{track.Name}.wav"); } private class Progress : IProgress { private string _prefix; public Progress(string prefix) { _prefix = prefix; } public void Report(double value) { Utils.Status(_prefix, Percent.FactoryPutInRange(value)); } } public override async Task Verify(Archive archive) { try { var client = new YoutubeClient(Common.Http.ClientFactory.Client); var video = await client.Videos.GetAsync(Key); return true; } catch (VideoUnavailableException) { return false; } } public override IDownloader GetDownloader() { return DownloadDispatcher.GetInstance(); } public override string GetManifestURL(Archive a) { return $"https://www.youtube.com/watch?v={Key}"; } public override string[] GetMetaIni() { IEnumerable start = new List {"[General]", $"directURL=https://www.youtube.com/watch?v={Key}"}; start = start.Concat(Tracks.SelectMany((track, idx) => { return new[] { $"\n[track/{idx}]", $"name={track.Name}", $"start={track.Start}", $"end={track.End}", $"format={track.Format}" }; })); return start.ToArray(); } } } }