// ==UserScript== // @name 123云盘解锁 // @author QingJ // @namespace https://github.com/QingJ01/123pan_unlock // @version 1.2.0 // @description 专业的123云盘增强脚本 - 完美解锁会员功能、突破下载限制、去广告、支持自定义用户信息。整合秒传链接功能,支持生成和保存秒传文件,快速分享和保存文件。界面精美,功能强大,让你的123云盘体验更美好! // @contributor Baoqing、Chaofan、lipkiat - 123FastLink秒传功能核心贡献者 // @contributor hmjz100 - 借鉴了部分适配代码 // @license Apache Licence 2 // @icon  // @match *://*.123pan.com/* // @match *://*.123pan.cn/* // @match *://*.123684.com/* // @match *://*.123865.com/* // @match *://*.123952.com/* // @match *://*.123912.com/* // @grant GM_setValue // @grant GM_getValue // @grant unsafeWindow // @grant GM_info // @grant GM_registerMenuCommand // @grant GM_setClipboard // @grant GM_unregisterMenuCommand // @grant GM_xmlhttpRequest // @run-at document-start // ==/UserScript== (function () { 'use strict'; // 检测unsafeWindow if (typeof (unsafeWindow) === 'undefined') window.unsafeWindow = window; // 配置验证和加载函数 function loadConfig() { const defaultConfig = { vip: 1, svip: 1, pvip: 0, ad: 1, name: "QingJ", photo: "http://q.qlogo.cn/headimg_dl?dst_uin=2903609300&spec=640&img_type=jpg", mail: "", phone: "", id: "", level: 128, endtime: 253402185600, debug: 0 }; const config = {}; for (const [key, defaultValue] of Object.entries(defaultConfig)) { let value = GM_getValue(key, defaultValue); // 类型验证 if (typeof value !== typeof defaultValue) { console.warn(`[123云盘解锁] 配置项 ${key} 类型错误,使用默认值`); value = defaultValue; } // 数值范围验证 if (key === 'level' && (typeof value !== 'number' || value < 0 || value > 128)) { console.warn(`[123云盘解锁] 等级值超出范围,使用默认值`); value = 128; } // VIP逻辑验证 if (key === 'vip' && (value !== 0 && value !== 1)) { value = 1; } if (key === 'svip' && (value !== 0 && value !== 1)) { value = 1; } if (key === 'pvip' && (value !== 0 && value !== 1)) { value = 0; } // URL验证(宽松检查) if (key === 'photo' && value && typeof value === 'string') { if (value.length > 2000) { console.warn(`[123云盘解锁] 头像URL过长,使用默认值`); value = defaultValue; } } config[key] = value; } return config; } // 从存储中读取配置(带验证) var user = loadConfig(); // ============================================================================= // 秒传功能模块 - 整合自 123FastLink // 特别鸣谢:@Baoqing、@Chaofan、@lipkiat // ============================================================================= // 秒传配置 const FastLinkConfig = { enabled: GM_getValue('fastlink_enabled', 1), getFileListPageDelay: 500, getFileInfoBatchSize: 100, getFileInfoDelay: 200, getFolderInfoDelay: 300, saveLinkDelay: 100, mkdirDelay: 100, usesBase62EtagsInExport: true, COMMON_PATH_LINK_PREFIX: "123FLCPV2$" }; // 1. API通信类 class PanApiClient { constructor() { this.host = 'https://' + window.location.host; this.authToken = null; this.loginUuid = null; this.progress = 0; this.progressDesc = ""; } init() { this.authToken = localStorage['authorToken']; this.loginUuid = localStorage['LoginUuid']; } buildURL(path, queryParams) { const queryString = new URLSearchParams(queryParams || {}).toString(); return `${this.host}${path}?${queryString}`; } async sendRequest(method, path, queryParams, body) { const headers = { 'Content-Type': 'application/json;charset=UTF-8', 'Authorization': 'Bearer ' + this.authToken, 'platform': 'web', 'App-Version': '3', 'LoginUuid': this.loginUuid, 'Origin': this.host, 'Referer': document.location.href, }; try { const response = await fetch(this.buildURL(path, queryParams), { method, headers, body, credentials: 'include' }); const data = await response.json(); if (data.code !== 0) { throw new Error(data.message); } return data; } catch (e) { console.error('[123云盘解锁] API请求失败:', e); throw e; } } async getOnePageFileList(parentFileId, page) { const urlParams = { driveId: '0', limit: '100', next: '0', orderBy: 'file_name', orderDirection: 'asc', parentFileId: parentFileId.toString(), trashed: 'false', SearchData: '', Page: page.toString(), OnlyLookAbnormalFile: '0', event: 'homeListFile', operateType: '1', inDirectSpace: 'false' }; const data = await this.sendRequest("GET", "/b/api/file/list/new", urlParams); return { data: { InfoList: data.data.InfoList }, total: data.data.Total }; } async getFileList(parentFileId) { let InfoList = []; this.progress = 0; this.progressDesc = `获取文件列表 ID:${parentFileId}`; const info = await this.getOnePageFileList(parentFileId, 1); InfoList.push(...info.data.InfoList); const total = info.total; if (total > 100) { const times = Math.ceil(total / 100); for (let i = 2; i < times + 1; i++) { this.progress = Math.ceil((i / times) * 100); const pageInfo = await this.getOnePageFileList(parentFileId, i); InfoList.push(...pageInfo.data.InfoList); await new Promise(resolve => setTimeout(resolve, FastLinkConfig.getFileListPageDelay)); } } this.progress = 100; return { data: { InfoList }, total: total }; } async getFileInfo(idList) { const fileIdList = idList.map(fileId => ({ fileId })); const data = await this.sendRequest("POST", "/b/api/file/info", {}, JSON.stringify({ fileIdList })); return { data: { InfoList: data.data.infoList } }; } async uploadRequest(fileInfo) { try { const response = await this.sendRequest('POST', '/b/api/file/upload_request', {}, JSON.stringify({ ...fileInfo, RequestSource: null })); const reuse = response['data']['Reuse']; if (response['code'] !== 0) { return [false, response['message']]; } if (!reuse) { return [false, "未能实现秒传"]; } else { return [true, null]; } } catch (error) { console.error('[123云盘解锁] 上传请求失败:', error); return [false, '请求失败']; } } async getParentFileId() { const homeFilePath = JSON.parse(sessionStorage['filePath'])['homeFilePath']; const parentFileId = (homeFilePath[homeFilePath.length - 1] || 0); return parentFileId.toString(); } async getFile(fileInfo, parentFileId) { if (!parentFileId) { parentFileId = await this.getParentFileId(); } return await this.uploadRequest({ driveId: 0, etag: fileInfo.etag, fileName: fileInfo.fileName, parentFileId, size: fileInfo.size, type: 0, duplicate: 1 }); } async mkdir(parentFileId, folderName = "New Folder") { let folderFileId = null; try { const response = await this.sendRequest('POST', '/b/api/file/upload_request', {}, JSON.stringify({ driveId: 0, etag: "", fileName: folderName, parentFileId, size: 0, type: 1, duplicate: 1, NotReuse: true, event: "newCreateFolder", operateType: 1, RequestSource: null })); folderFileId = response['data']['Info']['FileId']; } catch (error) { console.error('[123云盘解锁] 创建文件夹失败:', error); return { 'folderFileId': null, 'folderName': folderName, 'success': false }; } return { 'folderFileId': folderFileId, 'folderName': folderName, 'success': true }; } } // 2. 选中文件管理类 class TableRowSelector { constructor() { this.selectedRowKeys = []; this.unselectedRowKeys = []; this.isSelectAll = false; this._inited = false; } init() { if (this._inited) return; this._inited = true; const originalCreateElement = document.createElement; const self = this; document.createElement = function (tagName, options) { const element = originalCreateElement.call(document, tagName, options); const observer = new MutationObserver(() => { if (element.classList.contains('ant-table-row') && element.classList.contains('ant-table-row-level-0')) { const input = element.querySelector('input'); if (input) { input.addEventListener('click', function () { const rowKey = element.getAttribute('data-row-key'); if (self.isSelectAll) { if (!this.checked) { if (!self.unselectedRowKeys.includes(rowKey)) { self.unselectedRowKeys.push(rowKey); } } else { const idx = self.unselectedRowKeys.indexOf(rowKey); if (idx > -1) { self.unselectedRowKeys.splice(idx, 1); } } } else { if (this.checked) { if (!self.selectedRowKeys.includes(rowKey)) { self.selectedRowKeys.push(rowKey); } } else { const idx = self.selectedRowKeys.indexOf(rowKey); if (idx > -1) { self.selectedRowKeys.splice(idx, 1); } } } }); } observer.disconnect(); } else if (element.classList.contains('ant-checkbox-input') && element.getAttribute('aria-label') === 'Select all') { if (!(element.parentElement.classList.contains('ant-checkbox-indeterminate') || element.parentElement.classList.contains('ant-checkbox-checked'))) { self.unselectedRowKeys = []; self.selectedRowKeys = []; self.isSelectAll = false; } self._bindSelectAllEvent(element); } else if (element.classList.contains('ant-btn') && element.classList.contains('ant-btn-link')) { element.addEventListener('click', function () { self.selectedRowKeys = []; self.unselectedRowKeys = []; self.isSelectAll = false; }); } }); observer.observe(element, { attributes: true, attributeFilter: ['class', 'aria-label'] }); return element; }; } _bindSelectAllEvent(checkbox) { if (checkbox.dataset.selectAllBound) return; checkbox.dataset.selectAllBound = 'true'; checkbox.addEventListener('click', () => { if (checkbox.checked) { this.isSelectAll = true; this.unselectedRowKeys = []; this.selectedRowKeys = []; } else { this.isSelectAll = false; this.selectedRowKeys = []; this.unselectedRowKeys = []; } }); } getSelection() { return { isSelectAll: this.isSelectAll, selectedRowKeys: [...this.selectedRowKeys], unselectedRowKeys: [...this.unselectedRowKeys] }; } } // 3. 秒传链接管理类 class ShareLinkManager { constructor(apiClient) { this.apiClient = apiClient; this.progress = 0; this.progressDesc = ""; this.taskCancel = false; this.fileInfoList = []; this.commonPath = ""; } async _getAllFileInfoByFolderId(parentFileId, folderName = '', total) { this.progressDesc = `正在扫描文件夹:${folderName}`; let progress = this.progress; const progressUpdater = setInterval(() => { this.progress = progress + this.apiClient.progress / total; this.progressDesc = this.apiClient.progressDesc; if (this.progress > 100) { clearInterval(progressUpdater); } }, 500); const allFileInfoList = (await this.apiClient.getFileList(parentFileId)).data.InfoList.map(file => ({ fileName: file.FileName, etag: file.Etag, size: file.Size, type: file.Type, fileId: file.FileId })); clearInterval(progressUpdater); const fileInfo = allFileInfoList.filter(file => file.type !== 1); fileInfo.forEach(file => { file.path = folderName + file.fileName; }); this.fileInfoList.push(...fileInfo); const directoryFileInfo = allFileInfoList.filter(file => file.type === 1); for (const folder of directoryFileInfo) { await new Promise(resolve => setTimeout(resolve, FastLinkConfig.getFolderInfoDelay)); if (this.taskCancel) { this.progressDesc = "任务已取消"; return; } await this._getAllFileInfoByFolderId(folder.fileId, folderName + folder.fileName + "/", total * directoryFileInfo.length); } this.progress = progress + 100 / total; } async _getFileInfoBatch(idList) { const total = idList.length; let completed = 0; let allFileInfo = []; for (let i = 0; i < total; i += FastLinkConfig.getFileInfoBatchSize) { const batch = idList.slice(i, i + FastLinkConfig.getFileInfoBatchSize); try { const response = await this.apiClient.getFileInfo(batch); allFileInfo = allFileInfo.concat(response.data.InfoList || []); } catch (e) { console.error('[123云盘解锁] 获取文件信息失败:', e); } completed += batch.length; this.progress = Math.round((completed / total) * 100 - 1); this.progressDesc = `正在获取文件信息... (${completed} / ${total})`; await new Promise(resolve => setTimeout(resolve, FastLinkConfig.getFileInfoDelay)); } return allFileInfo.map(file => ({ fileName: file.FileName, etag: file.Etag, size: file.Size, type: file.Type, fileId: file.FileId })); } async _getCommonPath() { if (!this.fileInfoList || this.fileInfoList.length === 0) return ''; const pathArrays = this.fileInfoList.map(file => { const path = file.path || ''; const lastSlashIndex = path.lastIndexOf('/'); return lastSlashIndex === -1 ? [] : path.substring(0, lastSlashIndex).split('/'); }); let commonPrefix = []; const firstPath = pathArrays[0]; for (let i = 0; i < firstPath.length; i++) { const currentComponent = firstPath[i]; const allMatch = pathArrays.every(pathArray => pathArray.length > i && pathArray[i] === currentComponent); if (allMatch) { commonPrefix.push(currentComponent); } else { break; } } const commonPath = commonPrefix.length > 0 ? commonPrefix.join('/') + '/' : ''; this.commonPath = commonPath; return commonPath; } async _getSelectedFilesInfo(fileSelectionDetails) { this.fileInfoList = []; if (!fileSelectionDetails.isSelectAll && fileSelectionDetails.selectedRowKeys.length === 0) { return false; } let fileSelectFolderInfoList = []; if (fileSelectionDetails.isSelectAll) { this.progress = 10; this.progressDesc = "正在递归获取选择的文件..." let allFileInfo = (await this.apiClient.getFileList(await this.apiClient.getParentFileId())).data.InfoList.map(file => ({ fileName: file.FileName, etag: file.Etag, size: file.Size, type: file.Type, fileId: file.FileId })); let fileInfo = allFileInfo.filter(file => file.type !== 1); fileInfo.filter(file => !fileSelectionDetails.unselectedRowKeys.includes(file.fileId.toString())).forEach(file => { file.path = file.fileName; }); this.fileInfoList.push(...fileInfo); fileSelectFolderInfoList = allFileInfo.filter(file => file.type === 1).filter(file => !fileSelectionDetails.unselectedRowKeys.includes(file.fileId.toString())); } else { let fileSelectIdList = fileSelectionDetails.selectedRowKeys; if (!fileSelectIdList.length) { this.progress = 100; this.progressDesc = "未选择文件"; return false; } const allFileInfo = await this._getFileInfoBatch(fileSelectIdList); const fileInfo = allFileInfo.filter(info => info.type !== 1); fileInfo.forEach(file => { file.path = file.fileName; }); this.fileInfoList.push(...fileInfo); fileSelectFolderInfoList = allFileInfo.filter(info => info.type === 1); } for (let i = 0; i < fileSelectFolderInfoList.length; i++) { const folderInfo = fileSelectFolderInfoList[i]; this.progress = Math.round((i / fileSelectFolderInfoList.length) * 100); await new Promise(resolve => setTimeout(resolve, FastLinkConfig.getFolderInfoDelay)); if (this.taskCancel) { this.progressDesc = "任务已取消"; return true; } await this._getAllFileInfoByFolderId(folderInfo.fileId, folderInfo.fileName + "/", fileSelectFolderInfoList.length); } const commonPath = await this._getCommonPath(); if (commonPath) { this.fileInfoList.forEach(info => { info.path = info.path.slice(commonPath.length); }); } return true; } async generateShareLink(fileSelectionDetails) { this.progress = 0; this.progressDesc = "准备获取文件信息..."; const result = await this._getSelectedFilesInfo(fileSelectionDetails); if (!result) return ''; this.progressDesc = "秒传链接生成完成"; return this.buildShareLink(this.fileInfoList, this.commonPath); } buildShareLink(fileInfoList, commonPath) { const shareLinkFileInfo = fileInfoList.map(info => { return [FastLinkConfig.usesBase62EtagsInExport ? this._hexToBase62(info.etag) : info.etag, info.size, info.path.replace(/[%#$]/g, '')].join('#'); }).filter(Boolean).join('$'); const shareLink = `${FastLinkConfig.COMMON_PATH_LINK_PREFIX}${commonPath}%${shareLinkFileInfo}`; return shareLink; } _parseShareLink(shareLink, InputUsesBase62 = true) { let commonPath = ''; let shareFileInfo = ''; if (shareLink.slice(0, 4) === "123F") { const commonPathLinkPrefix = shareLink.split('$')[0]; shareLink = shareLink.replace(`${commonPathLinkPrefix}$`, ''); if (commonPathLinkPrefix + "$" === FastLinkConfig.COMMON_PATH_LINK_PREFIX) { commonPath = shareLink.split('%')[0]; shareFileInfo = shareLink.replace(`${commonPath}%`, ''); } else { return null; } } else { shareFileInfo = shareLink; InputUsesBase62 = false; } const shareLinkList = Array.from(shareFileInfo.replace(/\r?\n/g, '$').split('$')); this.commonPath = commonPath; return shareLinkList.map(singleShareLink => { const singleFileInfoList = singleShareLink.split('#'); if (singleFileInfoList.length < 3) return null; return { etag: InputUsesBase62 ? this._base62ToHex(singleFileInfoList[0]) : singleFileInfoList[0], size: singleFileInfoList[1], path: singleFileInfoList[2], fileName: singleFileInfoList[2].split('/').pop() }; }).filter(Boolean); } async _makeDirForFiles(shareFileList) { const total = shareFileList.length; this.progressDesc = `正在创建文件夹...`; let folder = {}; const rootFolderId = await this.apiClient.getParentFileId(); if (this.commonPath) { const commonPathParts = this.commonPath.split('/').filter(part => part !== ''); let currentParentId = rootFolderId; for (let i = 0; i < commonPathParts.length; i++) { const currentPath = commonPathParts.slice(0, i + 1).join('/'); const folderName = commonPathParts[i]; if (!folder[currentPath]) { const newFolder = await this.apiClient.mkdir(currentParentId, folderName); await new Promise(resolve => setTimeout(resolve, FastLinkConfig.mkdirDelay)); folder[currentPath] = newFolder.folderFileId; } currentParentId = folder[currentPath]; } } else { folder[''] = rootFolderId; } for (let i = 0; i < shareFileList.length; i++) { const item = shareFileList[i]; const itemPath = item.path.split('/').slice(0, -1); let nowParentFolderId = folder[this.commonPath.slice(0, -1)] || rootFolderId; for (let i = 0; i < itemPath.length; i++) { const path = itemPath.slice(0, i + 1).join('/'); if (!folder[path]) { const newFolderID = await this.apiClient.mkdir(nowParentFolderId, itemPath[i]); await new Promise(resolve => setTimeout(resolve, FastLinkConfig.mkdirDelay)); folder[path] = newFolderID.folderFileId; nowParentFolderId = newFolderID.folderFileId; } else { nowParentFolderId = folder[path]; } if (this.taskCancel) { this.progressDesc = "任务已取消"; return shareFileList; } } shareFileList[i].parentFolderId = nowParentFolderId; this.progress = Math.round((i / total) * 100); this.progressDesc = `正在创建文件夹... (${i + 1} / ${total})`; } return shareFileList; } async _saveFileList(shareFileList) { let completed = 0; let successList = []; let failedList = []; const total = shareFileList.length; for (let i = 0; i < shareFileList.length; i++) { if (this.taskCancel) { this.progressDesc = "任务已取消"; break; } const fileInfo = shareFileList[i]; if (i > 0) { await new Promise(resolve => setTimeout(resolve, FastLinkConfig.saveLinkDelay)); } const reuse = await this.apiClient.getFile({ etag: fileInfo.etag, size: fileInfo.size, fileName: fileInfo.fileName }, fileInfo.parentFolderId); if (reuse[0]) { successList.push(fileInfo); } else { fileInfo.error = reuse[1]; failedList.push(fileInfo); } completed++; this.progress = Math.round((completed / total) * 100); this.progressDesc = `正在保存第 ${completed} / ${total} 个文件...`; } return { success: successList, failed: failedList, commonPath: this.commonPath }; } async saveShareLink(content) { let saveResult = { success: [], failed: [] }; try { const jsonData = this.safeParse(content); if (jsonData && this.validateJson(jsonData)) { saveResult = await this.saveJsonShareLink(jsonData); } else { const shareFileList = this._parseShareLink(content); if (!shareFileList) return { success: [], failed: [] }; saveResult = await this._saveFileList(await this._makeDirForFiles(shareFileList)); } } catch (error) { console.error('[123云盘解锁] 保存失败:', error); saveResult = { success: [], failed: [] }; } return saveResult; } _base62chars() { return '0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ'; } _hexToBase62(hex) { if (!hex) return ''; let num = BigInt('0x' + hex); if (num === 0n) return '0'; let chars = []; const base62 = this._base62chars(); while (num > 0n) { chars.push(base62[Number(num % 62n)]); num = num / 62n; } return chars.reverse().join(''); } _base62ToHex(base62) { if (!base62) return ''; const chars = this._base62chars(); let num = 0n; for (let i = 0; i < base62.length; i++) { num = num * 62n + BigInt(chars.indexOf(base62[i])); } let hex = num.toString(16); if (hex.length % 2) hex = '0' + hex; while (hex.length < 32) hex = '0' + hex; return hex; } // JSON相关功能 safeParse(str) { try { return JSON.parse(str); } catch { return null; } } _formatSize(size) { if (size < 1024) return size + ' B'; if (size < 1024 * 1024) return (size / 1024).toFixed(2) + ' KB'; if (size < 1024 * 1024 * 1024) return (size / 1024 / 1024).toFixed(2) + ' MB'; return (size / 1024 / 1024 / 1024).toFixed(2) + ' GB'; } validateJson(json) { return (json && Array.isArray(json.files) && json.files.every(f => f.etag && f.size && f.path)); } shareLinkToJson(shareLink) { const fileInfo = this._parseShareLink(shareLink); if (fileInfo.length === 0) { console.error('[123云盘解锁] 解析秒传链接失败:', shareLink); return { error: '解析秒传链接失败' }; } if (FastLinkConfig.usesBase62EtagsInExport) { fileInfo.forEach(f => { f.etag = this._hexToBase62(f.etag); }); } const totalSize = fileInfo.reduce((sum, f) => sum + Number(f.size), 0); return { scriptVersion: "1.2.0", exportVersion: "1.0", usesBase62EtagsInExport: FastLinkConfig.usesBase62EtagsInExport, commonPath: this.commonPath, totalFilesCount: fileInfo.length, totalSize, formattedTotalSize: this._formatSize(totalSize), files: fileInfo.map(f => ({ etag: f.etag, size: f.size, path: f.path })) }; } _parseJsonShareLink(jsonData) { this.commonPath = jsonData['commonPath'] || ''; const shareFileList = jsonData['files']; if (jsonData['usesBase62EtagsInExport']) { shareFileList.forEach(file => { file.etag = this._base62ToHex(file.etag); }); } shareFileList.forEach(file => { file.fileName = file.path.split('/').pop(); }); return shareFileList; } async saveJsonShareLink(jsonContent) { const shareFileList = this._parseJsonShareLink(jsonContent); return this._saveFileList(await this._makeDirForFiles(shareFileList)); } async retrySaveFailed(FileList) { return this._saveFileList(FileList); } } // 秒传功能初始化 let panApiClient = null; let tableRowSelector = null; let shareLinkManager = null; // 任务队列系统 let taskList = []; let isTaskRunning = false; let taskIdCounter = 0; let currentTask = null; let isProgressMinimized = false; const minimizeWidgetId = 'fastlink-progress-minimize-widget'; function initFastLink() { if (!FastLinkConfig.enabled) return; panApiClient = new PanApiClient(); tableRowSelector = new TableRowSelector(); shareLinkManager = new ShareLinkManager(panApiClient); // 初始化 panApiClient.init(); tableRowSelector.init(); } // ============================================================================= // 秒传功能模块结束 // ============================================================================= // 保存原始方法 const originalXHR = unsafeWindow.XMLHttpRequest; const originalFetch = unsafeWindow.fetch; const originalOpen = XMLHttpRequest.prototype.open; const originalSend = XMLHttpRequest.prototype.send; const originalSetRequestHeader = XMLHttpRequest.prototype.setRequestHeader; // 创建唯一标识符 const requestURLSymbol = Symbol('requestURL'); const modifiedHeadersSymbol = Symbol('modifiedHeaders'); const handlerSymbol = Symbol('readyStateHandler'); // 规则配置 const rules = [ { // 用户信息 runat: "end", match: (url) => url.pathname.includes('api/user/info'), condition: () => user.vip === 1, action: (res) => { if (!res.data) return res; res.data.Vip = true; res.data.VipLevel = user.pvip ? 3 : (user.svip ? 2 : 1); if (user.ad === 1) res.data.IsShowAdvertisement = false; // 确保UserVipDetail存在 if (!res.data.UserVipDetail) { res.data.UserVipDetail = {}; } res.data.UserVipDetail.VipCode = res.data.VipLevel; if (user.pvip === 1) { // 长期会员 res.data.VipExpire = "永久有效"; res.data.UserVipDetail.UserPermanentVIPDetailInfos = [{ VipDesc: "长期VIP会员", TimeDesc: " 永久有效", IsUse: true }]; res.data.UserVipDetailInfos = []; } else if (user.svip === 1) { // 超级会员 let time = new Date(user.endtime * 1000); res.data.VipExpire = time.toLocaleString(); res.data.UserVipDetailInfos = [{ VipDesc: "SVIP 会员", TimeDesc: time.toLocaleDateString() + " 到期", IsUse: time >= new Date() }]; } else { // 普通会员 let time = new Date(user.endtime * 1000); res.data.VipExpire = time.toLocaleString(); res.data.UserVipDetailInfos = [{ VipDesc: "VIP 会员", TimeDesc: time.toLocaleDateString() + " 到期", IsUse: time >= new Date() }]; } if (user.name) res.data.Nickname = user.name; if (user.photo) res.data.HeadImage = user.photo; if (user.mail) res.data.Mail = user.mail; if (user.phone) res.data.Passport = Number(user.phone); if (user.id) res.data.UID = Number(user.id); if (user.level) res.data.GrowSpaceAddCount = Number(user.level); return res; } }, { // 用户报告信息 runat: "end", match: (url) => url.pathname.includes('user/report/info'), condition: () => user.vip === 1, action: (res) => { if (res && res.data) { res.data.vipType = user.pvip ? 3 : (user.svip ? 2 : 1); res.data.vipSub = user.pvip ? 3 : (user.svip ? 2 : 1); res.data.developSub = user.pvip ? 3 : (user.svip ? 2 : 1); } return res; } }, { // 下载请求头处理 runat: "header", match: (url) => [ 'file/download_info', 'file/batch_download_info', 'share/download/info', 'file/batch_download_share_info' ].some(path => url.pathname.includes(path)), condition: () => true, action: (headers) => { headers.platform = 'android'; return headers; } }, { // 下载信息处理 runat: "end", match: (url) => [ 'file/download_info', 'file/batch_download_info', 'share/download/info', 'file/batch_download_share_info' ].some(path => url.pathname.includes(path)), condition: () => true, action: (res, url) => { // 处理下载限制错误 if (res?.code === 5113 || res?.code === 5114 || res?.message?.includes("下载流量已超出")) { if (url.pathname.includes("batch_download")) { showFastLinkToast("请勿多选文件!已为您拦截支付下载窗口", 'warning', 3000); return { code: 400, message: "已拦截", data: null }; } else { showFastLinkToast("您今日下载流量已超出限制,已为您拦截支付窗口", 'warning', 3000); return { code: 400, message: "已拦截", data: null }; } } if (res.data && (res.data.DownloadUrl || res.data.DownloadURL)) { // 统一处理下载链接 let origKey = res.data.DownloadUrl ? 'DownloadUrl' : 'DownloadURL'; let origURL = new URL(res.data[origKey]); let finalURL; if (origURL.origin.includes("web-pro")) { let params = (() => { try { return decodeURIComponent(atob(origURL.searchParams.get('params'))); } catch { return atob(origURL.searchParams.get('params')); } })(); let directURL = new URL(params, origURL.origin); directURL.searchParams.set('auto_redirect', 0); origURL.searchParams.set('params', btoa(encodeURI(directURL.href))); finalURL = decodeURIComponent(origURL.href); } else { origURL.searchParams.set('auto_redirect', 0); let newURL = new URL('https://web-pro2.123952.com/download-v2/', origURL.origin); newURL.searchParams.set('params', btoa(encodeURI(origURL.href))); newURL.searchParams.set('is_s3', 0); finalURL = decodeURIComponent(newURL.href); } res.data[origKey] = finalURL; } return res; } }, { // 屏蔽数据收集请求 runat: "start", match: (url) => url.pathname.includes('web_logs') || url.pathname.includes('metrics'), condition: () => true, action: () => { throw new Error('【123云盘解锁】已屏蔽此数据收集器'); } } ]; // 工具函数 function findMatchingRule(url, phase) { try { return rules.find(rule => rule.match(url) && rule.condition() && rule.runat === phase ); } catch (error) { console.error('[123云盘解锁] 规则匹配失败:', error); if (user.debug) { console.error('错误详情:', { url: url.href, phase }); } return null; } } function processData(data) { if (typeof data === 'string') { try { return JSON.parse(data); } catch { return data; } } return data; } function debugLog(method, phase, url, original, modified) { if (user.debug) { console.log(`[123云盘解锁] ${method} ${phase}`, { url: url.href, original: original, modified: modified }); } } function applyRule(rule, data, url, method, phase) { const originalData = processData(data); let result = rule.action(originalData, url); // 处理header格式化 if (phase === 'header' && result && typeof result === 'object') { const headers = {}; Object.entries(result).forEach(([key, value]) => { const formattedKey = key.toLowerCase() .split('-') .map(word => word.charAt(0).toUpperCase() + word.slice(1)) .join('-'); headers[formattedKey] = value; }); result = headers; } debugLog(method, phase, url, originalData, result); // 非header返回字符串 if (phase !== 'header' && result && typeof result === 'object') { return JSON.stringify(result); } return result; } // 统一错误处理包装函数 function safeApplyRule(rule, data, url, method, phase) { try { return applyRule(rule, data, url, method, phase); } catch (error) { console.error(`[123云盘解锁] 规则执行失败 [${phase}]:`, error); if (user.debug) { console.error('错误详情:', { url: url.href, method: method, phase: phase, stack: error.stack }); } // 返回原始数据以保证功能不中断 return data; } } // 修复后的Fetch拦截(带统一错误处理) unsafeWindow.fetch = async function (input, init = {}) { try { const url = new URL(typeof input === 'string' ? input : input.url, location.origin); // 检查start规则 const startRule = findMatchingRule(url, 'start'); if (startRule) { try { const result = safeApplyRule(startRule, null, url, 'fetch', 'start'); return new Response(result, { status: 200, statusText: 'OK', headers: { 'Content-Type': 'application/json' } }); } catch (error) { console.warn('[123云盘解锁] fetch start错误:', error); if (user.debug) { console.error('错误堆栈:', error.stack); } } } // 检查header规则 const headerRule = findMatchingRule(url, 'header'); if (headerRule) { try { if (!init.headers) init.headers = {}; let headers = {}; if (init.headers instanceof Headers) { init.headers.forEach((value, key) => headers[key] = value); } else { headers = { ...init.headers }; } const modifiedHeaders = safeApplyRule(headerRule, headers, url, 'fetch', 'header'); init.headers = new Headers(modifiedHeaders); } catch (error) { console.warn('[123云盘解锁] fetch header错误:', error); if (user.debug) { console.error('错误堆栈:', error.stack); } } } // 执行原始请求 const response = await originalFetch.call(this, input, init); // 检查end规则 const endRule = findMatchingRule(url, 'end'); if (endRule) { try { const responseText = await response.clone().text(); const modifiedResponse = safeApplyRule(endRule, responseText, url, 'fetch', 'end'); return new Response(modifiedResponse, { status: response.status, statusText: response.statusText, headers: response.headers }); } catch (error) { console.warn('[123云盘解锁] fetch end错误:', error); if (user.debug) { console.error('错误堆栈:', error.stack); } } } return response; } catch (error) { console.error('[123云盘解锁] fetch拦截失败:', error); if (user.debug) { console.error('错误堆栈:', error.stack); } // 失败时调用原始fetch return originalFetch.call(this, input, init); } }; // 修复后的XMLHttpRequest拦截(修复内存泄漏) XMLHttpRequest.prototype.open = function (method, url, ...args) { try { const fullUrl = new URL(url, location.origin); this[requestURLSymbol] = fullUrl; // 移除旧的事件监听器(如果存在) if (this[handlerSymbol]) { this.removeEventListener('readystatechange', this[handlerSymbol]); } // 使用箭头函数保持this上下文 const handleStateChange = () => { if (this.readyState === 4) { const endRule = findMatchingRule(fullUrl, 'end'); if (endRule) { try { const modifiedResponse = safeApplyRule( endRule, this.responseText, fullUrl, 'XHR', 'end' ); Object.defineProperty(this, 'responseText', { value: modifiedResponse, writable: false, configurable: true }); Object.defineProperty(this, 'response', { value: modifiedResponse, writable: false, configurable: true }); } catch (error) { console.warn('[123云盘解锁] XHR响应处理错误:', error); if (user.debug) { console.error('错误堆栈:', error.stack); } } } } }; // 保存事件处理器引用,以便后续移除 this[handlerSymbol] = handleStateChange; this.addEventListener('readystatechange', handleStateChange); return originalOpen.call(this, method, url, ...args); } catch (error) { console.error('[123云盘解锁] XHR.open错误:', error); if (user.debug) { console.error('错误堆栈:', error.stack); } // 失败时调用原始方法 return originalOpen.call(this, method, url, ...args); } }; XMLHttpRequest.prototype.setRequestHeader = function (name, value) { try { const url = this[requestURLSymbol]; if (!url) return originalSetRequestHeader.call(this, name, value); const headerRule = findMatchingRule(url, 'header'); if (headerRule) { try { if (!this[modifiedHeadersSymbol]) this[modifiedHeadersSymbol] = {}; this[modifiedHeadersSymbol][name] = value; const modifiedHeaders = safeApplyRule(headerRule, this[modifiedHeadersSymbol], url, 'XHR', 'header'); this[modifiedHeadersSymbol] = modifiedHeaders; return; } catch (error) { console.warn('[123云盘解锁] XHR header处理错误:', error); if (user.debug) { console.error('错误堆栈:', error.stack); } } } return originalSetRequestHeader.call(this, name, value); } catch (error) { console.error('[123云盘解锁] XHR.setRequestHeader错误:', error); if (user.debug) { console.error('错误堆栈:', error.stack); } // 失败时调用原始方法 return originalSetRequestHeader.call(this, name, value); } }; XMLHttpRequest.prototype.send = function (data) { try { const url = this[requestURLSymbol]; if (!url) return originalSend.call(this, data); // 应用修改的headers const modifiedHeaders = this[modifiedHeadersSymbol]; if (modifiedHeaders) { try { Object.entries(modifiedHeaders).forEach(([name, value]) => { originalSetRequestHeader.call(this, name, value); }); } catch (error) { console.warn('[123云盘解锁] 应用修改的headers失败:', error); if (user.debug) { console.error('错误堆栈:', error.stack); } } } // 检查start规则 const startRule = findMatchingRule(url, 'start'); if (startRule) { try { const result = safeApplyRule(startRule, null, url, 'XHR', 'start'); // 设置响应属性 Object.defineProperty(this, 'readyState', { value: 4, configurable: true }); Object.defineProperty(this, 'status', { value: 200, configurable: true }); Object.defineProperty(this, 'responseText', { value: result, configurable: true }); Object.defineProperty(this, 'response', { value: result, configurable: true }); // 触发事件 setTimeout(() => { ['readystatechange', 'load', 'loadend'].forEach(eventType => { try { this.dispatchEvent(new Event(eventType)); const handler = this[`on${eventType}`]; if (typeof handler === 'function') handler.call(this); } catch (error) { console.warn(`[123云盘解锁] 事件错误 ${eventType}:`, error); if (user.debug) { console.error('错误堆栈:', error.stack); } } }); }, 0); return; } catch (error) { console.warn('[123云盘解锁] XHR start处理错误:', error); if (user.debug) { console.error('错误堆栈:', error.stack); } } } return originalSend.call(this, data); } catch (error) { console.error('[123云盘解锁] XHR.send错误:', error); if (user.debug) { console.error('错误堆栈:', error.stack); } // 失败时调用原始方法 return originalSend.call(this, data); } }; // 格式化设置项 const formatSetting = (key, value, comment) => { const item = document.createElement('div'); item.className = 'setting-item'; const content = document.createElement('div'); content.className = 'setting-content'; const keyElement = document.createElement('div'); keyElement.className = 'setting-key'; keyElement.textContent = key; content.appendChild(keyElement); // 判断设置类型 - 修复等级1被误判为开关的问题 const switchKeys = ['VIP状态', 'SVIP显示', '长期会员显示', '广告控制', '秒传功能', '调试模式']; const isSwitch = switchKeys.includes(key) && typeof value === 'number' && (value === 0 || value === 1); const isEditable = ['用户名', '头像', '等级', '过期时间'].includes(key); if (isSwitch) { // 创建开关按钮 const switchContainer = document.createElement('label'); switchContainer.className = 'switch'; const input = document.createElement('input'); input.type = 'checkbox'; input.checked = value === 1; const slider = document.createElement('span'); slider.className = 'slider round'; switchContainer.appendChild(input); switchContainer.appendChild(slider); // 添加点击事件 input.addEventListener('change', () => { let newValue = input.checked ? 1 : 0; // 更新用户配置 switch (key) { case 'VIP状态': user.vip = newValue; GM_setValue('vip', newValue); break; case 'SVIP显示': user.svip = newValue; GM_setValue('svip', newValue); // 如果SVIP关闭,长期会员也应该关闭 if (newValue === 0 && user.pvip === 1) { user.pvip = 0; GM_setValue('pvip', 0); } break; case '长期会员显示': user.pvip = newValue; GM_setValue('pvip', newValue); // 如果长期会员开启,SVIP必须开启 if (newValue === 1 && user.svip === 0) { user.svip = 1; GM_setValue('svip', 1); } break; case '广告控制': user.ad = newValue; GM_setValue('ad', newValue); break; case '秒传功能': FastLinkConfig.enabled = newValue; GM_setValue('fastlink_enabled', newValue); break; case '调试模式': user.debug = newValue; GM_setValue('debug', newValue); break; } // 刷新页面以应用更改 setTimeout(() => location.reload(), 300); }); content.appendChild(switchContainer); } else if (isEditable) { // 创建输入框 const inputContainer = document.createElement('div'); inputContainer.className = 'input-container'; const inputElement = document.createElement('input'); // 根据不同的设置项设置输入框类型和属性 if (key === '等级') { inputElement.type = 'number'; inputElement.min = 0; inputElement.max = 128; inputElement.value = value; } else if (key === '过期时间') { inputElement.type = 'datetime-local'; // 将时间戳转换为datetime-local格式 const date = new Date(value * 1000); const year = date.getFullYear(); const month = String(date.getMonth() + 1).padStart(2, '0'); const day = String(date.getDate()).padStart(2, '0'); const hours = String(date.getHours()).padStart(2, '0'); const minutes = String(date.getMinutes()).padStart(2, '0'); inputElement.value = `${year}-${month}-${day}T${hours}:${minutes}`; } else { inputElement.type = 'text'; inputElement.value = value; } inputElement.className = 'setting-input'; // 添加保存按钮 const saveButton = document.createElement('button'); saveButton.textContent = '保存'; saveButton.className = 'save-btn'; // 保存按钮点击事件 saveButton.addEventListener('click', () => { let newValue = inputElement.value; // 验证和转换不同类型的输入 if (key === '等级') { newValue = parseInt(newValue); if (isNaN(newValue) || newValue < 0 || newValue > 128) { alert('等级必须在 0-128 之间'); return; } } else if (key === '过期时间') { // 将datetime-local格式转换为时间戳 const date = new Date(newValue); if (isNaN(date.getTime())) { alert('请输入有效的日期时间'); return; } newValue = Math.floor(date.getTime() / 1000); } else if (key === '头像' && newValue && !newValue.match(/^https?:\/\/.+/)) { if (!confirm('头像URL似乎不是有效的HTTP/HTTPS地址,是否继续保存?')) { return; } } // 更新配置 switch (key) { case '用户名': user.name = newValue; GM_setValue('name', newValue); break; case '头像': user.photo = newValue; GM_setValue('photo', newValue); break; case '等级': user.level = newValue; GM_setValue('level', newValue); break; case '过期时间': user.endtime = newValue; GM_setValue('endtime', newValue); break; } // 显示保存成功提示 saveButton.textContent = '已保存'; saveButton.classList.add('saved'); setTimeout(() => { saveButton.textContent = '保存'; saveButton.classList.remove('saved'); location.reload(); }, 1500); }); inputContainer.appendChild(inputElement); inputContainer.appendChild(saveButton); content.appendChild(inputContainer); } else { // 非编辑项的显示 const valueElement = document.createElement('div'); valueElement.className = 'setting-value'; valueElement.textContent = key === '过期时间' ? new Date(value * 1000).toLocaleString() : value; content.appendChild(valueElement); } item.appendChild(content); if (comment) { const commentElement = document.createElement('div'); commentElement.className = 'setting-comment'; commentElement.textContent = comment; item.appendChild(commentElement); } return item; }; function createSettingsPanel() { // 检查是否已存在面板 if (document.getElementById('vip-settings-panel')) { return; } // 创建面板容器 const panel = document.createElement('div'); panel.id = 'vip-settings-panel'; panel.className = 'settings-panel'; // 创建标题栏 const header = document.createElement('div'); header.className = 'panel-header'; // 创建标题容器 const titleContainer = document.createElement('div'); titleContainer.className = 'title-container'; const title = document.createElement('h3'); title.textContent = '123云盘脚本设置'; titleContainer.appendChild(title); // 添加GitHub图标 const githubIcon = document.createElement('a'); githubIcon.href = 'https://github.com/QingJ01/123pan_unlock'; githubIcon.target = '_blank'; githubIcon.className = 'github-icon'; githubIcon.innerHTML = ` `; githubIcon.title = '访问GitHub项目'; titleContainer.appendChild(githubIcon); header.appendChild(titleContainer); // 添加关闭按钮 const closeButton = document.createElement('button'); closeButton.className = 'close-btn'; closeButton.innerHTML = '×'; closeButton.addEventListener('click', () => panel.remove()); header.appendChild(closeButton); panel.appendChild(header); // 创建设置列表 const settingsList = document.createElement('div'); settingsList.className = 'settings-list'; // 添加所有设置项 const settings = [ { key: 'VIP状态', value: user.vip, comment: '会员修改总开关' }, { key: 'SVIP显示', value: user.svip, comment: '显示为超级会员 (关闭将自动关闭长期会员)' }, { key: '长期会员显示', value: user.pvip, comment: '显示为长期会员 (开启将自动开启 SVIP 显示)' }, { key: '广告控制', value: user.ad, comment: '关闭广告' }, { key: '秒传功能', value: FastLinkConfig.enabled, comment: '启用秒传链接生成和保存功能' }, { key: '用户名', value: user.name, comment: '自定义用户名(支持中文、英文、数字)' }, { key: '头像', value: user.photo, comment: '自定义头像URL(建议使用HTTPS地址)' }, { key: '等级', value: user.level, comment: '成长容量等级(0-128,数字越大容量越大)' }, { key: '过期时间', value: user.endtime, comment: '会员过期时间(可自定义任意时间)' }, { key: '调试模式', value: user.debug, comment: '调试信息显示级别' } ]; settings.forEach(setting => { settingsList.appendChild(formatSetting(setting.key, setting.value, setting.comment)); }); panel.appendChild(settingsList); // 添加交流群按钮 const groupButton = document.createElement('a'); groupButton.href = 'http://qm.qq.com/cgi-bin/qm/qr?_wv=1027&k=GGU3-kUsPnz1bq-jwN7e8D41yxZ-DyI2&authKey=ujGsFKDnF5zD3j1z9krJR5xHlWWAKHOJV2oarfAgNmqZAl0xmTb45QwsqgYPPF7e&noverify=0&group_code=1035747022'; groupButton.target = '_blank'; groupButton.className = 'group-btn'; groupButton.innerHTML = ` 加入交流群 `; panel.appendChild(groupButton); document.body.appendChild(panel); } // 秒传功能UI函数 function showFastLinkToast(message, type = 'info', duration = 3000) { const toast = document.createElement('div'); toast.className = 'fastlink-toast'; toast.style.cssText = ` position: fixed; top: 20px; right: 20px; background: #fff; color: #333; padding: 12px 20px; border-radius: 8px; box-shadow: 0 4px 12px rgba(0, 0, 0, 0.15); z-index: 10002; font-size: 14px; max-width: 300px; animation: fastlinkToastSlideIn 0.3s ease-out; `; // 根据类型设置边框颜色 const borderColors = { success: '#4CAF50', error: '#f44336', warning: '#ff9800', info: '#2196F3' }; toast.style.borderLeft = `4px solid ${borderColors[type] || borderColors.info}`; toast.textContent = message; document.body.appendChild(toast); setTimeout(() => { toast.style.animation = 'fastlinkToastSlideOut 0.3s ease-out forwards'; setTimeout(() => { if (toast.parentNode) { toast.parentNode.removeChild(toast); } }, 300); }, duration); } function showFastLinkProgress(title, percent, desc) { let modal = document.getElementById('fastlink-progress-modal'); const taskQueueHtml = taskList.length > 0 ? ` - 队列 ${taskList.length}` : ''; if (!modal) { modal = document.createElement('div'); modal.id = 'fastlink-progress-modal'; modal.className = 'fastlink-modal-overlay'; modal.innerHTML = `