Skip to content

Latest commit

 

History

History
1277 lines (1080 loc) · 48.7 KB

README.md

File metadata and controls

1277 lines (1080 loc) · 48.7 KB

bilibili 直播弹幕机

Go Report Card codecov FOSSA Status 少年Pi的直播回放 Static Badge

commits GitHub all releases

see more

当前支持显示/功能

当前支持显示

以下内容可能过时,点击查看当前支持显示

  • 人气
  • 天选之人开始
  • 天选之人获奖
  • 直播间关注提示
  • 大航海购买
  • 节奏风暴
  • 大航海进入
  • 弹幕
  • 房间信息分区改变
  • 禁言
  • 礼物
  • 封禁
  • 下播
  • 开播
  • SC
  • 排行榜

当前支持功能

以下内容可能过时,点击查看功能配置

  • 直播流、弹幕回放服务
  • 每天自动发送将要过期的银瓜子礼物(默认发送3天内过期的)
  • 保持当前已点亮的粉丝牌总是点亮
  • 银瓜子自动兑换硬币
  • 发进房弹幕(可选有无粉丝牌(可选每日首次发送后不发))
  • 每日签到
  • 自定义私信
  • 自动切换粉丝牌
  • 扫码登录(qrcode in webServer and cmd)
  • 自定义语音提醒
  • GTK弹幕窗
  • GTK信息窗
  • 营收统计
  • 舰长数统计
  • 直播流、弹幕保存
  • ASS字幕生成
  • 节奏提示
  • 反射型弹幕机
  • 自动型弹幕机
  • 相同弹幕合并
  • 重复度高弹幕屏蔽
  • 弹幕开头字符相同缩减

其他特性

  • 使用http,https,socks5代理
  • cookie加密
  • 弹幕自动重连(30s无响应)
  • 直播流开播自动下载、断流再保存、故障转移、移除历史
  • 指定弹幕、标题切换重启录制
  • 命令行支持房间切换、弹幕发送、表情发送、启停录制、重载弹幕、查看历史记录、查看正在直播的主播、退出房间
  • GTK信息窗支持房间切换、弹幕格式化发送、时长统计
  • GTK弹幕窗支持自定义人/事件消息停留

说明

本项目使用github action自动构建,构建过程详见yml

关于离线构建,详见章节运行及其注意事项

环境变量覆盖配置项

添加配置项从环境变量覆盖(>v0.14.26)。将在配置文件都加载后,用环境变量覆盖配置项。

  • key为配置键名(例如:Web服务地址),为空时将忽略。
  • type为类型,可选string,bool,float64。为空(默认)为string。当为int等类型时,也填为float64
  • env为env名(例如:addr),为空时将忽略。

例子:

config_K_v.json
{
  "Web服务地址":"0.0.0.0:20000",
  "从环境变量覆盖": [
      {
          "key": "Web服务地址",
          "env": "addr"
      }
  ]
}

在当前命令行配置临时环境变量(此处为linux例子),并启动:

export addr=0.0.0.0:22000
./main -ckv config_K_v.json

启动后,程序将监听22000端口而非20000端口

注意:

  • 当要配置的键为数组时,使用a.[n]表示第n个,n为非负整数。当数组长度小于n时,将忽略。
  • 当要配置的键为map时,使用a.b表示a下的b键。当b不存在时,将忽略。
  • 注意核对配置的类型是正确的,否则可能会导致配置无效。

cookie自定义位置

配置文件添加配置项cookie路径(>v0.14.26),默认为./cookie.txt

获取视频切片

当请求http://{Web服务地址}{直播Web服务路径}stream?ref={录播文件夹名}&st={起始时间}&dur={片段时长}时,将返回从录播文件的切片视频(>v0.14.21)

切片将从大于{起始时间}的关键帧开始,{片段时长}之后的关键帧结束,故大多数情况不能获得精确时间的切片视频

切片不会进行渲染,仅参考文件中的时间戳进行切片

其中dur为空时,将返回全部时长。st参数可以为空或不传,此时从录播文件起始点开始。

{起始时间}{片段时长}格式使用time.ParseDuration进行转换。例:1m为1分钟、1h2m为1小时2分。

注意:当配置直播流回放连接检查启用时(默认不启用),你需要配置直播流回放连接检查忽略key(>v0.14.21)以避免检查,url加上参数&key={配置的key}

例子:

