百度网盘文件转存助手

使用百度网盘的时候经常要将别人分享的文件(夹)转存到自己网盘里。对于非会员用户有每次500个文件的限制,超过500个文件只能自己手动弄,比较麻烦,因此有了这个工具,希望能帮到需要的人,不喜轻喷。(目前支持保存到根目录)

  1. // ==UserScript==
  2. // @name 百度网盘文件转存助手
  3. // @namespace https://github.com/hyx00000000007
  4. // @version 1.0.2
  5. // @description 使用百度网盘的时候经常要将别人分享的文件(夹)转存到自己网盘里。对于非会员用户有每次500个文件的限制,超过500个文件只能自己手动弄,比较麻烦,因此有了这个工具,希望能帮到需要的人,不喜轻喷。(目前支持保存到根目录)
  6. // @author shimmer,Teng(samisold)
  7. // @license BSD
  8. // @match *://pan.baidu.com/disk/home*
  9. // @match *://yun.baidu.com/disk/home*
  10. // @match *://pan.baidu.com/disk/main*
  11. // @match *://yun.baidu.com/disk/main*
  12. // @require https://unpkg.com/jquery@3.7.0/dist/jquery.min.js
  13. // @connect baidu.com
  14. // @connect baidupcs.com
  15. // @icon data:image/gif;base64,R0lGODlhAQABAAAAACH5BAEKAAEALAAAAAABAAEAAAICTAEAOw==
  16. // @grant none
  17. // ==/UserScript==
  18.  
  19. (function() {
  20. 'use strict';
  21.  
  22. window.BaiduTransfer = function(rootPath) {
  23. this.ROOT_URL = 'https://pan.baidu.com';
  24. this.bdstoken = null;
  25. this.shareId = null;
  26. this.shareRoot = null;
  27. this.userId = null;
  28. this.dirList = [];
  29. this.fileList = [];
  30. this.rootPath = rootPath || "";
  31. };
  32.  
  33. BaiduTransfer.prototype = {
  34.  
  35. request: async function(path, method, params, data, checkErrno) {
  36. var url = this.ROOT_URL + path;
  37. if (params) {
  38. url += '?' + params;
  39. }
  40.  
  41. try {
  42. var response = await $.ajax({
  43. url: url,
  44. type: method,
  45. headers: {
  46. 'X-Requested-With': 'XMLHttpRequest'
  47. },
  48. data: data,
  49. xhrFields: {
  50. withCredentials: true
  51. }
  52. });
  53. if (checkErrno && response.errno && response.errno !== 0) {
  54. var errno = response.errno;
  55. var errmsg = response.show_msg || "过5分钟重试";
  56. var customError = new Error(errmsg);
  57. customError.errno = errno;
  58. throw customError;
  59. }
  60. return response;
  61. } catch (error) {
  62. throw error;
  63. }
  64. },
  65.  
  66. createDirectory: async function(dirPath) {
  67. try {
  68. await this.listDir(dirPath);
  69. return;
  70. } catch (error) {
  71. if (error.errno !== -9) {
  72. throw error;
  73. }
  74. }
  75. var path = "/api/create";
  76. var params = "a=commit&bdstoken=" + this.bdstoken;
  77. var data = "path=" + encodeURIComponent(dirPath) + "&isdir=1&block_list=[]";
  78. return await this.request(path, "POST", params, data, true);
  79. },
  80.  
  81. listDir: async function(dirPath) {
  82. var path = "/api/list";
  83. var params = "order=time&desc=1&showempty=0&page=1&num=1000&dir=" + this.customUrlEncode(dirPath) + "&bdstoken=" + this.bdstoken;
  84. return await this.request(path, "GET", params, null, true);
  85. },
  86.  
  87. transfer: async function(userId, shareId, fsidList, transferPath) {
  88. var path = "/share/transfer";
  89. var params = "shareid=" + shareId + "&from=" + userId + "&ondup=newcopy&channel=chunlei&bdstoken=" + this.bdstoken;
  90. var data = "fsidlist=[" + fsidList.join(",") + "]&path=" + (transferPath || "/");
  91. var response = await this.request(path, "POST", params, data, false);
  92. var errno = response.errno;
  93. if (errno !== 0) {
  94. if (errno === 2) {
  95. var error = new Error("APIParameterError: url=" + path + " param=" + params);
  96. throw error;
  97. } else if (errno === 12) {
  98. var limit = response.target_file_nums_limit
  99. var count = response.target_file_nums
  100. if(limit&&count){
  101. var error = new Error("TransferLimitExceededException: limit=" + limit + " count=" + count);
  102. throw error;
  103. }
  104. var error = new Error(response.show_msg);
  105. throw error;
  106. } else if (errno === 1504) {
  107. console.log(`Transfer path ${transferPath} exceeds deadline, retry later...`);
  108. await new Promise(resolve => setTimeout(resolve, 1000));
  109. this.transfer(userId, shareId, fsidList, transferPath);
  110. } else if (errno === 111) {
  111. console.log(`Transfer path ${transferPath} call api too fast , retry later...`);
  112. await new Promise(resolve => setTimeout(resolve, 10000));
  113. this.transfer(userId, shareId, fsidList, transferPath);
  114. } else {
  115. var error = new Error("BaiduYunPanAPIException: [" + errno + "] " + response.errmsg);
  116. throw error;
  117. }
  118. }
  119. },
  120.  
  121. getBdstoken: async function() {
  122. if (this.bdstoken) {
  123. return this.bdstoken;
  124. }
  125.  
  126. var path = "/api/gettemplatevariable";
  127. var params = "fields=[\"bdstoken\"]";
  128. var response = await this.request(path, "GET", params, null, true);
  129. this.bdstoken = response.result.bdstoken;
  130. return this.bdstoken;
  131. },
  132.  
  133. getRandsk: async function(shareKey, pwd) {
  134. var path = "/share/verify";
  135. var params = "surl=" + shareKey + "&bdstoken=" + this.bdstoken;
  136. var data = "pwd=" + pwd;
  137. var response = await this.request(path, "POST", params, data, true);
  138. return response.randsk;
  139. },
  140.  
  141. getShareData: async function(shareKey, pwd) {
  142. var path = "/s/1" + shareKey;
  143. var response = await this.request(path, "GET", null, null ,false);
  144. var startTag = 'locals.mset(';
  145. var endTag = '});';
  146.  
  147. var startIndex = response.indexOf(startTag);
  148. if (startIndex === -1) {
  149. throw new Error("Invalid response: unable to find locals.mset");
  150. }
  151. startIndex += startTag.length;
  152.  
  153. var endIndex = response.indexOf(endTag, startIndex);
  154. if (endIndex === -1) {
  155. throw new Error("Invalid response: unable to find end of locals.mset");
  156. }
  157.  
  158. var jsonStr = response.substring(startIndex, endIndex + 1);
  159. var data = JSON.parse(jsonStr);
  160. return {
  161. userId: data.share_uk,
  162. shareId: data.shareid,
  163. bdstoken: data.bdstoken,
  164. shareRoot: data.file_list[0].parent_path,
  165. dirList: data.file_list.filter(e => e.isdir === 1).map(function(file) {
  166. return {
  167. id: file.fs_id,
  168. name: file.server_filename,
  169. };
  170. }),
  171. fileList: data.file_list.filter(e => e.isdir !== 1).map(function(file) {
  172. return {
  173. id: file.fs_id,
  174. name: file.server_filename,
  175. };
  176. })
  177. };
  178. },
  179.  
  180. updateRandsk: async function(shareKey, pwd) {
  181. await this.getBdstoken();
  182. await this.getRandsk(shareKey, pwd);
  183. },
  184.  
  185. initShareData: async function(shareKey, pwd) {
  186. if (pwd) {
  187. await this.updateRandsk(shareKey, pwd);
  188. }
  189. try {
  190. var shareData = await this.getShareData(shareKey, pwd);
  191. this.userId = shareData.userId;
  192. this.shareId = shareData.shareId;
  193. this.bdstoken = shareData.bdstoken;
  194. this.shareRoot = shareData.shareRoot;
  195. this.dirList = shareData.dirList;
  196. this.fileList = shareData.fileList;
  197. } catch (error) {
  198. if (error.message.indexOf('/share/init')){
  199. if (pwd) {
  200. throw new Error("Wrong password: " + pwd);
  201. } else {
  202. throw new Error("Password not specified");
  203. }
  204. }
  205. }
  206. },
  207.  
  208. transferFiles: async function(fileList, targetPath) {
  209. if (targetPath) {
  210. await this.createDirectory(targetPath);
  211. }
  212.  
  213. var maxTransferCount = 100;
  214.  
  215. for (var i = 0; i < fileList.length; i += maxTransferCount) {
  216. var batch = fileList.slice(i, i + maxTransferCount);
  217. var fsidList = batch.map(function(file) { return file.id; });
  218. await this.transfer(this.userId, this.shareId, fsidList, targetPath);
  219. }
  220. console.log("Transfer " + fileList.length + " files under directory " + targetPath + " success");
  221. },
  222.  
  223. transferDirs: async function(dirList, targetPath) {
  224. if (targetPath) {
  225. await this.createDirectory(targetPath);
  226. }
  227.  
  228. if (dirList.length === 0) {
  229. return;
  230. }
  231.  
  232. var dirPaths = dirList.map(function(dir) {
  233. return targetPath + '/' + dir.name;
  234. });
  235.  
  236. try {
  237. await this.transfer(this.userId, this.shareId, dirList.map(dir => dir.id), targetPath);
  238. dirPaths.forEach(function(dirPath) {
  239. console.log(`Transfer directory ${dirPath} success`);
  240. });
  241. } catch (error) {
  242. if (error.message.includes('TransferLimitExceededException:')) {
  243. console.log(`Directory ${dirPaths.join(',')} ${error.message}`);
  244.  
  245. if (dirList.length >= 2) {
  246. var mid = Math.floor(dirList.length / 2);
  247. await this.transferDirs(dirList.slice(0, mid), targetPath);
  248. await this.transferDirs(dirList.slice(mid), targetPath);
  249. } else {
  250. var dir = dirList[0];
  251. var dirPath = this.shareRoot;
  252. if (targetPath.length > this.rootPath.length) {
  253. dirPath += targetPath.slice(this.rootPath.length);
  254. }
  255. dirPath += '/' + dir.name;
  256.  
  257. var subFiles = await this.listShareDir(this.userId, this.shareId, dirPath);
  258. var subDirList = subFiles.filter(function(file) { return file.isDirectory; });
  259. var subFileList = subFiles.filter(function(file) { return !file.isDirectory; });
  260.  
  261. if (subDirList.length > 0) {
  262. await this.transferDirs(subDirList, targetPath + '/' + dir.name);
  263. }
  264. if (subFileList.length > 0) {
  265. await this.transferFiles(subFileList, targetPath + '/' + dir.name);
  266. }
  267. }
  268. } else {
  269. throw error;
  270. }
  271. }
  272. },
  273.  
  274. listShareDir: async function(userId, shareId, dirPath) {
  275. var path = "/share/list";
  276. var page = 1;
  277. var limit = 100;
  278. var result = []
  279. while(true){
  280. // bug fix by Teng(samisold)
  281. var params = `uk=${userId}&shareid=${shareId}&order=name&desc=0&showempty=0&page=${page}&num=${limit}&dir=${this.customUrlEncode(dirPath)}`;
  282. var response = await this.request(path, "GET", params, null ,true);
  283. var list = response.list;
  284. list.forEach(function(item) {
  285. result.push({
  286. id: item.fs_id,
  287. name: item.server_filename,
  288. isDirectory: item.isdir === 1
  289. });
  290. });
  291. if(list.length < 100){
  292. break;
  293. }
  294. page++;
  295. }
  296. return result;
  297. },
  298.  
  299. extractShareKey: function(url) {
  300. try {
  301. var decodedUrl = decodeURIComponent(url);
  302. if (decodedUrl.includes("/s/1")) {
  303. return decodedUrl.split("/s/1")[1].split("?")[0];
  304. } else if (decodedUrl.includes("surl=")) {
  305. return decodedUrl.split("surl=")[1].split("&")[0];
  306. }
  307. } catch (e) {
  308. console.error("Error extracting share key:", e);
  309. }
  310. return null;
  311. },
  312.  
  313. customUrlEncode: function(input) {
  314. let encoded = '';
  315. for (let c of input) {
  316. if (c === ' ' || c === '"' || c === '\'') {
  317. encoded += encodeURIComponent(c);
  318. } else {
  319. encoded += c;
  320. }
  321. }
  322. return encoded;
  323. },
  324.  
  325.  
  326. transferFinal: async function(url, pwd) {
  327. var shareKey = this.extractShareKey(url);
  328. if (!shareKey) {
  329. throw new Error("Unable to extract share key from URL");
  330. }
  331.  
  332. await this.initShareData(shareKey, pwd);
  333.  
  334. if (this.dirList.length > 0) {
  335. await this.transferDirs(this.dirList, this.rootPath);
  336. }
  337.  
  338. if (this.fileList.length > 0) {
  339. await this.transferFiles(this.fileList, this.rootPath);
  340. }
  341. }
  342. };
  343.  
  344. var button = '<div id="shimmer-draggable-button" style="position: fixed; bottom: 20px; left: 20px; z-index: 1000; cursor: grab;">'
  345. +'<button style="padding: 10px 20px; font-size: 16px; border: none; background-color: #007bff; color: white; cursor: pointer; border-radius: 5px; box-shadow: 0 2px 5px rgba(0, 0, 0, 0.2); transition: background-color 0.3s ease; outline: none;" onmouseover="this.style.backgroundColor=\'#0056b3\';" onmouseout="this.style.backgroundColor=\'#007bff\';">转存助手</button>'
  346. +'</div>'
  347. $('body').append(button)
  348.  
  349. // 动态创建弹窗
  350. var modal = $('<div>', {
  351. id: 'shimmer-input-modal',
  352. style: 'display: none; position: fixed; top: 0; left: 0; width: 100%; height: 100%; background-color: rgba(0, 0, 0, 0.5); z-index: 1001;'
  353. }).append(
  354. $('<div>', {
  355. style: 'position: absolute; top: 50%; left: 50%; transform: translate(-50%, -50%); width: 600px; padding: 20px; background-color: white; border-radius: 10px; box-shadow: 0 2px 10px rgba(0, 0, 0, 0.2);'
  356. }).append(
  357. $('<h2>', {
  358. text: '转存',
  359. style: 'margin-top: 0; color: #007bff;'
  360. }),
  361. $('<input>', {
  362. type: 'text',
  363. id: 'shimmer-input-modal-url',
  364. placeholder: '分享链接',
  365. style: 'width: 100%; padding: 10px; margin-bottom: 10px; border: 1px solid #ccc; border-radius: 5px;'
  366. }),
  367. $('<input>', {
  368. type: 'text',
  369. id: 'shimmer-input-modal-pwd',
  370. placeholder: '密码',
  371. style: 'width: 100%; padding: 10px; margin-bottom: 10px; border: 1px solid #ccc; border-radius: 5px;'
  372. }),
  373. $('<button>', {
  374. id: 'shimmer-input-modal-confirm-button',
  375. text: '确认',
  376. style: 'width: 100%; padding: 10px; background-color: #007bff; color: white; border: none; border-radius: 5px; cursor: pointer; transition: background-color 0.3s ease;',
  377. on: {
  378. mouseover: function() {
  379. $(this).css('backgroundColor', '#0056b3');
  380. },
  381. mouseout: function() {
  382. $(this).css('backgroundColor', '#007bff');
  383. }
  384. }
  385. })
  386. )
  387. );
  388.  
  389. $('body').append(modal);
  390.  
  391. var buttonWidth = $('#shimmer-draggable-button').outerWidth();
  392. var buttonHeight = $('#shimmer-draggable-button').outerHeight();
  393. var edgeOffset = 50;
  394.  
  395. $('#shimmer-draggable-button').css('left', -buttonWidth + edgeOffset + 'px');
  396.  
  397. $('#shimmer-draggable-button').on('mouseenter', function() {
  398. $(this).css('left', '0');
  399. });
  400.  
  401. $('#shimmer-draggable-button').on('mouseleave', function() {
  402. $(this).css('left', -buttonWidth + edgeOffset + 'px');
  403. });
  404.  
  405. $('#shimmer-draggable-button').on('click', function(event) {
  406. $('#shimmer-input-modal').show();
  407. });
  408.  
  409. $('#shimmer-input-modal').on('click', function(event) {
  410. if (event.target === this) {
  411. $('#shimmer-input-modal').hide();
  412. }
  413. });
  414.  
  415. $('#shimmer-input-modal-confirm-button').on('click', async function(event) {
  416. var rootPath = "";
  417. var transfer = new BaiduTransfer(rootPath);
  418. var url = $("#shimmer-input-modal-url").val();
  419. var pwd = $("#shimmer-input-modal-pwd").val();
  420.  
  421. // 检查 url
  422. if (!url) {
  423. alert("请输入分享链接");
  424. return;
  425. }
  426.  
  427. $('#shimmer-draggable-button').css('left', '0');
  428. $('#shimmer-draggable-button button').text('转存中...').prop('disabled', true);
  429. $('#shimmer-input-modal').hide();
  430. alert("转存在后台运行中,请不要关闭浏览器和刷新当前页面,注意左下角按钮的状态(目前这个弹窗需要点击确认)");
  431.  
  432. try {
  433. await transfer.transferFinal(url, pwd);
  434. console.log("Transfer completed successfully.");
  435. alert("转存成功");
  436. location.reload();
  437. } catch (error) {
  438. console.error("Error during transfer:", error);
  439. alert("发生错误了..." + error);
  440. } finally {
  441. $('#shimmer-draggable-button').css('left', -buttonWidth + edgeOffset + 'px')
  442. $('#shimmer-draggable-button button').text('转存助手').prop('disabled', false);
  443. }
  444. });
  445.  
  446. })();
  447.  

QingJ © 2025

镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址