吾爱破解 - 52pojie.cn

 找回密码
 注册[Register]

QQ登录

只需一步,快速开始

查看: 8036|回复: 29
收起左侧

[其他原创] [油猴脚本]Pixiv系列小说自动爬虫

  [复制链接]
steven026 发表于 2022-7-24 20:55
本帖最后由 steven026 于 2022-7-25 09:02 编辑

【简介】
用来练手的爬虫脚本,放弃了引入第三方脚本,选择使用油猴内置函数,因此依赖油猴。
不需要设置cookies,只需要用浏览器登陆Pixiv保证对目标小说有访问权限即可。
下载内容存储在油猴脚本中,可通过控制台命令将指定小说导出到本地。

由于GM_xmlhttpRequest没有跨域限制,实际上可以将
// @match https://www.pixiv.net/novel/series/*
替换为
// @include *
从而在任意页面使用本脚本

【爬虫命令】
startMain(seriesID,"Charpters",isForce)
启动爬虫
--{number}  seriesID:  必填,系列小说ID,获取位置为:https://www.pixiv.net/novel/series/系列小说ID (数字,不含?)
--{string}  Charpters: 可选,指定章节,默认为全部,具体指定规则见后文
--{boolean} isForce:   可选,对于已存在章节是否强制下载,默认为否


downloadList()
获取已下载内容列表
无参数,返回2个内容,分别为全部章节和全部小说,全部小说中文本可用于download()参数中的NovelName


download(NovelName,Charpters)
导出已下载内容
--{string} NovelName: 必填,系列小说名,可通过downloadList()获取
--{string} Charpters: 可选,指定章节,默认为全部,具体指定规则见后文


Charpters 规则:
规则1:单一数字/^\d+$/       添加单一数字章节到下载队列
规则2:数字范围/^\d+-\d+$/   添加数字范围章节到下载队列
规则3:^单一数字/^\^\d+$/    从下载队列中删除单一数字章节

多个规则用,分隔
例:"1,3-10,^5,15" 代表下载第1、3、4、6、7、8、9、10、15章
例:"^5,3-10,1,15" 代表下载第1、3、4、5、6、7、8、9、10、15章
[JavaScript] 纯文本查看 复制代码
// ==UserScript==
// @name         Pixiv系列小说自动爬虫
// @version      1.0
// @description  根据Pixiv系列小说seriesID自动爬取指定章节或整本小说并可导出成.txt
// @Author       DreamNya
// @match        https://www.pixiv.net/novel/series/*
// @grant        unsafeWindow
// @grant        GM_xmlhttpRequest
// @grant        GM_setValue
// @grant        GM_getValue
// @grant        GM_listValues
// @grant        GM_download
// @connect      pixiv.net
// @license      MIT
// @namespace https://greasyfork.org/users/809466
// ==/UserScript==

//自定义章节分割名 用以小说阅读器自动拆分章节
function CustomCharpter(index) {
        return `第${index}章 `
}

function GMget(url) {
        return new Promise((resolve) => {
                GM_xmlhttpRequest({
                        method: "GET",
                        url: url,
                        onload: resolve
                })
        })
}

//获取所有章节信息
async function startCharpter(ID) {
        let CharpterRes = await GMget(`https://www.pixiv.net/ajax/novel/series/${ID}/content_titles`)
        if(CharpterRes.status != 200) {
                console.log(CharpterRes)
                throw new Error(JSON.parse(CharpterRes.responseText).message)
        }
        return JSON.parse(CharpterRes.responseText).body
}

//获取章节内容
async function getCharpter(Charpter, CharpterIndex) {
        let CharpterPage = await GMget("https://www.pixiv.net/novel/show.php?id=" + Charpter.id)
        let CharpterDoc = new DOMParser().parseFromString(CharpterPage.responseText, "text/html")
        let CharpterNovel = JSON.parse(CharpterDoc.querySelector("#meta-preload-data").content).novel
        let result = CharpterNovel[Object.getOwnPropertyNames(CharpterNovel)].content
        return CharpterIndex + Charpter.title + "\n" + result //自动补一行章节名用以分割章节
}

//主函数 启动爬虫
async function startMain(ID, Charpters = "ALL", force = false) {
        let mainRes = await GMget("https://www.pixiv.net/ajax/novel/series/" + ID)
        if(mainRes.status != 200) {
                console.log(mainRes)
                throw new Error(JSON.parse(mainRes.responseText).message)
        }
        let mainJSON = JSON.parse(mainRes.responseText).body
        let NovelName = mainJSON.title
        let NovelTotal = mainJSON.total
        let res = Charpters != "ALL" ? CharptersParse(Charpters) : "ALL"
        if(res != "ALL" && res[res.length - 1] > NovelTotal) throw new Error(`欲爬取章节${res[res.length-1]}大于可爬取最大章节数${NovelTotal}`)
        let CharpterInfos = await startCharpter(ID)
        if(res == "ALL") res = CharpterInfos.map((item, index) => index + 1)
        for(let item of res) {
                let index = item - 1
                let CharpterInfo = CharpterInfos[index]
                let CharpterIndex = CustomCharpter(item)
                if(CharpterInfo.available != true) {
                        console.log(`${CharpterIndex}无访问权限 无法爬取 已跳过 available!=true`, CharpterInfo)
                        continue
                }
                let CharpterKey = `${NovelName} # ${CharpterIndex}`
                if(force == false && GM_getValue(CharpterKey)) {
                        console.log(CharpterInfo.title + "已存在,已跳过爬取,如需强制爬取,请调用startMain(ID,Charpters,true)")
                        continue
                }
                let CharpterResult = await getCharpter(CharpterInfo, CharpterIndex)
                GM_setValue(CharpterKey, CharpterResult)
                console.log(CharpterInfo.title + "爬取成功")
        }
        console.log(ID, `${NovelName} ${Charpters} "已爬取完毕"`)
}
unsafeWindow.startMain = startMain

