mirror of
https://github.com/AynaLivePlayer/blivedm-go.git
synced 2025-12-06 11:22:50 +08:00
141 lines
6.1 KiB
Go
141 lines
6.1 KiB
Go
package client
|
|
|
|
import (
|
|
"github.com/Akegarasu/blivedm-go/message"
|
|
"github.com/Akegarasu/blivedm-go/packet"
|
|
"github.com/Akegarasu/blivedm-go/utils"
|
|
log "github.com/sirupsen/logrus"
|
|
"runtime/debug"
|
|
"strings"
|
|
)
|
|
|
|
var (
|
|
knownCMD = []string{"INTERACT_WORD", "HOT_RANK_SETTLEMENT", "DANMU_GIFT_LOTTERY_START", "WELCOME_GUARD", "PK_PROCESS", "PK_BATTLE_PRO_TYPE", "MATCH_TEAM_GIFT_RANK", "PK_BATTLE_CRIT", "LUCK_GIFT_AWARD_USER", "SCORE_CARD", "ONLINE_RANK_V2", "PK_BATTLE_SPECIAL_GIFT", "SEND_TOP", "SUPER_CHAT_MESSAGE_JPN", "ANIMATION", "GUARD_LOTTERY_START", "WEEK_STAR_CLOCK", "WELCOME", "WIN_ACTIVITY", "ROOM_KICKOUT", "CHANGE_ROOM_INFO", "ROOM_SKIN_MSG", "ROOM_BLOCK_MSG", "SUPER_CHAT_ENTRANCE", "PK_BATTLE_RANK_CHANGE", "ROOM_LOCK", "TV_END", "PK_PRE", "ROOM_SILENT_OFF", "SEND_GIFT", "DANMU_MSG", "ANCHOR_LOT_START", "ROOM_BOX_USER", "ONLINE_RANK_TOP3", "WIDGET_BANNER", "PK_BATTLE_START", "ACTIVITY_MATCH_GIFT", "PK_AGAIN", "PK_MATCH", "RAFFLE_START", "LIVE", "WISH_BOTTLE", "GUARD_ACHIEVEMENT_ROOM", "ONLINE_RANK_COUNT", "COMMON_NOTICE_DANMAKU", "LOL_ACTIVITY", "HOT_RANK_CHANGED", "ROOM_BLOCK_INTO", "ROOM_LIMIT", "PANEL", "RAFFLE_END", "ENTRY_EFFECT", "STOP_LIVE_ROOM_LIST", "TV_START", "WATCH_LPL_EXPIRED", "PK_BATTLE_PRE", "USER_TOAST_MSG", "BOX_ACTIVITY_START", "PK_MIC_END", "LIVE_INTERACTIVE_GAME", "ROOM_BANNER", "PK_BATTLE_GIFT", "MESSAGEBOX_USER_GAIN_MEDAL", "LITTLE_TIPS", "HOUR_RANK_AWARDS", "NOTICE_MSG", "ROOM_REAL_TIME_MESSAGE_UPDATE", "ANCHOR_LOT_END", "PREPARING", "GUARD_BUY", "ROOM_CHANGE", "room_admin_entrance", "CHASE_FRAME_SWITCH", "DANMU_GIFT_LOTTERY_AWARD", "PK_BATTLE_VOTES_ADD", "PK_BATTLE_END", "CUT_OFF", "PK_BATTLE_PROCESS", "PK_BATTLE_SETTLE_USER", "ANCHOR_LOT_AWARD", "WIN_ACTIVITY_USER", "VOICE_JOIN_STATUS", "DANMU_GIFT_LOTTERY_END", "ROOM_RANK", "SUPER_CHAT_MESSAGE", "ACTIVITY_BANNER_UPDATE_V2", "SPECIAL_GIFT", "ROOM_SILENT_ON", "WARNING", "ROOM_ADMINS", "COMBO_SEND", "HOT_RANK_SETTLEMENT_V2", "ANCHOR_LOT_CHECKSTATUS", "HOT_RANK_CHANGED_V2", "SUPER_CHAT_MESSAGE_DELETE", "PK_END", "PK_SETTLE", "ROOM_REFRESH", "PK_START", "COMBO_END", "PK_LOTTERY_START", "GUARD_WINDOWS_OPEN", "REENTER_LIVE_ROOM", "MESSAGEBOX_USER_MEDAL_CHANGE", "MESSAGEBOX_USER_MEDAL_COMPENSATION", "LITTLE_MESSAGE_BOX", "PK_BATTLE_PRE_NEW", "PK_BATTLE_START_NEW", "PK_BATTLE_PROCESS_NEW", "PK_BATTLE_FINAL_PROCESS", "PK_BATTLE_SETTLE_V2", "PK_BATTLE_SETTLE_NEW", "PK_BATTLE_PUNISH_END", "PK_BATTLE_VIDEO_PUNISH_BEGIN", "PK_BATTLE_VIDEO_PUNISH_END", "ENTRY_EFFECT_MUST_RECEIVE", "SUPER_CHAT_AUDIT", "VIDEO_CONNECTION_JOIN_START", "VIDEO_CONNECTION_JOIN_END", "VIDEO_CONNECTION_MSG", "VTR_GIFT_LOTTERY", "RED_POCKET_START", "FULL_SCREEN_SPECIAL_EFFECT", "POPULARITY_RED_POCKET_START", "POPULARITY_RED_POCKET_WINNER_LIST", "USER_PANEL_RED_ALARM", "SHOPPING_CART_SHOW", "THERMAL_STORM_DANMU_BEGIN", "THERMAL_STORM_DANMU_UPDATE", "THERMAL_STORM_DANMU_CANCEL", "THERMAL_STORM_DANMU_OVER", "MILESTONE_UPDATE_EVENT", "WEB_REPORT_CONTROL", "DANMU_TAG_CHANGE", "RANK_REM", "LIVE_PLAYER_LOG_RECYCLE", "LIVE_INTERNAL_ROOM_LOGIN", "LIVE_OPEN_PLATFORM_GAME", "WATCHED_CHANGE"}
|
|
knownCMDMap map[string]int
|
|
)
|
|
|
|
type eventHandlers struct {
|
|
danmakuMessageHandlers []func(*message.Danmaku)
|
|
superChatHandlers []func(*message.SuperChat)
|
|
giftHandlers []func(*message.Gift)
|
|
guardBuyHandlers []func(*message.GuardBuy)
|
|
liveHandlers []func(*message.Live)
|
|
}
|
|
|
|
type customEventHandlers map[string]func(s string)
|
|
|
|
func init() {
|
|
knownCMDMap = make(map[string]int)
|
|
for _, c := range knownCMD {
|
|
knownCMDMap[c] = 0
|
|
}
|
|
}
|
|
|
|
func (c *Client) RegisterCustomEventHandler(cmd string, handler func(s string)) {
|
|
(*c.customEventHandlers)[cmd] = handler
|
|
}
|
|
|
|
func (c *Client) OnDanmaku(f func(*message.Danmaku)) {
|
|
c.eventHandlers.danmakuMessageHandlers = append(c.eventHandlers.danmakuMessageHandlers, f)
|
|
}
|
|
|
|
func (c *Client) OnSuperChat(f func(*message.SuperChat)) {
|
|
c.eventHandlers.superChatHandlers = append(c.eventHandlers.superChatHandlers, f)
|
|
}
|
|
|
|
func (c *Client) OnGift(f func(gift *message.Gift)) {
|
|
c.eventHandlers.giftHandlers = append(c.eventHandlers.giftHandlers, f)
|
|
}
|
|
|
|
func (c *Client) OnGuardBuy(f func(*message.GuardBuy)) {
|
|
c.eventHandlers.guardBuyHandlers = append(c.eventHandlers.guardBuyHandlers, f)
|
|
}
|
|
|
|
func (c *Client) OnLive(f func(*message.Live)) {
|
|
c.eventHandlers.liveHandlers = append(c.eventHandlers.liveHandlers, f)
|
|
}
|
|
|
|
func (c *Client) Handle(p packet.Packet) {
|
|
switch p.Operation {
|
|
case packet.Notification:
|
|
cmd := parseCmd(p.Body)
|
|
sb := utils.BytesToString(p.Body)
|
|
// 新的弹幕 cmd 可能带参数
|
|
if ind := strings.Index(cmd, ":"); ind >= 0 {
|
|
cmd = cmd[:ind]
|
|
}
|
|
// 优先执行自定义 eventHandler ,会覆盖库内自带的 handler
|
|
f, ok := (*c.customEventHandlers)[cmd]
|
|
if ok {
|
|
go cover(func() { f(sb) })
|
|
return
|
|
}
|
|
switch cmd {
|
|
case "DANMU_MSG":
|
|
d := new(message.Danmaku)
|
|
d.Parse(p.Body)
|
|
for _, fn := range c.eventHandlers.danmakuMessageHandlers {
|
|
go cover(func() { fn(d) })
|
|
}
|
|
case "SUPER_CHAT_MESSAGE":
|
|
s := new(message.SuperChat)
|
|
s.Parse(p.Body)
|
|
for _, fn := range c.eventHandlers.superChatHandlers {
|
|
go cover(func() { fn(s) })
|
|
}
|
|
case "SEND_GIFT":
|
|
g := new(message.Gift)
|
|
g.Parse(p.Body)
|
|
for _, fn := range c.eventHandlers.giftHandlers {
|
|
go cover(func() { fn(g) })
|
|
}
|
|
case "GUARD_BUY":
|
|
g := new(message.GuardBuy)
|
|
g.Parse(p.Body)
|
|
for _, fn := range c.eventHandlers.guardBuyHandlers {
|
|
go cover(func() { fn(g) })
|
|
}
|
|
case "LIVE":
|
|
l := new(message.Live)
|
|
l.Parse(p.Body)
|
|
for _, fn := range c.eventHandlers.liveHandlers {
|
|
go cover(func() { fn(l) })
|
|
}
|
|
default:
|
|
if _, ok := knownCMDMap[cmd]; ok {
|
|
return
|
|
}
|
|
log.Debugf("unknown cmd(%s), body: %s", cmd, p.Body)
|
|
}
|
|
case packet.HeartBeatResponse:
|
|
case packet.RoomEnterResponse:
|
|
default:
|
|
log.WithField("protover", p.ProtocolVersion).
|
|
WithField("data", string(p.Body)).
|
|
Warn("unknown protover")
|
|
}
|
|
}
|
|
|
|
func parseCmd(d []byte) string {
|
|
// {"cmd":"DANMU_MSG", ...
|
|
l := len(d)
|
|
pos := 8
|
|
s := byte('"')
|
|
for pos <= l {
|
|
if d[pos] == s {
|
|
break
|
|
}
|
|
pos++
|
|
}
|
|
return utils.BytesToString(d[8:pos])
|
|
}
|
|
|
|
func cover(f func()) {
|
|
defer func() {
|
|
if pan := recover(); pan != nil {
|
|
log.Errorf("event error: %v\n%s", pan, debug.Stack())
|
|
}
|
|
}()
|
|
f()
|
|
}
|