深入研究websocket直播中signature这个参数怎么来的,模拟自己生成一个

news2024/12/26 0:42:32

上一节课我们已经找到了生成signature这个字段的代码位置,就是这个B函数,嗯......听起来好像有点奇怪,但是它确实叫B啊,笑死。不管了,看一下里面的逻辑是啥。

注意e参数的内容是:

{
    "app_name": "douyin_web",
    "version_code": "180800",
    "webcast_sdk_version": "1.0.14-beta.0",
    "update_version_code": "1.0.14-beta.0",
    "compress": "gzip",
    "device_platform": "web",
    "cookie_enabled": true,
    "screen_width": 1512,
    "screen_height": 982,
    "browser_language": "zh-CN",
    "browser_platform": "MacIntel",
    "browser_name": "Mozilla",
    "browser_version": "5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/126.0.0.0 Safari/537.36",
    "browser_online": true,
    "tz_name": "Asia/Shanghai",
    "cursor": "t-1718942296076_r-1_d-1_u-1_h-7382800685396382772",
    "internal_ext": "internal_src:dim|wss_push_room_id:7382777844734167858|wss_push_did:7347516590731134502|first_req_ms:1718942295989|fetch_time:1718942296076|seq:1|wss_info:0-1718942296076-0-0|wrds_v:7382800932146251064",
    "host": "https://live.douyin.com",
    "aid": "6383",
    "live_id": 1,
    "did_rule": 3,
    "endpoint": "live_pc",
    "support_wrds": 1,
    "user_unique_id": "7347516590731134502",
    "im_path": "/webcast/im/fetch/",
    "identity": "audience",
    "need_persist_msg_count": "15",
    "insert_task_id": "",
    "live_reason": "",
    "room_id": "7382777844734167858",
    "heartbeatDuration": "0"
}

注意t是很多参数的e里面的websocket_key数组,它里面是:

[
            {
                param_name: 'live_id',
                param_type: 'string',
            },
            {
                param_name: 'aid',
                param_type: 'string',
            },
            {
                param_name: 'version_code',
                param_type: 'string',
            },
            {
                param_name: 'webcast_sdk_version',
                param_type: 'string',
            },
            {
                param_name: 'room_id',
                param_type: 'string',
            },
            {
                param_name: 'sub_room_id',
                param_type: 'string',
            },
            {
                param_name: 'sub_channel_id',
                param_type: 'string',
            },
            {
                param_name: 'did_rule',
                param_type: 'string',
            },
            {
                param_name: 'user_unique_id',
                param_type: 'string',
            },
            {
                param_name: 'device_platform',
                param_type: 'string',
            },
            {
                param_name: 'device_type',
                param_type: 'string',
            },
            {
                param_name: 'ac',
                param_type: 'string',
            },
            {
                param_name: 'identity',
                param_type: 'string',
            },
        ]

有了这两个参数传递过来,那我们就可以安心研究B里面的逻辑了。

看一下这个for循环吧,它的逻辑就是取出e里面的参数(将t里面的参数名称),然后拼接到o这个字符串上,然后再把o传递给V()这个函数:

V()这个函数里面又做了什么事情呢?V()其实会返回一个函数,这个函数可以传递两个参数,

接下来继续看返回的这个函数里面代码逻辑: 

其实这里继续深入研究,会发现是把e参数转为Bytes数组了:

这个转换函数也可以自己写一个:

        const o =
            ',live_id=1,aid=6383,version_code=180800,webcast_sdk_version=1.0.14-beta.0,room_id=7382777844734167858,sub_room_id=,sub_channel_id=,did_rule=3,user_unique_id=7347516590731134502,device_platform=web,device_type=,ac=,identity=audience'
        const substr = o.substring(1)
        console.log('subStr----', substr)

        // 将字符串转Bytes数组
        const stringToBytes = (str) => {
            var array = new Uint8Array(str.length)
            for (var i = 0, l = str.length; i < l; i++) {
                array[i] = str.charCodeAt(i)
            }
            return array
        }

        console.log('字符串转为Bytes数组', stringToBytes(substr))

