微博备份

微博数据下载备份

目前為 2023-05-12 提交的版本,檢視 最新版本

// ==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或关注我们的公众号极客氢云获取最新地址