开启辅助访问 切换到宽版

精易论坛

 找回密码
 注册

QQ登录

只需一步,快速开始

用微信号发送消息登录论坛

新人指南 邀请好友注册 - 我关注人的新帖 教你赚取精币 - 每日签到


求职/招聘- 论坛接单- 开发者大厅

论坛版规 总版规 - 建议/投诉 - 应聘版主 - 精华帖总集 积分说明 - 禁言标准 - 有奖举报

查看: 64629|回复: 314
收起左侧

[易语言纯源码] [RFC 6455] WebSocket Client 模块纯源码实现以及分析

    [复制链接]
结帖率:100% (26/26)
发表于 2017-7-1 21:18:05 | 显示全部楼层 |阅读模式   安徽省池州市
分享源码
界面截图:
是否带模块: 纯源码
备注说明: -
[RFC 6455] WebSocket Client 模块纯源码实现以及分析


        WebSocket虽然很常见,但是我很少用到,源自某次群里的讨(吹)论(比),于是就实现了一下,一直没有整理代码,今天顺便写点分析,更多的也就是试着说一下自己是如何学习一个新接触的东西的。


一、简介
        网上的介绍相当多,我就简单地理解优势,相对于HTTP,服务端可以不需要客户Duan去主动请求就可以[推送]数据。所以在聊天室、客服、推送等场景中的应用特别多。但是归根到底,它和HTTP都是TCP。
        最权威的资料当然是RFC 6455 [The WebSocket Protocol],里面有各种标准定义,本源码以及分析也都是基于这个RFC:https://tools.ietf.org/html/rfc6455
        这也是我的学习习惯,尽量看原版的权威资料,翻译或者博客经常有很多谬误,容易被误导,反而耽误时间。

二、抓包
        ws:的抓包相当简单,任意一个可以抓取tcp数据的工具都可以,wss:则由于是SSL,都是密文,还是用网卡抓包工具的话分析起来就很麻烦。所以我是利用Chrome浏览器的开发者工具抓包。当然只是Opcode为Text的情况才可以利用这个来分析Frame信息,但是握手数据却是可以很简单看到的,二进制数据我没有做过,那大概就需要分析客户Duan代码(javascript)了。

抓包 1.jpg
抓包 2.jpg



