1
mirror of https://github.com/nilaoda/N_m3u8DL-CLI synced 2025-10-26 02:22:13 +02:00

Compare commits

..

74 Commits
2.7.2 ... 2.9.2

Author SHA1 Message Date
nilaoda
4a4bfae5ab 优化MPD下载行为 2021-01-18 02:00:19 +08:00
nilaoda
d586dddfcd Update changelog.txt 2021-01-18 01:59:31 +08:00
nilaoda
fca6b3ff6c Update changelog.txt 2020-12-29 23:19:39 +08:00
nilaoda
5d75626a36 Update Global.cs 2020-12-29 23:18:16 +08:00
nilaoda
a94271c244 mpd - xigua 2020-12-20 18:53:40 +08:00
nilaoda
c51118dce7 解密huke88 2020-12-20 18:53:02 +08:00
nilaoda
81b2e87bf7 处理同一ID分散在不同Period的情况 2020-12-12 02:07:41 +08:00
nilaoda
71a9878aaa Update changelog.txt 2020-12-06 21:32:48 +08:00
nilaoda
769fe4e926 Update Global.cs 2020-12-06 21:32:38 +08:00
nilaoda
1f57ba7c09 Update Parser.cs 2020-12-06 21:32:29 +08:00
nilaoda
71282bda30 Update N_m3u8DL-CLI.csproj 2020-12-02 20:35:44 +08:00
nilaoda
41ee8aebdf update project 2020-12-02 20:31:44 +08:00
nilaoda
a4537bc093 del xml 2020-12-02 20:23:50 +08:00
nilaoda
b8a60b3917 Update packages.config 2020-12-02 20:23:36 +08:00
nilaoda
8091dd290f Update N_m3u8DL-CLI.csproj 2020-12-02 20:15:29 +08:00
nilaoda
d48e84e611 Update FodyWeavers.xml 2020-12-02 20:10:29 +08:00
nilaoda
9f5423a437 Costura.Fody 2020-12-02 19:56:20 +08:00
nilaoda
ce7e38770a Update N_m3u8DL-CLI.csproj 2020-12-02 19:51:29 +08:00
nilaoda
8fdb2e918e Update packages.config 2020-12-02 19:49:10 +08:00
nilaoda
d4b7d240c1 修正多语言识别问题 2020-12-02 11:56:46 +08:00
nilaoda
484d2941ed Update Global.cs 2020-12-02 11:56:17 +08:00
nilaoda
a0f2b66575 Update changelog.txt 2020-12-02 11:56:04 +08:00
nilaoda
65cc0681e2 Update changelog.txt 2020-11-26 21:02:04 +08:00
nilaoda
7d980ec9a2 Update Global.cs 2020-11-26 21:01:57 +08:00
nilaoda
ec5892c05a 修复可能存在的溢出问题 2020-11-26 21:01:48 +08:00
nilaoda
9aed50fbf9 优化MPD识别 2020-11-26 21:01:03 +08:00
nilaoda
d657b455cd BUG FIX 2020-11-25 21:16:41 +08:00
nilaoda
cda6575605 BUG FIX 2020-11-25 21:16:18 +08:00
nilaoda
0d6377d41b Update changelog.txt 2020-11-25 17:34:53 +08:00
nilaoda
9c76bdcbce 支持选择音轨 2020-11-25 17:34:12 +08:00
nilaoda
83915ff606 Update Global.cs 2020-11-25 14:12:46 +08:00
nilaoda
9cd4746f33 Update changelog.txt 2020-11-25 14:11:31 +08:00
nilaoda
33a5b917ac 修正MPD判断最高清晰度的逻辑 2020-11-25 14:08:40 +08:00
nilaoda
f69978bd82 修改芒果TV请求头 2020-11-23 21:53:41 +08:00
nilaoda
d9fd526886 Update changelog.txt 2020-11-23 21:53:26 +08:00
nilaoda
1cc8ecfaaf Update Global.cs 2020-11-23 21:30:12 +08:00
nilaoda
2bc2dde2ad 修改默认UA 2020-11-23 21:29:48 +08:00
nilaoda
c3ddcf9e0e Update Program.cs 2020-11-23 21:29:31 +08:00
nilaoda
ba5d20dd02 Update Parser.cs 2020-11-23 21:29:03 +08:00
nilaoda
16f705fe66 Update Global.cs 2020-11-22 23:13:59 +08:00
nilaoda
d141cabc4a Update changelog.txt 2020-11-22 23:08:51 +08:00
nilaoda
adcf884a93 新的任务监控逻辑 2020-11-22 23:08:33 +08:00
nilaoda
9d903a025f Update DownloadManager.cs 2020-11-22 22:36:21 +08:00
nilaoda
0bd23ab641 更新版本号 2020-11-22 18:48:54 +08:00
nilaoda
a38f27ccd7 手动处理重定向
解决HTTPS协议自动重定向后,Referer丢失问题
2020-11-22 18:29:40 +08:00
nilaoda
3acec5efd3 Update Global.cs 2020-11-22 15:40:47 +08:00
nilaoda
90874e4bfe Update Parser.cs 2020-11-22 14:30:41 +08:00
nilaoda
b94768e3e8 修复自定义MPD的BaseURL 2020-11-22 00:40:17 +08:00
nilaoda
311f3b882e 更新版本号 2020-11-21 19:36:04 +08:00
nilaoda
4b4f537984 BUG FIX 2020-11-21 19:34:47 +08:00
nilaoda
8032d50b42 传递MPD_URL时处理302 2020-11-21 19:34:27 +08:00
nilaoda
f615764e55 Update build_latest.yml 2020-11-21 13:48:20 +08:00
nilaoda
ccaa200ef8 Update build_latest.yml 2020-11-21 13:45:56 +08:00
nilaoda
6058d878eb Update N_m3u8DL-CLI.csproj 2020-11-21 13:40:15 +08:00
nilaoda
c712c6dee0 Create changelog.txt 2020-11-21 13:30:56 +08:00
nilaoda
bb24bb998f Update Global.cs 2020-11-21 13:11:15 +08:00
nilaoda
e9d951efa5 检测GIF HEADER 2020-11-21 12:25:01 +08:00
nilaoda
6f88a805ef 修复PNG检测逻辑
多写了一个分号……
2020-11-21 12:08:28 +08:00
nilaoda
6aa6d63a8d Convert MPD to M3U8
通过将MPD转换为m3u8进行下载
2020-11-20 23:34:35 +08:00
nilaoda
147246caba Update Parser.cs 2020-11-18 16:05:18 +08:00
nilaoda
35c1ee4777 Update Parser.cs 2020-11-18 15:52:36 +08:00
nilaoda
8b32081b85 识别m3u8文件中的EXT-X-PROGRAM-DATE-TIME 2020-11-18 15:33:39 +08:00
nilaoda
c00de328d1 修改默认UA 修改音轨判断逻辑
修改UA为Mozilla/5.0 (Linux; U; Android 7.0; zh-cn; 15 Plus Build/NRD90M) AppleWebKit/537.36 (KHTML, like Gecko) Version/4.0 Chrome/66.0.3359.126 MQQBrowser/9.4 Mobile Safari/537.36

修改AAC滤镜的使用逻辑