转换完之后,又使用wordsToBytes函数将结果转成了另外一个形式:

然后再调用bytesToHex函数:

const bytesToHex = function (e) {
    for (var t = [], r = 0; r < e.length; r++)
        t.push((e[r] >>> 4).toString(16)), t.push((15 & e[r]).toString(16))
    return t.join('')
}

转换之后的结果格式为:a5faced0e2965a966b9fde2044e3ff1e

然后再调用frontierSign函数将上面这串字符串转为signature的值:

但是这个frontierSign是啥呢?这是一个webmssdk.es5.js包里面的函数,所以需要将这个webmssdk.es5.js包下载到本地,然后集成到window对象上,就可以调用这个函数了,我这里写了一个demo:可以看到已经生成了值

demo代码如下:

<!DOCTYPE html>
<html lang="en">
    <head>
        <meta charset="UTF-8" />
        <meta name="viewport" content="width=device-width, initial-scale=1.0" />
        <title>Document</title>
        <script src="./vFun.js"></script>
        <script src="./webmssdk.es5.js"></script>
    </head>
    <body>
        <div>测试代码</div>
    </body>
    <script>
        const config = [
            {
                param_name: 'live_id',
                param_type: 'string',
            },
            {
                param_name: 'aid',
                param_type: 'string',
            },
            {
                param_name: 'version_code',
                param_type: 'string',
            },
            {
                param_name: 'webcast_sdk_version',
                param_type: 'string',
            },
            {
                param_name: 'room_id',
                param_type: 'string',
            },
            {
                param_name: 'sub_room_id',
                param_type: 'string',
            },
            {
                param_name: 'sub_channel_id',
                param_type: 'string',
            },
            {
                param_name: 'did_rule',
                param_type: 'string',
            },
            {
                param_name: 'user_unique_id',
                param_type: 'string',
            },
            {
                param_name: 'device_platform',
                param_type: 'string',
            },
            {
                param_name: 'device_type',
                param_type: 'string',
            },
            {
                param_name: 'ac',
                param_type: 'string',
            },
            {
                param_name: 'identity',
                param_type: 'string',
            },
        ]

        // 使用for便利试试
        for (let { param_name: i } of config) {
            console.log('i----', i)
        }
        const o =
            ',live_id=1,aid=6383,version_code=180800,webcast_sdk_version=1.0.14-beta.0,room_id=7382772251994655488,sub_room_id=,sub_channel_id=,did_rule=3,user_unique_id=7347516590731134502,device_platform=web,device_type=,ac=,identity=audience'
        const substr = o.substring(1)
        console.log('subStr----', substr)
        // s函数就是stringToBytes
        const sResult = sFunc(substr)
        //  V()函数就是
        console.log('s函数stringToBytes结果', sResult)
        // 有了s的返回结果,再调用i.wordsToBytes
        // var r = i.wordsToBytes(s(e, t));
        const r = wordsToBytes(sResult)
        console.log('r----', r)
        // 最后调用bytesToHex;
        // return t && t.asBytes ? r : t && t.asString ? a.bytesToString(r) : i.bytesToHex(r)
        const bytesRes = bytesToHex(r)
        console.log('bytesRes----', bytesRes)
        const frontierSignRes = window.byted_acrawler.frontierSign({
            'X-MS-STUB': bytesRes,
        })
        console.log('frontierSignRes----', frontierSignRes)
    </script>
</html>

vFun.js的代码如下:

