一、自己实现websocket

网上流传的都是Python2的websocket实现

# coding=utf8
# !/usr/bin/python import struct, socket
import hashlib
import threading, random
import time
import struct
from base64 import b64encode, b64decode connectionlist = {}
g_code_length = 0
g_header_length = 0 def hex2dec(string_num):
return str(int(string_num.upper(), 16)) def get_datalength(msg):
global g_code_length
global g_header_length print(len(msg))
g_code_length = ord(msg[1]) & 127
received_length = 0;
if g_code_length == 126:
# g_code_length = msg[2:4]
# g_code_length = (ord(msg[2])<<8) + (ord(msg[3]))
g_code_length = struct.unpack('>H', str(msg[2:4]))[0]
g_header_length = 8
elif g_code_length == 127:
# g_code_length = msg[2:10]
g_code_length = struct.unpack('>Q', str(msg[2:10]))[0]
g_header_length = 14
else:
g_header_length = 6
g_code_length = int(g_code_length)
return g_code_length def parse_data(msg):
global g_code_length
g_code_length = ord(msg[1]) & 127
received_length = 0;
if g_code_length == 126:
g_code_length = struct.unpack('>H', str(msg[2:4]))[0]
masks = msg[4:8]
data = msg[8:]
elif g_code_length == 127:
g_code_length = struct.unpack('>Q', str(msg[2:10]))[0]
masks = msg[10:14]
data = msg[14:]
else:
masks = msg[2:6]
data = msg[6:] i = 0
raw_str = '' for d in data:
raw_str += chr(ord(d) ^ ord(masks[i % 4]))
i += 1 print(u"总长度是:%d" % int(g_code_length))
return raw_str def sendMessage(message):
global connectionlist message_utf_8 = message.encode('utf-8')
for connection in connectionlist.values():
back_str = []
back_str.append('\x81')
data_length = len(message_utf_8) if data_length <= 125:
back_str.append(chr(data_length))
elif data_length <= 65535:
back_str.append(struct.pack('b', 126))
back_str.append(struct.pack('>h', data_length))
# back_str.append(chr(data_length >> 8))
# back_str.append(chr(data_length & 0xFF))
# a = struct.pack('>h', data_length)
# b = chr(data_length >> 8)
# c = chr(data_length & 0xFF)
elif data_length <= (2 ^ 64 - 1):
# back_str.append(chr(127))
back_str.append(struct.pack('b', 127))
back_str.append(struct.pack('>q', data_length))
# back_str.append(chr(data_length >> 8))
# back_str.append(chr(data_length & 0xFF))
else:
print(u'太长了')
msg = ''
for c in back_str:
msg += c
back_str = str(msg) + message_utf_8 # .encode('utf-8')
# connection.send(str.encode(str(u"\x00%s\xFF\n\n" % message))) #这个是旧版
# print (u'send message:' + message)
if back_str != None and len(back_str) > 0:
print(back_str, '$backstr')
print(','.join([str(ord(i)) for i in back_str]))
connection.send(back_str) def deleteconnection(item):
global connectionlist
del connectionlist['connection' + item] class WebSocket(threading.Thread): # 继承Thread GUID = "258EAFA5-E914-47DA-95CA-C5AB0DC85B11" def __init__(self, conn, index, name, remote, path="/"):
threading.Thread.__init__(self) # 初始化父类Thread
self.conn = conn
self.index = index
self.name = name
self.remote = remote
self.path = path
self.buffer = ""
self.buffer_utf8 = ""
self.length_buffer = 0 def run(self): # 重载Thread的run
print('Socket%s Start!' % self.index)
headers = {}
self.handshaken = False while True:
if self.handshaken == False:
print('Socket%s Start Handshaken with %s!' % (self.index, self.remote))
self.buffer += bytes.decode(self.conn.recv(1024)) if self.buffer.find('\r\n\r\n') != -1:
header, data = self.buffer.split('\r\n\r\n', 1)
for line in header.split("\r\n")[1:]:
key, value = line.split(": ", 1)
headers[key] = value headers["Location"] = ("ws://%s%s" % (headers["Host"], self.path))
key = headers['Sec-WebSocket-Key']
token = b64encode(hashlib.sha1(str.encode(str(key + self.GUID))).digest()) handshake = "HTTP/1.1 101 Switching Protocols\r\n" \
"Upgrade: websocket\r\n" \
"Connection: Upgrade\r\n" \
"Sec-WebSocket-Accept: " + bytes.decode(token) + "\r\n" \
"WebSocket-Origin: " + str(
headers["Origin"]) + "\r\n" \
"WebSocket-Location: " + str(headers["Location"]) + "\r\n\r\n" self.conn.send(str.encode(str(handshake)))
self.handshaken = True
print('Socket %s Handshaken with %s success!' % (self.index, self.remote))
sendMessage(u'Welcome, ' + self.name + ' !')
self.buffer_utf8 = ""
g_code_length = 0 else:
global g_code_length
global g_header_length
mm = self.conn.recv(128)
if len(mm) <= 0:
continue
if g_code_length == 0:
get_datalength(mm)
# 接受的长度
self.length_buffer = self.length_buffer + len(mm)
self.buffer = self.buffer + mm
if self.length_buffer - g_header_length < g_code_length:
continue
else:
self.buffer_utf8 = parse_data(self.buffer) # utf8
msg_unicode = str(self.buffer_utf8).decode('utf-8', 'ignore') # unicode
if msg_unicode == 'quit':
print(u'Socket%s Logout!' % (self.index))
nowTime = time.strftime('%H:%M:%S', time.localtime(time.time()))
sendMessage(u'%s %s say: %s' % (nowTime, self.remote, self.name + ' Logout'))
deleteconnection(str(self.index))
self.conn.close()
break # 退出线程
else:
# print (u'Socket%s Got msg:%s from %s!' % (self.index, msg_unicode, self.remote))
nowTime = time.strftime(u'%H:%M:%S', time.localtime(time.time()))
sendMessage(u'%s %s say: %s' % (nowTime, self.remote, msg_unicode))
# 重置buffer和bufferlength
self.buffer_utf8 = ""
self.buffer = ""
g_code_length = 0
self.length_buffer = 0
self.buffer = "" class WebSocketServer(object):
def __init__(self):
self.socket = None def begin(self):
print('WebSocketServer Start!')
self.socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
self.socket.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
self.socket.bind(("127.0.0.1", 12345))
self.socket.listen(50) global connectionlist i = 0
while True:
connection, address = self.socket.accept() username = address[0]
newSocket = WebSocket(connection, i, username, address)
newSocket.start() # 开始线程,执行run函数
connectionlist['connection' + str(i)] = connection
i = i + 1 if __name__ == "__main__":
server = WebSocketServer()
server.begin()

