-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathupdate.py
361 lines (328 loc) · 13.6 KB
/
update.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
from asyncio import sleep, gather
from collections import deque
from types import SimpleNamespace
from math import ceil
from webbrowser import open as webbrowser_open
from aiohttp import ClientError
from bilibili_api.user import User, BangumiType
from bilibili_api.bangumi import get_meta
from bilibili_api.exceptions import ResponseCodeException
from config import (
PARSE_EPISODE_PROGRESS, SKIP_COLLECTED, READ_ONLY,
OPEN_FAILED_BANGUMI_BILI_PAGE
)
from utilities import (
loop, client, print_debug, print_status,
try_for_times_async_chain, try_get_json, try_post_json
)
from episode_progress import parse_episode_progress
async def get_one_bili_data(data: SimpleNamespace, page: int):
'''获取单个 Bilibili 追番数据'''
print_debug(f'获取追番数据 @ {page} ...')
bangumi_data = await try_for_times_async_chain( # 尝试三次
3,
lambda: data.user.get_subscribed_bangumis(
page,
BangumiType.BANGUMI
)
)
data.bangumi_total += len(bangumi_data['list'])
data.bangumi_remaining.extend(map(
lambda bangumi: (
bangumi['media_id'], bangumi['follow_status'], bangumi['progress']
),
bangumi_data['list']
))
data.bili_processed_count += 1
if page == 1:
return ceil(bangumi_data['total'] / 15) # 计算并返回总页数
async def get_bili_data(data: SimpleNamespace):
'''获取 Bilibili 追番数据'''
print_debug('获取第一页 Bilibili 追番数据...')
try:
data.bili_total_count = await get_one_bili_data(data, 1)
except ResponseCodeException:
print_status(
'** Bilibili 授权设置不正确,无法读取隐私设置'
'未公开的 Bilibili 追番数据!'
)
exit(1)
print_debug('创建并等待获取单个 Bilibili 追番数据任务...')
await gather(*(
loop.create_task(get_one_bili_data(data, page))
for page in range(2, data.bili_total_count + 1)
))
print_debug('完成!')
async def update_one_bgm_data(
data: SimpleNamespace,
bgm_id: int, status: str, progress: str
):
'''更新单个 Bangumi 动画数据'''
print_debug(f'开始更新 @ {bgm_id} -> {status} ...')
update_flag = True
eps_count = None
if SKIP_COLLECTED:
# 获取当前收藏状态
collection_status = await try_get_json( # 尝试三次
3, client,
f'https://api.bgm.tv/collection/{bgm_id}',
headers={'Authorization': data.bgm_auth_data}
)
status_previous = None
if 'status' in collection_status:
status_previous = collection_status['status']['type']
if status_previous == status:
update_flag = False
if PARSE_EPISODE_PROGRESS and status == 'do':
eps_count = parse_episode_progress(progress)
if eps_count is not None:
update_flag = True
if update_flag:
update_watched_eps_flag = False
# 更新在看分集进度
if PARSE_EPISODE_PROGRESS and status == 'do' and eps_count is not None:
update_watched_eps_flag = True
elif status == 'collect': # 更新看过分集进度
# 获取分集总数
subject_data = await try_get_json( # 尝试三次
3, client,
f'https://api.bgm.tv/subject/{bgm_id}',
headers={'Authorization': data.bgm_auth_data}
)
eps_count = subject_data['eps_count']
update_watched_eps_flag = True
if update_watched_eps_flag:
# 为更新分集进度预先更新收藏为在看
if (status_previous != 'do'):
print_debug(f'预先更新收藏 @ {bgm_id} -> do ...')
if not READ_ONLY:
result = await try_post_json( # 尝试三次
3, client,
f'https://api.bgm.tv/collection/{bgm_id}/update',
data={'status': 'do'},
headers={'Authorization': data.bgm_auth_data}
)
print_debug(f'预先更新收藏完成 @ {bgm_id}')
if status == 'do':
print_debug(f'更新在看分集进度 @ {bgm_id} -> {eps_count} ...')
if not READ_ONLY:
ep_info = await try_get_json( # 尝试三次
3, client,
f'https://api.bgm.tv/subject/{bgm_id}/ep',
headers={'Authorization': data.bgm_auth_data}
)
ep_id_raw = dict(map(
lambda ep: (ep['sort'], str(ep['id'])),
filter(lambda ep: ep['type'] == 0, ep_info['eps'])
))
ep_watched_info = await try_get_json( # 尝试三次
3, client,
f'https://api.bgm.tv/user/'
f'{data.bgm_user_id}/progress?subject_id={bgm_id}',
headers={'Authorization': data.bgm_auth_data}
)
if ep_watched_info is None:
ep_watched = set()
else:
ep_watched = set(map(
lambda ep: str(ep['id']),
filter(
lambda ep: ep['status']['id'] == 2,
ep_watched_info['eps']
)
))
ep_ids = deque()
for i in range(1, eps_count + 1):
if i in ep_id_raw:
ep_id = ep_id_raw[i]
if ep_id not in ep_watched:
ep_ids.append(ep_id)
if len(ep_ids) > 0:
result = await try_post_json( # 尝试三次
3, client,
f'https://api.bgm.tv'
f'/ep/{ep_ids[-1]}/status/watched',
data={
'ep_id': ','.join(ep_ids)
},
headers={'Authorization': data.bgm_auth_data}
)
code = result['code']
msg = result['error']
print_debug(
f'更新在看分集进度返回状态 {code} {msg} @ {bgm_id}'
)
if code > 400: # 忽略重复更新时的 400 Bad Request
print_status(
f'** 更新在看分集进度返回状态'
f' {code} {msg} @ {bgm_id}'
)
data.bangumi_failed_count += 1
data.bangumi_processed_count += 1
return
elif status == 'collect':
print_debug(f'更新看过分集进度 @ {bgm_id} -> {eps_count} ...')
if not READ_ONLY:
result = await try_post_json( # 尝试三次
3, client,
f'https://api.bgm.tv'
f'/subject/{bgm_id}/update/watched_eps',
data={'watched_eps': eps_count},
headers={'Authorization': data.bgm_auth_data}
)
code = result['code']
msg = result['error']
print_debug(
f'更新看过分集进度返回状态 {code} {msg} @ {bgm_id}'
)
if code > 400: # 忽略重复更新时的 400 Bad Request
print_status(
f'** 更新看过分集进度返回状态'
f' {code} {msg} @ {bgm_id}'
)
data.bangumi_failed_count += 1
data.bangumi_processed_count += 1
return
if (not SKIP_COLLECTED) or status_previous != status:
print_debug(f'更新收藏 @ {bgm_id} -> {status} ...')
if not READ_ONLY:
result = await try_post_json( # 尝试三次
3, client,
f'https://api.bgm.tv/collection/{bgm_id}/update',
data={'status': status},
headers={'Authorization': data.bgm_auth_data}
)
print_debug(f'更新收藏完成 @ {bgm_id}')
else:
print_debug(f'跳过 @ {bgm_id} -> {status} ...')
data.bangumi_processed_count += 1
print_debug(f'完成 @ {bgm_id}!')
async def check_and_update_bgm_data(data: SimpleNamespace):
'''检查剩余数据并更新 Bangumi 动画数据'''
while len(data.bangumi_remaining) > 0:
bangumi = data.bangumi_remaining.popleft()
if (
bangumi[0] in data.bili2bgm_map
and data.bili2bgm_map[bangumi[0]][0] is not None
):
try:
data.update_one_bgm_data_tasks.append(
loop.create_task(update_one_bgm_data(
data,
data.bili2bgm_map[bangumi[0]][0],
{1: 'wish', 2: 'do', 3: 'collect'}[bangumi[1]],
bangumi[2]
))
)
except ClientError:
pass
else:
continue
data.bangumi_failed.append(bangumi[0])
data.bangumi_failed_count += 1
data.bangumi_processed_count += 1
async def update_bgm_data(data: SimpleNamespace):
'''更新 Bangumi 动画数据'''
print_debug('创建更新单个 Bangumi 数据任务...')
while not data.get_bili_data_task.done():
await check_and_update_bgm_data(data)
await sleep(0.01)
await check_and_update_bgm_data(data)
print_debug('等待更新单个 Bangumi 数据任务...')
await gather(*data.update_one_bgm_data_tasks)
print_debug('完成!')
async def print_unknown(data: SimpleNamespace, bangumi: int):
'''打印没有对应的数据的 Bilibili 追番数据'''
bangumi_meta = await get_meta(bangumi, data.bili_auth_data)
print_status(
f'** {bangumi_meta["media"]["title"]}'
f'(Bilibili 编号 md{bangumi})没有对应的数据!'
)
async def print_progress(data: SimpleNamespace):
'''打印进度'''
while len(data.bangumi_failed) > 0:
bangumi = data.bangumi_failed.popleft()
if bangumi in data.bili2bgm_map:
if data.bili2bgm_map[bangumi][0] is not None:
print_status(
f'** {data.bili2bgm_map[bangumi][1]}'
f'(Bilibili 编号 md{bangumi},'
f'Bangumi 编号 {data.bili2bgm_map[bangumi][0]})更新失败!'
)
else:
print_status(
f'** {data.bili2bgm_map[bangumi][1]}'
f'(Bilibili 编号 md{bangumi})'
f'没有对应的 Bangumi 数据!'
)
else:
data.print_unknown_tasks.append(
loop.create_task(
print_unknown(data, bangumi)
)
)
if OPEN_FAILED_BANGUMI_BILI_PAGE:
webbrowser_open(
f'https://www.bilibili.com/bangumi/media/md{bangumi}/'
)
bili_status = '[Bilibili %d/%d %.1f%%]' % (
data.bili_processed_count,
data.bili_total_count,
(data.bili_processed_count) / data.bili_total_count * 100
)
bgm_status = '[Bangumi %d/%d %.1f%% (失败 %s)]' % (
data.bangumi_processed_count,
data.bangumi_total,
data.bangumi_processed_count / data.bangumi_total * 100,
data.bangumi_failed_count
)
print_status(
f'{bili_status} -> {bgm_status} {"." * data.animation_points}',
end='\r'
)
data.animation_points += 1
if data.animation_points > 3:
data.animation_points = 1
async def get_and_update(
bili2bgm_map, bili_auth_data, bili_uid, bgm_auth_data, bgm_user_id
):
'''获取 Bilibili 番剧数据并更新 Bangumi 动画数据'''
data = SimpleNamespace(
bili2bgm_map=bili2bgm_map,
user=User(bili_uid, bili_auth_data),
bili_auth_data=bili_auth_data,
bgm_auth_data=bgm_auth_data,
bgm_user_id=bgm_user_id,
bili_processed_count=0,
bili_total_count=None,
bangumi_total=0,
# [(bili media id, 1: 想看 | 2: 在看 | 3: 看过), ...]
bangumi_remaining=deque(),
bangumi_processed_count=0,
bangumi_failed=deque(),
bangumi_failed_count=0,
animation_points=1,
get_bili_data_task=None,
update_bgm_data_task=None,
update_one_bgm_data_tasks=deque(),
print_unknown_tasks=deque()
)
print_debug('创建获取 Bilibili 数据任务 -> [get_bili_data]')
data.get_bili_data_task = loop.create_task(get_bili_data(data))
print_debug('创建更新 Bangumi 数据任务 -> [update_bgm_data]')
data.update_bgm_data_task = loop.create_task(update_bgm_data(data))
print_debug('等待任务...')
while data.bili_total_count is None:
await sleep(0.01)
while not (
data.get_bili_data_task.done()
and data.update_bgm_data_task.done()
):
await print_progress(data)
await sleep(0.1)
await data.get_bili_data_task
await data.update_bgm_data_task
await print_progress(data)
await gather(*data.print_unknown_tasks)
await print_progress(data)
print()