/*
Charpters 规则:
规则1:单一数字/^\d+$/       添加该单一数字章节到下载队列
规则2:数字范围/^\d+-\d+$/   添加该数字范围章节到下载队列
规则3:^单一数字/^\^\d+$/    从下载队列中删除该单一数字

多个规则用,分隔
例:"1,3-10,^5,15" 代表下载第1、3、4、6、7、8、9、10、15章
例:"^5,3-10,1,15" 代表下载第1、3、4、5、6、7、8、9、10、15章
*/
//导出已下载内容
function download(NovelName, Charpters) {
        let keys = Charpters ? CharptersParse(Charpters) : "ALL"
        let NovelValues = []
        if(keys == "ALL") {
                let pattern = new RegExp("^" + NovelName + " # ")
                let Allkeys = GM_listValues().filter(i => pattern.test(i))
                if(Allkeys.length == 0) {
                        throw new Error(NovelName + "未下载任何章节")
                }
                NovelValues = Allkeys.map(i => GM_getValue(i))
                Charpters = "ALL"
        } else {
                for(let item of keys) {
                        let _value = GM_getValue(`${NovelName} # ${CustomCharpter(_value)}`)
                        if(_value == void 0) {
                                throw new Error(`${NovelName} # ${CustomCharpter(_value)} 未下载,无法保存到本地`)
                        }
                        NovelValues.push(_value)
                }
        }
        GM_download(URL.createObjectURL(new Blob([NovelValues.join("\n\n")])), `${NovelName} ${Charpters}.txt`)
}
unsafeWindow.download = download

//获取已下载内容列表
function downloadList() {
        let list = GM_listValues()
        console.log("全部章节", list)
        console.log("全部小说", [...new Set(list.map(i => i.split(" # ")[0]))])
}
unsafeWindow.downloadList = downloadList

//指定章节格式化
function CharptersParse(Charpters) {
        let res = []
        for(let item of Charpters.split(",")) {
                switch (true) {
                        case /^\d+$/.test(item):
                                res.push(item * 1)
                                break
                        case /^\d+-\d+$/.test(item):
                                {
                                        let con = item.split("-")
                                        for(let i = con[0] * 1; i <= con[1] * 1; i++) res.push(i)
                                        break
                                }
                        case /^\^\d+$/.test(item):
                                {
                                        let non = item.replace("^", "")
                                        res = res.filter(i => i != non)
                                        break
                                }
                        default:
                                throw new Error(item + "格式有误")
                }
        }
        return [...new Set(res)].sort()
}

免费评分

参与人数 2吾爱币 +2 热心值 +2 收起 理由
jlzoe + 1 + 1 谢谢@Thanks!
mincai + 1 + 1 谢谢@Thanks!

查看全部评分

本帖被以下淘专辑推荐:

发帖前要善用论坛搜索功能,那里可能会有你要找的答案或者已经有人发布过相同内容了,请勿重复发帖。

 楼主| steven026 发表于 2022-7-25 09:04
helh0275 发表于 2022-7-25 00:52
为啥显示创建无效啊?大佬

论坛自动把油猴元信息识别了……把 删除就行了
我已经编辑了,要等审核

另外我把代码发greasyfork托管了,你可以直接去gf(不知道论坛允不允许发gf链接,不允许发的话我删了)
https://greasyfork.org/scripts/448366
 楼主| steven026 发表于 2022-7-26 13:32
zhuyuanlun 发表于 2022-7-26 11:24
很多链接都是这样的啊https://www.pixiv.net/novel/show.php?id=******  。   不是这种系列的https://www.p ...

写这个脚本是为了下整本小说的
如果是单章的小说不如直接直接复制粘贴
枫叶江南 发表于 2022-7-24 23:28
我去,大神你这个早点出来就好了,我也不会花几百大洋去看小说了
RyesLai 发表于 2022-7-25 00:13
厉害的,给大佬点赞
Sameal 发表于 2022-7-25 00:39
这脚本好厉害
helh0275 发表于 2022-7-25 00:52
本帖最后由 helh0275 于 2022-7-25 00:54 编辑

为啥显示创建无效啊?大佬
批注 2022-07-25 005358.jpg
穷困潦倒待救济 发表于 2022-7-25 03:55
提示脚本无效
yousj2022 发表于 2022-7-25 07:19
有打包的成品不
雁渡LS寒潭 发表于 2022-7-25 08:24

// @name         Pixiv系列小说自动爬虫
有些版本 不兼容 在第二行加上这个
MI20220721 发表于 2022-7-25 08:27
谢谢楼主分享,试一下!
yjn866y 发表于 2022-7-25 08:45
您需要登录后才可以回帖 登录 | 注册[Register]

本版积分规则

返回列表

RSS订阅|小黑屋|处罚记录|联系我们|吾爱破解 - LCG - LSG ( 京ICP备16042023号 | 京公网安备 11010502030087号 )

GMT+8, 2024-11-25 01:20

Powered by Discuz!

Copyright © 2001-2020, Tencent Cloud.

快速回复 返回顶部 返回列表