var sFunc = function (e, t) {
    // 判断e是不是string类型,是的话,把t赋值给e,然后
    e.constructor == String
        ? stringToBytes(e)
        : oFunc(e)
        ? (e = Array.prototype.slice.call(e, 0))
        : Array.isArray(e) || e.constructor === Uint8Array || (e = e.toString())
    for (
        var r = bytesToWords(e),
            l = 8 * e.length,
            c = 1732584193,
            u = -271733879,
            p = -1732584194,
            d = 271733878,
            h = 0;
        h < r.length;
        h++
    )
        r[h] =
            (((r[h] << 8) | (r[h] >>> 24)) & 16711935) |
            (((r[h] << 24) | (r[h] >>> 8)) & 4278255360)
    ;(r[l >>> 5] |= 128 << l % 32), (r[(((l + 64) >>> 9) << 4) + 14] = l)
    for (var m = sff, f = sgg, g = shh, _ = sii, h = 0; h < r.length; h += 16) {
        var v = c,
            C = u,
            y = p,
            T = d
        ;(c = m(c, u, p, d, r[h + 0], 7, -680876936)),
            (d = m(d, c, u, p, r[h + 1], 12, -389564586)),
            (p = m(p, d, c, u, r[h + 2], 17, 606105819)),
            (u = m(u, p, d, c, r[h + 3], 22, -1044525330)),
            (c = m(c, u, p, d, r[h + 4], 7, -176418897)),
            (d = m(d, c, u, p, r[h + 5], 12, 1200080426)),
            (p = m(p, d, c, u, r[h + 6], 17, -1473231341)),
            (u = m(u, p, d, c, r[h + 7], 22, -45705983)),
            (c = m(c, u, p, d, r[h + 8], 7, 1770035416)),
            (d = m(d, c, u, p, r[h + 9], 12, -1958414417)),
            (p = m(p, d, c, u, r[h + 10], 17, -42063)),
            (u = m(u, p, d, c, r[h + 11], 22, -1990404162)),
            (c = m(c, u, p, d, r[h + 12], 7, 1804603682)),
            (d = m(d, c, u, p, r[h + 13], 12, -40341101)),
            (p = m(p, d, c, u, r[h + 14], 17, -1502002290)),
            (u = m(u, p, d, c, r[h + 15], 22, 1236535329)),
            (c = f(c, u, p, d, r[h + 1], 5, -165796510)),
            (d = f(d, c, u, p, r[h + 6], 9, -1069501632)),
            (p = f(p, d, c, u, r[h + 11], 14, 643717713)),
            (u = f(u, p, d, c, r[h + 0], 20, -373897302)),
            (c = f(c, u, p, d, r[h + 5], 5, -701558691)),
            (d = f(d, c, u, p, r[h + 10], 9, 38016083)),
            (p = f(p, d, c, u, r[h + 15], 14, -660478335)),
            (u = f(u, p, d, c, r[h + 4], 20, -405537848)),
            (c = f(c, u, p, d, r[h + 9], 5, 568446438)),
            (d = f(d, c, u, p, r[h + 14], 9, -1019803690)),
            (p = f(p, d, c, u, r[h + 3], 14, -187363961)),
            (u = f(u, p, d, c, r[h + 8], 20, 1163531501)),
            (c = f(c, u, p, d, r[h + 13], 5, -1444681467)),
            (d = f(d, c, u, p, r[h + 2], 9, -51403784)),
            (p = f(p, d, c, u, r[h + 7], 14, 1735328473)),
            (u = f(u, p, d, c, r[h + 12], 20, -1926607734)),
            (c = g(c, u, p, d, r[h + 5], 4, -378558)),
            (d = g(d, c, u, p, r[h + 8], 11, -2022574463)),
            (p = g(p, d, c, u, r[h + 11], 16, 1839030562)),
            (u = g(u, p, d, c, r[h + 14], 23, -35309556)),
            (c = g(c, u, p, d, r[h + 1], 4, -1530992060)),
            (d = g(d, c, u, p, r[h + 4], 11, 1272893353)),
            (p = g(p, d, c, u, r[h + 7], 16, -155497632)),
            (u = g(u, p, d, c, r[h + 10], 23, -1094730640)),
            (c = g(c, u, p, d, r[h + 13], 4, 681279174)),
            (d = g(d, c, u, p, r[h + 0], 11, -358537222)),
            (p = g(p, d, c, u, r[h + 3], 16, -722521979)),
            (u = g(u, p, d, c, r[h + 6], 23, 76029189)),
            (c = g(c, u, p, d, r[h + 9], 4, -640364487)),
            (d = g(d, c, u, p, r[h + 12], 11, -421815835)),
            (p = g(p, d, c, u, r[h + 15], 16, 530742520)),
            (u = g(u, p, d, c, r[h + 2], 23, -995338651)),
            (c = _(c, u, p, d, r[h + 0], 6, -198630844)),
            (d = _(d, c, u, p, r[h + 7], 10, 1126891415)),
            (p = _(p, d, c, u, r[h + 14], 15, -1416354905)),
            (u = _(u, p, d, c, r[h + 5], 21, -57434055)),
            (c = _(c, u, p, d, r[h + 12], 6, 1700485571)),
            (d = _(d, c, u, p, r[h + 3], 10, -1894986606)),
            (p = _(p, d, c, u, r[h + 10], 15, -1051523)),
            (u = _(u, p, d, c, r[h + 1], 21, -2054922799)),
            (c = _(c, u, p, d, r[h + 8], 6, 1873313359)),
            (d = _(d, c, u, p, r[h + 15], 10, -30611744)),
            (p = _(p, d, c, u, r[h + 6], 15, -1560198380)),
            (u = _(u, p, d, c, r[h + 13], 21, 1309151649)),
            (c = _(c, u, p, d, r[h + 4], 6, -145523070)),
            (d = _(d, c, u, p, r[h + 11], 10, -1120210379)),
            (p = _(p, d, c, u, r[h + 2], 15, 718787259)),
            (u = _(u, p, d, c, r[h + 9], 21, -343485551)),
            (c = (c + v) >>> 0),
            (u = (u + C) >>> 0),
            (p = (p + y) >>> 0),
            (d = (d + T) >>> 0)
    }
    return endian([c, u, p, d])
}

