1use reqrio::{json, HlsError, Method, ReqExt, ScReq, Timeout, ALPN, Cipher};
2use std::fs::File;
3use std::io::Write;
4
5struct M3u8DownEngine {
6 req: ScReq,
7 key_url: String,
8 sequence: u128,
9 index_url: String,
10 ts_urls: Vec<String>,
11 cipher: Cipher,
12}
13
14impl M3u8DownEngine {
15 fn new(index: impl ToString) -> M3u8DownEngine {
16 let mut req = ScReq::new().with_alpn(ALPN::Http20); req.set_headers_json(json::object! {
18 "Host": "",
19 "sec-ch-ua-platform": "Android",
20 "user-agent": "Mozilla/5.0 (Linux; Android 13; Pixel 7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/116.0.0.0 Mobile Safari/537.36",
21 "sec-ch-ua": r#""Android WebView";v="135", "Not-A.Brand";v="8", "Chromium";v="135""#,
22 "sec-ch-ua-mobile": "?1",
23 "accept": "*/*",
24 "origin": "",
25 "x-requested-with": "mark.via",
26 "sec-fetch-site": "cross-site",
27 "sec-fetch-mode": "cors",
28 "sec-fetch-dest": "empty",
29 "referer": "",
30 "accept-encoding": "gzip, deflate, br, zstd",
31 "accept-language": "en,zh-CN;q=0.9,zh;q=0.8,en-US;q=0.7",
32 "priority": "u=1, i"
33 }).unwrap();
34 M3u8DownEngine {
35 req,
36 index_url: index.to_string(),
37 sequence: 0,
38 key_url: "".to_string(),
39 ts_urls: vec![],
40 cipher: Cipher::aes_128_ecb(),
41 }
42 }
43
44 fn down_ts(&mut self) -> Result<(), HlsError> {
45 let mut file = File::create("4.mp4")?;
46 for (index, ts_url) in self.ts_urls.iter().enumerate() {
47 println!("Downloading: {:3}/{}; url:{}", index, self.ts_urls.len(), ts_url);
48 self.req.set_url(ts_url)?;
49 let body = self.req.get()?.bytes()?;
50 file.write_all(&if self.key_url.is_empty() { body } else { self.cipher.decrypt(body)? })?;
51 }
52 Ok(())
53 }
54
55 fn get_key(&mut self) -> Result<(), HlsError> {
56 println!("key url: {}", self.key_url);
57 self.req.set_url(self.key_url.as_str())?;
58 let key = self.req.send_check(Method::GET)?.text()?;
59 println!("key: {}; sequence: {}", key, self.sequence);
60 self.cipher.set_secret_key(key.into_bytes(), Some(self.sequence.to_be_bytes().to_vec()));
61 Ok(())
62 }
63
64 fn download(&mut self) -> Result<(), HlsError> {
65 self.req.set_url(self.index_url.as_str())?;
66 let body = self.req.send_check(Method::GET)?.text()?;
67 for line in body.split("\n") {
69 if line.starts_with("#EXT-X-MEDIA-SEQUENCE:") {
70 self.sequence = line.trim().replace("#EXT-X-MEDIA-SEQUENCE:", "").parse()?;
71 continue;
72 }
73 if line.starts_with("#EXT-X-KEY") {
74 let pos = line.find("URI=\"");
75 if let Some(pos) = pos {
76 self.key_url = line[pos + 4..].trim().replace("\"", "");
77 }
78 if line.contains("=AES-128,") {
79 self.cipher = Cipher::aes_128_cbc();
80 }
81 continue;
82 }
83 if line.starts_with("http") {
84 self.ts_urls.push(line.trim().to_string());
85 }
86 }
87 let mut timeout = Timeout::new_same(5000, 30000);
88 timeout.set_handle_times(10);
89 self.req.set_timeout(timeout);
90 if !self.key_url.is_empty() { self.get_key()?; }
91 self.down_ts()?;
92 Ok(())
93 }
94
95}
96
97
98fn main() {
99 let index = "";
100 let mut engine = M3u8DownEngine::new(index);
101 engine.download().unwrap();
102 println!("{:?} {:?}", engine.key_url, engine.ts_urls);
103}