python2前端

<!DOCTYPE html>
</html>
<head>
<meta charset="utf-8">
</head>
<body>
<h3>WebSocketTest</h3>
<div id="login">
<div>
<input id="serverIP" type="text" placeholder="服务器IP" value="127.0.0.1" autofocus="autofocus"/>
<input id="serverPort" type="text" placeholder="服务器端口" value="5000"/>
<input id="btnConnect" type="button" value="连接" onclick="connect()"/>
</div>
<div>
<input id="sendText" type="text" placeholder="发送文本" value="I'm WebSocket Client!"/>
<input id="btnSend" type="button" value="发送" onclick="send()"/>
</div>
<div>
<div>
来自服务端的消息
</div>
<textarea id="txtContent" cols="50" rows="10" readonly="readonly"></textarea>
</div>
</div>
</body>
<script>
var socket; function connect() {
var host = "ws://" + $("serverIP").value + ":" + $("serverPort").value + "/"
socket = new WebSocket(host);
try {
socket.onopen = function (msg) {
$("btnConnect").disabled = true;
alert("连接成功!");
}; socket.onmessage = function (msg) {
if (typeof msg.data == "string") {
displayContent(msg.data);
}
else {
alert("非文本消息");
}
}; socket.onclose = function (msg) {
alert("socket closed!")
};
socket.onerror = function (msg) {
alert("socket error" + msg)
}
}
catch (ex) {
log(ex);
}
} function send() {
var msg = $("sendText").value
socket.send(msg);
} window.onbeforeunload = function () {
try {
socket.close();
socket = null;
}
catch (ex) {
}
}; function $(id) {
return document.getElementById(id);
} Date.prototype.Format = function (fmt) { //author: meizz
var o = {
"M+": this.getMonth() + 1, //月份
"d+": this.getDate(), //日
"h+": this.getHours(), //小时
"m+": this.getMinutes(), //分
"s+": this.getSeconds(), //秒
"q+": Math.floor((this.getMonth() + 3) / 3), //季度
"S": this.getMilliseconds() //毫秒
};
if (/(y+)/.test(fmt)) fmt = fmt.replace(RegExp.$1, (this.getFullYear() + "").substr(4 - RegExp.$1.length));
for (var k in o)
if (new RegExp("(" + k + ")").test(fmt)) fmt = fmt.replace(RegExp.$1, (RegExp.$1.length == 1) ? (o[k]) : (("00" + o[k]).substr(("" + o[k]).length)));
return fmt;
} function displayContent(msg) {
$("txtContent").value += "\r\n" + new Date().Format("yyyy/MM/dd hh:mm:ss") + ": " + msg;
}
function onkey(event) {
if (event.keyCode == 13) {
send();
}
}
</script>
</html>

Python3的实现服务器发送消息有点问题