当m3u8文本大小大于50MB时应当放弃
2020-11-18 15:03:21 +08:00
nilaoda
d5193c1645 fix bug 2020-11-06 22:11:21 +08:00
nilaoda
c06fbf5820 update docs 2020-11-03 11:38:24 +08:00
nilaoda
e700edba56 Update Program.cs
修正处理文件名过长的逻辑
2020-10-14 22:18:56 +08:00
nilaoda
aad948da7c v2.7.5 2020-10-14 22:01:03 +08:00
nilaoda
14f7b20176 Merge pull request #246 from Suwmlee/master
Fix build error
2020-09-22 10:19:45 +08:00
Mathhew
a66a9a4096 Fix build error 2020-09-22 09:49:58 +08:00
nilaoda
c862f23a9c v2.7.4
支持ddyun m3u8解密
2020-09-20 13:41:49 +08:00
nilaoda
1a722e80de Delete DecryptNfmovies.cs 2020-09-20 13:41:02 +08:00
nilaoda
eff43e8ac3 Merge pull request #240 from Suwmlee/master
Add github action
2020-09-17 18:58:30 +08:00
Mathhew
7648f8f8dc Add github action 2020-09-17 10:05:35 +08:00
nilaoda
1edc1a43fe v2.7.3 2020-09-14 21:53:51 +08:00
24 changed files with 2333 additions and 547 deletions

34
.github/workflows/build_latest.yml vendored Normal file
View File

@@ -0,0 +1,34 @@
name: Build_Latest
on: [push]
jobs:
build:
runs-on: windows-latest
steps:
- uses: actions/checkout@v1
name: Checkout Code
- name: Setup MSBuild Path
uses: warrenbuckley/Setup-MSBuild@v1
env:
ACTIONS_ALLOW_UNSECURE_COMMANDS: 'true'
- name: Setup NuGet
uses: NuGet/setup-nuget@v1.0.2
env:
ACTIONS_ALLOW_UNSECURE_COMMANDS: 'true'
- name: Restore NuGet Packages
run: nuget restore N_m3u8DL-CLI.sln
- name: Build
run: msbuild N_m3u8DL-CLI.sln /p:Configuration=Release /p:DebugSymbols=false /p:DebugType=None
- name: Upload Artifact
uses: actions/upload-artifact@v1.0.0
with:
name: N_m3u8DL-CLI_latest
path: N_m3u8DL-CLI\bin\Release\

View File

@@ -0,0 +1,41 @@
using System.Security.Cryptography;
using System.Text;
using System.Text.RegularExpressions;
namespace N_m3u8DL_CLI
{
class DecodeDdyun
{
public static string DecryptM3u8(byte[] byteArray)
{
string tmp = DecodeNfmovies.DecryptM3u8(byteArray);
if (tmp.StartsWith("duoduo.key"))
{
tmp = Regex.Replace(tmp, @"#EXT-X-BYTERANGE:.*\s", "");
tmp = tmp.Replace("https:", "jump/https:")
.Replace("inews.gtimg.com", "puui.qpic.cn");
}
return tmp;
}
//https://player.ddyunp.com/jQuery.min.js?v1.5
public static string GetVaildM3u8Url(string url)
{
//url: https://hls.ddyunp.com/ddyun/id/1/key/playlist.m3u8
string id = Regex.Match(url, @"\w{20,}").Value;
string tm = Global.GetTimeStamp(false);
string t = ((long.Parse(tm) / 0x186a0) * 0x64).ToString();
string tmp = id + "duoduo" + "1" + t;
MD5 md5 = MD5.Create();
byte[] bs = Encoding.UTF8.GetBytes(tmp);
byte[] hs = md5.ComputeHash(bs);
StringBuilder sb = new StringBuilder();
foreach (byte b in hs)
{
sb.Append(b.ToString("x2"));
}
string key = sb.ToString();
return Regex.Replace(url, @"1/\w{20,}", "1/" + key);
}
}
}

View File

@@ -0,0 +1,53 @@
using Newtonsoft.Json.Linq;
using System;
using System.Collections.Generic;
using System.Security.Cryptography;
using System.Text;
using System.Text.RegularExpressions;
namespace N_m3u8DL_CLI
{
//https://js.huke88.com/assets/revision/js/plugins/tcplayer/tcplayer.v4.1.min.js?v=930
//https://js.huke88.com/assets/revision/js/plugins/tcplayer/libs/hls.min.0.13.2m.js?v=930
class DecodeHuke88Key
{
private static string[] GetOverlayInfo(string url)
{
var enc = new Regex("eyJ\\w{100,}").Match(url).Value;
var json = Encoding.UTF8.GetString(Convert.FromBase64String(enc));
JObject jObject = JObject.Parse(json);
var key = jObject["overlayKey"].ToString();
var iv = jObject["overlayIv"].ToString();
return new string[] { key, iv };
}
public static string DecodeKey(string url, byte[] data)
{
var info = GetOverlayInfo(url);
var overlayKey = info[0];
var overlayIv = info[1];
var l = new List<byte>();
var c = new List<byte>();
for (int h = 0; h < 16; h++)
{
var f = overlayKey.Substring(2 * h, 2);
var g = overlayIv.Substring(2 * h, 2);
l.Add(Convert.ToByte(f, 16));
c.Add(Convert.ToByte(g, 16));
}
var _lastCipherblock = c.ToArray();
var t = new byte[data.Length];
var r = data;
r = Decrypter.AES128Decrypt(data, l.ToArray(), Decrypter.HexStringToBytes("00000000000000000000000000000000"), CipherMode.CBC, PaddingMode.Zeros);
for (var o = 0; o < 16; o++)
t[o] = (byte)(r[o] ^ _lastCipherblock[o]);
var key = Convert.ToBase64String(t);
return key;
}
}
}

View File

@@ -0,0 +1,36 @@
using System;
using System.IO;
using System.Linq;
using System.Text;
namespace N_m3u8DL_CLI
{
class DecodeNfmovies
{
//https://jx.nfmovies.com/hls.min.js
public static string DecryptM3u8(byte[] byteArray)
{
var t = byteArray;
var decrypt = "";
if (137 == t[0] && 80 == t[1] && 130 == t[354] && 96 == t[353]) t = t.Skip(355).ToArray();
else
{
if (137 != t[0] || 80 != t[1] || 130 != t[394] || 96 != t[393])
{
for (var i = 0; i < t.Length; i++) decrypt += Convert.ToChar(t[i]);
return decrypt;
}
t = t.Skip(395).ToArray();
}
using (var zipStream =
new System.IO.Compression.GZipStream(new MemoryStream(t), System.IO.Compression.CompressionMode.Decompress))
{
using (StreamReader sr = new StreamReader(zipStream, Encoding.UTF8))
{
decrypt = sr.ReadToEnd();
}
}
return decrypt;
}
}
}

View File

