import req from './req.js'; import {ENV} from './env.js'; import COOKIE from './cookieManager.js'; import '../libs_drpy/crypto-js.js'; import {join} from 'path'; import fs from 'fs'; import {PassThrough} from 'stream'; class QuarkHandler { constructor() { this.regex = /https:\/\/pan\.quark\.cn\/s\/([^\\|#/]+)/; this.pr = 'pr=ucpro&fr=pc'; this.baseHeader = { 'User-Agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) quark-cloud-drive/2.5.20 Chrome/100.0.4896.160 Electron/18.3.5.4-b478491100 Safari/537.36 Channel/pckk_other_ch', Referer: 'https://pan.quark.cn', }; this.apiUrl = 'https://drive.quark.cn/1/clouddrive/'; this.shareTokenCache = {}; this.saveDirName = 'drpy'; this.saveDirId = null; this.saveFileIdCaches = {}; this.currentUrlKey = ''; this.cacheRoot = (process.env['NODE_PATH'] || '.') + '/quark_cache'; this.maxCache = 1024 * 1024 * 100; this.urlHeadCache = {}; this.subtitleExts = ['.srt', '.ass', '.scc', '.stl', '.ttml']; } // 使用 getter 定义动态属性 get cookie() { // console.log('env.cookie.quark:',ENV.get('quark_cookie')); return ENV.get('quark_cookie'); } getShareData(url) { let matches = this.regex.exec(url); if (matches.indexOf("?") > 0) { matches = matches.split('?')[0]; } if (matches) { return { shareId: matches[1], folderId: '0', }; } return null; } async initQuark(db, cfg) { if (this.cookie) { console.log("cookie 获取成功"); } else { console.log("cookie 获取失败") } } lcs(str1, str2) { if (!str1 || !str2) { return { length: 0, sequence: '', offset: 0, }; } var sequence = ''; var str1Length = str1.length; var str2Length = str2.length; var num = new Array(str1Length); var maxlen = 0; var lastSubsBegin = 0; for (var i = 0; i < str1Length; i++) { var subArray = new Array(str2Length); for (var j = 0; j < str2Length; j++) { subArray[j] = 0; } num[i] = subArray; } var thisSubsBegin = null; for (i = 0; i < str1Length; i++) { for (j = 0; j < str2Length; j++) { if (str1[i] !== str2[j]) { num[i][j] = 0; } else { if (i === 0 || j === 0) { num[i][j] = 1; } else { num[i][j] = 1 + num[i - 1][j - 1]; } if (num[i][j] > maxlen) { maxlen = num[i][j]; thisSubsBegin = i - num[i][j] + 1; if (lastSubsBegin === thisSubsBegin) { sequence += str1[i]; } else { lastSubsBegin = thisSubsBegin; sequence = ''; // clear it sequence += str1.substr(lastSubsBegin, i + 1 - lastSubsBegin); } } } } } return { length: maxlen, sequence: sequence, offset: thisSubsBegin, }; } findBestLCS(mainItem, targetItems) { const results = []; let bestMatchIndex = 0; for (let i = 0; i < targetItems.length; i++) { const currentLCS = this.lcs(mainItem.name, targetItems[i].name); results.push({target: targetItems[i], lcs: currentLCS}); if (currentLCS.length > results[bestMatchIndex].lcs.length) { bestMatchIndex = i; } } const bestMatch = results[bestMatchIndex]; return {allLCS: results, bestMatch: bestMatch, bestMatchIndex: bestMatchIndex}; } delay(ms) { return new Promise((resolve) => setTimeout(resolve, ms)); } async api(url, data, headers, method, retry) { let cookie = this.cookie || ''; headers = headers || {}; Object.assign(headers, this.baseHeader); Object.assign(headers, { Cookie: cookie }); method = method || 'post'; const resp = method === 'get' ? await req.get(`${this.apiUrl}/${url}`, { headers: headers, }).catch((err) => { console.error(err.message); return err.response || {status: 500, data: {}}; }) : await req.post(`${this.apiUrl}/${url}`, data, { headers: headers, }).catch((err) => { console.error(err.message); return err.response || {status: 500, data: {}}; }); const leftRetry = retry || 3; if (resp.headers['set-cookie']) { const puus = resp.headers['set-cookie'].join(';;;').match(/__puus=([^;]+)/); if (puus) { if (cookie.match(/__puus=([^;]+)/)[1] !== puus[1]) { cookie = cookie.replace(/__puus=[^;]+/, `__puus=${puus[1]}`); console.log('[quark] api:更新cookie:', cookie); ENV.set('quark_cookie', cookie); } } } if (resp.status === 429 && leftRetry > 0) { await this.delay(1000); return await this.api(url, data, headers, method, leftRetry - 1); } return resp.data || {}; } async clearSaveDir() { const listData = await this.api(`file/sort?${this.pr}&pdir_fid=${this.saveDirId}&_page=1&_size=200&_sort=file_type:asc,updated_at:desc`, {}, {}, 'get'); if (listData.data && listData.data.list && listData.data.list.length > 0) { const del = await this.api(`file/delete?${this.pr}`, { action_type: 2, filelist: listData.data.list.map((v) => v.fid), exclude_fids: [], }); // console.log(del); } } async createSaveDir(clean) { if (this.saveDirId) { if (clean) await this.clearSaveDir(); return; } const listData = await this.api(`file/sort?${this.pr}&pdir_fid=0&_page=1&_size=200&_sort=file_type:asc,updated_at:desc`, {}, {}, 'get'); if (listData.data && listData.data.list) for (const item of listData.data.list) { if (item.file_name === this.saveDirName) { this.saveDirId = item.fid; await this.clearSaveDir(); break; } } if (!this.saveDirId) { const create = await this.api(`file?${this.pr}`, { pdir_fid: '0', file_name: this.saveDirName, dir_path: '', dir_init_lock: false, }); console.log(create); if (create.data && create.data.fid) { this.saveDirId = create.data.fid; } } } async getShareToken(shareData) { if (!this.shareTokenCache[shareData.shareId]) { delete this.shareTokenCache[shareData.shareId]; const shareToken = await this.api(`share/sharepage/token?${this.pr}`, { pwd_id: shareData.shareId, passcode: shareData.sharePwd || '', }); if (shareToken.data && shareToken.data.stoken) { this.shareTokenCache[shareData.shareId] = shareToken.data; } } } async getFilesByShareUrl(shareInfo) { const shareData = typeof shareInfo === 'string' ? this.getShareData(shareInfo) : shareInfo; if (!shareData) return []; await this.getShareToken(shareData); if (!this.shareTokenCache[shareData.shareId]) return []; const videos = []; const subtitles = []; const listFile = async (shareId, folderId, page) => { const prePage = 200; page = page || 1; const listData = await this.api(`share/sharepage/detail?${this.pr}&pwd_id=${shareId}&stoken=${encodeURIComponent(this.shareTokenCache[shareId].stoken)}&pdir_fid=${folderId}&force=0&_page=${page}&_size=${prePage}&_sort=file_type:asc,file_name:asc`, {}, {}, 'get'); if (!listData.data) return []; const items = listData.data.list; if (!items) return []; const subDir = []; for (const item of items) { if (item.dir === true) { subDir.push(item); } else if (item.file === true && item.obj_category === 'video') { if (item.size < 1024 * 1024 * 5) continue; item.stoken = this.shareTokenCache[shareData.shareId].stoken; videos.push(item); } else if (item.type === 'file' && this.subtitleExts.some((x) => item.file_name.endsWith(x))) { subtitles.push(item); } } if (page < Math.ceil(listData.metadata._total / prePage)) { const nextItems = await listFile(shareId, folderId, page + 1); for (const item of nextItems) { items.push(item); } } for (const dir of subDir) { const subItems = await listFile(shareId, dir.fid); for (const item of subItems) { items.push(item); } } return items; }; await listFile(shareData.shareId, shareData.folderId); if (subtitles.length > 0) { videos.forEach((item) => { var matchSubtitle = this.findBestLCS(item, subtitles); if (matchSubtitle.bestMatch) { item.subtitle = matchSubtitle.bestMatch.target; } }); } return videos; } async save(shareId, stoken, fileId, fileToken, clean) { await this.createSaveDir(clean); if (clean) { const saves = Object.keys(this.saveFileIdCaches); for (const save of saves) { delete this.saveFileIdCaches[save]; } } if (!this.saveDirId) return null; if (!stoken) { await this.getShareToken({ shareId: shareId, }); if (!this.shareTokenCache[shareId]) return null; } const saveResult = await this.api(`share/sharepage/save?${this.pr}`, { fid_list: [fileId], fid_token_list: [fileToken], to_pdir_fid: this.saveDirId, pwd_id: shareId, stoken: stoken || this.shareTokenCache[shareId].stoken, pdir_fid: '0', scene: 'link', }); if (saveResult.data && saveResult.data.task_id) { let retry = 0; while (true) { const taskResult = await this.api(`task?${this.pr}&task_id=${saveResult.data.task_id}&retry_index=${retry}`, {}, {}, 'get'); if (taskResult.data && taskResult.data.save_as && taskResult.data.save_as.save_as_top_fids && taskResult.data.save_as.save_as_top_fids.length > 0) { return taskResult.data.save_as.save_as_top_fids[0]; } retry++; if (retry > 5) break; await this.delay(1000); } } return true; } async refreshQuarkCookie(from = '') { const nowCookie = this.cookie; const cookieSelfRes = await axios({ url: "https://drive-pc.quark.cn/1/clouddrive/file/sort?pr=ucpro&fr=pc&uc_param_str=&pdir_fid=0&_page=1&_size=50&_fetch_total=1&_fetch_sub_dirs=0&_sort=file_type:asc,updated_at:desc", method: "GET", headers: { "User-Agent": 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) quark-cloud-drive/2.5.20 Chrome/100.0.4896.160 Electron/18.3.5.4-b478491100 Safari/537.36 Channel/pckk_other_ch', Origin: 'https://pan.quark.cn', Referer: 'https://pan.quark.cn/', Cookie: nowCookie } }); const cookieResDataSelf = cookieSelfRes.headers; const resCookie = cookieResDataSelf['set-cookie']; if (!resCookie) { console.log(`${from}自动更新夸克 cookie: 没返回新的cookie`); return } const cookieObject = COOKIE.parse(resCookie); // console.log(cookieObject); if (cookieObject.__puus) { const oldCookie = COOKIE.parse(nowCookie); const newCookie = COOKIE.stringify({ __pus: oldCookie.__pus, __puus: cookieObject.__puus, }); console.log(`${from}自动更新夸克 cookie: ${newCookie}`); ENV.set('quark_cookie', newCookie); } } async getLiveTranscoding(shareId, stoken, fileId, fileToken) { if (!this.saveFileIdCaches[fileId]) { const saveFileId = await this.save(shareId, stoken, fileId, fileToken, true); if (!saveFileId) return null; this.saveFileIdCaches[fileId] = saveFileId; } const transcoding = await this.api(`file/v2/play?${this.pr}`, { fid: this.saveFileIdCaches[fileId], resolutions: 'normal,low,high,super,2k,4k', supports: 'fmp4', }); if (transcoding.data && transcoding.data.video_list) { const low_url = transcoding.data.video_list.slice(-1)[0].video_info.url; const low_cookie = this.cookie; const low_headers = { 'User-Agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/131.0.0.0 Safari/537.36', 'origin': 'https://pan.quark.cn', 'referer': 'https://pan.quark.cn/', 'Cookie': low_cookie }; // console.log('low_url:', low_url); // console.log('low_cookie:', low_cookie); const test_result = await this.testSupport(low_url, low_headers); // console.log(test_result); if (!test_result[0]) { try { await this.refreshQuarkCookie('getLiveTranscoding'); } catch (e) { console.log(`getLiveTranscoding:自动刷新夸克cookie失败:${e.message}`); console.error(e); } } return transcoding.data.video_list; } return null; } async getDownload(shareId, stoken, fileId, fileToken, clean) { if (!this.saveFileIdCaches[fileId]) { const saveFileId = await this.save(shareId, stoken, fileId, fileToken, clean); if (!saveFileId) return null; this.saveFileIdCaches[fileId] = saveFileId; } const down = await this.api(`file/download?${this.pr}`, { fids: [this.saveFileIdCaches[fileId]], }); if (down.data) { const low_url = down.data[0].download_url; const low_cookie = this.cookie; const low_headers = { 'User-Agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/131.0.0.0 Safari/537.36', 'origin': 'https://pan.quark.cn', 'referer': 'https://pan.quark.cn/', 'Cookie': low_cookie }; // console.log('low_url:', low_url); // console.log('low_cookie:', low_cookie); const test_result = await this.testSupport(low_url, low_headers); // console.log('test_result:', test_result); if (!test_result[0]) { try { await this.refreshQuarkCookie('getDownload'); } catch (e) { console.log(`getDownload:自动刷新Quark cookie失败:${e.message}`) } } return down.data[0]; } return null; } async testSupport(url, headers) { const resp = await req .get(url, { responseType: 'stream', headers: Object.assign( { Range: 'bytes=0-0', }, headers, ), }) .catch((err) => { // console.error(err); console.error('[testSupport] error:', err.message); return err.response || {status: 500, data: {}}; }); if (resp && (resp.status === 206 || resp.status === 200)) { const isAccept = resp.headers['accept-ranges'] === 'bytes'; const contentRange = resp.headers['content-range']; const contentLength = parseInt(resp.headers['content-length']); const isSupport = isAccept || !!contentRange || contentLength === 1 || resp.status === 200; const length = contentRange ? parseInt(contentRange.split('/')[1]) : contentLength; delete resp.headers['content-range']; delete resp.headers['content-length']; if (length) resp.headers['content-length'] = length.toString(); return [isSupport, resp.headers]; } else { console.log('[testSupport] resp.status:', resp.status); return [false, null]; } } delAllCache(keepKey) { try { fs.readdir(this.cacheRoot, (_, files) => { if (files) for (const file of files) { if (file === keepKey) continue; const dir = join(this.cacheRoot, file); fs.stat(dir, (_, stats) => { if (stats && stats.isDirectory()) { fs.readdir(dir, (_, subFiles) => { if (subFiles) for (const subFile of subFiles) { if (!subFile.endsWith('.p')) { fs.rm(join(dir, subFile), {recursive: true}, () => { }); } } }); } }); } }); } catch (error) { console.error(error); } } async chunkStream(inReq, outResp, url, urlKey, headers, option) { urlKey = urlKey || CryptoJS.enc.Hex.stringify(CryptoJS.MD5(url)).toString(); if (this.currentUrlKey !== urlKey) { this.delAllCache(urlKey); this.currentUrlKey = urlKey; } if (!this.urlHeadCache[urlKey]) { const [isSupport, urlHeader] = await this.testSupport(url, headers); if (!isSupport || !urlHeader['content-length']) { outResp.redirect(url); return; } this.urlHeadCache[urlKey] = urlHeader; } let exist = true; await fs.promises.access(join(this.cacheRoot, urlKey)).catch((_) => (exist = false)); if (!exist) { await fs.promises.mkdir(join(this.cacheRoot, urlKey), {recursive: true}); } const contentLength = parseInt(this.urlHeadCache[urlKey]['content-length']); let byteStart = 0; let byteEnd = contentLength - 1; const streamHeader = {}; if (inReq.headers.range) { const ranges = inReq.headers.range.trim().split(/=|-/); if (ranges.length > 2 && ranges[2]) { byteEnd = parseInt(ranges[2]); } byteStart = parseInt(ranges[1]); Object.assign(streamHeader, this.urlHeadCache[urlKey]); streamHeader['content-length'] = (byteEnd - byteStart + 1).toString(); streamHeader['content-range'] = `bytes ${byteStart}-${byteEnd}/${contentLength}`; outResp.code(206); } else { Object.assign(streamHeader, this.urlHeadCache[urlKey]); outResp.code(200); } option = option || {chunkSize: 1024 * 256, poolSize: 5, timeout: 1000 * 10}; const chunkSize = option.chunkSize; const poolSize = option.poolSize; const timeout = option.timeout; let chunkCount = Math.ceil(contentLength / chunkSize); let chunkDownIdx = Math.floor(byteStart / chunkSize); let chunkReadIdx = chunkDownIdx; let stop = false; const dlFiles = {}; for (let i = 0; i < poolSize && i < chunkCount; i++) { new Promise((resolve) => { (async function doDLTask(spChunkIdx) { if (stop || chunkDownIdx >= chunkCount) { resolve(); return; } if (spChunkIdx === undefined && (chunkDownIdx - chunkReadIdx) * chunkSize >= this.maxCache) { setTimeout(doDLTask, 5); return; } const chunkIdx = spChunkIdx || chunkDownIdx++; const taskId = `${inReq.id}-${chunkIdx}`; try { const dlFile = join(this.cacheRoot, urlKey, `${inReq.id}-${chunkIdx}.p`); let exist = true; await fs.promises.access(dlFile).catch((_) => (exist = false)); if (!exist) { const start = chunkIdx * chunkSize; const end = Math.min(contentLength - 1, (chunkIdx + 1) * chunkSize - 1); console.log(inReq.id, chunkIdx); const dlResp = await req.get(url, { responseType: 'stream', timeout: timeout, headers: Object.assign( { Range: `bytes=${start}-${end}`, }, headers, ), }); const dlCache = join(this.cacheRoot, urlKey, `${inReq.id}-${chunkIdx}.dl`); const writer = fs.createWriteStream(dlCache); const readTimeout = setTimeout(() => { writer.destroy(new Error(`${taskId} read timeout`)); }, timeout); const downloaded = new Promise((resolve) => { writer.on('finish', async () => { if (stop) { await fs.promises.rm(dlCache).catch((e) => console.error(e)); } else { await fs.promises.rename(dlCache, dlFile).catch((e) => console.error(e)); dlFiles[taskId] = dlFile; } resolve(true); }); writer.on('error', async (e) => { console.error(e); await fs.promises.rm(dlCache).catch((e1) => console.error(e1)); resolve(false); }); }); dlResp.data.pipe(writer); const result = await downloaded; clearTimeout(readTimeout); if (!result) { setTimeout(() => { doDLTask(chunkIdx); }, 15); return; } } setTimeout(doDLTask, 5); } catch (error) { console.error(error); setTimeout(() => { doDLTask(chunkIdx); }, 15); } })(); }); } outResp.headers(streamHeader); const stream = new PassThrough(); new Promise((resolve) => { let writeMore = true; (async function waitReadFile() { try { if (chunkReadIdx >= chunkCount || stop) { stream.end(); resolve(); return; } if (!writeMore) { setTimeout(waitReadFile, 5); return; } const taskId = `${inReq.id}-${chunkReadIdx}`; if (!dlFiles[taskId]) { setTimeout(waitReadFile, 5); return; } const chunkByteStart = chunkReadIdx * chunkSize; const chunkByteEnd = Math.min(contentLength - 1, (chunkReadIdx + 1) * chunkSize - 1); const readFileStart = Math.max(byteStart, chunkByteStart) - chunkByteStart; const dlFile = dlFiles[taskId]; delete dlFiles[taskId]; const fd = await fs.promises.open(dlFile, 'r'); const buffer = Buffer.alloc(chunkByteEnd - chunkByteStart - readFileStart + 1); await fd.read(buffer, 0, chunkByteEnd - chunkByteStart - readFileStart + 1, readFileStart); await fd.close().catch((e) => console.error(e)); await fs.promises.rm(dlFile).catch((e) => console.error(e)); writeMore = stream.write(buffer); if (!writeMore) { stream.once('drain', () => { writeMore = true; }); } chunkReadIdx++; setTimeout(waitReadFile, 5); } catch (error) { setTimeout(waitReadFile, 5); } })(); }); stream.on('close', async () => { Object.keys(dlFiles).forEach((reqKey) => { if (reqKey.startsWith(inReq.id)) { fs.rm(dlFiles[reqKey], {recursive: true}, () => { }); delete dlFiles[reqKey]; } }); stop = true; }); return stream; } } export const Quark = new QuarkHandler();