import base64
import hashlib
import socket
import threading
import traceback """
基于socket协议手写Websocket协议
"""
clients = {} # 通知客户端,广播,这个函数有点问题
def notify(message):
if len(message) > 127:
print('message too long')
return
print(len(clients), 'len(clients)')
for connection in list(clients.values()):
print('before')
content = bytearray(message, 'utf8')
bs = bytearray([0x81])
bs.extend(content)
print(bytes(bs))
# connection.send(bytes(bs))
connection.send(bytes(b'\x81\x05hello')) # 客户端处理线程,每个客户端都启动一个线程
class websocket_thread(threading.Thread):
def __init__(self, connection, username):
super(websocket_thread, self).__init__()
self.connection = connection
self.username = username def run(self):
print('new websocket client joined!')
data = self.connection.recv(1024)
# 解析用户请求的头部,获取sec-websocket-key,并对其进行sha1、base64
headers = self.parse_headers(data)
token = self.generate_token(headers['Sec-WebSocket-Key'])
headers = "HTTP/1.1 101 WebSocket Protocol Hybi-10\r\n" \
"Upgrade: WebSocket\r\n" \
"Connection: Upgrade\r\n" \
"Sec-WebSocket-Accept: %s\r\n\r\n" % token
self.connection.send(headers.encode('ascii'))
print('loop start')
while True:
try:
data = self.connection.recv(1024)
except socket.error as e:
traceback.print_exc(e)
print("unexpected error: ", e)
clients.pop(self.username)
break
if len(data) == 0:
continue
data = self.parse_data(data)
message = self.username + ": " + data
notify(message)
print(message) # 解析用户请求中的数据
def parse_data(self, msg):
print(msg, type(msg), len(msg))
v = msg[1] & 0x7f
if v == 0x7e:
p = 4
elif v == 0x7f:
p = 10
else:
p = 2
mask = msg[p:p + 4]
data = msg[p + 4:]
ans = bytearray([v ^ mask[k % 4] for k, v in enumerate(data)])
ans = bytes(ans).decode('utf8')
return ans def parse_headers(self, msg):
headers = {}
msg = msg.decode('ascii')
header, data = msg.split('\r\n\r\n', 1)
for line in header.split('\r\n')[1:]:
key, value = line.split(': ', 1)
headers[key] = value
headers['data'] = data
return headers def generate_token(self, msg):
# 下面这个字符串是websocket的magicstring
key = msg + '258EAFA5-E914-47DA-95CA-C5AB0DC85B11'
ser_key = hashlib.sha1(key.encode('ascii')).digest()
return base64.encodebytes(ser_key).decode('ascii') # 服务端
class websocket_server(threading.Thread):
def __init__(self, port):
super(websocket_server, self).__init__()
self.port = port def run(self):
sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
sock.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
sock.bind(('127.0.0.1', self.port))
sock.listen(5)
print('websocket server started!')
while True:
connection, address = sock.accept()
try:
username = "ID" + str(address[1])
thread = websocket_thread(connection, username)
thread.start()
clients[username] = connection
except socket.timeout:
print('websocket connection timeout!') if __name__ == '__main__':
server = websocket_server(5000)
server.start()

二、Flask关于websocket的库

Flask关于websocket有两个库:

  • flask-sockets
  • flask-socketIO

这两个库的官方例子我都没有运行成功。打开浏览器一直显示正在加载,不知何故。

三、gevent-socket

gevent-socket是实现websocket的另一种方式。gevent-socket官方例子给的挺好。

可是,当我使用Nginx配置成wss之后,将ws请求转发到gevent-socket,就无法正常工作了。不知道为什么。

from geventwebsocket import WebSocketServer, WebSocketApplication, Resource

class EchoApplication(WebSocketApplication):
def on_open(self):
print("Connection opened") def on_message(self, message):
self.ws.send(message[::-1]) def on_close(self, reason):
print(reason) WebSocketServer(
('', 5000),
Resource({'/': EchoApplication})
).serve_forever()

html

<html>
<head>
<style>
* {
font-size: 20px;
}
</style>
</head>
<body>
<div style="text-align: center">
<textarea style="width: 80%;height: 70%;" id="all" readonly></textarea>
<br>
<input id="me" type="text" style="height: 10%;width: 80%;">
</div>
</body>
<script>
ws = new WebSocket("ws://weiyinfu.cn:5000/message/");
ws.onmessage = function (msg) {
output('server: ' + msg.data)
};
ws.onopen = function (msg) {
output('welcome,websocket opened')
}
ws.onerror = function (msg) {
output('bad ! websocket error')
}
function output(s) {
var all = document.getElementById('all')
all.value += s + '\n'
all.scrollTop = all.scrollHeight
}
document.getElementById('me').onkeydown = function (e) {
if (e.keyCode == 13) {
var s = document.getElementById('me').value
output('Me: ' + s )
ws.send(s)
document.getElementById('me').value = ''
e.preventDefault()
}
}
</script>
</html>

