您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
微博数据下载备份
当前为
// ==UserScript== // @name 微博备份 // @namespace https://github.com/Shapooo/ // @version 0.1 // @description 微博数据下载备份 // @author Shapooo // @match *://weibo.com/* // @icon https://www.google.com/s2/favicons?sz=64&domain=weibo.com // @grant none // @license GPL // ==/UserScript== "use strict"; function download(content, fileName, contentType) { let a = document.createElement("a"); let file = new Blob([content], { type: contentType }); a.href = URL.createObjectURL(file); a.download = fileName; a.click(); } async function fetchMyContent(page = 1, type = "myblog") { let uid = globalConfig.uid; return await fetchContent(uid, page, type); } async function fetchContent(uid = 0, page = 1, type = "myblog") { let api = `https://weibo.com/ajax/statuses/mymblog?uid=${uid}&page=${page}&feature=0&with_total=true`; if (type === "fav") { api = `https://weibo.com/ajax/favorites/all_fav?uid=${uid}&page=${page}&with_total=true`; } if (type === "like") { api = `https://weibo.com/ajax/statuses/likelist?uid=${uid}&page=${page}&with_total=true`; } console.log(`request ${api}`); const req = await fetch(api, { headers: { "accept": "application/json, text/plain, */*", "accept-language": "zh-CN,zh;q=0.9,en-IN;q=0.8,en;q=0.7,ar;q=0.6", "sec-ch-ua": '" Not A;Brand";v="99", "Chromium";v="101", "Google Chrome";v="101"', "sec-ch-ua-mobile": "?0", "sec-ch-ua-platform": '"macOS"', "sec-fetch-dest": "empty", "sec-fetch-mode": "cors", "sec-fetch-site": "same-origin", "x-requested-with": "XMLHttpRequest", }, referrer: `https://weibo.com/u/${uid}`, referrerPolicy: "strict-origin-when-cross-origin", body: null, method: "GET", mode: "cors", credentials: "include", }); const data = await req.json(); return data; } async function fetchAll(type = "myblog", range) { console.log(`fetching ${type} post ${range}`) let uid = globalConfig.uid; let page = 1; let allPageData = []; let noMore = false; for (let index = range[0]; index <= range[1]; index++) { console.log("scan", "page", page); showTip(`正在备份第 ${page} 页<br>因微博速率限制,过程可能较长,先干点别的吧`); for (let index = 0; index < 10; index++) { const pageData = await fetchContent(uid, page, type); if (pageData.ok) { const dataList = type === "fav" ? pageData.data.status : pageData.data.list; allPageData.push(dataList); if (dataList.length === 0) noMore = true; break; } await new Promise((resolve) => { setTimeout(resolve, 8 * 1000); }); console.log("retry", index); showTip( `[重试]备份第 ${page} 页,错误内容: ${JSON.stringify(pageData)}` ); } page++; if (noMore) break; await new Promise((resolve) => { setTimeout(resolve, 5 * 1000); }); } showTip(`数据拉取完成,等待下载到本地`); let rawData = allPageData.flat(); download( JSON.stringify(rawData, null, 2), "weibo-" + Date.now() + "-" + type + ".json", "text/plain" ); console.log("all done"); showTip(`完成,可以进行其它操作`); }let bkBoxStyle = `.bkBox{position:fixed;border-radius:5px;background:#fff;top:80px;right:20px;z-index:100000;padding:10px 15px;text-align:center;border:1px solid #a7a6a6}.bkBox-title{font-size:20px;color:#ff8200;margin:8px 0;font-weight:400;border:1px solid transparent;border-bottom-color:#a7a6a6}.bkBox-button{border-radius:2.25rem;display:block;background:#ffd3a3;padding:5px 8px;color:#fff;text-align:center;cursor:pointer;margin:16px auto 7px;border:0;font:white}.bkBox-button:hover{background:#ff8200}.box,.nav{width:300px;margin:auto;display:flex}.nav{height:30px;background-color:#fff;border:1px gray}.box{overflow:hidden}.bk-naviItem{color:#000;height:30px;width:100px;margin:auto;border:8px #000;float:left}.bk-naviItem,.bk-naviItem:hover{text-decoration:none}.bk-navi-inner{border-radius:8px;height:30px}.bk-navi-inner:hover{background-color:#f2f0ef}.bk-pageNumIndicator{margin:16px}.bk-settingPage{border-radius:5px;width:300px;height:150px;background-color:#f2f0ef;flex-shrink:0}.bk-input{border-radius:1em;border:2px solid #ffd3a3;height:20px;width:50px;text-align:center}.bk-input:hover{border:2px solid #ff8200}.bkBox-tip{height:100px;width:300px;display:flex;align-items:center;justify-content:center}`; const SETTINGS_TEMP = `<div class="bk-pageNumIndicator"></div><div>下载 <input type="number" value="1" min="1" max="22" class="bk-input"> 至 <input type="number" value="22" min="1" max="22" class="bk-input"> 页</div><div><button class="bkBox-button">开始下载</button></div>`; const TITLE_PIC = `<img src="">`; const BKBOX_TEMP = `<div class="bkBox-title">${TITLE_PIC}</div><nav><div class="navi"></div></nav><section><div class="box"></div></section>`; let globalConfig = { uid: $CONFIG.uid, } const TT = (type, name) => { return (async () => { let a = await getPageAmount(type); console.log(`get amount of ${type} post: ${a}`); return { type: type, name: name, pageAmount: a } })(); } let bkBox = document.createElement("div"); bkBox.className = "bkBox"; bkBox.innerHTML = BKBOX_TEMP; let bkBoxStyleSheet = document.createElement("style"); bkBoxStyleSheet.innerText = bkBoxStyle; bkBox.appendChild(bkBoxStyleSheet); let bkBoxTip = document.createElement("div"); bkBoxTip.className = 'bkBox-tip'; bkBox.appendChild(bkBoxTip); let bkTypes = [TT('fav', '我的收藏'), TT('like', '我的点赞'), TT('myblog', '我的发布')]; Promise.all(bkTypes).then((values) => { values.map((t) => { return { type: t.type, name: t.name, settingsPage: (() => { let type = t.type; let name = t.name; let pageAmount = t.pageAmount; let page = document.createElement('div'); page.innerHTML = SETTINGS_TEMP; let pageNumIndicator = page.getElementsByClassName("bk-pageNumIndicator")[0]; if (type == 'like') { pageNumIndicator.innerHTML = '无法获知点赞总页数'; pageNumIndicator.title = "微博的接口并不返回全部点赞的数量;同样每页二十条"; } else { pageNumIndicator.innerHTML = name + "的总页数为 " + pageAmount; pageNumIndicator.title = "每页二十条博文"; } let pageEndInput = page.querySelectorAll('input')[1]; pageEndInput.max = pageAmount; pageEndInput.value = pageAmount; let btn = page.getElementsByClassName("bkBox-button")[0]; btn.addEventListener("click", async () => { let dlRange_ = page.querySelectorAll("input"); let dlRange = [dlRange_[0].value, dlRange_[1].value]; hideAllButton(); await fetchAll(type, dlRange); showAllButton(); }) return page; })() }; }).forEach((c) => { settleDownBkTypes(c.type, c.name, c.settingsPage); console.log(`settled post type ${c.type} to bkBox`); }) document.body.appendChild(bkBox); }) function hideAllButton() { let buttons = document.getElementsByClassName('bkBox-button'); buttons.forEach(btn => { btn.style.display = 'none'; }) } function showAllButton() { let buttons = document.getElementsByClassName('bkBox-button'); buttons.forEach(btn => { btn.style.display = 'block'; }) } function settleDownBkTypes(type, name, settingsPage) { let nav = bkBox.getElementsByClassName("navi")[0]; let settingBox = bkBox.getElementsByClassName("box")[0]; let naviButton = document.createElement("a"); naviButton.innerHTML = "<div class='bk-navi-inner'>" + name + "</div>"; naviButton.className = "bk-naviItem" naviButton.href = "#" + type; nav.appendChild(naviButton); let page = document.createElement("div"); page.id = type; page.className = "bk-settingPage" page.appendChild(settingsPage); settingBox.appendChild(page); nav.append } async function getPageAmount(type) { if (type == 'like') { return Infinity; } let data = (await fetchMyContent(1, type)).data; let amount = 0; if (type == 'fav') { amount = data.total_number; } else { amount = data.total; } return Math.ceil(amount / 20); } function showTip(msg) { bkBoxTip.innerHTML = msg; }
QingJ © 2025
镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址