using DownKyi.Core.Aria2cNet; using DownKyi.Core.Aria2cNet.Client; using DownKyi.Core.Aria2cNet.Client.Entity; using DownKyi.Core.Aria2cNet.Server; using DownKyi.Core.BiliApi.Login; using DownKyi.Core.BiliApi.VideoStream.Models; using DownKyi.Core.Logging; using DownKyi.Core.Settings; using DownKyi.Core.Utils; using DownKyi.Models; using DownKyi.Utils; using DownKyi.ViewModels.DownloadManager; using System; using System.Collections.Generic; using System.Collections.ObjectModel; using System.IO; using System.Linq; using System.Threading.Tasks; namespace DownKyi.Services.Download { /// /// 音视频采用Aria下载,其余采用WebClient下载 /// public class AriaDownloadService : DownloadService, IDownloadService { public AriaDownloadService(ObservableCollection downloadingList, ObservableCollection downloadedList) : base(downloadingList, downloadedList) { Tag = "AriaDownloadService"; } #region 音视频 /// /// 下载音频,返回下载文件路径 /// /// /// public override string DownloadAudio(DownloadingItem downloading) { PlayUrlDashVideo downloadAudio = BaseDownloadAudio(downloading); return DownloadVideo(downloading, downloadAudio); } /// /// 下载视频,返回下载文件路径 /// /// /// public override string DownloadVideo(DownloadingItem downloading) { PlayUrlDashVideo downloadVideo = BaseDownloadVideo(downloading); return DownloadVideo(downloading, downloadVideo); } /// /// 将下载音频和视频的函数中相同代码抽象出来 /// /// /// /// private string DownloadVideo(DownloadingItem downloading, PlayUrlDashVideo downloadVideo) { // 如果为空,说明没有匹配到可下载的音频视频 if (downloadVideo == null) { return null; } // 下载链接 List urls = new List(); if (downloadVideo.BaseUrl != null) { urls.Add(downloadVideo.BaseUrl); } if (downloadVideo.BackupUrl != null) { urls.AddRange(downloadVideo.BackupUrl); } // 路径 downloading.DownloadBase.FilePath = downloading.DownloadBase.FilePath.Replace("\\", "/"); string[] temp = downloading.DownloadBase.FilePath.Split('/'); //string path = downloading.DownloadBase.FilePath.Replace(temp[temp.Length - 1], ""); string path = downloading.DownloadBase.FilePath.TrimEnd(temp[temp.Length - 1].ToCharArray()); // 下载文件名 string fileName = Guid.NewGuid().ToString("N"); string key = $"{downloadVideo.Id}_{downloadVideo.Codecs}"; // 老版本数据库没有这一项,会变成null if (downloading.Downloading.DownloadedFiles == null) { downloading.Downloading.DownloadedFiles = new List(); } if (downloading.Downloading.DownloadFiles.ContainsKey(key)) { // 如果存在,表示下载过, // 则继续使用上次下载的文件名 fileName = downloading.Downloading.DownloadFiles[key]; // 还要检查一下文件有没有被人删掉,删掉的话重新下载 // 如果下载视频之后音频文件被人删了。此时gid还是视频的,会下错文件 if (downloading.Downloading.DownloadedFiles.Contains(key) && File.Exists(Path.Combine(path, fileName))) { return Path.Combine(path, fileName); } } else { // 记录本次下载的文件 try { downloading.Downloading.DownloadFiles.Add(key, fileName); } catch (ArgumentException) { } // Gid最好能是每个文件单独存储,现在复用有可能会混 // 不过好消息是下载是按固定顺序的,而且下载了两个音频会混流不过 downloading.Downloading.Gid = null; } // 启用https AllowStatus useSSL = SettingsManager.GetInstance().UseSSL(); if (useSSL == AllowStatus.YES) { for (int i = 0; i < urls.Count; i++) { string url = urls[i]; if (url.StartsWith("http://")) { urls[i] = url.Replace("http://", "https://"); } } } else { for (int i = 0; i < urls.Count; i++) { string url = urls[i]; if (url.StartsWith("https://")) { urls[i] = url.Replace("https://", "http://"); } } } // 开始下载 DownloadResult downloadStatus = DownloadByAria(downloading, urls, path, fileName); switch (downloadStatus) { case DownloadResult.SUCCESS: downloading.Downloading.DownloadedFiles.Add(key); downloading.Downloading.Gid = null; return Path.Combine(path, fileName); case DownloadResult.FAILED: case DownloadResult.ABORT: default: return nullMark; } } #endregion /// /// 下载封面 /// /// public override string DownloadCover(DownloadingItem downloading, string coverUrl, string fileName) { return BaseDownloadCover(downloading, coverUrl, fileName); } /// /// 下载弹幕 /// /// public override string DownloadDanmaku(DownloadingItem downloading) { return BaseDownloadDanmaku(downloading); } /// /// 下载字幕 /// /// public override List DownloadSubtitle(DownloadingItem downloading) { return BaseDownloadSubtitle(downloading); } /// /// 混流音频和视频 /// /// /// /// /// public override string MixedFlow(DownloadingItem downloading, string audioUid, string videoUid) { if (videoUid == nullMark) { return null; } return BaseMixedFlow(downloading, audioUid, videoUid); } /// /// 解析视频流的下载链接 /// /// public override void Parse(DownloadingItem downloading) { BaseParse(downloading); } /// /// 停止下载服务(转换await和Task.Wait两种调用形式) /// private async Task EndTask() { // 停止基本任务 await BaseEndTask(); // 关闭Aria服务器 await CloseAriaServer(); } /// /// 停止下载服务 /// public void End() { Task.Run(EndTask).Wait(); } /// /// 启动下载服务 /// public void Start() { // 启动Aria服务器 StartAriaServer(); // 启动基本服务 BaseStart(); } /// /// 强制暂停 /// /// /// protected override void Pause(DownloadingItem downloading) { cancellationToken.ThrowIfCancellationRequested(); downloading.DownloadStatusTitle = DictionaryResource.GetString("Pausing"); if (downloading.Downloading.DownloadStatus == DownloadStatus.PAUSE) { throw new OperationCanceledException("Stop thread by pause"); } // 是否存在 var isExist = IsExist(downloading); if (!isExist.Result) { throw new OperationCanceledException("Task is deleted"); } } /// /// 是否存在于下载列表中 /// /// /// private async Task IsExist(DownloadingItem downloading) { bool isExist = downloadingList.Contains(downloading); if (isExist) { return true; } else { // 先恢复为waiting状态,暂停状态下Remove会导致文件重新下载,原因暂不清楚 await AriaClient.UnpauseAsync(downloading.Downloading.Gid); // 移除下载项 var ariaRemove = await AriaClient.RemoveAsync(downloading.Downloading.Gid); if (ariaRemove == null || ariaRemove.Result == downloading.Downloading.Gid) { // 从内存中删除下载项 await AriaClient.RemoveDownloadResultAsync(downloading.Downloading.Gid); } return false; } } /// /// 启动Aria服务器 /// private async void StartAriaServer() { List header = new List { $"Cookie: {LoginHelper.GetLoginInfoCookiesString()}", $"Origin: https://www.bilibili.com", $"Referer: https://www.bilibili.com", $"User-Agent: Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/97.0.4692.99 Safari/537.36" }; AriaConfig config = new AriaConfig() { ListenPort = SettingsManager.GetInstance().GetAriaListenPort(), Token = "downkyi", LogLevel = SettingsManager.GetInstance().GetAriaLogLevel(), MaxConcurrentDownloads = SettingsManager.GetInstance().GetMaxCurrentDownloads(), MaxConnectionPerServer = 8, // 最大取16 Split = SettingsManager.GetInstance().GetAriaSplit(), //MaxTries = 5, MinSplitSize = 10, // 10MB MaxOverallDownloadLimit = SettingsManager.GetInstance().GetAriaMaxOverallDownloadLimit() * 1024L, // 输入的单位是KB/s,所以需要乘以1024 MaxDownloadLimit = SettingsManager.GetInstance().GetAriaMaxDownloadLimit() * 1024L, // 输入的单位是KB/s,所以需要乘以1024 MaxOverallUploadLimit = 0, MaxUploadLimit = 0, ContinueDownload = true, FileAllocation = SettingsManager.GetInstance().GetAriaFileAllocation(), Headers = header }; var task = await AriaServer.StartServerAsync(config); if (task) { Console.WriteLine("Start ServerAsync Completed"); } for (int i = 0; i < 10; i++) { var globOpt = await AriaClient.GetGlobalOptionAsync(); if (globOpt != null) break; await Task.Delay(1000); } Console.WriteLine("Start ServerAsync end"); } /// /// 关闭Aria服务器 /// private async Task CloseAriaServer() { // 暂停所有下载 var ariaPause = await AriaClient.PauseAllAsync(); #if DEBUG Core.Utils.Debugging.Console.PrintLine(ariaPause.ToString()); #endif // 关闭服务器 bool close = AriaServer.CloseServer(); #if DEBUG Core.Utils.Debugging.Console.PrintLine(close); #endif } /// /// 采用Aria下载文件 /// /// /// private DownloadResult DownloadByAria(DownloadingItem downloading, List urls, string path, string localFileName) { // path已斜杠结尾,去掉斜杠 path = path.TrimEnd('/').TrimEnd('\\'); //检查gid对应任务,如果已创建那么直接使用 //但是代理设置会出现不能随时更新的问题 if (downloading.Downloading.Gid != null) { Task status = AriaClient.TellStatus(downloading.Downloading.Gid); if (status == null || status.Result == null) downloading.Downloading.Gid = null; else if (status.Result.Result == null && status.Result.Error != null) { if (status.Result.Error.Message.Contains("is not found")) { downloading.Downloading.Gid = null; } } } if (downloading.Downloading.Gid == null) { AriaSendOption option = new AriaSendOption { //HttpProxy = $"http://{Settings.GetAriaHttpProxy()}:{Settings.GetAriaHttpProxyListenPort()}", Dir = path, Out = localFileName, //Header = $"cookie: {LoginHelper.GetLoginInfoCookiesString()}\nreferer: https://www.bilibili.com", //UseHead = "true", UserAgent = "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/101.0.4951.54 Safari/537.36", }; // 如果设置了代理,则增加HttpProxy if (SettingsManager.GetInstance().IsAriaHttpProxy() == AllowStatus.YES) { option.HttpProxy = $"http://{SettingsManager.GetInstance().GetAriaHttpProxy()}:{SettingsManager.GetInstance().GetAriaHttpProxyListenPort()}"; } // 添加一个下载 Task ariaAddUri = AriaClient.AddUriAsync(urls, option); if (ariaAddUri == null || ariaAddUri.Result == null || ariaAddUri.Result.Result == null) { return DownloadResult.FAILED; } // 保存gid string gid = ariaAddUri.Result.Result; downloading.Downloading.Gid = gid; } else { Task ariaUnpause = AriaClient.UnpauseAsync(downloading.Downloading.Gid); } // 管理下载 AriaManager ariaManager = new AriaManager(); ariaManager.TellStatus += AriaTellStatus; ariaManager.DownloadFinish += AriaDownloadFinish; return ariaManager.GetDownloadStatus(downloading.Downloading.Gid, new Action(() => { cancellationToken.ThrowIfCancellationRequested(); switch (downloading.Downloading.DownloadStatus) { case DownloadStatus.PAUSE: Task ariaPause = AriaClient.PauseAsync(downloading.Downloading.Gid); // 通知UI,并阻塞当前线程 Pause(downloading); break; case DownloadStatus.DOWNLOADING: break; } })); } private void AriaTellStatus(long totalLength, long completedLength, long speed, string gid) { // 当前的下载视频 DownloadingItem video = null; try { video = downloadingList.FirstOrDefault(it => it.Downloading.Gid == gid); } catch (InvalidOperationException e) { Core.Utils.Debugging.Console.PrintLine("AriaTellStatus()发生异常: {0}", e); LogManager.Error("AriaTellStatus()", e); } if (video == null) { return; } // 下载进度百分比 float percent = 0; if (totalLength != 0) { percent = (float)completedLength / totalLength * 100; } // 根据进度判断本次是否需要更新UI if (Math.Abs(percent - video.Progress) < 0.01) { return; } // 下载进度 video.Progress = percent; // 下载大小 video.DownloadingFileSize = Format.FormatFileSize(completedLength) + "/" + Format.FormatFileSize(totalLength); // 下载速度 video.SpeedDisplay = Format.FormatSpeed(speed); // 最大下载速度 if (video.Downloading.MaxSpeed < speed) { video.Downloading.MaxSpeed = speed; } } private void AriaDownloadFinish(bool isSuccess, string downloadPath, string gid, string msg) { //throw new NotImplementedException(); } } }