三、握手
        RFC6455 4.1 Client Requirements :https://tools.ietf.org/html/rfc6455#section-4.1
        请求:
        下面就是一个普通的握手请求,标注星号的不是必须。
                 GET / HTTP/1.1
                 Accept-Language: zh-CN
                 Host: 121.40.165.18:8088//不是默认端口则跟上端口
                 Sec-WebSocket-Version: 13
                 Upgrade: websocket
                 Sec-WebSocket-Key: 2mzaxLsKR++Hp0c5q2ufwg==//随机16byte的base64编码
                 Connection: Upgrade
                *User-Agent:  Mozilla/5.0 (Windows NT 6.1; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/58.0.3029.96 Safari/537.36
                *Origin: http://121.40.165.18
                *Sec-WebSocket-Protocol:
                *Sec-WebSocket-Extensions:
                *Cookie:
                *Authorization:

        响应:
        下面就是一个普通的握手响应。
                HTTP/1.1 101 Switching Protocols
                Upgrade: websocket
                Connection: Upgrade
                Sec-WebSocket-Accept: Gk9vEODnOs/Kmjxy9leCLqw+9f8=

                并不是每个ws的握手的响应的状态码都是101,如果不是101,则根据RFC rfc2616  https://tools.ietf.org/html/rfc2616 来进行处理。如果状态码是401,则可能需要执行身份验证。

                Sec-WebSocket-Accept 应当为Sec-WebSocket-Key + “258EAFA5-E914-47DA-95CA-C5AB0DC85B1” 的SHA1值的base64

四丶数据包
        Base Framing Protocol:https://tools.ietf.org/html/rfc6455#section-5.2

        数据包构造和解析应该是WebSocket学习中最重要的部分了,其实也就是看一看RFC就出来了。
        下面我配合着RFC挨个解释下,也方便英文不太好的朋友:
                1 byte(字节) = 8 bit(位)  这种基础知识就不说了

snipaste20170701_201030.jpg


                Fin:1 bit
                        标识着这个 Frame 是不是一个消息中的最后一条
                        有的消息可能被拆成了若干个Frame
                RSV1, RSV2, RSV3: 每个都是 1 bit  
                        就是 Reserved(保留)的意思,在一些扩展协yi以外一般赋0。
                Opcode:4 bits
                        操作码,标识着 Payload Data 的类型:
                                OPCODE_CONTINUATION0继续
                                OPCODE_TEXT1文本
                                OPCODE_BINARY2二进制
                                OPCODE_CLOSE8关闭
                                OPCODE_PING9ping
                                OPCODE_PONG10pong
                        以及一堆保留的操作码
                Mask:1 bit
                        标识着是否需要进行掩码计算
                Payload length:7 或者 7+16 或者 7+64 bits
                        这儿是 Payload Data 的长度,要注意,这个长度不一样,占用的位数也不一样。
                        其实这个Frame的设计相当令人佩服,尽可能地减少空间,这才是网络传输中睿智的选择。
                        假设 DataLen 是 Payload Data 的长度,那么:
                                当 DataLen ≤ 125 的时候
                                        Payload length 占用的位数为 7 bits,直接表示了 DataLen 。
                        对位运算不熟悉的朋友可以下载源码看 EncryptData 和 DecryptData 部分的写法。
                                当 126 ≤ DataLen ≤ 65535 的时候
                                        Payload length 占用的位数为 7+16 bits,
                                        前 7 bits 置为126,紧跟的 16 bits (相当于易的短整数型,但是是无符号的。)来表示 DataLen
                                当 DataLen > 65535 的时候
                                        Payload length 占用的位数为 7+64 bits,
                                        前 7 bits 置为127,紧跟的 64 bits (相当于易的长整数型,但是是无符号的。)来表示 DataLen
                Masking-key:0 bit 或者 4 bytes
                        这儿是 掩码计算的Key。
                        如果 Mask 位为0 这儿就不需要 为 0 bit。否则则需要随机一个4字节的key。
                        严格按照规范的话,普通的随机数生成的key不合格,需要足够安全的随机数。
                Payload data:x+y  bytes
                        扩展:x bytes
                                在没有协商扩展的情况下,为0
                        应用:y bytes
                                例程中发送或者读取的也就是这儿的数据。

四丶Masked 掩码计算
        就是简单的异或,将那 4 bytes 的 Masking-key 每 byte 地和 Payload Data 进行异或运算。

五、异步
        异步当然是最适合WebSocket的写法,这个实现里为了符合自己的习惯,没有用异步。

六、wss
        wss和ws没有很大区别,但是需要SSL。
        由于个人洁癖不喜欢用第三方库,所以此处用到心冷丶鱼儿开源的用CryptoAPI实现的SSL。
        而且发给我一些SSL的资料,受益很多,此处感谢。

七、关于模块
        找来了一些在线聊天室的ws和wss进行了实现,写在了demo里。
        更多的兼容没有去做,但是我相信所有的疑问都可以在这个RFC6455中找到答案。
        调用方法也相当简单,
                Init → 传入URL
                HandShake → 握手
                Send → 发送
                Recv → 接收
                SetHeader → 设置协yi头


源码、资料以及本篇的doc下载 WebSocketClient.zip (1.39 MB, 下载次数: 7524)

点评

@兔子君   美国  发表于 2017-7-1 21:44
我只想咨询一下兔子君我不明白的地方,该帖刚申请精华帖评审,看到很多评审团成员还没有点评,为何就已经加精了?是否有其他直通渠道?   美国  发表于 2017-7-1 21:43

评分

参与人数 14好评 +13 精币 +128 收起 理由
皮皮驴 + 1 + 2 感谢发布原创作品,精易因你更精彩!
刘海斌 + 1 + 2 很棒!
精易客服 + 1 + 5 奉上小小红包希望笑纳
拉面 + 1 + 20 感谢分享,很给力!~
冯古屋 + 1 + 1 支持开源~!感谢分享
ANJIANTEST + 1 + 2 支持开源~!感谢分享
铅笔刀 + 1 + 2 厉害
落款hMZ + 1 + 2 支持开源~!感谢分享
大飛 + 1 + 8 支持开源~!感谢分享
AdGame + 2 支持开源~!感谢分享
单身汪 + 1 + 8 最近也在研究这个,学习一下
Hoibben + 1 + 11 很好很强悍,坚持下去哦~
兔子君 + 1 + 60 摸BB,撅PY
代码之殇 + 1 + 3 支持开源~!感谢分享

查看全部评分


本帖被以下淘专辑推荐:

结帖率:100% (16/16)
发表于 2017-7-1 21:56:54 | 显示全部楼层   四川省成都市
雪山凌狐 发表于 2017-7-1 21:41
来看一下,学习学习~

客服或者本版版主发现优秀帖可以直接加精无需申请。
对于没有申请的,客服或本版版主不定期巡查,发现优秀的也是直接加精无需申请。
回复 支持 反对

使用道具 举报

结帖率:25% (1/4)
发表于 2024-7-19 15:55:27 | 显示全部楼层   重庆市重庆市
感谢分享,很给力!~
回复 支持 反对

使用道具 举报

结帖率:88% (7/8)

签到天数: 5 天

发表于 2024-6-22 23:06:42 | 显示全部楼层   贵州省黔东南苗族侗族自治州
学习下。找了好多个都没满足我要的
回复 支持 反对

使用道具 举报

发表于 2024-5-30 12:42:07 | 显示全部楼层   四川省德阳市
我只想咨询一下兔子君我不明白的地方,该帖刚申请精华帖评审,
回复 支持 反对

使用道具 举报

结帖率:25% (1/4)

签到天数: 5 天

发表于 2024-4-15 20:53:57 | 显示全部楼层   山东省滨州市
感谢焚香呢亲
回复 支持 反对

使用道具 举报

发表于 2024-1-31 11:22:03 | 显示全部楼层   河南省新乡市
学习了大佬666
回复 支持 反对

使用道具 举报

发表于 2023-12-28 14:56:35 | 显示全部楼层   湖南省长沙市
回复 支持 反对

使用道具 举报

发表于 2023-11-5 07:54:06 | 显示全部楼层   江苏省常州市
来学习一下
回复 支持 反对

使用道具 举报

签到天数: 14 天

发表于 2023-10-14 00:06:53 | 显示全部楼层   北京市北京市
感谢分享
回复 支持 反对

使用道具 举报

结帖率:100% (12/12)

签到天数: 3 天

发表于 2023-9-25 14:31:24 | 显示全部楼层   江苏省镇江市
需要这个
回复 支持 反对

使用道具 举报

发表于 2023-9-3 16:48:24 | 显示全部楼层   四川省凉山彝族自治州
厉害,谢谢分享
回复 支持 反对

使用道具 举报

您需要登录后才可以回帖 登录 | 注册

本版积分规则 致发广告者

发布主题 收藏帖子 返回列表

sitemap| 易语言源码| 易语言教程| 易语言论坛| 易语言模块| 手机版| 广告投放| 精易论坛
拒绝任何人以任何形式在本论坛发表与中华人民共和国法律相抵触的言论,本站内容均为会员发表,并不代表精易立场!
论坛帖子内容仅用于技术交流学习和研究的目的,严禁用于非法目的,否则造成一切后果自负!如帖子内容侵害到你的权益,请联系我们!
防范网络诈骗,远离网络犯罪 违法和不良信息举报电话0663-3422125,QQ: 793400750,邮箱:wp@125.la
Powered by Discuz! X3.4 揭阳市揭东区精易科技有限公司 ( 粤ICP备12094385号-1) 粤公网安备 44522102000125 增值电信业务经营许可证 粤B2-20192173

快速回复 返回顶部 返回列表