using AaxDecrypter; using AAXClean; using Dinah.Core; using DataLayer; using LibationFileManager; using System.Threading.Tasks; using System; using System.IO; using ApplicationServices; using LibationFileManager.Templates; #nullable enable namespace FileLiberator { public partial class DownloadOptions : IDownloadOptions, IDisposable { public event EventHandler? DownloadSpeedChanged; public LibraryBook LibraryBook { get; } public LibraryBookDto LibraryBookDto { get; } public string DownloadUrl { get; } public KeyData[]? DecryptionKeys { get; } public required TimeSpan RuntimeLength { get; init; } public OutputFormat OutputFormat { get; } public required ChapterInfo ChapterInfo { get; init; } public string Title => LibraryBook.Book.Title; public string Subtitle => LibraryBook.Book.Subtitle; public string Publisher => LibraryBook.Book.Publisher; public string Language => LibraryBook.Book.Language; public string? AudibleProductId => LibraryBookDto.AudibleProductId; public string? SeriesName => LibraryBookDto.FirstSeries?.Name; public float? SeriesNumber => LibraryBookDto.FirstSeries?.Number; public NAudio.Lame.LameConfig? LameConfig { get; } public string UserAgent => AudibleApi.Resources.Download_User_Agent; public bool TrimOutputToChapterLength => Config.AllowLibationFixup && Config.StripAudibleBrandAudio; public bool StripUnabridged => Config.AllowLibationFixup && Config.StripUnabridged; public bool CreateCueSheet => Config.CreateCueSheet; public bool DownloadClipsBookmarks => Config.DownloadClipsBookmarks; public long DownloadSpeedBps => Config.DownloadSpeedLimit; public bool RetainEncryptedFile => Config.RetainAaxFile; public bool FixupFile => Config.AllowLibationFixup; public bool Downsample => Config.AllowLibationFixup && Config.LameDownsampleMono; public bool MatchSourceBitrate => Config.AllowLibationFixup && Config.LameMatchSourceBR && Config.LameTargetBitrate; public bool MoveMoovToBeginning => Config.MoveMoovToBeginning; public AAXClean.FileType? InputType { get; } public AudibleApi.Common.DrmType DrmType { get; } public AudibleApi.Common.ContentMetadata ContentMetadata { get; } public string GetMultipartFileName(MultiConvertFileProperties props) { var baseDir = Path.GetDirectoryName(props.OutputFileName); var extension = Path.GetExtension(props.OutputFileName); return Templates.ChapterFile.GetFilename(LibraryBookDto, props, baseDir!, extension); } public string GetMultipartTitle(MultiConvertFileProperties props) => Templates.ChapterTitle.GetName(LibraryBookDto, props); public async Task SaveClipsAndBookmarksAsync(string fileName) { if (DownloadClipsBookmarks) { var format = Config.ClipsBookmarksFileFormat; var formatExtension = format.ToString().ToLowerInvariant(); var filePath = Path.ChangeExtension(fileName, formatExtension); var api = await LibraryBook.GetApiAsync(); var records = await api.GetRecordsAsync(LibraryBook.Book.AudibleProductId); switch(format) { case Configuration.ClipBookmarkFormat.CSV: RecordExporter.ToCsv(filePath, records); break; case Configuration.ClipBookmarkFormat.Xlsx: RecordExporter.ToXlsx(filePath, records); break; case Configuration.ClipBookmarkFormat.Json: RecordExporter.ToJson(filePath, LibraryBook, records); break; } return filePath; } return string.Empty; } public Configuration Config { get; } private readonly IDisposable cancellation; public void Dispose() { cancellation?.Dispose(); GC.SuppressFinalize(this); } private DownloadOptions(Configuration config, LibraryBook libraryBook, LicenseInfo licInfo) { Config = ArgumentValidator.EnsureNotNull(config, nameof(config)); LibraryBook = ArgumentValidator.EnsureNotNull(libraryBook, nameof(libraryBook)); ArgumentValidator.EnsureNotNull(licInfo, nameof(licInfo)); if (licInfo.ContentMetadata.ContentUrl.OfflineUrl is not string licUrl) throw new InvalidDataException("Content license doesn't contain an offline Url"); DownloadUrl = licUrl; DecryptionKeys = licInfo.DecryptionKeys; DrmType = licInfo.DrmType; ContentMetadata = licInfo.ContentMetadata; InputType = licInfo.DrmType is AudibleApi.Common.DrmType.Widevine ? AAXClean.FileType.Dash : licInfo.DrmType is AudibleApi.Common.DrmType.Adrm && licInfo.DecryptionKeys?.Length == 1 && licInfo.DecryptionKeys[0].KeyPart1.Length == 8 && licInfo.DecryptionKeys[0].KeyPart2 is null ? AAXClean.FileType.Aax : licInfo.DrmType is AudibleApi.Common.DrmType.Adrm && licInfo.DecryptionKeys?.Length == 1 && licInfo.DecryptionKeys[0].KeyPart1.Length == 32 && licInfo.DecryptionKeys[0].KeyPart2?.Length == 32 ? AAXClean.FileType.Aaxc : null; //If DrmType is not Adrm or Widevine, the delivered file is an unencrypted mp3. OutputFormat = licInfo.DrmType is not AudibleApi.Common.DrmType.Adrm and not AudibleApi.Common.DrmType.Widevine || (config.AllowLibationFixup && config.DecryptToLossy && licInfo.ContentMetadata.ContentReference.Codec != Ac4Codec) ? OutputFormat.Mp3 : OutputFormat.M4b; LameConfig = OutputFormat == OutputFormat.Mp3 ? GetLameOptions(config) : null; // no null/empty check for key/iv. unencrypted files do not have them LibraryBookDto = LibraryBook.ToDto(); LibraryBookDto.Codec = licInfo.ContentMetadata.ContentReference.Codec; cancellation = config .ObservePropertyChanged( nameof(Configuration.DownloadSpeedLimit), newVal => DownloadSpeedChanged?.Invoke(this, newVal)); } } }