@@ -1,16 +1,12 @@
using System; using System;
using System.Collections.Generic;
using System.IO; using System.IO;
using System.Linq;
using System.Security.Cryptography; using System.Security.Cryptography;
using System.Text;
using System.Threading.Tasks;
namespace N_m3u8DL_CLI namespace N_m3u8DL_CLI
{ {
class Decrypter class Decrypter
{ {
public static byte[] AES128Decrypt(string filePath, byte[] keyByte, byte[] ivByte, CipherMode mode = CipherMode.CBC) public static byte[] AES128Decrypt(string filePath, byte[] keyByte, byte[] ivByte, CipherMode mode = CipherMode.CBC, PaddingMode padding = PaddingMode.PKCS7)
{ {
FileStream fs = new FileStream(filePath, FileMode.Open); FileStream fs = new FileStream(filePath, FileMode.Open);
//获取文件大小 //获取文件大小
@@ -25,14 +21,14 @@ namespace N_m3u8DL_CLI
dcpt.Key = keyByte; dcpt.Key = keyByte;
dcpt.IV = ivByte; dcpt.IV = ivByte;
dcpt.Mode = mode; dcpt.Mode = mode;
dcpt.Padding = PaddingMode.PKCS7; dcpt.Padding = padding;
ICryptoTransform cTransform = dcpt.CreateDecryptor(); ICryptoTransform cTransform = dcpt.CreateDecryptor();
Byte[] resultArray = cTransform.TransformFinalBlock(inBuff, 0, inBuff.Length); Byte[] resultArray = cTransform.TransformFinalBlock(inBuff, 0, inBuff.Length);
return resultArray; return resultArray;
} }
public static byte[] AES128Decrypt(byte[] encryptedBuff, byte[] keyByte, byte[] ivByte, CipherMode mode = CipherMode.CBC) public static byte[] AES128Decrypt(byte[] encryptedBuff, byte[] keyByte, byte[] ivByte, CipherMode mode = CipherMode.CBC, PaddingMode padding = PaddingMode.PKCS7)
{ {
byte[] inBuff = encryptedBuff; byte[] inBuff = encryptedBuff;
@@ -42,7 +38,7 @@ namespace N_m3u8DL_CLI
dcpt.Key = keyByte; dcpt.Key = keyByte;
dcpt.IV = ivByte; dcpt.IV = ivByte;
dcpt.Mode = mode; dcpt.Mode = mode;
dcpt.Padding = PaddingMode.PKCS7; dcpt.Padding = padding;
ICryptoTransform cTransform = dcpt.CreateDecryptor(); ICryptoTransform cTransform = dcpt.CreateDecryptor();
Byte[] resultArray = cTransform.TransformFinalBlock(inBuff, 0, inBuff.Length); Byte[] resultArray = cTransform.TransformFinalBlock(inBuff, 0, inBuff.Length);
@@ -56,7 +52,7 @@ namespace N_m3u8DL_CLI
return new byte[0]; return new byte[0];
} }
if (hexStr.StartsWith("0x") || hexStr.StartsWith("0X")) if (hexStr.StartsWith("0x") || hexStr.StartsWith("0X"))
{ {
hexStr = hexStr.Remove(0, 2); hexStr = hexStr.Remove(0, 2);
} }

View File

@@ -11,25 +11,12 @@ namespace N_m3u8DL_CLI
{ {
class DownloadManager class DownloadManager
{ {
private static int calcTime = 1; //计算文件夹大小的间隔
private int stopCount = 0; //速度为零的停止 private int stopCount = 0; //速度为零的停止
private int timeOut = 10000; //超时设置
private static double downloadedSize = 0; //已下载大小
private static bool disableIntegrityCheck = false; //关闭完整性检查
private string jsonFile = string.Empty; private string jsonFile = string.Empty;
private string headers = string.Empty;
private string downDir = string.Empty;
private string downName = string.Empty;
private string muxSetJson = string.Empty;
private int threads = 1;
private int retryCount = 5;
private static int count = 0;
private static int partsCount = 0;
private int total = 0; private int total = 0;
public static string partsPadZero = string.Empty; public static string partsPadZero = string.Empty;
string segsPadZero = string.Empty; string segsPadZero = string.Empty;
bool delAfterDone = false;
private bool isVTT = false; private bool isVTT = false;
bool externalAudio = false; //额外的音轨 bool externalAudio = false; //额外的音轨
string externalAudioUrl = ""; string externalAudioUrl = "";
@@ -37,28 +24,56 @@ namespace N_m3u8DL_CLI
string externalSubUrl = ""; string externalSubUrl = "";
string fflogName = "_ffreport.log"; string fflogName = "_ffreport.log";
public static bool BinaryMerge = false; public static bool BinaryMerge = false;
private bool noMerge = false;
private bool muxFastStart = true;
private string muxFormat = "mp4";
private static bool hasSetDir = false;
public int Threads { get => threads; set => threads = value; } public int Threads { get; set; } = 1;
public int RetryCount { get => retryCount; set => retryCount = value; } public int RetryCount { get; set; } = 5;
public string Headers { get => headers; set => headers = value; } public string Headers { get; set; } = string.Empty;
public string DownDir { get => downDir; set => downDir = value; } public string DownDir { get; set; } = string.Empty;
public string DownName { get => downName; set => downName = value; } public string DownName { get; set; } = string.Empty;
public bool DelAfterDone { get => delAfterDone; set => delAfterDone = value; } public bool DelAfterDone { get; set; } = false;
public string MuxFormat { get => muxFormat; set => muxFormat = value; } public string MuxFormat { get; set; } = "mp4";
public bool MuxFastStart { get => muxFastStart; set => muxFastStart = value; } public bool MuxFastStart { get; set; } = true;
public string MuxSetJson { get => muxSetJson; set => muxSetJson = value; } public string MuxSetJson { get; set; } = string.Empty;
public int TimeOut { get => timeOut; set => timeOut = value; } public int TimeOut { get; set; } = 10000; //超时设置
public static double DownloadedSize { get => downloadedSize; set => downloadedSize = value; } public static double DownloadedSize { get; set; } = 0; //已下载大小
public static bool HasSetDir { get => hasSetDir; set => hasSetDir = value; } public static bool HasSetDir { get; set; } = false;
public bool NoMerge { get => noMerge; set => noMerge = value; } public bool NoMerge { get; set; } = false;
public static int CalcTime { get => calcTime; set => calcTime = value; } public static int CalcTime { get; set; } = 1; //计算速度的间隔
public static int Count { get => count; set => count = value; } public static int Count { get; set; } = 0;
public static int PartsCount { get => partsCount; set => partsCount = value; } public static int PartsCount { get; set; } = 0;
public static bool DisableIntegrityCheck { get => disableIntegrityCheck; set => disableIntegrityCheck = value; } public static bool DisableIntegrityCheck { get; set; } = false; //关闭完整性检查
static CancellationTokenSource cts = new CancellationTokenSource();
//计算下载速度
static System.Timers.Timer timer = new System.Timers.Timer(1000 * CalcTime); //实例化Timer类
public DownloadManager()
{
timer.AutoReset = true;
timer.Elapsed += delegate
{
Console.SetCursorPosition(0, 1);
Console.Write("Speed: " + Global.FormatFileSize((Global.BYTEDOWN) / CalcTime) + " / s".PadRight(70));
if (Global.HadReadInfo && Global.BYTEDOWN <= Global.STOP_SPEED * 1024 * CalcTime)
{
stopCount++;
Console.SetCursorPosition(0, 1);
Console.Write("Speed: " + Global.FormatFileSize((Global.BYTEDOWN) / CalcTime) + " / s [" + stopCount + "]".PadRight(70));
if (stopCount >= 12)
{
Global.ShouldStop = true;
cts.Cancel();
}
}
else
{
stopCount = 0;
Global.BYTEDOWN = 0;
}
};
}
public void DoDownload() public void DoDownload()
{ {
@@ -94,7 +109,6 @@ namespace N_m3u8DL_CLI
PartsCount = parts.Count; PartsCount = parts.Count;
segsPadZero = string.Empty.PadRight(oriCount.Length, '0'); segsPadZero = string.Empty.PadRight(oriCount.Length, '0');
partsPadZero = string.Empty.PadRight(Convert.ToString(parts.Count).Length, '0'); partsPadZero = string.Empty.PadRight(Convert.ToString(parts.Count).Length, '0');
CancellationTokenSource cts = new CancellationTokenSource();
//是直播视频 //是直播视频
if (isVOD == "False") if (isVOD == "False")
@@ -104,7 +118,6 @@ namespace N_m3u8DL_CLI
Global.ShouldStop = false; //是否该停止下载 Global.ShouldStop = false; //是否该停止下载
//监控文件夹变化
if (!Directory.Exists(DownDir)) if (!Directory.Exists(DownDir))
Directory.CreateDirectory(DownDir); //新建文件夹 Directory.CreateDirectory(DownDir); //新建文件夹
Watcher watcher = new Watcher(DownDir); Watcher watcher = new Watcher(DownDir);
@@ -112,41 +125,9 @@ namespace N_m3u8DL_CLI
watcher.PartsCount = PartsCount; watcher.PartsCount = PartsCount;
watcher.WatcherStrat(); watcher.WatcherStrat();
//监控文件夹大小变化 via https://stackoverflow.com/questions/2869561/what-is-the-fastest-way-to-calculate-a-windows-folders-size/12665904#12665904 //开始计算速度
System.Timers.Timer timer = new System.Timers.Timer(1000 * CalcTime); //实例化Timer类
timer.AutoReset = true;
timer.Enabled = true; timer.Enabled = true;
//Scripting.FileSystemObject fso = new Scripting.FileSystemObject(); cts = new CancellationTokenSource();
//Scripting.Folder folder = fso.GetFolder(DownDir);
timer.Elapsed += delegate
{
//采用COM组件获取文件夹的大小需要引入 "Microsoft Scripting Runtime"
//double sizeInBytes = folder.Size;
Console.SetCursorPosition(0, 1);
//Console.WriteLine("Speed: " + Global.FormatFileSize((sizeInBytes - lastSizeInBytes) / calcTime) + " / s ");
Console.Write("Speed: " + Global.FormatFileSize((Global.BYTEDOWN) / CalcTime) + " / s".PadRight(70));
if (Global.HadReadInfo && Global.BYTEDOWN <= Global.STOP_SPEED * 1024 * CalcTime)
{
stopCount++;
Console.SetCursorPosition(0, 1);
Console.Write("Speed: " + Global.FormatFileSize((Global.BYTEDOWN) / CalcTime) + " / s [" + stopCount + "]".PadRight(70));
if (stopCount >= 12)
{
Global.ShouldStop = true;
cts.Cancel();
GC.Collect();
return;
}
}
else
{
stopCount = 0;
Global.BYTEDOWN = 0;
}
};
//开始调用下载 //开始调用下载
LOGGER.WriteLine(strings.startDownloading); LOGGER.WriteLine(strings.startDownloading);
@@ -200,16 +181,10 @@ namespace N_m3u8DL_CLI
sd.Key = firstSeg["key"].Value<string>(); sd.Key = firstSeg["key"].Value<string>();
sd.Iv = firstSeg["iv"].Value<string>(); sd.Iv = firstSeg["iv"].Value<string>();
} }
try if (firstSeg["expectByte"] != null)
{
sd.ExpectByte = firstSeg["expectByte"].Value<long>(); sd.ExpectByte = firstSeg["expectByte"].Value<long>();
} if (firstSeg["startByte"] != null)
catch (Exception) { }
try
{
sd.StartByte = firstSeg["startByte"].Value<long>(); sd.StartByte = firstSeg["startByte"].Value<long>();
}
catch (Exception) { }
sd.Headers = Headers; sd.Headers = Headers;
sd.SavePath = DownDir + "\\Part_" + 0.ToString(partsPadZero) + "\\" + firstSeg["index"].Value<int>().ToString(segsPadZero) + ".tsdownloading"; sd.SavePath = DownDir + "\\Part_" + 0.ToString(partsPadZero) + "\\" + firstSeg["index"].Value<int>().ToString(segsPadZero) + ".tsdownloading";
if (File.Exists(sd.SavePath)) if (File.Exists(sd.SavePath))
@@ -290,16 +265,10 @@ namespace N_m3u8DL_CLI
sd.Key = info["key"].Value<string>(); sd.Key = info["key"].Value<string>();
sd.Iv = info["iv"].Value<string>(); sd.Iv = info["iv"].Value<string>();
} }
try if (firstSeg["expectByte"] != null)
{
sd.ExpectByte = info["expectByte"].Value<long>(); sd.ExpectByte = info["expectByte"].Value<long>();
} if (firstSeg["startByte"] != null)
catch (Exception) { }
try
{
sd.StartByte = info["startByte"].Value<long>(); sd.StartByte = info["startByte"].Value<long>();
}
catch (Exception) { }
sd.Headers = Headers; sd.Headers = Headers;
sd.SavePath = DownDir + "\\Part_" + info["part"].Value<int>().ToString(partsPadZero) + "\\" + info["index"].Value<int>().ToString(segsPadZero) + ".tsdownloading"; sd.SavePath = DownDir + "\\Part_" + info["part"].Value<int>().ToString(partsPadZero) + "\\" + info["index"].Value<int>().ToString(segsPadZero) + ".tsdownloading";
if (File.Exists(sd.SavePath)) if (File.Exists(sd.SavePath))
@@ -327,12 +296,8 @@ namespace N_m3u8DL_CLI
watcher.WatcherStop(); watcher.WatcherStop();
//监控文件夹大小变化的收尾工作 //停止速度监测
timer.Enabled = false; timer.Enabled = false;
timer.Close();
// cleanup COM
//System.Runtime.InteropServices.Marshal.ReleaseComObject(folder);
//System.Runtime.InteropServices.Marshal.ReleaseComObject(fso);
//检测是否下完 //检测是否下完
IsComplete(Convert.ToInt32(segCount)); IsComplete(Convert.ToInt32(segCount));
@@ -363,14 +328,13 @@ namespace N_m3u8DL_CLI
Count++; Count++;
LOGGER.WriteLine(strings.retryCount + Count + " / " + RetryCount); LOGGER.WriteLine(strings.retryCount + Count + " / " + RetryCount);
LOGGER.PrintLine(strings.retryCount + Count + " / " + RetryCount, LOGGER.Warning); LOGGER.PrintLine(strings.retryCount + Count + " / " + RetryCount, LOGGER.Warning);
Thread.Sleep(3000); Thread.Sleep(6000);
GC.Collect(); //垃圾回收
DoDownload(); DoDownload();
} }
} }
else //开始合并 else //开始合并
{ {
LOGGER.PrintLine(strings.downloadComplete + (DisableIntegrityCheck ? "("+strings.disableIntegrityCheck+")" : "")); LOGGER.PrintLine(strings.downloadComplete + (DisableIntegrityCheck ? "(" + strings.disableIntegrityCheck + ")" : ""));
Console.WriteLine(); Console.WriteLine();
if (NoMerge == false) if (NoMerge == false)
{ {
@@ -539,7 +503,7 @@ namespace N_m3u8DL_CLI
FFmpeg.OutPutPath = Path.Combine(Directory.GetParent(DownDir).FullName, DownName); FFmpeg.OutPutPath = Path.Combine(Directory.GetParent(DownDir).FullName, DownName);
FFmpeg.ReportFile = driverName + "\\:" + exePath.Remove(0, exePath.IndexOf(':') + 1).Replace("\\", "/") + "/Logs/" + Path.GetFileNameWithoutExtension(LOGGER.LOGFILE) + fflogName; FFmpeg.ReportFile = driverName + "\\:" + exePath.Remove(0, exePath.IndexOf(':') + 1).Replace("\\", "/") + "/Logs/" + Path.GetFileNameWithoutExtension(LOGGER.LOGFILE) + fflogName;
//合并分段 //合并分段
LOGGER.PrintLine(strings.startMerging); LOGGER.PrintLine(strings.startMerging);
for (int i = 0; i < PartsCount; i++) for (int i = 0; i < PartsCount; i++)
@@ -640,7 +604,7 @@ namespace N_m3u8DL_CLI
parser.DownDir = Path.Combine(Path.GetDirectoryName(DownDir), parser.DownName); parser.DownDir = Path.Combine(Path.GetDirectoryName(DownDir), parser.DownName);
LOGGER.WriteLine(strings.startParsing + externalAudioUrl); LOGGER.WriteLine(strings.startParsing + externalAudioUrl);
LOGGER.WriteLine(strings.downloadingExternalAudioTrack); LOGGER.WriteLine(strings.downloadingExternalAudioTrack);
DownName = DownName + "(Audio)"; DownName = parser.DownName;
fflogName = "_ffreport(Audio).log"; fflogName = "_ffreport(Audio).log";
DownDir = parser.DownDir; DownDir = parser.DownDir;
parser.Parse(); //开始解析 parser.Parse(); //开始解析

View File

@@ -10,19 +10,20 @@ namespace N_m3u8DL_CLI
{ {
class FFmpeg class FFmpeg
{ {
private static string outPutPath = string.Empty; public static string FFMPEG_PATH = "ffmpeg";
private static string reportFile = string.Empty; public static string REC_TIME = ""; //录制日期
private static bool useAACFilter = false; //是否启用滤镜
private static bool writeDate = true; //是否写入录制日期 public static string OutPutPath { get; set; } = string.Empty;
public static string OutPutPath { get => outPutPath; set => outPutPath = value; } public static string ReportFile { get; set; } = string.Empty;
public static string ReportFile { get => reportFile; set => reportFile = value; } public static bool UseAACFilter { get; set; } = false; //是否启用滤镜
public static bool UseAACFilter { get => useAACFilter; set => useAACFilter = value; } public static bool WriteDate { get; set; } = true; //是否写入录制日期
public static bool WriteDate { get => writeDate; set => writeDate = value; }
public static void Merge(string[] files, string muxFormat, bool fastStart, public static void Merge(string[] files, string muxFormat, bool fastStart,
string poster = "", string audioName = "", string title = "", string poster = "", string audioName = "", string title = "",
string copyright = "", string comment = "", string encodingTool = "") string copyright = "", string comment = "", string encodingTool = "")
{ {
string dateString = string.IsNullOrEmpty(REC_TIME) ? DateTime.Now.ToString("o") : REC_TIME;
//同名文件已存在的共存策略 //同名文件已存在的共存策略
if (File.Exists($"{OutPutPath}.{muxFormat.ToLower()}")) if (File.Exists($"{OutPutPath}.{muxFormat.ToLower()}"))
{ {
@@ -50,7 +51,7 @@ namespace N_m3u8DL_CLI
command += " " + (string.IsNullOrEmpty(ddpAudio) ? "" : "-i \"" + ddpAudio + "\""); command += " " + (string.IsNullOrEmpty(ddpAudio) ? "" : "-i \"" + ddpAudio + "\"");
command += command +=
$" -map 0:v? {(string.IsNullOrEmpty(ddpAudio) ? "-map 0:a?" : $"-map {(string.IsNullOrEmpty(poster) ? "1" : "2")}:a -map 0:a?")} -map 0:s? " + (string.IsNullOrEmpty(poster) ? "" : addPoster) $" -map 0:v? {(string.IsNullOrEmpty(ddpAudio) ? "-map 0:a?" : $"-map {(string.IsNullOrEmpty(poster) ? "1" : "2")}:a -map 0:a?")} -map 0:s? " + (string.IsNullOrEmpty(poster) ? "" : addPoster)
+ (writeDate ? " -metadata date=\"" + DateTime.Now.ToString("o") + "\"" : "") + + (WriteDate ? " -metadata date=\"" + dateString + "\"" : "") +
" -metadata encoding_tool=\"" + encodingTool + "\" -metadata title=\"" + title + " -metadata encoding_tool=\"" + encodingTool + "\" -metadata title=\"" + title +
"\" -metadata copyright=\"" + copyright + "\" -metadata comment=\"" + comment + "\" -metadata copyright=\"" + copyright + "\" -metadata comment=\"" + comment +
$"\" -metadata:s:a:{(string.IsNullOrEmpty(ddpAudio) ? "0" : "1")} handler_name=\"" + audioName + $"\" -metadata:s:a:{(string.IsNullOrEmpty(ddpAudio) ? "0" : "1")} handler=\"" + audioName + "\" "; $"\" -metadata:s:a:{(string.IsNullOrEmpty(ddpAudio) ? "0" : "1")} handler_name=\"" + audioName + $"\" -metadata:s:a:{(string.IsNullOrEmpty(ddpAudio) ? "0" : "1")} handler=\"" + audioName + "\" ";
@@ -83,7 +84,7 @@ namespace N_m3u8DL_CLI
} }
Run("ffmpeg", command, Path.GetDirectoryName(files[0])); Run(FFMPEG_PATH, command, Path.GetDirectoryName(files[0]));
LOGGER.WriteLine(strings.ffmpegDone); LOGGER.WriteLine(strings.ffmpegDone);
//Console.WriteLine(command); //Console.WriteLine(command);
} }
@@ -92,7 +93,7 @@ namespace N_m3u8DL_CLI
{ {
if (Global.VIDEO_TYPE == "H264") if (Global.VIDEO_TYPE == "H264")
{ {
Run("ffmpeg", Run(FFMPEG_PATH,
"-loglevel quiet -i \"" + file + "\" -map 0 -c copy -copy_unknown -f mpegts -bsf:v h264_mp4toannexb \"" "-loglevel quiet -i \"" + file + "\" -map 0 -c copy -copy_unknown -f mpegts -bsf:v h264_mp4toannexb \""
+ Path.GetFileNameWithoutExtension(file) + "[MPEGTS].ts\"", + Path.GetFileNameWithoutExtension(file) + "[MPEGTS].ts\"",
Path.GetDirectoryName(file)); Path.GetDirectoryName(file));
@@ -104,7 +105,7 @@ namespace N_m3u8DL_CLI
} }
else if (Global.VIDEO_TYPE == "H265") else if (Global.VIDEO_TYPE == "H265")
{ {
Run("ffmpeg", Run(FFMPEG_PATH,
"-loglevel quiet -i \"" + file + "\" -map 0 -c copy -copy_unknown -f mpegts -bsf:v hevc_mp4toannexb \"" "-loglevel quiet -i \"" + file + "\" -map 0 -c copy -copy_unknown -f mpegts -bsf:v hevc_mp4toannexb \""
+ Path.GetFileNameWithoutExtension(file) + "[MPEGTS].ts\"", + Path.GetFileNameWithoutExtension(file) + "[MPEGTS].ts\"",
Path.GetDirectoryName(file)); Path.GetDirectoryName(file));

View File

@@ -30,8 +30,9 @@ namespace N_m3u8DL_CLI
/*===============================================================================*/ /*===============================================================================*/
static string nowVer = "2.5.7"; static Version ver = System.Reflection.Assembly.GetExecutingAssembly().GetName().Version;
static string nowDate = "20200809"; static string nowVer = $"{ver.Major}.{ver.Minor}.{ver.Build}";
static string nowDate = "20210118";
public static void WriteInit() public static void WriteInit()
{ {
Console.Clear(); Console.Clear();
@@ -101,23 +102,25 @@ namespace N_m3u8DL_CLI
public static string GetWebSource(String url, string headers = "", int TimeOut = 60000) public static string GetWebSource(String url, string headers = "", int TimeOut = 60000)
{ {
string htmlCode = string.Empty; string htmlCode = string.Empty;
for(int i = 0; i < 10; i++) for (int i = 0; i < 10; i++)
{ {
try try
{ {
reProcess:
HttpWebRequest webRequest = (System.Net.HttpWebRequest)System.Net.WebRequest.Create(url); HttpWebRequest webRequest = (System.Net.HttpWebRequest)System.Net.WebRequest.Create(url);
webRequest.Method = "GET"; webRequest.Method = "GET";
if (NoProxy) webRequest.Proxy = null; if (NoProxy) webRequest.Proxy = null;
webRequest.UserAgent = "Mozilla/4.0"; webRequest.UserAgent = "Mozilla/5.0 (Windows NT 10.0; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/78.0.3904.108 Safari/537.36";
webRequest.Headers.Add("Accept-Encoding", "gzip, deflate"); webRequest.Accept = "*/*";
webRequest.Headers.Add("Accept-Encoding", "gzip, deflate, br");
webRequest.Timeout = TimeOut; //设置超时 webRequest.Timeout = TimeOut; //设置超时
webRequest.KeepAlive = false; webRequest.KeepAlive = false;
webRequest.AllowAutoRedirect = true; //自动跳转 webRequest.AllowAutoRedirect = false; //手动处理重定向否则会丢失Referer
if (url.Contains("pcvideo") && url.Contains(".titan.mgtv.com")) if (url.Contains("pcvideo") && url.Contains(".titan.mgtv.com"))
{ {
webRequest.UserAgent = ""; webRequest.UserAgent = "";
if (!url.Contains("/internettv/")) if (!url.Contains("/internettv/"))
webRequest.Referer = "https://player.mgtv.com/mgtv_v6_player/PlayerCore.swf"; webRequest.Referer = "https://www.mgtv.com";
webRequest.Headers.Add("Cookie", "MQGUID"); webRequest.Headers.Add("Cookie", "MQGUID");
} }
//添加headers //添加headers
@@ -145,6 +148,18 @@ namespace N_m3u8DL_CLI
} }
} }
HttpWebResponse webResponse = (HttpWebResponse)webRequest.GetResponse(); HttpWebResponse webResponse = (HttpWebResponse)webRequest.GetResponse();
//302
if (webResponse.Headers.Get("Location") != null)
{
url = webResponse.Headers.Get("Location");
webResponse.Close();
goto reProcess;
}
//文件过大则认为不是m3u8
if (webResponse.ContentLength != -1 && webResponse.ContentLength > 50 * 1024 * 1024) return "";
if (webResponse.ContentEncoding != null if (webResponse.ContentEncoding != null
&& webResponse.ContentEncoding.ToLower() == "gzip") //如果使用了GZip则先解压 && webResponse.ContentEncoding.ToLower() == "gzip") //如果使用了GZip则先解压
{ {
@@ -402,14 +417,17 @@ namespace N_m3u8DL_CLI
} }
} }
reProcess:
byte[] arraryByte; byte[] arraryByte;
HttpWebRequest req = (HttpWebRequest)WebRequest.Create(url); HttpWebRequest req = (HttpWebRequest)WebRequest.Create(url);
req.Method = "GET"; req.Method = "GET";
req.Timeout = timeOut; req.Timeout = timeOut;
req.ReadWriteTimeout = timeOut; //重要 req.ReadWriteTimeout = timeOut; //重要
req.AllowAutoRedirect = false; //手动处理重定向否则会丢失Referer
if (NoProxy) req.Proxy = null; if (NoProxy) req.Proxy = null;
req.Headers.Add("Accept-Encoding", "gzip, deflate"); req.Headers.Add("Accept-Encoding", "gzip, deflate");
req.Accept = "*/*"; req.Accept = "*/*";
req.UserAgent = "Mozilla/5.0 (Windows NT 10.0; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/78.0.3904.108 Safari/537.36";
//添加headers //添加headers
if (headers != "") if (headers != "")
{ {
@@ -437,6 +455,13 @@ namespace N_m3u8DL_CLI
using (HttpWebResponse wr = (HttpWebResponse)req.GetResponse()) using (HttpWebResponse wr = (HttpWebResponse)req.GetResponse())
{ {
//302
if (wr.Headers.Get("Location") != null)
{
url = wr.Headers.Get("Location");
wr.Close();
goto reProcess;
}
if (wr.ContentEncoding != null && wr.ContentEncoding.ToLower() == "gzip") //如果使用了GZip则先解压 if (wr.ContentEncoding != null && wr.ContentEncoding.ToLower() == "gzip") //如果使用了GZip则先解压
{ {
using (Stream streamReceive = wr.GetResponseStream()) using (Stream streamReceive = wr.GetResponseStream())
@@ -497,10 +522,11 @@ namespace N_m3u8DL_CLI
if (shouldStop) if (shouldStop)
return; return;
reProcess:
HttpWebRequest request = WebRequest.Create(url) as HttpWebRequest; HttpWebRequest request = WebRequest.Create(url) as HttpWebRequest;
request.Timeout = timeOut; request.Timeout = timeOut;
request.ReadWriteTimeout = timeOut; //重要 request.ReadWriteTimeout = timeOut; //重要
request.AllowAutoRedirect = true; request.AllowAutoRedirect = false; //手动处理重定向否则会丢失Referer
request.KeepAlive = false; request.KeepAlive = false;
request.Method = "GET"; request.Method = "GET";
if (NoProxy) request.Proxy = null; if (NoProxy) request.Proxy = null;
@@ -510,11 +536,11 @@ namespace N_m3u8DL_CLI
{ {
request.UserAgent = ""; request.UserAgent = "";
if (!url.Contains("/internettv/")) if (!url.Contains("/internettv/"))
request.Referer = "https://player.mgtv.com/mgtv_v6_player/PlayerCore.swf"; request.Referer = "https://www.mgtv.com";
request.Headers.Add("Cookie", "MQGUID"); request.Headers.Add("Cookie", "MQGUID");
} }
else else
request.UserAgent = "VLC/2.2.1 LibVLC/2.2.1"; request.UserAgent = "Mozilla/5.0 (Windows NT 10.0; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/78.0.3904.108 Safari/537.36";
//下载部分字节 //下载部分字节
if (expectByte != -1) if (expectByte != -1)
request.AddRange("bytes", startByte, startByte + expectByte - 1); request.AddRange("bytes", startByte, startByte + expectByte - 1);
@@ -545,8 +571,16 @@ namespace N_m3u8DL_CLI
long totalLen = 0; long totalLen = 0;
long downLen = 0; long downLen = 0;
bool pngHeader = false; //PNG HEADER检测
using (var response = (HttpWebResponse)request.GetResponse()) using (var response = (HttpWebResponse)request.GetResponse())
{ {
//302
if (response.Headers.Get("Location") != null)
{
url = response.Headers.Get("Location");
response.Close();
goto reProcess;
}
using (var responseStream = response.GetResponseStream()) using (var responseStream = response.GetResponseStream())
{ {
using (var stream = new FileStream(path, FileMode.Create, FileAccess.Write, FileShare.Write)) using (var stream = new FileStream(path, FileMode.Create, FileAccess.Write, FileShare.Write))
@@ -555,6 +589,17 @@ namespace N_m3u8DL_CLI
totalLen = response.ContentLength; totalLen = response.ContentLength;
byte[] bArr = new byte[1024]; byte[] bArr = new byte[1024];
int size = responseStream.Read(bArr, 0, (int)bArr.Length); int size = responseStream.Read(bArr, 0, (int)bArr.Length);
if (!pngHeader && size > 3 && 137 == bArr[0] && 80 == bArr[1] && 78 == bArr[2] && 71 == bArr[3])
{
pngHeader = true;
}
//GIF HEADER检测
if (!pngHeader && size > 3 && 0x47 == bArr[0] && 0x49 == bArr[1] && 0x46 == bArr[2] && 0x38 == bArr[3])
{
bArr = bArr.Skip(42).ToArray();
size -= 42;
downLen += 42;
}
while (size > 0) while (size > 0)
{ {
stream.Write(bArr, 0, size); stream.Write(bArr, 0, size);
@@ -577,8 +622,11 @@ namespace N_m3u8DL_CLI
} }
if (shouldStop) if (shouldStop)
try { File.Delete(path); } catch (Exception) { } try { File.Delete(path); } catch (Exception) { }
if (totalLen != -1 && downLen != totalLen) if (totalLen != -1 && downLen != totalLen)
try { File.Delete(path); } catch (Exception) { } try { File.Delete(path); } catch (Exception) { }
if (pngHeader)
TrySkipPngHeader(path);
} }
catch (Exception e) catch (Exception e)
{ {
@@ -587,6 +635,51 @@ namespace N_m3u8DL_CLI
} }
} }
/// <summary>
/// 用于处理利用图床上传TS导致前面被插入PNG Header的情况
/// </summary>
/// <param name="filePath"></param>
public static void TrySkipPngHeader(string filePath)
{
var u = File.ReadAllBytes(filePath);
if (0x47 == u[0])
{
return;
}
else if (137 == u[0] && 80 == u[1] && 78 == u[2] && 71 == u[3] && 96 == u[118] && 130 == u[119])
{
u = u.Skip(120).ToArray();
}
else if (137 == u[0] && 80 == u[1] && 78 == u[2] && 71 == u[3] && 96 == u[6100] && 130 == u[6101])
{
u = u.Skip(6102).ToArray();
}
else if (137 == u[0] && 80 == u[1] && 78 == u[2] && 71 == u[3] && 96 == u[67] && 130 == u[68])
{
u = u.Skip(69).ToArray();
}
else if (137 == u[0] && 80 == u[1] && 78 == u[2] && 71 == u[3] && 96 == u[769] && 130 == u[770])
{
u = u.Skip(771).ToArray();
}
else if (137 == u[0] && 80 == u[1] && 78 == u[2] && 71 == u[3])
{
//确定是PNG但是需要手动查询结尾标记(0x60 0x82 0x47)
int skip = 0;
for (int i = 4; i < u.Length - 3; i++)
{
if (u[i] == 0x60 && u[i + 1] == 0x82 && u[i + 2] == 0x47)
{
skip = i + 2;
break;
}
}
u = u.Skip(skip).ToArray();
}
File.WriteAllBytes(filePath, u);
}
//格式化json字符串 //格式化json字符串
public static string ConvertJsonString(string str) public static string ConvertJsonString(string str)
{ {
@@ -748,11 +841,17 @@ namespace N_m3u8DL_CLI
} }
} }
if(res.Contains("Audio aac")) if (res.Contains("Audio aac"))
{ {
FFmpeg.UseAACFilter = true; FFmpeg.UseAACFilter = true;
} }
//有非AAC音轨则关闭UseAACFilter
if (res.Contains("Audio") && !res.Contains("Audio aac"))
{
FFmpeg.UseAACFilter = false;
}
if ((VIDEO_TYPE == "" || VIDEO_TYPE == "IGNORE") && res.Contains("Audio eac3")) if ((VIDEO_TYPE == "" || VIDEO_TYPE == "IGNORE") && res.Contains("Audio eac3"))
{ {
AUDIO_TYPE = "eac3"; AUDIO_TYPE = "eac3";
@@ -1048,6 +1147,8 @@ namespace N_m3u8DL_CLI
if (i > 0) if (i > 0)
{ {
int newTime = Convert.ToInt32(Regex.Match(tmp, "X-TIMESTAMP-MAP.*MPEGTS:(\\d+)").Groups[1].Value); int newTime = Convert.ToInt32(Regex.Match(tmp, "X-TIMESTAMP-MAP.*MPEGTS:(\\d+)").Groups[1].Value);
if (newTime == 900000)
continue;
//计算偏移量 //计算偏移量
//LOGGER.PrintLine((newTime - baseTime).ToString()); //LOGGER.PrintLine((newTime - baseTime).ToString());
addTime = addTime + ((newTime - baseTime) / 100); addTime = addTime + ((newTime - baseTime) / 100);

783
N_m3u8DL-CLI/MPDParser.cs Normal file

File diff suppressed because it is too large Load Diff

View File

@@ -1,5 +1,6 @@
<?xml version="1.0" encoding="utf-8"?> <?xml version="1.0" encoding="utf-8"?>
<Project ToolsVersion="15.0" xmlns="http://schemas.microsoft.com/developer/msbuild/2003"> <Project ToolsVersion="15.0" xmlns="http://schemas.microsoft.com/developer/msbuild/2003">
<Import Project="..\packages\Costura.Fody.4.1.0\build\Costura.Fody.props" Condition="Exists('..\packages\Costura.Fody.4.1.0\build\Costura.Fody.props')" />
<Import Project="$(MSBuildExtensionsPath)\$(MSBuildToolsVersion)\Microsoft.Common.props" Condition="Exists('$(MSBuildExtensionsPath)\$(MSBuildToolsVersion)\Microsoft.Common.props')" /> <Import Project="$(MSBuildExtensionsPath)\$(MSBuildToolsVersion)\Microsoft.Common.props" Condition="Exists('$(MSBuildExtensionsPath)\$(MSBuildToolsVersion)\Microsoft.Common.props')" />
<PropertyGroup> <PropertyGroup>
<Configuration Condition=" '$(Configuration)' == '' ">Debug</Configuration> <Configuration Condition=" '$(Configuration)' == '' ">Debug</Configuration>
@@ -12,6 +13,8 @@
<FileAlignment>512</FileAlignment> <FileAlignment>512</FileAlignment>
<AutoGenerateBindingRedirects>true</AutoGenerateBindingRedirects> <AutoGenerateBindingRedirects>true</AutoGenerateBindingRedirects>
<TargetFrameworkProfile /> <TargetFrameworkProfile />
<NuGetPackageImportStamp>
</NuGetPackageImportStamp>
</PropertyGroup> </PropertyGroup>
<PropertyGroup Condition=" '$(Configuration)|$(Platform)' == 'Debug|AnyCPU' "> <PropertyGroup Condition=" '$(Configuration)|$(Platform)' == 'Debug|AnyCPU' ">
<PlatformTarget>x86</PlatformTarget> <PlatformTarget>x86</PlatformTarget>
@@ -38,13 +41,16 @@
<ApplicationIcon>logo_3Iv_icon.ico</ApplicationIcon> <ApplicationIcon>logo_3Iv_icon.ico</ApplicationIcon>
</PropertyGroup> </PropertyGroup>
<ItemGroup> <ItemGroup>
<Reference Include="Costura, Version=4.1.0.0, Culture=neutral, PublicKeyToken=9919ef960d84173d, processorArchitecture=MSIL">
<HintPath>..\packages\Costura.Fody.4.1.0\lib\net40\Costura.dll</HintPath>
</Reference>
<Reference Include="Microsoft.JScript" /> <Reference Include="Microsoft.JScript" />
<Reference Include="netstandard, Version=2.0.0.0, Culture=neutral, PublicKeyToken=cc7b13ffcd2ddd51" /> <Reference Include="netstandard, Version=2.0.0.0, Culture=neutral, PublicKeyToken=cc7b13ffcd2ddd51" />
<Reference Include="Newtonsoft.Json"> <Reference Include="Newtonsoft.Json, Version=12.0.0.0, Culture=neutral, PublicKeyToken=30ad4fe6b2a6aeed, processorArchitecture=MSIL">
<HintPath>..\packages\Newtonsoft.Json.dll</HintPath> <HintPath>..\packages\Newtonsoft.Json.12.0.3\lib\net45\Newtonsoft.Json.dll</HintPath>
</Reference> </Reference>
<Reference Include="NiL.JS, Version=2.5.1428.0, Culture=neutral, PublicKeyToken=fa941a7c2a4de689, processorArchitecture=MSIL"> <Reference Include="NiL.JS, Version=2.5.1428.0, Culture=neutral, PublicKeyToken=fa941a7c2a4de689, processorArchitecture=MSIL">
<HintPath>..\packages\NiL.JS.dll</HintPath> <HintPath>..\packages\NiL.JS.2.5.1428\lib\net45\NiL.JS.dll</HintPath>
</Reference> </Reference>
<Reference Include="PresentationFramework" /> <Reference Include="PresentationFramework" />
<Reference Include="System" /> <Reference Include="System" />
@@ -63,6 +69,10 @@
<Compile Include="CommandLineArgument.cs" /> <Compile Include="CommandLineArgument.cs" />
<Compile Include="CommandLineArgumentParser.cs" /> <Compile Include="CommandLineArgumentParser.cs" />
<Compile Include="Decode51CtoKey.cs" /> <Compile Include="Decode51CtoKey.cs" />
<Compile Include="DecodeDdyun.cs" />
<Compile Include="DecodeHuke88Key.cs" />
<Compile Include="DecodeImooc.cs" />
<Compile Include="DecodeNfmovies.cs" />
<Compile Include="Decrypter.cs" /> <Compile Include="Decrypter.cs" />
<Compile Include="FFmpeg.cs" /> <Compile Include="FFmpeg.cs" />
<Compile Include="Global.cs" /> <Compile Include="Global.cs" />
@@ -70,10 +80,26 @@
<Compile Include="HLSTags.cs" /> <Compile Include="HLSTags.cs" />
<Compile Include="LOGGER.cs" /> <Compile Include="LOGGER.cs" />
<Compile Include="DownloadManager.cs" /> <Compile Include="DownloadManager.cs" />
<Compile Include="MPDParser.cs" />
<Compile Include="Parser.cs" /> <Compile Include="Parser.cs" />
<Compile Include="Program.cs" /> <Compile Include="Program.cs" />
<Compile Include="Properties\AssemblyInfo.cs" /> <Compile Include="Properties\AssemblyInfo.cs" />
<Compile Include="Downloader.cs" /> <Compile Include="Downloader.cs" />
<Compile Include="strings.Designer.cs">
<AutoGen>True</AutoGen>
<DesignTime>True</DesignTime>
<DependentUpon>strings.resx</DependentUpon>
</Compile>
<Compile Include="strings.en-US.Designer.cs">
<AutoGen>True</AutoGen>
<DesignTime>True</DesignTime>
<DependentUpon>strings.en-US.resx</DependentUpon>
</Compile>
<Compile Include="strings.zh-TW.Designer.cs">
<AutoGen>True</AutoGen>
<DesignTime>True</DesignTime>
<DependentUpon>strings.zh-TW.resx</DependentUpon>
</Compile>
<Compile Include="Watcher.cs" /> <Compile Include="Watcher.cs" />
</ItemGroup> </ItemGroup>
<ItemGroup> <ItemGroup>
@@ -82,9 +108,6 @@
</None> </None>
<None Include="packages.config" /> <None Include="packages.config" />
</ItemGroup> </ItemGroup>
<ItemGroup>
<None Include="bin\Debug\Newtonsoft.Json.dll" />
</ItemGroup>
<ItemGroup> <ItemGroup>
<COMReference Include="Scripting"> <COMReference Include="Scripting">
<Guid>{420B2830-E718-11CF-893D-00A0C9054228}</Guid> <Guid>{420B2830-E718-11CF-893D-00A0C9054228}</Guid>
@@ -100,16 +123,26 @@
<Content Include="logo_3Iv_icon.ico" /> <Content Include="logo_3Iv_icon.ico" />
</ItemGroup> </ItemGroup>
<ItemGroup> <ItemGroup>
<EmbeddedResource Include="Properties\Resources.resx"> <EmbeddedResource Include="strings.en-US.resx">
<Generator>ResXFileCodeGenerator</Generator> <Generator>ResXFileCodeGenerator</Generator>
<LastGenOutput>Resources.Designer.cs</LastGenOutput> <LastGenOutput>strings.en-US.Designer.cs</LastGenOutput>
</EmbeddedResource> </EmbeddedResource>
<EmbeddedResource Include="strings.en-US.resx" />
<EmbeddedResource Include="strings.resx"> <EmbeddedResource Include="strings.resx">
<Generator>ResXFileCodeGenerator</Generator> <Generator>ResXFileCodeGenerator</Generator>
<LastGenOutput>strings.Designer.cs</LastGenOutput> <LastGenOutput>strings.Designer.cs</LastGenOutput>
</EmbeddedResource> </EmbeddedResource>
<EmbeddedResource Include="strings.zh-TW.resx" /> <EmbeddedResource Include="strings.zh-TW.resx">
<Generator>ResXFileCodeGenerator</Generator>
<LastGenOutput>strings.zh-TW.Designer.cs</LastGenOutput>
</EmbeddedResource>
</ItemGroup> </ItemGroup>
<Import Project="$(MSBuildToolsPath)\Microsoft.CSharp.targets" /> <Import Project="$(MSBuildToolsPath)\Microsoft.CSharp.targets" />
</Project> <Import Project="..\packages\Fody.6.0.0\build\Fody.targets" Condition="Exists('..\packages\Fody.6.0.0\build\Fody.targets')" />
<Target Name="EnsureNuGetPackageBuildImports" BeforeTargets="PrepareForBuild">
<PropertyGroup>
<ErrorText>这台计算机上缺少此项目引用的 NuGet 程序包。使用“NuGet 程序包还原”可下载这些程序包。有关更多信息,请参见 http://go.microsoft.com/fwlink/?LinkID=322105。缺少的文件是 {0}。</ErrorText>
</PropertyGroup>
<Error Condition="!Exists('..\packages\Fody.6.0.0\build\Fody.targets')" Text="$([System.String]::Format('$(ErrorText)', '..\packages\Fody.6.0.0\build\Fody.targets'))" />
<Error Condition="!Exists('..\packages\Costura.Fody.4.1.0\build\Costura.Fody.props')" Text="$([System.String]::Format('$(ErrorText)', '..\packages\Costura.Fody.4.1.0\build\Costura.Fody.props'))" />
</Target>
</Project>

File diff suppressed because it is too large Load Diff

File diff suppressed because it is too large Load Diff

311
N_m3u8DL-CLI/changelog.txt Normal file

File diff suppressed because it is too large Load Diff

View File

@@ -1,4 +1,7 @@
<?xml version="1.0" encoding="utf-8"?> <?xml version="1.0" encoding="utf-8"?>
<packages> <packages>
<package id="Fody" version="6.0.0" targetFramework="net46" developmentDependency="true" />
<package id="Costura.Fody" version="4.1.0" targetFramework="net46" />
<package id="Newtonsoft.Json" version="12.0.3" targetFramework="net46" />
<package id="NiL.JS" version="2.5.1428" targetFramework="net46" /> <package id="NiL.JS" version="2.5.1428" targetFramework="net46" />
</packages> </packages>

File diff suppressed because one or more lines are too long

555
docs/GetM3u8.html Normal file

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long