前端直连升级弹幕协议版本

This commit is contained in:
John Smith 2022-01-03 23:35:10 +08:00
parent 23ebaca372
commit e318bdfcac
4 changed files with 2400 additions and 66 deletions

2
frontend/.eslintignore Normal file
View File

@ -0,0 +1,2 @@
brotli_decode.js
pronunciation/dict*.js

View File

@ -13,7 +13,6 @@
"downloadjs": "^1.4.7", "downloadjs": "^1.4.7",
"element-ui": "^2.9.1", "element-ui": "^2.9.1",
"lodash": "^4.17.19", "lodash": "^4.17.19",
"pako": "^1.0.11",
"vue": "^2.6.10", "vue": "^2.6.10",
"vue-i18n": "^8.11.2", "vue-i18n": "^8.11.2",
"vue-router": "^3.0.6" "vue-router": "^3.0.6"

File diff suppressed because one or more lines are too long

View File

@ -1,14 +1,15 @@
import axios from 'axios' import axios from 'axios'
import * as pako from 'pako'
import {BrotliDecode} from './brotli_decode'
import {getUuid4Hex} from '@/utils' import {getUuid4Hex} from '@/utils'
import * as avatar from './avatar' import * as avatar from '../avatar'
const HEADER_SIZE = 16 const HEADER_SIZE = 16
// const WS_BODY_PROTOCOL_VERSION_INFLATE = 0 // const WS_BODY_PROTOCOL_VERSION_NORMAL = 0
// const WS_BODY_PROTOCOL_VERSION_NORMAL = 1 // const WS_BODY_PROTOCOL_VERSION_HEARTBEAT = 1
const WS_BODY_PROTOCOL_VERSION_DEFLATE = 2 // const WS_BODY_PROTOCOL_VERSION_DEFLATE = 2
const WS_BODY_PROTOCOL_VERSION_BROTLI = 3
// const OP_HANDSHAKE = 0 // const OP_HANDSHAKE = 0
// const OP_HANDSHAKE_REPLY = 1 // const OP_HANDSHAKE_REPLY = 1
@ -32,6 +33,9 @@ const OP_AUTH_REPLY = 8
// const MinBusinessOp = 1000 // const MinBusinessOp = 1000
// const MaxBusinessOp = 10000 // const MaxBusinessOp = 10000
const AUTH_REPLY_CODE_OK = 0
// const AUTH_REPLY_CODE_TOKEN_ERROR = -101
const HEARTBEAT_INTERVAL = 10 * 1000 const HEARTBEAT_INTERVAL = 10 * 1000
const RECEIVE_TIMEOUT = HEARTBEAT_INTERVAL + 5 * 1000 const RECEIVE_TIMEOUT = HEARTBEAT_INTERVAL + 5 * 1000
@ -105,9 +109,8 @@ export default class ChatClientDirect {
let authParams = { let authParams = {
uid: 0, uid: 0,
roomid: this.roomId, roomid: this.roomId,
protover: 2, protover: 3,
platform: 'web', platform: 'web',
clientver: '1.14.3',
type: 2 type: 2
} }
this.websocket.send(this.makePacket(authParams, OP_AUTH)) this.websocket.send(this.makePacket(authParams, OP_AUTH))
@ -145,7 +148,14 @@ export default class ChatClientDirect {
onReceiveTimeout() { onReceiveTimeout() {
window.console.warn('接收消息超时') window.console.warn('接收消息超时')
this.receiveTimeoutTimerId = null this.discardWebsocket()
}
discardWebsocket() {
if (this.receiveTimeoutTimerId) {
window.clearTimeout(this.receiveTimeoutTimerId)
this.receiveTimeoutTimerId = null
}
// 直接丢弃阻塞的websocket不等onclose回调了 // 直接丢弃阻塞的websocket不等onclose回调了
this.websocket.onopen = this.websocket.onclose = this.websocket.onmessage = null this.websocket.onopen = this.websocket.onclose = this.websocket.onmessage = null
@ -167,88 +177,124 @@ export default class ChatClientDirect {
if (this.isDestroying) { if (this.isDestroying) {
return return
} }
window.console.warn(`掉线重连中${++this.retryCount}`) this.retryCount++
window.console.warn('掉线重连中', this.retryCount)
window.setTimeout(this.wsConnect.bind(this), 1000) window.setTimeout(this.wsConnect.bind(this), 1000)
} }
onWsMessage (event) { onWsMessage (event) {
this.refreshReceiveTimeoutTimer() this.refreshReceiveTimeoutTimer()
this.retryCount = 0
if (!(event.data instanceof ArrayBuffer)) { if (!(event.data instanceof ArrayBuffer)) {
window.console.warn('未知的websocket消息', event.data) window.console.warn('未知的websocket消息类型data=', event.data)
return return
} }
let data = new Uint8Array(event.data) let data = new Uint8Array(event.data)
this.handlerMessage(data) this.parseWsMessage(data)
// 至少成功处理1条消息
this.retryCount = 0
} }
handlerMessage (data) { parseWsMessage (data) {
let offset = 0 let offset = 0
while (offset < data.byteLength) { let dataView = new DataView(data.buffer)
let dataView = new DataView(data.buffer, offset) let packLen = dataView.getUint32(0)
let packLen = dataView.getUint32(0) let rawHeaderSize = dataView.getUint16(4)
// let rawHeaderSize = dataView.getUint16(4) // let ver = dataView.getUint16(6)
let ver = dataView.getUint16(6) let operation = dataView.getUint32(8)
let operation = dataView.getUint32(8) // let seqId = dataView.getUint32(12)
// let seqId = dataView.getUint32(12)
switch (operation) {
switch (operation) { case OP_AUTH_REPLY:
case OP_HEARTBEAT_REPLY: { case OP_SEND_MSG_REPLY: {
// 人气值没用 // 业务消息,可能有多个包一起发,需要分包
break while (true) { // eslint-disable-line no-constant-condition
let body = new Uint8Array(data.buffer, offset + rawHeaderSize, packLen - rawHeaderSize)
this.parseBusinessMessage(dataView, body)
offset += packLen
if (offset >= data.byteLength) {
break
}
dataView = new DataView(data.buffer, offset)
packLen = dataView.getUint32(0)
rawHeaderSize = dataView.getUint16(4)
} }
case OP_SEND_MSG_REPLY: { break
let body = new Uint8Array(data.buffer, offset + HEADER_SIZE, packLen - HEADER_SIZE) }
if (ver == WS_BODY_PROTOCOL_VERSION_DEFLATE) { case OP_HEARTBEAT_REPLY: {
body = pako.inflate(body) // 服务器心跳包,包含人气值,这里没用
this.handlerMessage(body) break
} else { }
default: {
// 未知消息
let body = new Uint8Array(data.buffer, offset + rawHeaderSize, packLen - rawHeaderSize)
window.console.warn('未知包类型operation=', operation, dataView, body)
break
}
}
}
parseBusinessMessage (dataView, body) {
let ver = dataView.getUint16(6)
let operation = dataView.getUint32(8)
switch (operation) {
case OP_SEND_MSG_REPLY: {
// 业务消息
if (ver == WS_BODY_PROTOCOL_VERSION_BROTLI) {
// 压缩过的先解压
body = BrotliDecode(body)
this.parseWsMessage(body)
} else {
// 没压缩过的直接反序列化
if (body.length !== 0) {
try { try {
body = JSON.parse(textDecoder.decode(body)) body = JSON.parse(textDecoder.decode(body))
this.handlerCommand(body) this.handlerCommand(body)
} catch (e) { } catch (e) {
window.console.warn('body:', body) window.console.error('body=', body)
throw e throw e
} }
} }
break
} }
case OP_AUTH_REPLY: { break
this.sendHeartbeat() }
break case OP_AUTH_REPLY: {
// 认证响应
body = JSON.parse(textDecoder.decode(body))
if (body.code !== AUTH_REPLY_CODE_OK) {
window.console.error('认证响应错误body=', body)
// 这里应该重新获取token再重连的但前端没有用到token所以不重新init了
this.discardWebsocket()
throw new Error('认证响应错误')
} }
default: { this.sendHeartbeat()
let body = new Uint8Array(data.buffer, offset + HEADER_SIZE, packLen - HEADER_SIZE) break
window.console.warn('未知包类型operation=', operation, body) }
break default: {
} // 未知消息
} window.console.warn('未知包类型operation=', operation, dataView, body)
break
offset += packLen }
} }
} }
handlerCommand (command) { handlerCommand (command) {
if (command instanceof Array) {
for (let oneCommand of command) {
this.handlerCommand(oneCommand)
}
return
}
let cmd = command.cmd || '' let cmd = command.cmd || ''
let pos = cmd.indexOf(':') let pos = cmd.indexOf(':')
if (pos != -1) { if (pos != -1) {
cmd = cmd.substr(0, pos) cmd = cmd.substr(0, pos)
} }
let handler = COMMAND_HANDLERS[cmd] let callback = CMD_CALLBACK_MAP[cmd]
if (handler) { if (callback) {
handler.call(this, command) callback.call(this, command)
} }
} }
async onReceiveDanmaku (command) { async danmuMsgCallback (command) {
if (!this.onAddText) { if (!this.onAddText) {
return return
} }
@ -295,7 +341,7 @@ export default class ChatClientDirect {
this.onAddText(data) this.onAddText(data)
} }
onReceiveGift (command) { sendGiftCallback (command) {
if (!this.onAddGift) { if (!this.onAddGift) {
return return
} }
@ -316,7 +362,7 @@ export default class ChatClientDirect {
this.onAddGift(data) this.onAddGift(data)
} }
async onBuyGuard (command) { async guardBuyCallback (command) {
if (!this.onAddMember) { if (!this.onAddMember) {
return return
} }
@ -332,7 +378,7 @@ export default class ChatClientDirect {
this.onAddMember(data) this.onAddMember(data)
} }
onSuperChat (command) { superChatMessageCallback (command) {
if (!this.onAddSuperChat) { if (!this.onAddSuperChat) {
return return
} }
@ -350,7 +396,7 @@ export default class ChatClientDirect {
this.onAddSuperChat(data) this.onAddSuperChat(data)
} }
onSuperChatDelete (command) { superChatMessageDeleteCallback (command) {
if (!this.onDelSuperChat) { if (!this.onDelSuperChat) {
return return
} }
@ -363,10 +409,10 @@ export default class ChatClientDirect {
} }
} }
const COMMAND_HANDLERS = { const CMD_CALLBACK_MAP = {
DANMU_MSG: ChatClientDirect.prototype.onReceiveDanmaku, DANMU_MSG: ChatClientDirect.prototype.danmuMsgCallback,
SEND_GIFT: ChatClientDirect.prototype.onReceiveGift, SEND_GIFT: ChatClientDirect.prototype.sendGiftCallback,
GUARD_BUY: ChatClientDirect.prototype.onBuyGuard, GUARD_BUY: ChatClientDirect.prototype.guardBuyCallback,
SUPER_CHAT_MESSAGE: ChatClientDirect.prototype.onSuperChat, SUPER_CHAT_MESSAGE: ChatClientDirect.prototype.superChatMessageCallback,
SUPER_CHAT_MESSAGE_DELETE: ChatClientDirect.prototype.onSuperChatDelete SUPER_CHAT_MESSAGE_DELETE: ChatClientDirect.prototype.superChatMessageDeleteCallback
} }