-
Notifications
You must be signed in to change notification settings - Fork 16
/
Copy pathvideo.go
749 lines (682 loc) · 41.7 KB
/
video.go
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
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
package bilibili
import (
"github.com/go-resty/resty/v2"
)
type VideoParam struct {
Aid int `json:"aid,omitempty" request:"query,omitempty"` // 稿件avid。avid与bvid任选一个
Bvid string `json:"bvid,omitempty" request:"query,omitempty"` // 稿件bvid。avid与bvid任选一个
}
type CardVip struct {
Type int `json:"type"` // 会员类型。0:无。1:月大会员。2:年度及以上大会员
Status int `json:"status"` // 会员状态。0:无。1:有
DueDate int `json:"due_date"` // 会员过期时间。Unix时间戳(毫秒)
VipPayType int `json:"vip_pay_type"` // 支付类型。0:未支付(常见于官方账号)。1:已支付(以正常渠道获取的大会员均为此值)
ThemeType int `json:"theme_type"` // 0。作用尚不明确
Label Label `json:"label"` // 会员标签
AvatarSubscript int `json:"avatar_subscript"` // 是否显示会员图标。0:不显示。1:显示
NicknameColor string `json:"nickname_color"` // 会员昵称颜色。颜色码,一般为#FB7299,曾用于愚人节改变大会员配色
Role int `json:"role"` // 大角色类型。1:月度大会员。3:年度大会员。7:十年大会员。15:百年大会员
AvatarSubscriptUrl string `json:"avatar_subscript_url"` // 大会员角标地址
TvVipStatus int `json:"tv_vip_status"` // 电视大会员状态。0:未开通
TvVipPayType int `json:"tv_vip_pay_type"` // 电视大会员支付类型
}
type VideoCard struct {
Mid string `json:"mid"` // 用户mid
Name string `json:"name"` // 用户昵称
Approve bool `json:"approve"` // false。作用尚不明确
Sex string `json:"sex"` // 用户性别。男 女 保密
Rank string `json:"rank"` // 10000。作用尚不明确
Face string `json:"face"` // 用户头像链接
FaceNft int `json:"face_nft"` // 是否为 nft 头像。0不是nft头像。1是 nft 头像
Displayrank string `json:"DisplayRank"` // 0。作用尚不明确
Regtime int `json:"regtime"` // 0。作用尚不明确
Spacesta int `json:"spacesta"` // 0。作用尚不明确
Birthday string `json:"birthday"` // 空。作用尚不明确
Place string `json:"place"` // 空。作用尚不明确
Description string `json:"description"` // 空。作用尚不明确
Article int `json:"article"` // 0。作用尚不明确
Attentions []any `json:"attentions"` // 空。作用尚不明确
Fans int `json:"fans"` // 粉丝数
Friend int `json:"friend"` // 关注数
Attention int `json:"attention"` // 关注数
Sign string `json:"sign"` // 签名
LevelInfo LevelInfo `json:"level_info"` // 等级
Pendant Pendant `json:"pendant"` // 挂件
Nameplate Nameplate `json:"nameplate"` // 勋章
Official Official `json:"Official"` // 认证信息
OfficialVerify OfficialVerify `json:"official_verify"` // 认证信息2
Vip CardVip `json:"vip"` // 大会员状态
IsSeniorMember int `json:"is_senior_member"` // 是否为硬核会员。0:否。1:是
}
type VideoDetailInfoCard struct {
Card VideoCard `json:"card"` // UP主名片信息
Space CardSpace `json:"space"` // 主页头图
Following bool `json:"following"` // 是否关注此用户。true:已关注。false:未关注。需要登录(Cookie) 。未登录为false
ArchiveCount int `json:"archive_count"` // 用户稿件数
ArticleCount int `json:"article_count"` // 用户专栏数
Follower int `json:"follower"` // 粉丝数
LikeNum int `json:"like_num"` // UP主获赞次数
}
type VideoDetailInfo struct {
View VideoInfo `json:"View"` // 视频基本信息
Card VideoDetailInfoCard `json:"Card"` // 视频UP主信息
Tags []VideoTag `json:"Tags"` // 视频TAG信息
Reply CommentsHotReply `json:"Reply"` // 视频热评信息
Related []VideoInfo `json:"Related"` // 推荐视频信息
Spec any `json:"Spec"` // ?。作用尚不明确
HotShare any `json:"hot_share"` // ?。作用尚不明确
Elec any `json:"elec"` // ?。作用尚不明确
Recommend any `json:"recommend"` // ?。作用尚不明确
ViewAddit any `json:"view_addit"` // ?。作用尚不明确
}
// GetVideoDetailInfo 获取视频超详细信息
func (c *Client) GetVideoDetailInfo(param VideoParam) (*VideoDetailInfo, error) {
const (
method = resty.MethodGet
url = "https://api.bilibili.com/x/web-interface/view/detail"
)
return execute[*VideoDetailInfo](c, method, url, param)
}
// GetVideoRecommendList 获取单视频推荐列表
func (c *Client) GetVideoRecommendList(param VideoParam) ([]VideoInfo, error) {
const (
method = resty.MethodGet
url = "https://api.bilibili.com/x/web-interface/archive/related"
)
return execute[[]VideoInfo](c, method, url, param)
}
type DescV2 struct {
RawText string `json:"raw_text"` // 简介内容。type=1时显示原文。type=2时显示'@'+raw_text+' '并链接至biz_id的主页
Type int `json:"type"` // 类型。1:普通,2:@他人
BizId int `json:"biz_id"` // 被@用户的mid。=0,当type=1
}
type VideoRights struct {
Bp int `json:"bp"` // 是否允许承包
Elec int `json:"elec"` // 是否支持充电
Download int `json:"download"` // 是否允许下载
Movie int `json:"movie"` // 是否电影
Pay int `json:"pay"` // 是否PGC付费
Hd5 int `json:"hd5"` // 是否有高码率
NoReprint int `json:"no_reprint"` // 是否显示“禁止转载”标志
Autoplay int `json:"autoplay"` // 是否自动播放
UgcPay int `json:"ugc_pay"` // 是否UGC付费
IsCooperation int `json:"is_cooperation"` // 是否为联合投稿
UgcPayPreview int `json:"ugc_pay_preview"` // 0。作用尚不明确
NoBackground int `json:"no_background"` // 0。作用尚不明确
CleanMode int `json:"clean_mode"` // 0。作用尚不明确
IsSteinGate int `json:"is_stein_gate"` // 是否为互动视频
Is360 int `json:"is_360"` // 是否为全景视频
NoShare int `json:"no_share"` // 0。作用尚不明确
ArcPay int `json:"arc_pay"` // 0。作用尚不明确
FreeWatch int `json:"free_watch"` // 0。作用尚不明确
}
type Owner struct {
Mid int `json:"mid"` // UP主mid
Name string `json:"name"` // UP主昵称
Face string `json:"face"` // UP主头像
}
type VideoStat struct {
Aid int `json:"aid"` // 稿件avid
View int `json:"view"` // 播放数
Danmaku int `json:"danmaku"` // 弹幕数
Reply int `json:"reply"` // 评论数
Favorite int `json:"favorite"` // 收藏数
Coin int `json:"coin"` // 投币数
Share int `json:"share"` // 分享数
NowRank int `json:"now_rank"` // 当前排名
HisRank int `json:"his_rank"` // 历史最高排行
Like int `json:"like"` // 获赞数
Dislike int `json:"dislike"` // 点踩数。恒为0
Evaluation string `json:"evaluation"` // 视频评分
Vt int `json:"vt"` // 作用尚不明确。恒为0
}
type VideoSubtitleAuthor struct {
Mid int `json:"mid"` // 字幕上传者mid
Name string `json:"name"` // 字幕上传者昵称
Sex string `json:"sex"` // 字幕上传者性别。男 女 保密
Face string `json:"face"` // 字幕上传者头像url
Sign string `json:"sign"` // 字幕上传者签名
Rank int `json:"rank"` // 10000。作用尚不明确
Birthday int `json:"birthday"` // 0。作用尚不明确
IsFakeAccount int `json:"is_fake_account"` // 0。作用尚不明确
IsDeleted int `json:"is_deleted"` // 0。作用尚不明确
}
type VideoSubtitle struct {
Id int `json:"id"` // 字幕id
Lan string `json:"lan"` // 字幕语言
LanDoc string `json:"lan_doc"` // 字幕语言名称
IsLock bool `json:"is_lock"` // 是否锁定
AuthorMid int `json:"author_mid"` // 字幕上传者mid
SubtitleUrl string `json:"subtitle_url"` // json格式字幕文件url
Author VideoSubtitleAuthor `json:"author"` // 字幕上传者信息
}
type VideoSubtitles struct {
AllowSubmit bool `json:"allow_submit"` // 是否允许提交字幕
List []VideoSubtitle `json:"list"` // 字幕列表
}
type StaffVip struct {
Type int `json:"type"` // 成员会员类型。0:无。1:月会员。2:年会员
Status int `json:"status"` // 会员状态。0:无。1:有
ThemeType int `json:"theme_type"` // 0
}
type Staff struct {
Mid int `json:"mid"` // 成员mid
Title string `json:"title"` // 成员名称
Name string `json:"name"` // 成员昵称
Face string `json:"face"` // 成员头像url
Vip StaffVip `json:"vip"` // 成员大会员状态
Official Official `json:"official"` // 成员认证信息
Follower int `json:"follower"` // 成员粉丝数
LabelStyle int `json:"label_style"`
}
type UserGarb struct {
UrlImageAniCut string `json:"url_image_ani_cut"` // 某url?
}
type Honor struct {
Aid int `json:"aid"` // 当前稿件aid
Type int `json:"type"` // 1:入站必刷收录。2:第?期每周必看。3:全站排行榜最高第?名。4:热门
Desc string `json:"desc"` // 描述
WeeklyRecommendNum int `json:"weekly_recommend_num"`
}
type HonorReply struct {
Honor []Honor `json:"honor"`
}
type ArgueInfo struct {
ArgueLink string `json:"argue_link"` // 作用尚不明确
ArgueMsg string `json:"argue_msg"` // 警告/争议提示信息
ArgueType int `json:"argue_type"` // 作用尚不明确
}
type TopRecommendVideoList struct {
BusinessCard any `json:"business_card"` //无意义
FloorInfo any `json:"floor_info"` //无意义
Item []TopRecommendVideoItem `json:"item"` //推荐列表
Mid int `json:"mid"` //用户mid,未登录为0
PreloadExposePct float64 `json:"preload_expose_pct"` //用于预加载?
PreloadFloorExposePct float64 `json:"preload_floor_expose_pct"` //用于预加载?
SideBarColumn []any `json:"side_bar_column"` //边栏列表? 可参考字段 item 及对应功能文档
UserFeature any `json:"user_feature"` //无意义
}
// 推荐理由
type RcmdReason struct {
ReasonType int `json:"reason_type"` // 原因类型
Content string `json:"content"` // 原因描述(仅当 reason_type 为 3 时存在)
}
type TopRecommendVideoItem struct {
AvFeature any `json:"av_feature"` // 暂无参考意义
BusinessInfo any `json:"business_info"` // 商业推广信息,通常为 null
Bvid string `json:"bvid"` // 视频 bvid
Cid int `json:"cid"` // 视频 cid
DislikeSwitch int `json:"dislike_switch"` // 不感兴趣开关
DislikeSwitchPC int `json:"dislike_switch_pc"` // PC端不感兴趣开关
Duration int `json:"duration"` // 视频时长
EnableVt int `json:"enable_vt"` // 未知作用
Goto string `json:"goto"` // 目标类型 (av, ogv, live)
Id int `json:"id"` // 视频 avid / 直播间 id
IsFollowed int `json:"is_followed"` // 是否已关注
IsStock int `json:"is_stock"` // 未知作用
OgvInfo any `json:"ogv_info"` // 通常为 null
Owner Owner `json:"owner"` // 视频 UP 主信息
Pic string `json:"pic"` // 视频封面
Pic43 string `json:"pic_4_3"` // 4:3 比例封面
Pos int `json:"pos"` // 位置
Pubdate int `json:"pubdate"` // 发布时间(秒级时间戳)
RcmdReason RcmdReason `json:"rcmd_reason"` // 推荐理由
RoomInfo any `json:"room_info"` // 通常为 null
ShowInfo int `json:"show_info"` // 展示信息(1: 普通视频, 0: 直播)
Stat any `json:"stat"` // 视频状态信息
Title string `json:"title"` // 视频标题
TrackId string `json:"track_id"` // 跟踪标识
Uri string `json:"uri"` // 目标页 URI
VtDisplay string `json:"vt_display"` // 未知作用
}
type VideoInfo struct {
Bvid string `json:"bvid"` // 稿件bvid
Aid int `json:"aid"` // 稿件avid
Videos int `json:"videos"` // 稿件分P总数。默认为1
Tid int `json:"tid"` // 分区tid
Tname string `json:"tname"` // 子分区名称
Copyright int `json:"copyright"` // 视频类型。1:原创。2:转载
Pic string `json:"pic"` // 稿件封面图片url
Title string `json:"title"` // 稿件标题
Pubdate int `json:"pubdate"` // 稿件发布时间。秒级时间戳
Ctime int `json:"ctime"` // 用户投稿时间。秒级时间戳
Desc string `json:"desc"` // 视频简介
DescV2 []DescV2 `json:"desc_v2"` // 新版视频简介
State int `json:"state"` // 视频状态。详情见[属性数据文档](attribute_data.md#state字段值(稿件状态))
Duration int `json:"duration"` // 稿件总时长(所有分P)。单位为秒
Forward int `json:"forward"` // 撞车视频跳转avid。仅撞车视频存在此字段
MissionId int `json:"mission_id"` // 稿件参与的活动id
RedirectUrl string `json:"redirect_url"` // 重定向url。仅番剧或影视视频存在此字段。用于番剧&影视的av/bv->ep
Rights VideoRights `json:"rights"` // 视频属性标志
Owner Owner `json:"owner"` // 视频UP主信息
Stat VideoStat `json:"stat"` // 视频状态数
Dynamic string `json:"dynamic"` // 视频同步发布的的动态的文字内容
Cid int `json:"cid"` // 视频1P cid
Dimension Dimension `json:"dimension"` // 视频1P分辨率
Premiere any `json:"premiere"` // null
TeenageMode int `json:"teenage_mode"`
IsChargeableSeason bool `json:"is_chargeable_season"`
IsStory bool `json:"is_story"`
NoCache bool `json:"no_cache"` // 作用尚不明确
Pages []VideoPage `json:"pages"` // 视频分P列表
Subtitle VideoSubtitle `json:"subtitle"` // 视频CC字幕信息
Staff []Staff `json:"staff"` // 合作成员列表。非合作视频无此项
IsSeasonDisplay bool `json:"is_season_display"`
UserGarb UserGarb `json:"user_garb"` // 用户装扮信息
HonorReply HonorReply `json:"honor_reply"`
LikeIcon string `json:"like_icon"`
ArgueInfo ArgueInfo `json:"argue_info"` // 争议/警告信息
}
// GetVideoInfo 获取视频详细信息
func (c *Client) GetVideoInfo(param VideoParam) (*VideoInfo, error) {
const (
method = resty.MethodGet
url = "https://api.bilibili.com/x/web-interface/view"
)
return execute[*VideoInfo](c, method, url, param)
}
// GetVideoDesc 获取视频简介
func (c *Client) GetVideoDesc(param VideoParam) (string, error) {
const (
method = resty.MethodGet
url = "https://api.bilibili.com/x/web-interface/archive/desc"
)
return execute[string](c, method, url, param)
}
type Dimension struct {
Width int `json:"width"` // 当前分P 宽度
Height int `json:"height"` // 当前分P 高度
Rotate int `json:"rotate"` // 是否将宽高对换。0:正常。1:对换
}
type VideoPage struct {
Cid int `json:"cid"` // 当前分P cid
Page int `json:"page"` // 当前分P
From string `json:"from"` // 视频来源。vupload:普通上传(B站)。hunan:芒果TV。qq:腾讯
Part string `json:"part"` // 当前分P标题
Duration int `json:"duration"` // 当前分P持续时间。单位为秒
Vid string `json:"vid"` // 站外视频vid
Weblink string `json:"weblink"` // 站外视频跳转url
Dimension Dimension `json:"dimension"` // 当前分P分辨率。有部分视频无法获取分辨率
FirstFrame string `json:"first_frame"` // 分P封面
}
// GetVideoPageList 获取视频分P列表
func (c *Client) GetVideoPageList(param VideoParam) ([]VideoPage, error) {
const (
method = resty.MethodGet
url = "https://api.bilibili.com/x/player/pagelist"
)
return execute[[]VideoPage](c, method, url, param)
}
type StatusCount struct {
View int `json:"view"` // 0。作用尚不明确
Use int `json:"use"` // 视频添加TAG数
Atten int `json:"atten"` // TAG关注
}
type VideoTag struct {
TagId int `json:"tag_id"` // tag_id
TagName string `json:"tag_name"` // TAG名称
Cover string `json:"cover"` // TAG图片url
HeadCover string `json:"head_cover"` // TAG页面头图url
Content string `json:"content"` // TAG介绍
ShortContent string `json:"short_content"` // TAG简介
Type int `json:"type"` // ???
State int `json:"state"` // 0
Ctime int `json:"ctime"` // 创建时间。时间戳
Count StatusCount `json:"count"` // 状态数
IsAtten int `json:"is_atten"` // 是否关注。0:未关注。1:已关注。需要登录(Cookie) 。未登录为0
Likes int `json:"likes"` // 0。作用尚不明确
Hates int `json:"hates"` // 0。作用尚不明确
Attribute int `json:"attribute"` // 0。作用尚不明确
Liked int `json:"liked"` // 是否已经点赞。0:未点赞。1:已点赞。需要登录(Cookie) 。未登录为0
Hated int `json:"hated"` // 是否已经点踩。0:未点踩。1:已点踩。需要登录(Cookie) 。未登录为0
ExtraAttr int `json:"extra_attr"` // ? ? ?
}
// GetVideoTags 获取视频TAG
func (c *Client) GetVideoTags(param VideoParam) ([]VideoTag, error) {
const (
method = resty.MethodGet
url = "https://api.bilibili.com/x/tag/archive/tags"
)
return execute[[]VideoTag](c, method, url, param)
}
type VideoTagParam struct {
Aid int `json:"aid"` // 稿件avid
TagId int `json:"tag_id"` // tag_id
}
// LikeVideoTag 点赞视频TAG,重复请求为取消
func (c *Client) LikeVideoTag(param VideoTagParam) error {
const (
method = resty.MethodPost
url = "https://api.bilibili.com/x/tag/archive/like2"
)
_, err := execute[any](c, method, url, param, fillCsrf(c))
return err
}
// HateVideoTag 点踩视频TAG,重复访问为取消
func (c *Client) HateVideoTag(param VideoTagParam) error {
const (
method = resty.MethodPost
url = "https://api.bilibili.com/x/tag/archive/hate2"
)
_, err := execute[any](c, method, url, param, fillCsrf(c))
return err
}
type LikeVideoParam struct {
Aid int `json:"aid,omitempty" request:"query,omitempty"` // 稿件 avid。avid 与 bvid 任选一个
Bvid string `json:"bvid,omitempty" request:"query,omitempty"` // 稿件 bvid。avid 与 bvid 任选一个
Like int `json:"like"` // 操作方式。1:点赞。2:取消赞
}
// LikeVideo 点赞视频
func (c *Client) LikeVideo(param LikeVideoParam) error {
const (
method = resty.MethodPost
url = "https://api.bilibili.com/x/web-interface/archive/like"
)
_, err := execute[any](c, method, url, param, fillCsrf(c))
return err
}
type CoinVideoParam struct {
Aid int `json:"aid,omitempty" request:"query,omitempty"` // 稿件 avid。avid 与 bvid 任选一个
Bvid string `json:"bvid,omitempty" request:"query,omitempty"` // 稿件 bvid。avid 与 bvid 任选一个
Multiply int `json:"multiply"` // 投币数量。上限为2
SelectLike int `json:"select_like,omitempty" request:"query,omitempty"` // 是否附加点赞。0:不点赞。1:同时点赞。默认为0
}
type CoinVideoResult struct {
Like bool `json:"like"` // 是否点赞成功。true:成功。false:失败。已赞过则附加点赞失败
}
// CoinVideo 投币视频
func (c *Client) CoinVideo(param CoinVideoParam) (*CoinVideoResult, error) {
const (
method = resty.MethodPost
url = "https://api.bilibili.com/x/web-interface/coin/add"
)
return execute[*CoinVideoResult](c, method, url, param, fillCsrf(c))
}
type FavourVideoParam struct {
Rid int `json:"rid"` // 稿件 avid
Type int `json:"type"` // 必须为2
AddMediaIds []int `json:"add_media_ids,omitempty" request:"query,omitempty"` // 需要加入的收藏夹 mlid。同时添加多个,用,(%2C)分隔
DelMediaIds []int `json:"del_media_ids,omitempty" request:"query,omitempty"` // 需要取消的收藏夹 mlid。同时取消多个,用,(%2C)分隔
}
type FavourVideoResult struct {
Prompt bool `json:"prompt"` // 是否为未关注用户收藏。false:否。true:是
}
// FavourVideo 收藏视频
func (c *Client) FavourVideo(param FavourVideoParam) (*FavourVideoResult, error) {
const (
method = resty.MethodPost
url = "https://api.bilibili.com/medialist/gateway/coll/resource/deal"
)
return execute[*FavourVideoResult](c, method, url, param, fillCsrf(c))
}
type LikeCoinFavourResult struct {
Like bool `json:"like"` // 是否点赞成功。true:成功。false:失败
Coin bool `json:"coin"` // 是否投币成功。true:成功。false:失败
Fav bool `json:"fav"` // 是否收藏成功。true:成功。false:失败
Multiply int `json:"multiply"` // 投币枚数。默认为2
}
// LikeCoinFavourVideo 一键三连视频
func (c *Client) LikeCoinFavourVideo(param VideoParam) (*LikeCoinFavourResult, error) {
const (
method = resty.MethodPost
url = "https://api.bilibili.com/x/web-interface/archive/like/triple"
)
return execute[*LikeCoinFavourResult](c, method, url, param, fillCsrf(c))
}
type VideoCidParam struct {
Aid int `json:"aid,omitempty" request:"query,omitempty"` // 稿件avid。avid与bvid任选一个
Bvid string `json:"bvid,omitempty" request:"query,omitempty"` // 稿件bvid。avid与bvid任选一个
Cid int `json:"cid"` // 视频cid。用于选择目标分P
}
type ShowSwitch struct {
Total bool `json:"total"` // 展示所有终端总计人数
Count bool `json:"count"` // 展示web端实时在线人数
}
type VideoOnlineInfo struct {
Total string `json:"total"` // 所有终端总计人数。例如10万+
Count string `json:"count"` // web端实时在线人数
ShowSwitch ShowSwitch `json:"show_switch"` // 数据显示控制
}
// GetVideoOnlineInfo 获取视频在线人数
func (c *Client) GetVideoOnlineInfo(param VideoCidParam) (*VideoOnlineInfo, error) {
const (
method = resty.MethodGet
url = "https://api.bilibili.com/x/player/online/total"
)
return execute[*VideoOnlineInfo](c, method, url, param)
}
type VideoStatusNumber struct {
Aid int `json:"aid"` // 稿件avid
Bvid string `json:"bvid"` // 稿件bvid
View any `json:"view"` // 正常:播放次数(num)。屏蔽:"--"(str)
Danmaku int `json:"danmaku"` // 弹幕条数
Reply int `json:"reply"` // 评论条数
Favorite int `json:"favorite"` // 收藏人数
Coin int `json:"coin"` // 投币枚数
Share int `json:"share"` // 分享次数
NowRank int `json:"now_rank"` // 0。作用尚不明确
HisRank int `json:"his_rank"` // 历史最高排行
Like int `json:"like"` // 获赞次数
Dislike int `json:"dislike"` // 0。作用尚不明确
NoReprint int `json:"no_reprint"` // 禁止转载标志。0:无。1:禁止
Copyright int `json:"copyright"` // 版权标志。1:自制。2:转载
ArgueMsg string `json:"argue_msg"` // 警告信息。默认为空
Evaluation string `json:"evaluation"` // 视频评分。默认为空
}
// GetVideoStatusNumber 获取视频状态数视频
func (c *Client) GetVideoStatusNumber(param VideoParam) (*VideoStatusNumber, error) {
const (
method = resty.MethodGet
url = "https://api.bilibili.com/x/web-interface/archive/stat"
)
return execute[*VideoStatusNumber](c, method, url, param)
}
type GetTopRecommendVideoParam struct {
FreshType int `json:"fresh_type,omitempty" request:"query,omitempty"` // 相关性。默认为3 。 值越大推荐内容越相关
Version int `json:"version,omitempty" request:"query,omitempty"` // web端新旧版本:0为旧版本1为新版本。默认为 0 。1,0分别为新旧web端
Ps int `json:"ps,omitempty" request:"query,omitempty"` // pagesize 单页返回的记录条数默认为10或8。默认为10 。当version为1时默认为8
FreshIdx int `json:"fresh_idx,omitempty" request:"query,omitempty"` // 翻页相关。默认为1 。 与翻页相关
FreshIdx1H int `json:"fresh_idx_1h,omitempty" request:"query,omitempty"` // 翻页相关。默认为1 。 与翻页相关
}
// GetTopRecommendVideo 获取首页视频推荐列表
func (c *Client) GetTopRecommendVideo(param GetTopRecommendVideoParam) (*TopRecommendVideoList, error) {
const (
method = resty.MethodGet
url = "https://api.bilibili.com/x/web-interface/wbi/index/top/feed/rcmd"
)
return execute[*TopRecommendVideoList](c, method, url, param)
}
type GetVideoCollectionInfoParam struct {
Mid int `json:"mid"` // UP 主 ID
SeasonId int `json:"season_id"` // 视频合集 ID
SortReverse bool `json:"sort_reverse,omitempty" request:"query,omitempty"` // 未知
PageNum int `json:"page_num,omitempty" request:"query,omitempty"` // 页码索引
PageSize int `json:"page_size,omitempty" request:"query,omitempty"` // 单页内容数量
}
type CollectionVideoStat struct {
View int `json:"view"` // 稿件播放量
Vt int `json:"vt"` // 0
}
type CollectionVideo struct {
Aid int `json:"aid"` // 稿件avid
Bvid string `json:"bvid"` // 稿件bvid
Ctime int `json:"ctime"` // 创建时间。Unix 时间戳
Duration int `json:"duration"` // 视频时长。单位为秒
EnableVt any `json:"enable_vt"` // int or bool
InteractiveVideo bool `json:"interactive_video"` // false
Pic string `json:"pic"` // 封面 URL
PlaybackPosition int `json:"playback_position"` // 会随着播放时间增长,播放完成后为 -1 。单位未知
Pubdate int `json:"pubdate"` // 发布日期。Unix 时间戳
Stat CollectionVideoStat `json:"stat"` // 稿件信息
State int `json:"state"` // 0
Title string `json:"title"` // 稿件标题
UgcPay int `json:"ugc_pay"` // 0
VtDisplay string `json:"vt_display"`
}
type CollectionMeta struct {
Category int `json:"category"` // 0
Covr string `json:"covr"` // 合集封面 URL
Description string `json:"description"` // 合集描述
Mid int `json:"mid"` // UP 主 ID
Name string `json:"name"` // 合集标题
Ptime int `json:"ptime"` // 发布时间。Unix 时间戳
SeasonId int `json:"season_id"` // 合集 ID
Total int `json:"total"` // 合集内视频数量
}
type CollectionPage struct {
PageNum int `json:"page_num"` // 分页页码
PageSize int `json:"page_size"` // 单页个数
Total int `json:"total"` // 合集内视频数量
}
type VideoCollectionInfo struct {
Aids []int `json:"aids"` // 稿件avid。对应下方数组中内容 aid
Archives []CollectionVideo `json:"archives"` // 合集中的视频
Meta CollectionMeta `json:"meta,omitempty"` // 合集元数据
Page CollectionPage `json:"page"` // 分页信息
}
// GetVideoCollectionInfo 获取视频合集信息 https://github.com/SocialSisterYi/bilibili-API-collect/blob/master/docs/video/collection.md#%E8%8E%B7%E5%8F%96%E8%A7%86%E9%A2%91%E5%90%88%E9%9B%86%E4%BF%A1%E6%81%AF
func (c *Client) GetVideoCollectionInfo(param GetVideoCollectionInfoParam) (*VideoCollectionInfo, error) {
const (
method = resty.MethodGet
url = "https://api.bilibili.com/x/polymer/web-space/seasons_archives_list"
)
return execute[*VideoCollectionInfo](c, method, url, param)
}
type VideoCollectionByKeywordsInfo struct {
Archives []CollectionVideo `json:"archives"` // 视频列表
Page CollectionPage `json:"page"` // 页码信息
}
type GetVideoByKeywordsParam struct {
Mid int `json:"mid"` // 用户 mid
Keywords string `json:"keywords"` // 关键词。可为空, 即获取所有视频
Ps int `json:"ps,omitempty" request:"query,omitempty"` // 每页视频数。默认为 0, 留空为 20
Pn int `json:"pn,omitempty" request:"query,omitempty"` // 页码。留空为 1
Orderby string `json:"orderby,omitempty" request:"query,omitempty"` // 排序方式。最新发布: pubdate(默认)。最多播放: views。senddate: 最新发布
SeriesId int `json:"series_id,omitempty" request:"query,omitempty"` // 系列 ID。用于过滤结果, 即若某一视频包含在系列内则不返回该视频
}
// GetVideoByKeywords根据关键词查找视频
// https://github.com/SocialSisterYi/bilibili-API-collect/blob/master/docs/video/collection.md#%E6%A0%B9%E6%8D%AE%E5%85%B3%E9%94%AE%E8%AF%8D%E6%9F%A5%E6%89%BE%E8%A7%86%E9%A2%91
func (c *Client) GetVideoByKeywords(param GetVideoByKeywordsParam) (*VideoCollectionByKeywordsInfo, error) {
const (
method = resty.MethodGet
url = "https://api.bilibili.com/x/series/recArchivesByKeywords"
)
return execute[*VideoCollectionByKeywordsInfo](c, method, url, param)
}
type GetVideoSeriesInfoParam struct {
Mid int `json:"mid"` // UP 主 ID
SeriesId int `json:"series_id"` // 视频合集 ID
Sort string `json:"sort,omitempty" request:"query,omitempty"` // 未知
Pn int `json:"pn,omitempty" request:"query,omitempty"` // 页码索引
Ps int `json:"ps,omitempty" request:"query,omitempty"` // 单页内容数量
CurrentMid int `json:"current_mid,omitempty" request:"query,omitempty"` // 单页内容数量
}
// GetVideoSeriesInfo 获取视频列表信息(在个人空间里创建的叫做视频列表,在创作中心里创建的叫合集,注意区分)
func (c *Client) GetVideoSeriesInfo(param GetVideoSeriesInfoParam) (*VideoCollectionInfo, error) {
const (
method = resty.MethodGet
url = "https://api.bilibili.com/x/series/archives"
)
return execute[*VideoCollectionInfo](c, method, url, param)
}
type GetVideoStreamParam struct {
Avid int `json:"avid,omitempty" request:"query,omitempty"` // 稿件 avid。avid 与 bvid 任选一个
Bvid string `json:"bvid,omitempty" request:"query,omitempty"` // 稿件 bvid。avid 与 bvid 任选一个
Cid int `json:"cid"` // 视频 cid
Qn int `json:"qn,omitempty" request:"query,omitempty"` // 视频清晰度选择。未登录默认 32(480P),登录后默认 64(720P)。含义见 [上表](#qn视频清晰度标识)。**DASH 格式时无效**
Fnval int `json:"fnval,omitempty" request:"query,omitempty"` // 视频流格式标识。默认值为1(MP4 格式)。含义见 [ 上表](#fnval视频流格式标识)
Fnver int `json:"fnver,omitempty" request:"query,omitempty"` // 0
Fourk int `json:"fourk,omitempty" request:"query,omitempty"` // 是否允许 4K 视频。画质最高 1080P:0(默认)。画 质最高 4K:1
Session string `json:"session,omitempty" request:"query,omitempty"` // 从视频播放页的 HTML 中获取
Otype string `json:"otype,omitempty" request:"query,omitempty"` // 固定为json
Type string `json:"type,omitempty" request:"query,omitempty"` // 目前为空
Platform string `json:"platform,omitempty" request:"query,omitempty"` // pc:web播放(默认值,视频流存在 referer鉴权)。html5:移动端 HTML5 播放(仅支持 MP4 格式,无 referer 鉴权可以直接使用video标签播放)
HighQuality int `json:"high_quality,omitempty" request:"query,omitempty"` // 是否高画质。platform=html5时,此值 为1可使画质为1080p
}
type SupportFormat struct {
Quality int `json:"quality"` // 视频清晰度代码。含义见 [上表](#qn视频清晰度标识)
Format string `json:"format"` // 视频格式
NewDescription string `json:"new_description"` // 格式描述
DisplayDesc string `json:"display_desc"` // 格式描述
Superscript string `json:"superscript"` // (?)
Codecs []string `json:"codecs"` // 可用编码格式列表 例:av01.0.13M.08.0.110.01.01.01.0 使用AV1编码, avc1.640034 使用AVC编码, hev1.1.6.L153.90 使用HEVC编码
}
type Durl struct {
Order int `json:"order"` // 视频分段序号。某些视频会分为多个片段(从1顺序增长)
Length int `json:"length"` // 视频长度。单位为毫秒
Size int `json:"size"` // 视频大小。单位为 Byte
Ahead string `json:"ahead"` // (?)
Vhead string `json:"vhead"` // (?)
Url string `json:"url"` // 默认流 URL。**注意 unicode 转义符**。有效时间为120min
BackupUrl []string `json:"backup_url"` // 备用视频流 **注意 unicode 转义符**。有效时间为120min
}
type Dash struct {
Duration int `json:"duration"` // 视频长度。秒值
Minbuffertime int `json:"minBufferTime"` // 1.5?
MinBufferTime int `json:"min_buffer_time"` // 1.5?
Video []AudioOrVideo `json:"video"` // 视频流信息 同一清晰度可拥有 H.264 / H.265 / AV1 多种码流<br />**HDR 仅支持 H.265** |
Audio []AudioOrVideo `json:"audio"` // 伴音流信息。当视频没有音轨时,此项为 null
Dolby Dolby `json:"dolby"` // 杜比全景声伴音信息
Flac Flac `json:"flac"` // 无损音轨伴音信息。当视频没有无损音轨时,此项为 null
}
type Dolby struct {
Type int `json:"type"` // 杜比音效类型。1:普通杜比音效。2:全景杜比音效
Audio []AudioOrVideo `json:"audio"` // 杜比伴音流列表
}
type Flac struct {
Display bool `json:"display"` // 是否在播放器显示切换Hi-Res无损音轨按钮
Audio AudioOrVideo `json:"audio"` // 音频流信息。同上文 DASH 流中video及audio数组中的对象
}
type AudioOrVideo struct {
Id int `json:"id"` // 音视频清晰度代码。参考上表。[qn视频清晰度标识](#qn视频清晰度标识)。[视频伴音音质代码](#视 频伴音音质代码)
Baseurl string `json:"baseUrl"` // 默认流 URL。**注意 unicode 转义符**。有效时间为 120min
BaseUrl string `json:"base_url"` // **同上**
Backupurl []string `json:"backupUrl"` // 备用流 URL
BackupUrl []string `json:"backup_url"` // **同上**
Bandwidth int `json:"bandwidth"` // 所需最低带宽。单位为 Byte
Mimetype string `json:"mimeType"` // 格式 mimetype 类型
MimeType string `json:"mime_type"` // **同上**
Codecs string `json:"codecs"` // 编码/音频类型。eg:avc1.640032
Width int `json:"width"` // 视频宽度。单位为像素。**仅视频流存在该字段**
Height int `json:"height"` // 视频高度。单位为像素。**仅视频流存在该字段**
Framerate string `json:"frameRate"` // 视频帧率。**仅视频流存在该字段**
FrameRate string `json:"frame_rate"` // **同上**
Sar string `json:"sar"` // Sample Aspect Ratio(单个像素的宽高比)。音频流该值恒为空
Startwithsap int `json:"startWithSap"` // Stream Access Point(流媒体访问位点)。音频流该值恒为空
StartWithSap int `json:"start_with_sap"` // **同上**
Segmentbase SegmentBase `json:"SegmentBase"` // 见下表。url 对应 m4s 文件中,头部的位置。音频流该值恒为空
SegmentBase SegmentBase `json:"segment_base"` // **同上**
Codecid int `json:"codecid"` // 码流编码标识代码。含义见 [上表](#视频编码代码)。音频流该值恒为0
}
type SegmentBase struct {
Initialization string `json:"initialization"` // ${init_first}-${init_last}。eg:0-821。ftyp (file type) box 加 上 moov box 在 m4s 文件中的范围(单位为 bytes)。如 0-821 表示开头 820 个字节
IndexRange string `json:"index_range"` // ${sidx_first}-${sidx_last}。eg:822-1309。sidx (segment index) box 在 m4s 文件中的范围(单位为 bytes)。sidx 的核心是一个数组,记录了各关键帧的时间戳及其在文件中的位置,。其作用是索引 (拖进 度条)
}
type GetVideoStreamResult struct {
From string `json:"from"` // local?
Result string `json:"result"` // suee?
Message string `json:"message"` // 空?
Quality int `json:"quality"` // 清晰度标识。含义见 [上表](#qn视频清晰度标识)
Format string `json:"format"` // 视频格式。mp4/flv
Timelength int `json:"timelength"` // 视频长度。单位为毫秒。不同分辨率 / 格式可能有略微差异
AcceptFormat string `json:"accept_format"` // 支持的全部格式。每项用,分隔
AcceptDescription []string `json:"accept_description"` // 支持的清晰度列表(文字说明)
AcceptQuality []int `json:"accept_quality"` // 支持的清晰度列表(代码)。含义见 [上表](#qn视频清晰度标识)
VideoCodecid int `json:"video_codecid"` // 默认选择视频流的编码id。含义见 [上表](#视频编码代码)
SeekParam string `json:"seek_param"` // start?
SeekType string `json:"seek_type"` // offset(DASH / FLV)?。 second(MP4)?
Durl []Durl `json:"durl"` // 视频分段流信息。**注:仅 FLV / MP4 格式存在此字段**
Dash Dash `json:"dash"` // DASH 流信息。**注:仅 DASH 格式存在此字段**
SupportFormats []SupportFormat `json:"support_formats"` // 支持格式的详细信息
HighFormat *string `json:"high_format"` // (?)null
LastPlayTime int `json:"last_play_time"` // 上次播放进度。毫秒值
LastPlayCid int `json:"last_play_cid"` // 上次播放分P的 cid
}
// GetVideoStream 获取视频流地址_web端
func (c *Client) GetVideoStream(param GetVideoStreamParam) (*GetVideoStreamResult, error) {
const (
method = resty.MethodGet
url = "https://api.bilibili.com/x/player/wbi/playurl"
)
return execute[*GetVideoStreamResult](c, method, url, param, fillWbiHandler(c.wbi, c.GetCookies()))
}