{
  "Web服务地址":"0.0.0.0:20000",
  "直播Web服务路径":"/web/",
  "直播流回放连接检查": 10,
  "直播流回放连接检查忽略key-help": "字符串数组,默认空,空字符串将忽略,当不为空时,将不会定时检查指定key值的请求",
  "直播流回放连接检查忽略key": ["cut"],
}
curl -v "http://192.168.31.230:20000/web/stream?ref=2024_11_04-01_29_47-47867-250-edd590-JdB&key=cut&dur=1m"
*   Trying 192.168.31.230:20000...
* Connected to 192.168.31.230 (192.168.31.230) port 20000
> GET /web/stream?ref=2024_11_04-01_29_47-47867-250-edd590-JdB&key=cut&dur=1m HTTP/1.1
> Host: 192.168.31.230:20000
> User-Agent: curl/8.9.1
> Accept: */*
> 
* Request completely sent off
< HTTP/1.1 200 OK
< Access-Control-Allow-Credentials: true
< Access-Control-Allow-Headers: *
< Access-Control-Allow-Methods: POST, GET, OPTIONS
< Access-Control-Allow-Origin: *
< Connection: keep-alive
< Content-Disposition: inline; filename="2024_11_04-01_29_47-47867-250-edd590-JdB.1731342591.mp4"
< Content-Transfer-Encoding: binary
< Content-Type: video/mp4
< Date: Mon, 11 Nov 2024 16:29:51 GMT
< Transfer-Encoding: chunked
< 
Warning: Binary output can mess up your terminal. Use "--output -" to tell curl to output it to your terminal anyway, or consider "--output <FILE>" to save to a file.
* client returned ERROR on write of 1004 bytes
* Failed reading the chunked-encoded stream
* closing connection #0

添加快速索引文件生成,将在录制完成后,自定义回调命令执行前,读取视频文件,并将关键帧的时间戳和对应的下标值记录在.fastSeed文件,用于加快后续切片请求响应。(>v0.14.28)

可以设置禁用快速索引生成(默认为false)为true来禁用这个动作(>v0.14.28)

相较于之前的请求时进行查找,效率提升如下

mp4:
旧:
// 10s-30s 110.962896ms
// 10m-10m20s 1.955749395s
// 30m-30m20s 5.791614855s

新:
// 10s-30s 90.05983ms
// 10m-10m20s 88.769475ms
// 30m-30m20s 104.381225ms

flv:
旧:
// 10s-30s 184.852917ms
// 10m-10m20s 3.278605875s

新:
// 10s-30s 215.815423ms
// 10m-10m20s 174.918508ms

.fastSeed文件格式:

+--------+-----------+
|Ms int64|index int64|
+--------+-----------+
|8bytes  |8bytes     |
+--------+-----------+

Ms:毫秒时间戳
index:相对于文件起始位置的byte下标值

文件由n组上述16bytes组成,时间戳升序

Web自定义响应头

配置文件中添加配置项Web自定义响应头(>v0.14.19)。默认为空,当不为空时,将在所有响应中添加指定头。 例子:

{
  "Web服务地址":"0.0.0.0:11000",
  "直播Web服务路径":"/web/",
  "Web自定义响应头": {
    "Access-Control-Allow-Origin":"*"
  }
}
curl -i http://{主机名}:11000/web/emots/4e9621239e8a349b1ad198af90458e06.png
HTTP/1.1 200 OK
Access-Control-Allow-Origin: *
Cache-Control: private
Etag: 2024-09-24T01:53:03+08:00
Last-Modified: Tue, 24 Sep 2024 01:53:03 CST
Date: Sat, 05 Oct 2024 14:02:18 GMT
Content-Type: image/png
Transfer-Encoding: chunked

ip路径

配置文件中添加配置项ip路径(>v0.14.16)。不为空时,将会在指定路径显示出当前主机ip,从而可以使用主机名来获取主机ipv6,以便于ddns-go等通过命令获取ipv6。默认空。

例子:

{
  "Web服务地址":"0.0.0.0:11000",
  "ip路径":"/ip/"
}
curl -s http://{主机名}:11000/ip/
192.168.31.230
172.16.0.1
172.17.144.244
172.17.0.1
172.17.0.10
172.17.121.35
2409:xxxx:xxxx:xxxx:xxxx:xxxx:xxxx:8ce5
fd55:d4a5:a9e7:0:aab8:e0ff:fe03:8ce5
fe80::aab8:e0ff:fe03:8ce5
fe80::d8dc:35ff:fe21:d71b
fe80::785f:8dff:fe34:9443
fe80::440a:9fff:fe6d:5da6
fe80::60ff:4cff:fee7:c226
fe80::4c24:e3ff:fe65:b955
fe80::8099:ecff:fefa:f36c
fe80::68fe:63ff:fe74:6e35
curl -s http://{主机名}:11000/ip/ | awk '/240:?/'

2409:xxxx:xxxx:xxxx:xxxx:xxxx:xxxx:8ce5

保持粉丝牌点亮

配置文件中添加配置项保持牌子亮着_指定时间(>v0.14.11)。将会在指定时间启动保持。默认00:00:00

保持期间,可能会频繁发送弹幕(间隔5秒/条),此时可能会影响其他使用。

直播回放显示表情

配置文件中添加配置项弹幕表情(>v0.14.9)。默认为true,当为true时,将会保存弹幕中的表情png到emots目录下,并在回放时显示表情。

为了能顺利保存,会将某些字符进行转换(<=v0.14.18),如:[dog?]=>[dog?].png。

但之后(>v0.14.18)存储通过md5作为文件名,如:[dog?]=>md5("[dog?]")+".png"=>0c8427bdb9854d85e9cfe59c45d70583.png

从而避免对原表情产生修改,且避免不同环境对文件名支持的差异。

注意:需要配套更新demo/html/artPlayer下的资源

保存的表情可以通过http://{Web服务地址}{直播Web服务路径}emots/{md5}.png获取。

直播流停用服务器

配置文件中添加配置项直播流停用服务器(>v0.14.3)。默认为空,编写正则字符串,当获取到的服务器链接与字符串匹配时,将会停用。

{
  "直播流不使用mcdn": false,
  "直播流停用服务器": [
      "\\.mcdn\\."
  ],
}

fmp4相关优化

配置文件中添加配置项fmp4获取更多服务器(>v0.14.2)。当直播流类型为fmp4,为true时,会在可用切片服务器仅剩1个时,获取更多服务器,但总数超过5个后,将不会再获取。默认为true

配置文件中添加配置项fmp4跳过解码出错的帧(>v0.14.2)。当直播流类型为fmp4,为true时,会在跳过解码错误的帧(常见于音视频时间戳差过大),这将导致错误的部分缺失,时间轴发生跳越,但能获得一个完整记录。默认为false

{
  "fmp4获取更多服务器": true,
  "fmp4跳过解码出错的帧-help": "fmp4跳过解码出错的帧、但可能导致关键帧时间上的跳越",
  "fmp4跳过解码出错的帧": false,
}

自定义命令行登陆二维码块

配置文件中添加配置项登陆二维码-白登陆二维码-黑

默认:

{
  "登陆二维码-白":"OO",
  "登陆二维码-黑":"  "
}

在某些平台,命令行可能被设置为等宽字符,此时默认配置会导致二维码显示比例错误,可以调整到:

{
  "登陆二维码-白":"OOO",
  "登陆二维码-黑":"   "
}

或其他字符,或者使用浏览器打开扫码登录路径配置的路径进行扫码登陆

指定用户弹幕切片

配置文件中添加配置项指定弹幕重启录制

  • uid:进入个人空间,可以在地址栏看到你的uid
  • danmu:正则匹配表达式
  • 多个相同uid的规则存在时,则会覆盖,取最后一个规则,一个用户最多一条规则
  • uiddanmu为""时,将忽略这个规则

当有匹配分组时,第一个分组将作为新标题,例:t(.*),接收到t暗潮t炒时,则会将暗潮t炒作为标题。

匹配分组时,将以原标题切片,例:t,接收到t暗潮t炒时,将会切片。

{
  "指定弹幕重启录制":[
      {
          "uid":"29183321",
          "danmu":"t(.*)"
      }
  ]
}

指定房间录制区间

配置文件中添加配置项指定房间录制区间

指定roomid的房间在指定时间段内将会开启录制。注意:注意先配置主机时区

  • start时检查是否在直播,是则开始录制,如已在录制则切片。
  • end时如已在录制则停止录制。
  • 存在同时有startendfromTo,则开播时,若在startend之间,则录制。不存在同时有startendfromTo,则一开播就录制。
  • 5s内只能触发一个fromTo,所以同房间各startend之间间隔不要少于5s。
{
  "指定房间录制区间":[
      {
          "roomid":0,
          "fromTo":[
              {
                  "start": "12:01:00",
                  "end": "12:03:00"
              },
              {
                  "start": "12:02:00"
              }
          ]
      }
  ]
}

保存日志至DB

配置文件中添加配置项保存日志至db大部分 可以参考保存弹幕至db ,但有些许不同

insert语句中不能使用{...}的占位符,默认占位符1为Prefix,2为Base,3为具体内容。见下面的mysql实例。

使用{...}占位符,而非?。可选字段见下方示例(>v0.14.28)

{
  "保存日志至db":{
    "dbname": "postgres",
    "url":"postgres://postgres:qydysky@192.168.31.103:5432/postgres?sslmode=disable",
    "字段help":"{Date} time.Now().Format(time.DateTime), {Unix} time.Now().Unix(), {Prefix} string,{Base} string,{Msgs} string",
    "create":"create table log2 (created varchar(20), createdunix varchar(20), Prefix varchar(10), Base varchar(30), Msgs varchar(500))",
    "insert":"insert into log2 (created, createdunix, Prefix, Base, Msgs) values ({Date},{Unix},{Prefix},{Base},{Msgs})"
  }
}

log:

"created" "createdunix" "prefix" "base" "msgs"
"2025-01-15 01:42:25" "1736876545" T: 功能 更少弹幕 每秒弹幕数: 1
"2025-01-15 01:42:25" "1736876545" I: 功能 直播Web服务 启动于 http://0.0.0.0:20000/web/
"2025-01-15 01:42:25" "1736876545" I: bilidanmu 当前PID: 485073
"2025-01-15 01:42:25" "1736876545" I: bilidanmu version: 20250105171800
"2025-01-15 01:42:25" "1736876545" I: bilidanmu "3s内2次ctrl+c退出"
"2025-01-15 01:42:26" "1736876546" T: api IsConnected 已连接
"2025-01-15 01:42:26" "1736876546" T: api Get Get Cookie
"2025-01-15 01:42:26" "1736876546" I: api 获取Cookie 已登录
"2025-01-15 01:42:27" "1736876547" T: api Get Get LIVE_BUVID
"2025-01-15 01:42:27" "1736876547" I: api LIVE_BUVID 获取到LIVE_BUVID,保存cookie
"2025-01-15 01:42:27" "1736876547" T: api Get Get Uid
"2025-01-15 01:42:27" "1736876547" T: 命令行操作 回车查看帮助
"2025-01-15 01:42:28" "1736876548" T: api 银瓜子=>硬币 现在有银瓜子 518 个
"2025-01-15 01:42:28" "1736876548" I: api 银瓜子=>硬币 当前银瓜子数量不足
"2025-01-15 01:42:28" "1736876548" I: 功能 保持亮牌 将在 00:00:00 启动
"2025-01-15 01:42:28" "1736876548" T: 指定弹幕重启录制 加载 1 条规则
"2025-01-15 01:42:51" "1736876571" I: bilidanmu 等待1m0s协程结束
"2025-01-15 01:42:51" "1736876571" I: bilidanmu 结束

保存弹幕至DB

配置文件中添加配置项保存弹幕至db。参考以下实例:

postgreSql:

{
  "保存弹幕至db": {
      "dbname": "postgres",
      "url":"postgres://postgres:qydysky@192.168.31.103:5432/postgres?sslmode=disable",
      "字段help":"time.Now().Format(time.DateTime), time.Now().Unix(), item.msg, item.color, item.auth, item.uid, item.roomid",
      "create":"create table danmu (created varchar(20), createdunix varchar(20), msg varchar(100), color varchar(20), auth varchar(50), uid varchar(30), roomid varchar(30))",
      "insert":"insert into danmu (created, createdunix, msg, color, auth, uid, roomid) values ({Date},{Unix},{Msg},{Color},{Auth},{Uid},{Roomid})"
  }
}

mysql:

{
  "保存弹幕至db": {
      "dbname": "mysql",
      "url":"root:root@(192.168.31.103:10836)/test",
      "字段help":"time.Now().Format(time.DateTime), time.Now().Unix(), item.msg, item.color, item.auth, item.uid, item.roomid",
      "create":"create table danmu (created varchar(20), createdunix varchar(20), msg varchar(100), color varchar(20), auth varchar(50), uid varchar(30), roomid varchar(30))",
      "insert":"insert into danmu (created, createdunix, msg, color, auth, uid, roomid) values ({Date},{Unix},{Msg},{Color},{Auth},{Uid},{Roomid})"
  }
}

sqlite3:

注意:不要并发连接数据库,可能会导致失败

{
  "保存弹幕至db": {
      "dbname": "sqlite",
      "url":"danmu.sqlite3",
      "字段help":"time.Now().Format(time.DateTime), time.Now().Unix(), item.msg, item.color, item.auth, item.uid, item.roomid",
      "create":"create table danmu (created text, createdunix text, msg text, color text, auth text, uid text, roomid text)",
      "insert":"insert into danmu  values ({Date},{Unix},{Msg},{Color},{Auth},{Uid},{Roomid})"
  }
}

字段create可以为空字符串,此时将不会在初次接收到弹幕时执行create语句,你必须要预先创建好表。

数据实例:

created createdunix msg color auth uid roomid
2023-04-26 03:20:33 1682450433 可能走位配合了他的压枪 #e33fff 畏未 96767379 92613
2023-04-26 03:20:45 1682450445 =。= #54eed8 青江知暖 282800070 92613
2023-04-26 03:20:52 1682450452 这不是铁挂吗 #ffffff 瑶少溪 43048863 92613
2023-04-26 03:20:55 1682450455 开个箱? #58c1de 修阡宇 433678545 92613
2023-04-26 03:20:59 1682450459 来了来了 #00fffc 我肚子是吃饱了撑大的 830140 92613

指定房间录制回调

配置文件添加了如下配置

{
    "指定房间录制回调-help":"当指定roomid的房间结束录制后触发对应的命令,命令执行目录为录播目录,占位符({type}:视频类型),durationS:录制时长超过指定秒数才触发",
    "指定房间录制回调":[
        {
            "例子":"windows转为正常的视频",
            "roomid":0,
            "durationS":60,
            "after":["cmd","/c","ffmpeg","-i","0.{type}","-y","-c","copy","-movflags","+faststart","1.{type}"]
        },
        {
            "例子":"linux转为正常的视频",
            "roomid":0,
            "durationS":60,
            "after":["ffmpeg","-i","0.{type}","-y","-c","copy","-movflags","+faststart","1.{type}"]
        },
        {
            "例子":"linux调整为统一分辨率(一个视频中出现分辨率改变eg:连麦)",
            "roomid":0,
            "durationS":60,
            "after":["ffmpeg","-i","0.{type}","-vf","scale=1920:1080:force_original_aspect_ratio=decrease:eval=frame,pad=1920:1080:-1:-1:color=black","1.{type}"]
        }
    ]
}

上述例子中演示了windows、linux下使用ffmpeg 进行视频转换。

注意:命令运行是异步的,如同步执行多个命令,应使用脚本。

性能检查

当配置了Web服务地址性能路径时,运行中的性能信息将可以通过http获取。
例如有如下配置:
config_K_v.json

{
  ...
  "Web服务地址":"0.0.0.0:10000",
  "性能路径":"/state/"
  ...
}

此时GET http://127.0.0.1:10000/state/

{
  "code": 0,
  "message": "ok",
  "data": {
    "currentTime": "2023-03-11 15:49:06", //当前时间
    "startTime": "2023-03-11 15:48:26",   //启动时间
    "version": "c797128",                 //版本
    "state": {
      "base": {
        "goVersion": "go1.20.1",          //编译使用的golang版本
        "numGoroutine": 53,               //goroutine数量
        "reqPoolState": {
            "inuse": 0,                   //全局请求池-正在使用数量
            "nopooled": 0,
            "nouse": 4,
            "pooled": 4,
            "qts": 1.8,                   //全局请求池-每秒请求数
            "sum": 4                      //全局请求池-总数量
        }
      },
      "common": {
        "92613": {                        //正在录制的房间
            "pid": 9,
            "version": "c797128",
            "live": [                     //流服务器状态>v0.14.5
                {
                    "Host": "xy111x59x162x229xy.mcdn.bilivideo.cn:486", //流服务器HOST
                    "Up": true,                                         //是否启用
                    "Codec": "avc",                                     //编码格式
                    "ReUpTime": "0001-01-01 00:00:00",                  //重新启用时间
                    "Expires": "2078-12-03 11:53:58",
                    "DisableCount": 0                                   //被禁用次数
                },
                {
                    "Host": "d1--cn-gotcha204-4.bilivideo.com",
                    "Up": true,
                    "Codec": "avc",
                    "ReUpTime": "0001-01-01 00:00:00",
                    "Expires": "2078-12-03 11:53:58",
                    "DisableCount": 0
                },
                {
                    "Host": "d1--cn-gotcha209.bilivideo.com",
                    "Up": true,
                    "Codec": "avc",
                    "ReUpTime": "0001-01-01 00:00:00",
                    "Expires": "2078-12-03 11:53:58",
                    "DisableCount": 0
                },
                {
                    "Host": "d1--cn-gotcha208.bilivideo.com",
                    "Up": true,
                    "Codec": "avc",
                    "ReUpTime": "0001-01-01 00:00:00",
                    "Expires": "2078-12-03 11:53:58",
                    "DisableCount": 0
                }
            ],
            "liveQn": 10000,                                //画质
            "roomid": 92613,                                //房间号
            "title": "补作业,没得D了",                     //标题
            "uname": "少年Pi",
            "upUid": 13046,
            "rev": 0,
            "renqi": 1,
            "watched": 36272,
            "onlineNum": 4615,
            "guardNum": 720,
            "parentAreaID": 6,
            "areaID": 235,
            "locked": false,
            "note": "人气榜 100+",
            "liveStartTime": "2024-06-17T15:55:05+08:00",
            "liveing": true
        }
    },
      "gc": {
        "gcAvgS": 6.73,                   //平均gc间隔 单位秒
        "gcCPUFractionPpm": 4.74,         //gc的STC耗时占总CPU时间比值 单位百万分之
        "lastGC": "2023-03-11 15:49:01",  //最后gc时间
        "numGC": 6                        //总gc数量
      },
      "mem": { "memInUse": "7.2 MB" }     //总使用内存
    }
  }
}

另外,当配置文件中的debug路径不为空时, 访问此路径可以获取调试信息,为空时关闭,需要/结尾,默认为空

  • 标准包net/http/pprof将在debug路径路径可用,从而可以使用go tool pprof工具进行性能调试。

另外,当配置文件中的debug模式true时,默认为false, 启用额外调试信息

  • 当录制fmp4时,每1分钟打印m4s池状态

自定义config_K_v.json

当启动时使用-ckv 路径,将从此路径(或http地址)加载config_K_v.json并覆盖默认config_K_v.json中的配置项。 使用此配置,可以在有新配置项时,默认使用新配置项而保持之前其他的配置。

表情发送

demo/config/config_K_v.json弹幕_识别表情代码true时,发送特定的文字将发送表情。

demo/config/config_danmu_official.json中可找到支持的特殊文字

如,命令行输入点赞回车:

点赞

I: 2022/09/15 02:23:21 弹幕发送 [发送 official_147 至 92613]
赞

danmu.log

I: 2022/09/15 02:23:21 弹幕发送 [发送 official_147 至 92613]
I: 2022/09/15 02:23:23 Msg [qydysky丶 : 赞]

流保存、弹幕ass、弹幕回放文件

以下内容可能过时,点击查看配置

"直播流清晰度-help": "清晰度可选-1:不保存 0:默认 20000:4K 10000:原画 400:蓝光 250:超清 150:高清 80:流畅,无提供所选清晰度时,使用低一档清晰度",
"直播流清晰度": 400,
"直播流类型-help": "flv,fmp4,flvH,fmp4H,带H后缀的为Hevc格式编码",
"直播流类型": "flv",
"直播流保存位置": "./live",
"直播hls流故障转移-help":"true:hls服务器故障时,使用其他",
"直播hls流故障转移": true,
"直播hls流保存为MP4": true,
"仅保存当前直播间流-help": "启用此项,才会保存Ass",
"仅保存当前直播间流": true,
"直播Web服务口":10000,
"直播Web缓冲长度-help":"非负整数,越长直播流延迟越高 内存占用越高",
"直播Web缓冲长度":5,
"直播Web可以发送弹幕":true,
"弹幕回放-help": "仅保存当前直播间流为true时才有效",
"弹幕回放": true,
"ass-help": "只有保存直播流时才考虑生成ass,ass编码默认GB18030(可选utf-8)",
"生成Ass弹幕": true,
"Ass编码": "GB18030",

当直播流类型为fmp4fmp4H时,默认将下载的切片合并保存为mp4。

当所选类型在当前直播中不可用时,会按以下顺序[fmp4,flv]尝试。

ass编码GB18030支持中文

- GB18030(默认) - utf-8

Ass默认utf-8编码,在录播结束时生成,可以配合指定房间录制回调生成硬编码弹幕的视频,配置项如下(>v0.15.9)

{
  "指定房间录制回调":[
    {
      "roomid-help":"0替换为指定的房间号",
      "roomid":0,
      "durationS":10,
      "after":["ffmpeg","-i","0.{type}","-y","-vf","ass=0.ass","1.{type}"]
    }
  ],
  "Ass": {
    "fontname-help":"指定字体",
    "fontname":"",
    "fontsize-help":"字体大小int",
    "fontsize":40,
    "showSec-help":"弹幕显示时长int,秒",
    "showSec":10,
    "area-help":"弹幕显示区域,float64,0-1(全屏)",
    "area":1.0,
    "alpha-help":"弹幕透明度,float64,0(不透明)-1(透明)",
    "alpha":1.0
  }
}

弹幕回放(仅直播流Web服务)

在保存直播流时,如弹幕回放true,则会将会将弹幕同时保存为csv文件。切片或停止录制时,根据csv文件生成xml文件

直播流Web服务中,点击进入任意一个目录,进行回放,如弹幕回放true,则会将会将弹幕回放出来。

添加配置项弹幕回放_隐藏发送人(>v0.14.19),为true时,回放弹幕文件类型记录(ass,xml,csv...)不记录发送人,当前直播的即时回放不显示发送人,但不影响保存弹幕至db等日志类型记录。由于回放依赖于csv类型文件,故回放历史录播亦不显示发送人。默认为false

直播流保存天数大于1时(默认4),当t日有1录播时,会尝试删除t-n日及之前的1or2个最早的录播。目录下有.keep文件或目录名不以yyyy_mm_dd-hh24_mi_ss开头将忽略(v0.12.7+)。

直播流回放Web服务

启动Web流服务,为下载的直播流提供局域网内的流服务,提供flv、hls/mp4格式流。

demo/config/config_K_v.json中可找到配置项,0.0.0.0:开放至局域网 127.0.0.1:仅本机。

    "Web服务地址":"0.0.0.0:10000",
    "直播Web服务路径":"/web/",

下述路径以上面这个配置作为实例。

直接进入直播Web服务路径,http://127.0.0.1:10000/web/,将进入由Vueelement-plus创建的简易web目录。

添加ref参数,从而可以从指定位置加载目录(>v0.14.27)。

例如:

{
  "直播流保存位置":"/bili"
}

当有录播文件夹存放在/bili/a时,通过http://127.0.0.1:10000/web/?ref=a/可以读取a下的列表

https://github.com/qydysky/bili_danmu_streamList

点击目录项,将进入由artplayermpegts创建的简易回放界面

https://github.com/qydysky/bili_danmu_artplayer

特殊

  • 路径为http://127.0.0.1:10000/web/stream?ref=now

    当前正在获取的流,播放此链接时进度将保持当前流进度,在登陆情况下,可以发送弹幕。流格式为fmp4(Hevc)或flv(Hevc)

    使用?ref={目录}参数来获取流,当为?ref=now时,为当前直播流

  • Hevc格式,可能需要硬件解码支持

  • 预览92613少年Pi的直播回放,此预览地址已应用如下配置

    config_K_v.json
    {
      "ip路径":"/xxxxxxx/",
      "日志文件输出-help": "为空时不保存日志",
      "日志文件输出": "",
      "指定弹幕重启录制":[
          {
              "uid":"29183321",
              "danmu":"1"
          }
      ],
      "弹幕输出到日志": false,
      "直播流清晰度-help": "清晰度可选-1:不保存 0:默认 20000:4K 10000:原画 400:蓝光 250:超清 150:高清 80:流畅,无提供所选清晰度时,使用低一档清晰度",
      "直播流清晰度": 10000,
      "直播流类型-help": "flv,fmp4,flvH,fmp4H,带H后缀的为Hevc格式编码",
      "直播流类型": "fmp4",
      "直播流保存位置": "/bilirec",
      "Web服务地址":"0.0.0.0:10000",
      "直播Web服务路径":"/web/",
      "直播Web可以发送弹幕":false,
      "直播流不使用mcdn":false,
      "fmp4获取更多服务器": true,
      "fmp4跳过解码出错的帧": true,
      "flv使用内置头": true,
      "直播流回放速率": "3 MB",
      "分段时长min": 60,
      "标题修改检测s": 1200,
      "直播流回放连接检查": 100,
      "Web服务连接限制": [
          {
              "cidr":"0.0.0.0/0",
              "max":10
          }
      ],
      "直播流回放连接限制": [
          {
              "help":"内网网段",
              "cidr":"192.168.31.1/24",
              "max":-1
          },
          {
              "help":"公网网段",
              "cidr":"127.0.0.1/32",
              "max":2
          },
          {
              "help":"公网网段",
              "cidr":"::/0",
              "max":2
          }
      ],
      "cookie保护": "下填路径",
      "cookie加密公钥": "public.pem",
      "cookie解密私钥": "private.pem",
      "debug模式-help":"启用额外调试信息",
      "debug模式":true,
      "debug路径-help":"当Web服务地址不为空时, 访问此路径可以获取调试信息,为空时关闭,需要/结尾",
      "debug路径":"/xxxxxxx/",
      "生成pgo":"",
      "扫码登录":true,
      "保存弹幕至db": {
          "dbname": "postgres",
          "url":"postgres://postgres:qydysky@192.168.31.103:5432/postgres?sslmode=disable",
          "字段help":"time.Now().Format(time.DateTime), time.Now().Unix(), item.msg, item.color, item.auth, item.uid, item.roomid",
          "create":"create table danmu (created varchar(20), createdunix varchar(20), msg varchar(100), color varchar(20), auth varchar(50), uid varchar(30), roomid varchar(30))",
          "insert":"insert into danmu (created, createdunix, msg, color, auth, uid, roomid) values ({Date},{Unix},{Msg},{Color},{Auth},{Uid},{Roomid})"
      },
      "保存日志至db":{
          "dbname": "postgres",
          "url":"postgres://postgres:qydysky@192.168.31.103:5432/postgres?sslmode=disable",
          "字段help":"$1:Prefix $2:Base $3:Msgs, use ? or $%d not placeholder",
          "create":"create table log (created timestamp, Prefix varchar(10), Base varchar(30), Msgs varchar(500))",
          "insert":"insert into log (created, Prefix, Base, Msgs) values (now(),$1,$2,$3)"
      },
      "服务器时区":28800,
      "弹幕表情": true
    }

测试可用项目(测试可连续播放10min+):

命令行操作

在准备动作完成(T: 2021/03/06 16:22:39 命令行操作 [回车查看帮助])后,输入回车将显示帮助

切换房间->输入' 数字'回车
发送弹幕->输入'字符串'回车
查看直播中主播->输入' liv'回车
重载弹幕->输入' reload'回车
搜索主播->输入' sea关键词'回车
房间信息->输入' room'回车
开始结束录制->输入' rec'回车
退出当前房间->输入' exit'回车
其他输出隔断不影响

用例:

  • 直播间切换
 his
T: 2023/02/01 02:34:00 命令行操作 [指令( his)]

T: 2023/02/01 02:34:00 api 历史直播主播 [获取中]
T: 2023/02/01 02:34:00 api 历史直播主播 [完成]
0       ☁       少年Pi(92613)
                        MC
1               C酱です(47867)
                        阿C今天又在玩什么捏
2               逍遥散人(1017)
                        日常闲聊 麻将上分训练
3               烤鱼子Official(22259479)
                        !!!!新年快乐!!!!
4       ☁       Asaki大人(6154037)
                        开心鸭鸭杀
5               观察者网(948391)
                        一周军情观察:海的那边是敌人
6               一米八的坤儿(4350043)
                        单排下饭主播
7               数字猫(3839755)
                        石村号上的朋友真是热情好客
8       ☁       环球时报(796281)
                        家乡的风景 家乡的年
9               千鈴鳴_Official(26724557)
                        【B限】原神やるぞおおおお【千鈴鳴】
回复' to(序号)'进入直播间

3 x 9
满脑子都是小c
 to4
T: 2023/02/01 02:34:07 命令行操作 [指令( to4)]

I: 2023/02/01 02:34:07 bilidanmu [停止,等待服务器断开连接]
I: 2023/02/01 02:34:07 bilidanmu [服务器连接关闭]
I: 2023/02/01 02:34:07 直播流保存 [正在等待下载完成...]
T: 2023/02/01 02:34:08 弹幕回放 [停止]
I: 2023/02/01 02:34:08 直播流保存 [结束录制(92613)]
T: 2023/02/01 02:34:09 功能 弹幕合并 [房间更新: 6154037]
T: 2023/02/01 02:34:09 bilidanmu [准备]
T: 2023/02/01 02:34:09 api IsConnected [已连接]
I: 2023/02/01 02:34:10 bilidanmu [连接到房间 6154037]
I: 2023/02/01 02:34:16 api 切换粉丝牌 [无主播粉丝牌]
I: 2023/02/01 02:34:17 bilidanmu [直播中]
T: 2023/02/01 02:34:17 bilidanmu [连接 wss://hw-sh-live-comet-03.chat.bilibili.com/sub]
I: 2023/02/01 02:34:17 bilidanmu [已连接到房间 Asaki大人 ( 6154037 )]
I: 2023/02/01 02:34:17 bilidanmu [开心鸭鸭杀]
T: 2023/02/01 02:34:18 bilidanmu [获取人气]
 120
I: 2021/03/06 16:21:35 命令行操作 [进入 120]
  • 发送弹幕
1
I: 2021/03/06 16:21:17 弹幕发送 [发送 1 至 7734200]
  • 查看房间信息
 room
T: 2023/02/01 02:35:38 命令行操作 [指令( room)]

当前直播间(6154037)信息
Asaki大人 开心鸭鸭杀 直播中
已直播时长: 08:12:18
营收: ¥0.00
舰长数: 1287
分区排行: 人气榜 2 人气: 2049772 观看人数: 278848 在线人数: 10000
直播Web服务: http://0.0.0.0:10000
正在录制的房间:
        Asaki大人(6154037) 开心鸭鸭杀
输入` rec` 来启停当前房间录制

还支持登录、搜索主播直播间、查看历史记录、查看关注的直播间、保存直播流等功能

cookie加密

使用X25519和chacha20poly1305(>v0.14.15) 保护cookie.txt

demo/config/config_K_v.json中可找到配置项

"cookie加密公钥":"public.pem",
"cookie解密私钥":"private.pem"
  • 当配置了公钥路径后,cookie将被加密(若公钥无效,将会导致cookie无法储存)。若未配置私钥路径,则每次启动都会要求输入私钥路径。(若私钥无效,将会导致cookie被清除)
  • 当未配置公钥路径(空字符串),cookie将明文储存。
  • 默认使用了demo/下的(public.pem)(private.pem)进行加密,使用时注意自行生成公私钥并按照上述说明使用

注意,每次更换设置(设置或未设置公钥),cookie会失效。

附:创建公(public.pem)私(private.pem)钥

使用 -genKey 生成(>v0.14.15)
main(main.exe) -genKey

公钥:
-----BEGIN ECDH PUBLIC KEY-----
tvdVdbI7DTlRcyE44va7zXhi5rewxcm44/Dmp8DMnGY=
-----END ECDH PUBLIC KEY-----

私钥:
-----BEGIN ECDH PRIVATE KEY-----
xrWweTO5upvzDha6WrEBQKkToUYLyMCI7An2btRqop0=
-----END ECDH PRIVATE KEY-----

请复制以上公私钥并另存为文件,可以在cookie加密公钥、cookie解密私钥中使用

私信

在登录后,可以使用私信

私信配置在demo/config/config_K_v.json有说明

语音

调用tts默认使用ffplay,安装ffmpeg

或使用其他程序:可在demo/config/config_K_v.json中编辑调用的程序及附加选项

config_K_v.json
默认
    "TTS_使用程序路径":"ffplay",
    "TTS_使用程序参数":"-autoexit -nodisp"

使用mpv
    "TTS_使用程序路径":"mpv",
    "TTS_使用程序参数":"--no-video"

使用potplayer(例程序位置D:\potplayer\PotPlayerMini64.exe)
    "TTS_使用程序路径":"D:\\potplayer\\PotPlayerMini64.exe",
    "TTS_使用程序参数":"/current /autoplay"

release默认编译tts

总开关,自定义响应的事件可在demo/config/config_tts.json中编辑。{}为传递过来的变量,将会按设定替换。最后未使用的{}会全部删除。下例:ABC购买 1个月舰长

...
"0buyguide-help": "大航海 {username}:用户 {op_name}:购买方式 {role_name}:大航海类型 {num}:个数 {unit}:时间单位",
"0buyguide": "{username}{op_name} {num}个{unit}{role_name}",
...

特别说明的,下面结构为全局替换。

"replace":{
    "?":"问号",
    "?":"问号"
}

demo/config/config_K_v.json中可选使用的服务api

    "TTS_服务器-help": "baidu:百度翻译合成 youdao:有道TTS",
    "TTS_服务器": "youdao",
    "TTS_服务器_youdaoId": "",
    "TTS_服务器_youdaoKey": "",

支持baidu、有道讯飞

使用有道则需要Id和Key。

    "TTS_服务器_youdaoId": "7xxxxxxxxxxxxxxa",
    "TTS_服务器_youdaoKey": "yxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxP",

使用讯飞则需要应用Id、Key、Secret。可设置发音人。

    "TTS_服务器_xfId": "6xxxxxxb",
    "TTS_服务器_xfKey": "4xxxxxxxx9",
    "TTS_服务器_xfSecret": "YxxxxxxxBh",
    "TTS_服务器_xfVoice-help": "讯飞发音人 xiaoyan:小燕甜美女声 aisjiuxu:许久亲切男声 aisxping:小萍知性女声 aisjinger:小婧亲切女声 aisbabyxu:许小宝可爱童声 random:随机",
    "TTS_服务器_xfVoice": "random",

弹幕窗

构建gtk需要gtk3,先行安装gtk release 由于某些原因,Linux默认不编译gtk界面 Windows默认不编译

编译命令
cd demo
go build -v -tags `gtk` -o demo.exe -i main.go

弹幕处理/响应

默认开启了

  • 反射弹幕机

启动时加载,当弹幕内容与demo/config_auto_reply.json中所设键名相同时,在登录的情况下,会自动发送对应值的弹幕

  • 相同合并

当短时间存在大量完全相同的弹幕时,他们将合并显示。

  • 更少弹幕

过滤掉自身重复度及最近弹幕重复度高的弹幕

  • 更短弹幕

当与上条弹幕具有相同开头的开头时,重复的部分会用...替代

仅对显示效果进行处理,而不处理输出到日志。更多设置见demo/config/config_F.json

运行

方法

  1. 预先编译

下载安装golang/golangCN

clone/下载本项目。进入demo目录(文件夹),运行:

linux: CGO_ENABLED=0 go build .
windows: set CGO_ENABLED=0;go build .

再运行生成的demo.exedemo

  1. 即时编译

下载安装golang/golangCN

clone/下载本项目。进入demo目录(文件夹),运行:

linux: CGO_ENABLED=0 go run . [-r 房间ID] [-ckv 自定义config_K_v.json] [-genKey]
windows: set CGO_ENABLED=0;go run . [-r 房间ID] [-ckv 自定义config_K_v.json] [-genKey]
  1. docker部署

经测试可以部署到debian镜像上,注意首先得更新ca

apt-get update && apt-get install -y ca-certificates openssl

如果你日常使用windows,但在debian运行,那还需先编译linux版本

下载安装golang/golangCN

clone/下载本项目。进入demo目录(文件夹),运行:

set GOOS=linux
set CGO_ENABLED=0
go build .

注意实际使用时,在config_K_v.json关闭tts等需要关闭界面及音频的功能,配置好录播存放位置,做好直播流服务的端口映射

  1. github编译

前往releases页下载对应系统版本。解压后进入demo目录(文件夹),运行main(main.exe)。

./main [-r 房间ID] [-ckv 自定义config_K_v.json] [-genKey]
./main.exe [-r 房间ID] [-ckv 自定义config_K_v.json] [-genKey]

注意事项

  • 其中[]内的内容为可选项

  • -r,-ckv为空时,将尝试从环境变量中获取,分别对应键值r,ckv(>v0.14.21)

  • 法1,2,3建议使用最新提交

  • 程序受主机时区配置影响,注意正确配置主机时区

  • 弹幕及礼物默认会记录于danmu.log中

  • 部分功能(如签到、发送弹幕、获取原画等)需要在cookie路径下对应位置放置有效cookie.txt才可用运行时按提示使用扫码登录成功后才可用(登录信息会保存在cookie路径中)(cookie路径(>v0.14.26)默认为程序目录下的cookie.txt)

  • 在golang1.20+,由于某些原因,你可能需要在构建时添加CGO_ENABLED=0

  • 由于通常是发现功能不正常时,才会检查b站是否更新,又因日常录播并不会使用到全部功能,所以并不能确保全部功能都能正常运行

  • 离线编译:release版本将包含vendor压缩文件(>v0.15.7)。下载源代码及vendor压缩文件,解压后将vendor文件夹放置在go.mod同级目录时,你可以在demo文件夹使用go build -mod=vendor -v .进行离线构建

    源代码链接: https://github.com/qydysky/bili_danmu/archive/refs/tags/{version}.zip

    vendor链接: https://github.com/qydysky/bili_danmu/releases/download/{version}/vendor.zip

关于更新版本v0.{y}.{z}

  • 当bilibili停止服务时,将会发布v1版本。
  • 当发生不兼容的改动(删减原来有效、修改有效项的含义)时,将会发布v0.{y+1}.0版。更新前,你需要关注更新说明,以确认变动的配置项的影响。
  • 当发生向下兼容的改动(修复、新增功能)时,将会发布v0.{y}.{z+1}版。通常只需要覆盖主体mainmain.exe)即可更新。
  • 另外releases页面中,Pre-release表示此版本为尚未经过验证的版本,或存在重大bug,Latest表示此版本经过验证,不存在明显的重大bug。

效果展示

以下内容可能过时,以实际运行为准

命令窗口(以下为截取)

//启动
qydysky@DESKTOP-5CV1EFA:~/程序/git/go/src/github.com/qydysky/bili_danmu/demo$ go run -tags "gtk" main.go -r 21320551
I: 2021/02/18 20:33:09 api 小心心加密 [如需加密,会自动打开 http://127.0.0.1:33673]
I: 2021/02/18 20:33:09 api 小心心加密 [启动]
PID:14544
房间号: 21320551
T: 2021/02/18 20:33:09 api 新建 [ok]
T: 2021/02/18 20:33:09 api 获取房号 [获取房号]
T: 2021/02/18 20:33:10 api LIVE_BUVID [获取LIVE_BUVID]
I: 2021/02/18 20:33:10 api LIVE_BUVID [存在]
T: 2021/02/18 20:33:11 api 获取Token [ok]
I: 2021/02/18 20:33:13 api 获取直播流 [轮播中]
T: 2021/02/18 20:33:13 api 银瓜子=>硬币 [银瓜子=>硬币]
I: 2021/02/18 20:33:15 api 银瓜子=>硬币 [现在有银瓜子 540 个]
W: 2021/02/18 20:33:15 api 银瓜子=>硬币 [当前银瓜子数量不足]
T: 2021/02/18 20:33:15 api 签到 [签到]
I: 2021/02/18 20:33:19 api 获取客户版本 [api version 2.6.25]
I: 2021/02/18 20:33:21 api 获取热门榜 [热门榜: 虚拟主播 50+]
I: 2021/02/18 20:33:23 bilidanmu Demo [连接到房间 21320551]
I: 2021/02/18 20:33:23 bilidanmu Demo [连接 wss://tx-bj-live-comet-02.chat.bilibili.com/sub]
T: 2021/02/18 20:33:23 api 小心心 [获取小心心]
I: 2021/02/18 20:33:23 bilidanmu Demo [已连接到房间 乙女音Official ( 21320551 )]
I: 2021/02/18 20:33:23 bilidanmu Demo [【b限】学《巴啦啦小魔仙》]
I: 2021/02/18 20:33:24 bilidanmu Demo [获取人气]
T: 2021/02/18 20:33:25 api 礼物列表 [获取礼物列表]
I: 2021/02/18 20:33:27 api 获取舰长数 [舰长数获取成功 471]
I: 2021/02/18 20:33:27 弹幕发送 [发送   至 21320551]
I: 2021/02/18 20:33:27 功能 [营收 ¥0.00]
 
I: 2021/02/18 20:33:29 api 礼物列表 [成功]
I: 2021/02/18 20:33:29 api 小心心 [今天小心心已满!]
//普通弹幕
老鸡捉小鹰
你快扒拉他
你这好像是补刀
吓人
//大航海
>>> 欢迎舰长 Mana_单推... 进入直播间
//礼物
====
超级角击 投喂 1 个 摩天轮
====
//同字符串合并
7 x 原神公测B服冲冲冲
//同字符忽略
原神公测B站冲冲冲
...B服冲冲冲
//SC
====
SC:  凪穗 

有了OTO能量,我们才能够坚强,prprpr
OTOエネルギーがあれば、私たちは強くなれる。prprprpr
====
//gtk的弹幕格式化发送
2020/11/20 15:39:57 弹幕格式已设置为 [{D}]
INFO: 2020/11/20 15:40:05 [弹幕发送] [发送 [就是这样] 至 394988]
[就是这样]
INFO: 2020/11/20 15:40:15 [弹幕发送] [发送 [你知道么] 至 394988]
[你知道么]
2020/11/20 15:42:38 弹幕长度大于20,不做格式处理
INFO: 2020/11/20 15:42:38 [弹幕发送] [发送 11111111111111111111 至 394988]
11111111111111111111
//其他会出现在命令行的信息
//热门榜
I: 2021/02/18 14:59:00 Msg 房 [热门榜 虚拟 4]
//人气
I: 2021/02/18 14:58:51 Reply 人气 [当前人气 450869]
//营收
I: 2021/02/18 14:58:24 功能 [营收 ¥247.80]
//语音
I: 2021/02/18 14:59:00 TTS [0superchat SC:  三得笠·阿克曼茶   8888888  天才oto天才 ]

ctrl+c退出,会同时追加记录到文件danmu.log中(文件记录完整信息,不会减少附加功能作用的弹幕)

danmu.log

基本同命令行显示,不同下列:

//弹幕 上述合并、忽略都不会起作用
I: 2021/02/18 07:24:55 Msg [从天上掉下来的骚年 : 秀才]
//礼物 超过设定限额的将会在命令行中显示,级别为I
T: 2021/02/18 07:30:30 Msg 礼 [正道的光博航同志 投喂 1 个 上上签 ¥1.0]
I: 2021/02/18 14:52:31 Msg 礼 [三千千千千千千 投喂 1 个 爱之魔力 ¥28.0]
//sc
I: 2021/02/18 14:40:54 Msg 礼 [SC:  加拉入我心 ¥ 30 关注了乙女音,我才能够得到快乐 乙女音符に注目してこそ、私は幸せになれるのです。]
I: 2021/02/18 21:48:49 Msg 房 [欢迎舰长 Mana_单推... 进入直播间]

Gtk弹幕窗(Linux Only)

更多内容详见注释,如有疑问请发issues,欢迎pr