// (s._blocksize = 16)
// (s._digestsize = 16)

const sff = function (e, t, r, i, n, o, a) {
    var s = e + ((t & r) | (~t & i)) + (n >>> 0) + a
    return ((s << o) | (s >>> (32 - o))) + t
}

const sgg = function (e, t, r, i, n, o, a) {
    var s = e + ((t & i) | (r & ~i)) + (n >>> 0) + a
    return ((s << o) | (s >>> (32 - o))) + t
}

const shh = function (e, t, r, i, n, o, a) {
    var s = e + (t ^ r ^ i) + (n >>> 0) + a
    return ((s << o) | (s >>> (32 - o))) + t
}

const sii = function (e, t, r, i, n, o, a) {
    var s = e + (r ^ (t | ~i)) + (n >>> 0) + a
    return ((s << o) | (s >>> (32 - o))) + t
}

const stringToBytes = function (str) {
    var array = new Uint8Array(str.length)
    for (var i = 0, l = str.length; i < l; i++) {
        array[i] = str.charCodeAt(i)
    }
    return array
}

const oFunc = function (e) {
    return (
        null != e &&
        (t(e) ||
            ('function' == typeof e.readFloatLE &&
                'function' == typeof e.slice &&
                t(e.slice(0, 0))) ||
            !!e._isBuffer)
    )
}

const bytesToWords = function (e) {
    for (var t = [], r = 0, i = 0; r < e.length; r++, i += 8)
        t[i >>> 5] |= e[r] << (24 - (i % 32))
    return t
}

const wordsToBytes = function (e) {
    for (var t = [], r = 0; r < 32 * e.length; r += 8)
        t.push((e[r >>> 5] >>> (24 - (r % 32))) & 255)
    return t
}

const endian = function (e) {
    if (e.constructor == Number)
        return (16711935 & rotl(e, 8)) | (4278255360 & rotl(e, 24))
    for (var t = 0; t < e.length; t++) e[t] = endian(e[t])
    return e
}

const rotl = function (e, t) {
    return (e << t) | (e >>> (32 - t))
}

