Repository: ICE99125/BiliBili_Checkin
Branch: main
Commit: b2369543e10c
Files: 7
Total size: 17.5 KB
Directory structure:
gitextract_duf56czo/
├── .gitignore
├── .gitmodules
├── bilibili.py
├── config.py
├── index.py
├── readme.md
└── tools.py
================================================
FILE CONTENTS
================================================
================================================
FILE: .gitignore
================================================
__pycache__
================================================
FILE: .gitmodules
================================================
[submodule "push"]
path = push
url = https://github.com/arcturus-script/push.git
================================================
FILE: bilibili.py
================================================
import requests as req
from re import compile
from tools import failed, handler, info, success
from datetime import datetime
import time
# 获取视频信息地址
VIDEO_INFO = "https://api.bilibili.com/x/web-interface/view"
# 获取用户信息
PERSONAL_INFO = "https://api.bilibili.com/x/space/myinfo"
# 直播签到
LIVE_BROADCAST = "https://api.live.bilibili.com/sign/doSign"
# 漫画签到
COMICS = "https://manga.bilibili.com/twirp/activity.v1.Activity/ClockIn"
# 漫画签到信息
COMICS_INFO = "https://manga.bilibili.com/twirp/activity.v1.Activity/GetClockInInfo"
# 获取热门推荐
RECOMMAND = "https://api.bilibili.com/x/web-interface/popular"
# 客户端分享视频
VIDEO_SHARE = "https://api.bilibili.com/x/web-interface/share/add"
# 投币
COIN = "https://api.bilibili.com/x/web-interface/coin/add"
# 看视频
VIDEO_CLICK = "https://api.bilibili.com/x/click-interface/click/web/h5"
VIDEO_HEARTBEAT = "https://api.bilibili.com/x/click-interface/web/heartbeat"
# 兑换硬币
TO_COIN = "https://api.live.bilibili.com/xlive/revenue/v1/wallet/silver2coin"
# 获取当日投币情况
COIN_LOG = " https://api.bilibili.com/x/member/web/coin/log"
headers = {
"user-agent": "Mozilla/5.0",
"Content-Type": "application/x-www-form-urlencoded",
"Referer": "https://www.bilibili.com/",
}
class BiliBili:
headers = headers.copy()
def __init__(self, **config) -> None:
self.cookie = config.get("cookie")
self.options = config.get("options", {})
self.sid = BiliBili.extract("sid", self.cookie)
self.csrf = BiliBili.extract("bili_jct", self.cookie)
self.uid = BiliBili.extract("DedeUserID", self.cookie)
self.headers.update({"Cookie": self.cookie})
@staticmethod
def extract(key: str, cookie: str):
"""根据键从 cookie 中抽取数据
Args:
key: 需要抽取数据的键, 可能值 bili_jct, sid, DedeUserID
cookie (str): BiliBili 的 cookie
"""
regEx = compile(f"(?<={key}=).+?(?=;)|(?<={key}=).+")
csrf = regEx.findall(cookie)
if len(csrf) != 0:
return csrf[0]
else:
return ""
# 获取视频信息
@staticmethod
def get_video_info(bv):
try:
rep = req.get(
VIDEO_INFO,
params={"bvid": bv},
headers=BiliBili.headers,
).json()
if rep["code"] == 0:
data = rep["data"]
return {
"bvid": data["bvid"], # 视频 BV 号
"aid": data["aid"], # 视频 AV 号
"duration": data["duration"],
"cid": data["cid"],
"title": data["title"], # 视频标题
}
else:
failed(f"获取视频信息失败, 原因: {rep['message']}")
except Exception as ex:
failed(f"获取视频信息时出错, 原因: {ex}")
# 获取用户信息
def get_user_info(self):
try:
rep = req.get(PERSONAL_INFO, headers=self.headers).json()
if rep["code"] == 0:
data = rep["data"]
current_exp = data["level_exp"]["current_exp"]
next_exp = data["level_exp"]["next_exp"]
self.name = data["name"] # 用户名
self.level = data["level"] # 等级
self.coin = data["coins"] # 硬币数
self.exp = f"{current_exp}/{next_exp}" # 经验
self.silence = data["silence"] # 不知道是什么
success(f"获取用户信息成功, 用户: {self.name}")
else:
raise Exception(rep["message"])
except Exception as ex:
failed(f"获取用户信息时出错, 原因: {ex}")
self.name = "Unkown"
self.level = "lv0"
self.coin = 0
self.exp = "0/0"
self.silence = "Unkown"
# 直播签到
def live_broadcast_checkin(self):
if not self.options.get("lb", False):
return
try:
rep = req.get(LIVE_BROADCAST, headers=self.headers).json()
if rep["code"] == 0:
# 签到成功
data = rep["data"]
success(f"直播签到: 奖励 {data['text']}")
return {
"raward": data["text"],
"specialText": data["specialText"],
}
else:
raise Exception(rep["message"])
except Exception as ex:
failed(f"直播签到失败, {ex}")
# 漫画签到
def comics_checkin(self):
if not self.options.get("comics", False):
return
try:
rep = req.post(
COMICS,
headers=self.headers,
data={
"platform": "android",
},
).json()
if rep["code"] == 0:
success("漫画签到完成")
result = self.comics_checkin_info()
if result is not None:
return result
else:
return "unkown"
else:
raise Exception(rep.get("msg", "Unknown error"))
except Exception as ex:
failed(f"漫画签到失败, {ex}")
def comics_checkin_info(self):
rep = req.post(COMICS_INFO, headers=self.headers).json()
if rep["code"] == 0:
success(f"获取漫画签到信息成功, 您已经连续签到 {rep['data']['day_count']} 天")
return rep["data"]["day_count"]
else:
failed(f"获取漫画签到信息失败, 原因: {rep['msg']}")
# 获取推荐视频
@staticmethod
def video_suggest(ps: int = 50, pn: int = 1) -> list or None:
"""
Args:
ps (int): 视频个数
pn (int): 第几页数据
Returns:
video_list: 一个列表, 例如
[
{"aid": 551162867, "title": "2022我的世界拜年纪", "bvid": xxx},
{"aid": 508722277, "title": "B站UP主, 办了个电影节", "bvid": yyy},
...
]
"""
rep = req.get(RECOMMAND, params={"ps": ps, "pn": pn},headers=headers).json()
if rep["code"] == 0:
res = []
videos = rep["data"]["list"]
for video in videos:
# 将视频主要信息保存到字典里
res.append(
{
"aid": video["aid"],
"bvid": video["bvid"],
"title": video["title"],
}
)
return res
else:
failed(f"获取视频推荐列表失败, 原因: {rep['message']}")
return [{"bvid": "BV1LS4y1C7Pa"}]
# 投币
def give_coin(self, videos, per_coin_num=1, select_like=0):
coined = self.getCoinLog() # 已经投币数
max_coin = self.options.get("coins", 0)
if max_coin == 0:
return
surplus = max_coin - coined
surplus = 0 if surplus < 0 else surplus
info(f"还需投币 {surplus} 个")
coin_videos = []
for video in videos:
# 当已投币数超过想投币数时退出
if coined < max_coin:
data = {
"aid": str(video["aid"]),
"multiply": per_coin_num, # 每次投币多少个, 默认 1 个
"select_like": select_like, # 是否同时点赞, 默认不点赞
"cross_domain": "true",
"csrf": self.csrf,
}
rep = req.post(COIN, headers=self.headers, data=data).json()
if rep["code"] == 0:
# 投币成功
success(f"给[{video['title']}]投币成功")
coin_videos.append(video["title"])
coined += 1 # 投币次数加 1
else:
# 投币失败
failed(f"给[{video['title']}]投币失败, 原因: {rep['message']}")
else:
success(f"投币完成, 今日共投了 {coined} 个硬币")
break
return coin_videos
# 分享视频
def share_video(self, videos):
if not self.options.get("share", False):
return
for video in videos:
# 分享视频
data = {
"aid": video["aid"],
"csrf": self.csrf,
}
rep = req.post(VIDEO_SHARE, data=data, headers=self.headers).json()
if rep["code"] == 0:
# 如果分享成功, 退出循环, 并返回分享的视频名
success(f"分享视频完成, [{video['title']}]")
return video["title"]
else:
failed(f"分享视频[{video['title']}]失败, {rep['message']}")
# 每日看视频
def watch(self, bvid):
if not self.options.get("watch", False):
return
video_info = BiliBili.get_video_info(bvid)
# 获取视频信息成功
if video_info:
data = {
"aid": video_info["aid"],
"cid": video_info["cid"],
"part": 1,
"ftime": int(time.time()),
"jsonp": "jsonp",
"mid": self.uid,
"csrf": self.csrf,
"stime": int(time.time()),
}
rep = req.post(VIDEO_CLICK, data=data, headers=self.headers).json()
# 进入视频页
if rep["code"] == 0:
data = {
"aid": video_info["aid"],
"cid": video_info["cid"],
"jsonp": "jsonp",
"mid": self.uid,
"csrf": self.csrf,
"played_time": 0,
"pause": False,
"play_type": 1,
"realtime": video_info["duration"],
"start_ts": int(time.time()),
}
rep = req.post(VIDEO_HEARTBEAT, data=data, headers=self.headers).json()
if rep["code"] == 0:
# 模拟观看视频
time.sleep(5)
data["played_time"] = video_info["duration"] - 1
data["play_type"] = 0
data["start_ts"] = int(time.time())
rep = req.post(
VIDEO_HEARTBEAT,
data=data,
headers=self.headers,
).json()
if rep["code"] == 0:
success(f"观看视频完成, [{video_info['title']}]")
return f"观看视频[{video_info['title']}]成功"
failed(f"观看视频失败, [{video_info['title']}]")
# 银瓜子兑换银币
def toCoin(self):
if not self.options.get("toCoin", False):
return
resp = req.post(
TO_COIN,
headers=self.headers,
data={
"csrf_token": self.csrf,
"csrf": self.csrf,
},
).json()
return resp.get("message", "兑换失败")
def getCoinLog(self):
resp = req.get(
COIN_LOG,
headers=self.headers,
params={
"csrf": self.csrf,
"jsonp": "jsonp",
},
).json()
res = 0
if resp.get("code") == 0:
coin_log = resp.get("data").get("list")
today = datetime.today().date()
for i in coin_log:
t = datetime.strptime(i["time"], "%Y-%m-%d %H:%M:%S")
if t.date() == today:
if i["delta"] < 0:
res += -i["delta"]
success(f"获取硬币投递情况成功, 当前已投币 {res} 个")
else:
failed("获取投币情况失败")
return res
@handler
def start(self):
self.get_user_info() # 获取用户信息
videos = self.video_suggest() # 获取热门视频
return {
"name": self.name,
"level": self.level,
"coin": self.coin,
"exp": self.exp,
"coins": self.give_coin(videos), # 投币
"share": self.share_video(videos), # 视频分享
"comics": self.comics_checkin(), # 漫画签到
"lb": self.live_broadcast_checkin(), # 直播签到
"watch": self.watch(videos[0]["bvid"]), # 观看视频
"toCoin": self.toCoin(), # 银瓜子兑换硬币,
}
================================================
FILE: config.py
================================================
config = {
"multi": [
{
"cookie": "xxx",
"options": {
"watch": True, # 每日观看视频
"coins": 1, # 投币个数
"share": True, # 视频分享
"comics": True, # 漫画签到
"lb": True, # 直播签到
"threshold": 100, # 仅剩多少币时不再投币(不写默认100)
"toCoin": False, # 银瓜子兑换硬币
},
# "push": {
# "type": "pushplus",
# "key": "xxx",
# },
},
{
"cookie": "xxx",
"options": {
"watch": True, # 每日观看视频
"coins": 2, # 投币个数
"share": True, # 视频分享
"comics": True, # 漫画签到
"lb": True, # 直播签到
"toCoin": False, # 银瓜子兑换硬币
},
# "push": [
# # 以数组的形式填写, 则会向多个服务推送消息
# {
# "type": "pushplus",
# "key": "xxx",
# },
# {
# "type": "workWechat",
# "key": {
# "agentid": 1000002,
# "corpSecret": "xxx",
# "corpid": "xxx",
# },
# },
# ],
},
],
"push": {
# 只作用于在multi并未配置 push 的组
"type": "pushplus",
"key": "xxx",
},
}
================================================
FILE: index.py
================================================
from bilibili import BiliBili
from config import config
from push import PushSender, parse
def parse_message(message, push_type):
if push_type == "pushplus":
return parse(message, template="html")
else:
return parse(message, template="markdown")
def pushMessage(message, config):
if isinstance(config, list):
for item in config:
t = item.get("type")
p = PushSender(t, item.get("key"))
p.send(parse_message(message, t), title="Bilibili")
else:
t = config.get("type")
p = PushSender(config.get("type"), config.get("key"))
p.send(parse_message(message, t), title="Bilibili")
def main(*args):
accounts = config.get("multi")
push_together = config.get("push")
messages = []
for item in accounts:
obj = BiliBili(**item)
res = obj.start()
push = item.get("push")
if push is None:
if push_together is not None:
messages.extend(res)
else:
pushMessage(res, push)
if len(messages) != 0 and push_together is not None:
pushMessage(messages, push_together)
if __name__ == "__main__":
main()
================================================
FILE: readme.md
================================================
## BiliBili(云函数版)
### 实现功能
- [x] 获取用户信息
- [x] 直播签到
- [x] 漫画签到
- [x] 投币
- [x] 分享视频
- [x] 每日看视频
- [x] 多账户支持
- [x] 自动兑换银瓜子
- [ ] 大会员积分签到(等我有大会员了再搞(╹ڡ╹ ))
### 步骤
注意把子模块也一起下载
```bash
git clone --recursive https://github.com/arcturus-script/bilibili.git
```
直接配置 config.py 即可, 入口改为 index.main
================================================
FILE: tools.py
================================================
def handler(fn):
def inner(*args, **kwargs):
res = fn(*args, **kwargs)
content = [
{
"h4": {
"content": res["name"],
},
},
{
"txt": {
"content": f"等级: {res['level']}",
},
},
{
"txt": {
"content": f"硬币: {res['coin']}",
},
},
{
"txt": {
"content": f"经验: {res['exp']}",
},
},
]
watch = res.get("watch")
if watch is not None:
content.append(
{
"txt": {
"content": watch,
}
}
)
share = res.get("share")
if share is not None:
content.append(
{
"txt": {
"content": f"分享视频: {share}",
}
}
)
coins = res.get("coins")
if coins is not None:
content.append(
{
"h5": {
"content": "投币",
},
"orderedList": {
"content": coins,
},
}
)
comics = res.get("comics")
if comics is not None:
content.extend(
[
{
"h5": {
"content": "漫画签到",
},
"txt": {
"content": f"连续签到 {comics} 天",
},
},
]
)
lb = res.get("lb")
if lb is not None:
content.extend(
[
{
"h5": {
"content": "直播",
},
"txt": {
"content": lb["raward"],
},
},
]
)
toCoin = res.get("toCoin")
if toCoin is not None:
content.append(
{
"h5": {
"content": "银瓜子兑换硬币",
},
"txt": {
"content": toCoin,
},
}
)
return content
return inner
def failed(*args, **kwargs):
print("[\033[31mfailed\033[0m] ", end="")
print(*args, **kwargs)
def success(*args, **kwargs):
print("[\033[32msuccess\033[0m] ", end="")
print(*args, **kwargs)
def info(*args, **kwargs):
print("[\033[34minfo\033[0m] ", end="")
print(*args, **kwargs)
gitextract_duf56czo/ ├── .gitignore ├── .gitmodules ├── bilibili.py ├── config.py ├── index.py ├── readme.md └── tools.py
SYMBOL INDEX (22 symbols across 3 files)
FILE: bilibili.py
class BiliBili (line 48) | class BiliBili:
method __init__ (line 51) | def __init__(self, **config) -> None:
method extract (line 61) | def extract(key: str, cookie: str):
method get_video_info (line 77) | def get_video_info(bv):
method get_user_info (line 101) | def get_user_info(self):
method live_broadcast_checkin (line 130) | def live_broadcast_checkin(self):
method comics_checkin (line 154) | def comics_checkin(self):
method comics_checkin_info (line 181) | def comics_checkin_info(self):
method video_suggest (line 193) | def video_suggest(ps: int = 50, pn: int = 1) -> list or None:
method give_coin (line 232) | def give_coin(self, videos, per_coin_num=1, select_like=0):
method share_video (line 278) | def share_video(self, videos):
method watch (line 300) | def watch(self, bvid):
method toCoin (line 360) | def toCoin(self):
method getCoinLog (line 375) | def getCoinLog(self):
method start (line 405) | def start(self):
FILE: index.py
function parse_message (line 6) | def parse_message(message, push_type):
function pushMessage (line 13) | def pushMessage(message, config):
function main (line 29) | def main(*args):
FILE: tools.py
function handler (line 1) | def handler(fn):
function failed (line 115) | def failed(*args, **kwargs):
function success (line 120) | def success(*args, **kwargs):
function info (line 125) | def info(*args, **kwargs):
Condensed preview — 7 files, each showing path, character count, and a content snippet. Download the .json file or copy for the full structured content (21K chars).
[
{
"path": ".gitignore",
"chars": 11,
"preview": "__pycache__"
},
{
"path": ".gitmodules",
"chars": 83,
"preview": "[submodule \"push\"]\n\tpath = push\n\turl = https://github.com/arcturus-script/push.git\n"
},
{
"path": "bilibili.py",
"chars": 11962,
"preview": "import requests as req\nfrom re import compile\nfrom tools import failed, handler, info, success\nfrom datetime import date"
},
{
"path": "config.py",
"chars": 1438,
"preview": "config = {\n \"multi\": [\n {\n \"cookie\": \"xxx\",\n \"options\": {\n \"watch\": True,"
},
{
"path": "index.py",
"chars": 1207,
"preview": "from bilibili import BiliBili\nfrom config import config\nfrom push import PushSender, parse\n\n\ndef parse_message(message, "
},
{
"path": "readme.md",
"chars": 293,
"preview": "## BiliBili(云函数版)\n\n### 实现功能\n\n- [x] 获取用户信息\n- [x] 直播签到\n- [x] 漫画签到\n- [x] 投币\n- [x] 分享视频\n- [x] 每日看视频\n- [x] 多账户支持\n- [x] 自动兑换银瓜"
},
{
"path": "tools.py",
"chars": 2933,
"preview": "def handler(fn):\n def inner(*args, **kwargs):\n res = fn(*args, **kwargs)\n\n content = [\n {\n "
}
]
About this extraction
This page contains the full source code of the ICE99125/BiliBili_Checkin GitHub repository, extracted and formatted as plain text for AI agents and large language models (LLMs). The extraction includes 7 files (17.5 KB), approximately 4.6k tokens, and a symbol index with 22 extracted functions, classes, methods, constants, and types. Use this with OpenClaw, Claude, ChatGPT, Cursor, Windsurf, or any other AI tool that accepts text input. You can copy the full output to your clipboard or download it as a .txt file.
Extracted by GitExtract — free GitHub repo to text converter for AI. Built by Nikandr Surkov.