四、总结

本文提到四种websocket实现方式

  • 自己实现,这种方式肯定有不完善的地方,最终只能用于让自己明白一下websocket实现原理
  • flask-sockets
  • flask-socketIO
  • geventwebsocket

我最终使用了tomcat的websocket。

Python websocket的更多相关文章

  1. Python WebSocket长连接心跳与短连接

    python websocket 安装 pip install websocket-client 先来看一下,长连接调用方式: ws = websocket.WebSocketApp("ws ...

  2. python websocket学习使用

    前言 今天看了一些资料,记录一下心得. websocket是html5引入的一个新特性,传统的web应用是通过http协议来提供支持,如果要实时同步传输数据,需要轮询,效率低下 websocket是类 ...

  3. python websocket网页实时显示远程服务器日志信息

    功能:用websocket技术,在运维工具的浏览器上实时显示远程服务器上的日志信息 一般我们在运维工具部署环境的时候,需要实时展现部署过程中的信息,或者在浏览器中实时显示程序日志给开发人员看.你还在用 ...

  4. 运维开发:python websocket网页实时显示远程服务器日志信息

    功能:用websocket技术,在运维工具的浏览器上实时显示远程服务器上的日志信息 一般我们在运维工具部署环境的时候,需要实时展现部署过程中的信息,或者在浏览器中实时显示程序日志给开发人员看.你还在用 ...

  5. python websocket Django 实时消息推送

    概述: WebSocket 是什么? WebSocket 是 HTML5 提供的一种浏览器与服务器间进行全双工通讯的协议.依靠这种协议可以实现客户端和服务器端 ,一次握手,双向实时通信. WebSoc ...

  6. python websocket 再线聊天室的 Demo

    服务端 import tornado.web import tornado.ioloop import tornado.httpserver import tornado.options import ...

  7. python websocket 客户端连接

    # -*- coding: utf-8 -*-import jsonimport websocketimport _thread as thread # try:# import thread# ex ...

  8. python websocket client 使用

    import websocket ws = websocket.WebSocket() ws.connect("xx.xx.xx") ws.send("string&qu ...

  9. Python基于websocket实时通信的实现—GoEasy

    Python websocket实时消息推送 在这里我记录一下之前如何实现服务器端与客户端实时通信: 实现步骤如下: 1.        获取GoEasy appkey. 在goeasy官网上注册一个 ...

随机推荐

  1. JS 中div内容的显示和隐藏

    1. document.getElementById("dialog-auclot-status").style.display="none";//页面加载时隐 ...

  2. APP的上线和推广——线上推广渠道

    本文版权归个人所有,如需转载请注明出处http://www.cnblogs.com/PengLee/p/4637080.html 目录 应用商店 互联网开放平台 软件下载中心 媒体社交平台 刷榜推广 ...

  3. react random key generator;react如何产生随机不重复的key

    1.<div key={+new Date() + Math.random()}> 2.使用数组的索引 3.使用uuid:https://www.npmjs.com/package/uui ...

  4. 【BZOJ】【3437】小P的牧场

    DP/斜率优化 斜率优化基本题……等等,好像就没啥变化啊= = 嗯目测这题跟仓库建设差不多?写题的时候倒是没想这么多……直接推了公式. $$f[i]=min\{f[j]+cal(j,i)+a[i]\} ...

  5. 混沌数学之logistic模型

    logistic回归又称logistic回归分析,主要在流行病学中应用较多,比较常用的情形是探索某疾病的危险因素,根据危险因素预测某疾病发生的概率. 相关DEMO参见:混沌数学之离散点集图形DEMO ...

  6. [15] 星星(Star)图形的生成算法

    顶点数据的生成 bool YfBuildStarVertices ( Yreal radius, Yreal assistRadius, Yreal height, Yuint slices, YeO ...

  7. DeDeCMS(织梦)变量覆盖0day getshell

    测试方法: @Sebug.net   dis本站提供程序(方法)可能带有攻击性,仅供安全研究与教学之用,风险自负! #!usr/bin/php -w <?php error_reporting( ...

  8. JS操作JSON常用方法

    一.JSON字符串的替换 工作经常遇到这样的字符串,如下: 需要经过替换后,才能从字符串转化成JSON对象.这里我们需要用JS实现replaceAll的功能, 将所有的 ' \\" ' 替换 ...

  9. IE与Cognos的那些事

    问题描述1:打开报表设计页面的时候,即打开reportstudio的时候报IE阻止了一个来自XX.XX.XX.XX的弹出窗口程序 IE设置:关闭弹出窗口阻止程序即可 问题描述2:无法下载文件,例如Ex ...

  10. const char * 转换为char*

    可以用const_cast     const char* aa = "this is a const string.";     char* bb = const_cast< ...