const bytesToHex = function (e) {
    for (var t = [], r = 0; r < e.length; r++)
        t.push((e[r] >>> 4).toString(16)), t.push((15 & e[r]).toString(16))
    return t.join('')
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1846525.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

PS系统教学24

去水印 仿制图章工具修复画笔工具选区-编辑-填充-内容识别 人体皮肤亮度 人体皮肤发亮 减淡工具 缺点&#xff1a;不能对所有图层取样 选择减淡工具选择高光打完高光用中间调中和一下也可以用历史记录画笔工具进行修饰也可以用加深工具进行微调 图层中的模式 滤色 曝光过…

go语言day2

使用cmd 中的 go install &#xff1b; go build 命令出现 go cannot find main module 错误怎么解决&#xff1f; go学习-问题记录(开发环境)go: cannot find main module&#xff1b; see ‘go help modules‘_go: no flags specified (see go help mod edit)-CSDN博客 在本…

docker环境部署ruoyi系统前后端分离项目

创建局域网 docker network create net-ry 安装Redis 1 安装 创建两个目录 mkdir -p /data/redis/{conf,data} 上传redis.conf文件到/data/redis/conf文件夹中 cd /data/redis/conf 3.2 配置redis.conf文件 配置redis.conf文件&#xff1a; redis.conf文件配置注意&…

函数栈帧的创建和销毁,带动图详细解析,带你大致分析汇编代码

目录 1.什么是函数栈帧 2.理解函数栈帧有什么用&#xff1f; 3.函数栈帧的创建和销毁解析 3.1什么是栈&#xff1f; 3.2 认识相关寄存器和汇编指令 3.3函数栈帧的创建和销毁解析过程 3.4函数的调用 3.5汇编代码 3.5.1函数栈帧的创建 3.5.2main函数部分 3.5.3Add函数…

无线模块433MHz和2.4GHz的功能与适用性比较

433MHz和2.4GHz这两个频段常用于无线通信中的模块&#xff0c;今天我们就来介绍这两种频段无线模块各自的特点。433MHz和2.4GHz无线模块工作频段都属于国内免许可的ISM开放频段&#xff0c;因此二者使用较为广泛。 433MHz频段无线模块位于超高频(UHF)范围内&#xff0c;具体频…

人类启发的一区新算法|旅行徒步优化算法HOA原理及代码实现(Matlab/Python)

文章来源于我的个人公众号&#xff1a;KAU的云实验台&#xff0c;主要更新智能优化算法的原理、应用、改进 MATLAB PYTHON 目前&#xff0c;元启发式算法按其灵感来源可以分为&#xff1a; &#xff08;i&#xff09;群体启发&#xff0c;如粒子群PSO和灰狼优化算法GWO等&a…

wvp-GB28181-pro 源码分析-点播流程(三)

文章目录 一 、28181-2016标准文档中的点播流程二 、点播流程源码分析2.1 页面发起点播请求2.2 与ZLM协商SSRC信息2.3 订阅zlmediakit的hook消息及发送invite信令2.4 处理invite信令响应并应答2.5 收到ZLM的推流通知2.6 播放成功2.7 停止点播流程2024年6月20日下载的wvp-GB2818…

如何在华为 Ascend 设备上运行模型

模型转换:使用华为的模型转换工具 ATC ATC 在 ascend-cann-toolkit 包里 环境 Docker Image: ascendhub.huawei.com/public-ascendhub/ascend-pytorch:24.0.RC1-A2-2.1.0-ubuntu20.04 镜像版本CANN版本Pytorch版本变更项24.0.RC18.0.RC12.1.0基础镜像变更为 ubuntu20.04。p…

SAP ScreenPersonas

https://developers.sap.com/mission.screen-personas.html 跟着这个练习做一遍就了解了Personas 访问SAP提供的Personas练习系统 申请用户 登录练习系统 随便找一个可以支持Personas的程序搞起来&#xff0c;比如IW51 执行后等它出现这个图标就可以开始了.

Rocky9使用cockpitweb登陆时root用户无法登陆

Rocky9使用cockpitweb登陆时root用户无法登陆 [rootlvs ~]# vim /etc/cockpit/disallowed-users [rootlvs ~]# systemctl restart cockpit 取消disallowed-users中的root&#xff0c;即可访问 ip:9090 登陆。

模式分解的概念(下)-无损连接分解的与保持函数依赖分解的定义和判断、损失分解

一、无损连接分解 1、定义 2、检验一个分解是否是无损连接分解的算法 输入与输出 输入&#xff1a; 关系模式R&#xff08;U&#xff0c;F&#xff09;&#xff0c;F是最小函数依赖集 R上的一个分解 输出&#xff1a; 判断分解是否为无损连接分解 &#xff08;1&#x…

leetcode 动态规划(基础版)三角形最小路径和

题目&#xff1a; 题解&#xff1a; 一种可行的方案是从下到上&#xff0c;避免了从上到下的下标特判。走到每一个位置的最小值等于该位置的上两个位置中的最小值加上该位置的值。 int minimumTotal(vector<vector<int>>& triangle) {int dp[205][205]{0};f…

Origin电源维修高压发生器GEXUS-3 GEXUS-15R-02U

GEXUS-3电源维修 Origin高压发生器维修 Origin电源维修 光电设备电源维修 Origin高压电源维修故障分析应注意两点&#xff1a; 故障分析检测和故障硬件更换&#xff0c;根据高压电源故障提示和工作表现初步判断故障的类型和哪些硬件出了问题&#xff0c;再根据初步判断缩小检测…

rk3588 cpu npu gpu 定频 变频

网上的资料 cpu: npu&#xff1a; GPU: DDR:

现货黄金应用价格行为交易所需要的环境

在现货黄金投资中&#xff0c;投资者常用价格行为交易法来分析走势。简单来说&#xff0c;这种方法就是只看K线和支撑阻力位&#xff0c;顶多加一些简单的指标&#xff0c;以此构建分析和交易的系统。由于价格行为简单易学&#xff0c;现在的投资者或多或少都在使用这个方法。但…

云服务器可以从哪些方面降低开发运维难度

开发和运维工作面临着诸多挑战&#xff0c;如果说现在市场上有可以快速有效解决的方案&#xff0c;那么云服务器绝对是首选&#xff0c;云服务器从多个方面显著降低了其难度。 具象到云服务器的特质中&#xff0c;不得不提的还是云服务器的弹性伸缩&#xff0c;之前的文章里有…

busybox的基本使用记录壹

内核如何启动init进程 init/main.c static int __ref kernel_init(void *unused) {int ret;kernel_init_freeable();/* need to finish all async __init code before freeing the memory */async_synchronize_full();ftrace_free_init_mem();jump_label_invalidate_initmem()…

docker进阶篇

docker进阶篇 (重点) 1、docker run2、docker ps3、docker restart4、docker pull5、docker stop6、docker logs7、docker stats8、docker rm(重点) 9、docker exec10、查看本机镜像docker image11、发布镜像docker commit12、docker save(例如保存为tar给别人用)13、删除镜像1…

家居行业大洗牌,中国品牌“九牧现象”逆势进阶

两件印字白T&#xff0c;一件可能几十块就能买到&#xff0c;Gucci的售价却可达上千元&#xff1b;款式类似的珍珠耳环&#xff0c;有的可能上百元就能买到&#xff0c;香奈儿的却要过万…… 他们最大的差异在哪里&#xff1f;在品牌&#xff0c;以及品牌背后代表的品质与服务…

ArcGIS实现不同地块分类与面积汇总

​ 点击下方全系列课程学习 点击学习—>ArcGIS全系列实战视频教程——9个单一课程组合系列直播回放 点击学习——>遥感影像综合处理4大遥感软件ArcGISENVIErdaseCognition 我们要做一个不同地块面积汇总&#xff01; 你有一批地块&#xff0c;不同面积&#xff0c;我们需…