- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
我在制作 WebSocket 服务器/客户端时遇到问题。
这是我的 WebSocket 类:
wslist = []
class LoginLiveWebSocketHandler(WebSocket): # /live/ws/zones
loggedon = False
def opened(self):
log("Opened!")
wslist.append(self)
def received_message(self, m):
log("Received Message!")
log(m)
log(m.data)
m = str(m.data)
if not m.startswith("Login: ") & (not self.loggedon):
return
if m.startswith("Login: ") & self.loggedon:
return
if m.startswith("Login: ") & (not self.loggedon):
split = m.replace("Login: ", "").split(":")
try:
id = split[0]
key = split[1]
except KeyError:
self.send("Login: 0", False)
return
try:
usr = users.getuser(id, key)
nick = usr["nick"]
loggedin = True
token = usr["groupme_token"]
active = usr["active"]
except KeyError or TypeError:
self.send("Login: 0", False)
return
if (not loggedin) | (not active) | (not local.token) | (local.user["faction"] != "f"):
self.send("Login: 0", False)
return
self.send("Login: 1", False)
self.loggedon = True
return
def closed(self, code, reason=""):
log("Closed!")
wslist.remove(self)
还有我的 JavaScript 页面(注意,不是全部):
function init() {
showBlocking("Connecting...", "progress-bar-success");
var wsprotocol = (location.protocol == "https:") ? "wss://" : "ws://";
var loc = location.pathname;
loc = loc.replace(/live\u002flive$/i, "");
var wsurl = wsprotocol + location.host + loc + "/live/ws_zones";
console.log(wsurl);
var ws = new WebSocket(wsurl);
var key = getCookie("key");
var id = getCookie("id");
var closed = true;
ws.onopen = function(evt) {
console.log("onopen");
$('#pbheader').text("Logging in...");
ws.send("Login: " + id + ":" + key);
closed = false;
}
ws.onmessage = function(evt) {
console.log("OnMessage");
var channel = evt.data.substring(0, 7);
var data = evt.data.substr(6);
if (channel == "Login: ") {
if (data == "0") {
ws.close();
closed = true;
alert("Invalid Login, please re-login.");
$('#pbheader').text("Invalid Login!");
history.back();
} else if (data == "1") {
hideBlocking();
}
} else if (channel == "ZoneA: ") {
// add new thing...
// %gridref% %zoneid% %zonename% %faction% %legion% %faceless% %swarm%
var datadict = JSON.parse(data);
var replitem = itemtemplate.replace("%count%", count).replace("%gridref%", datadict.gridref).replace("%zoneid%", datadict.zoneid).replace("%zonename%", datadict.zonename).replace("%faction%", datadict.faction).replace("%legion%", datadict.legion).replace("%faceless%", datadict.faceless).replace("%swarm%", datadict.swarm);
$('#live').prepend(replitem);
var curritem = $('#item' + count);
setTimeout(function () { fadeoutremove(curritem); }, 60*1000);
count = count + 1;
}
}
ws.onclose = function(evt) {
console.log("Closed");
closed = true;
}
ws.onerror = function(evt) {
console.log("error");
closed = true;
}
}
window.addEventListener("load", init, false);
安装服务:
log("Starting WebServer (cherrypy)")
cherrypy.server.unsubscribe()
if config.DO_SSL is True:
server1 = cherrypy._cpserver.Server()
server1.socket_port = config.LISTEN_PORT_SSL
server1._socket_host = config.LISTEN
server1.thread_pool = 30
server1.ssl_certificate = config.SSL_CERTIFICATE
server1.ssl_private_key = config.SSL_PRIVATE_KEY
server1.ssl_certificate_chain = config.SSL_CHAIN
server1.ssl_module = config.SSL_HANDLER
server1.subscribe()
server2 = cherrypy._cpserver.Server()
server2.socket_port = config.LISTEN_PORT
server2._socket_host = config.LISTEN
server2.thread_pool = 30
server2.subscribe()
WebSocketPlugin(cherrypy.engine).subscribe()
cherrypy.tools.websocket = WebSocketTool()
cherrypy.tree.mount(APIService.APIService(), "/api", config={"/": {'error_page.404': APIService.ep}})
cherrypy.tree.mount(JSONService.JSONService(), "/json", config={"/": {'error_page.404': JSONService.ep}})
cherrypy.tree.mount(USRAPIService.USRAPIService(), "/usrapi", config={"/": {'error_page.404': USRAPIService.ep}})
cherrypy.tree.mount(USRAPIAdminService.USRAPIAdminService(), "/usrapi/admin", config={
"/": {'error_page.404': USRAPIService.ep}})
cherrypy.tree.mount(RootService.RootService(), "/", config={"/": {'error_page.404': RootService.ep}})
cherrypy.tree.mount(AdminService.AdminService(), "/admin", config={"/": {'error_page.404': AdminService.ep}})
cherrypy.tree.mount(HelperPanel.HelperPanel(), "/helper", config={"/": {'error_page.404': HelperPanel.ep}})
cherrypy.tree.mount(LiveService.LiveService(), "/live", config={"/": {'error_page.404': LiveService.ep},
"/ws_zones": {'tools.websocket.on': True,
'tools.websocket.handler_cls':
LiveService.LoginLiveWebSocketHandler
}})
cherrypy.tree.mount(None, "/static", config={"/": {'error_page.404': AdminService.ep,
'tools.staticdir.on': True,
'tools.staticdir.dir':
os.path.join(__current_dir, "web/static")}})
cherrypy.engine.start()
log("Started WebServer (cherrypy)")
实时服务:
class LiveService(object):
@cherrypy.expose
@require(level=0)
def index(self, user, usrname, usrhelper, usradmin):
return render("index", usrname, usrhelper, usradmin)
@cherrypy.expose
@require(level=4) # Require login, Groupme, active and faceless
def live(self, user, usrname, usrhelper, usradmin):
return render("websocket/zonesdata", usrname, usrhelper, usradmin)
@cherrypy.expose
def ws_zones(self):
log("Handler created: %s" % repr(cherrypy.request.ws_handler))
cherrypy.request.ws_handler.send(b"Login 0", False)
因此,安装和启动工作正常,整个生产环境已经运行了几个月,但是带有 ws 的新 LiveService 无法正常工作。在日志中,我收到 [21/Jul/2015:16:55:28] ENGINE 开始 WebSocket 处理
和 Opened
消息,但我从未收到或无法发送消息。当我将 LiveService.LoginLiveServiceWebSocketHandler 更改为 EchoWebSocket 时,他获取消息并发送它们,但在 ws_zones 处理程序上
cherrypy.request.ws_handler.send(b"登录0", False)
不工作。无论是否带有 b。
最佳答案
首先要说的是,您应该注意可用的调试工具。例如,当您在 Firefox 中尝试应用程序时,Firebug(或 Web 控制台)会告诉您:
Firefox can't establish a connection to the server at ws://127.0.0.1:8080/live/ws_zones.
Chromium 控制台有更多有用的错误消息:
WebSocket connection to 'ws://127.0.0.1:8080/live/ws_zones' failed: Error during WebSocket handshake: Invalid status line
好的,我们遇到了握手问题。
如果您打开 Wireshark(或其他嗅探器),您可以看到我们的 Login 0
确实出现在 ws4py
回复握手响应之前(或者可能存在竞争条件) )。
我可以理解您的困惑,因为在多个地方 ws4py
文档要么暗示或直接引导您错误的方式。让我们看看:
alludes — 您无法从普通处理程序发送。
@cherrypy.expose
def ws(self):
# you can access the class instance through
handler = cherrypy.request.ws_handler
guides incorrectly — 您无法从 opened
钩子(Hook)发送。
def opened(self):
"""
Called by the server when the upgrade handshake
has succeeeded.
"""
pass
所以我认为这要么是过时的文档,要么是错误。
现在我建议您在客户端发起通信。如果您有一些登录前的“聊天”,您可以在网络套接字之上制定一个小协议(protocol)(无论如何您最终都会这样做)。像这样的事情:
#!/usr/bin/env python3
import json
import cherrypy
from ws4py.server.cherrypyserver import WebSocketPlugin, WebSocketTool
from ws4py.websocket import WebSocket
class LoginLiveWebSocketHandler(WebSocket):
wsset = set()
'''Class level attr, do not rebind'''
def opened(self):
'''You can not send yet, because handshake has not been completed.
Yes, a documentation is not always correct ;-)
'''
cherrypy.log('Opened!')
self.wsset.add(self)
def closed(self, code, reason = ''):
cherrypy.log('Closed!')
self.wsset.remove(self)
def received_message(self, message):
cherrypy.log('Received Message!')
message = json.loads(message.data.decode())
if message['cmd'] == 'init':
self.send('Login 0')
else:
self.send(message['payload'] + ' pong')
class LiveService(object):
@cherrypy.expose
def index(self):
return '''<!DOCTYPE html>
<html>
<body>
<div id='log'></div>
<script type='application/javascript'>
var ws = new WebSocket('ws://127.0.0.1:8080/live/ws_zones');
ws.onopen = function(event)
{
ws.send(JSON.stringify({'cmd': 'init'}));
};
ws.onmessage = function(event)
{
document.getElementById('log').innerHTML += event.data + '<br/>';
};
var handle = setInterval(function()
{
if(ws.readyState == WebSocket.CLOSING || ws.readyState == WebSocket.CLOSED)
{
clearInterval(handle);
}
else
{
ws.send(JSON.stringify({'cmd': 'chat', 'payload': 'ping'}));
}
}, 1000);
</script>
</body>
</html>
'''
@cherrypy.expose
def ws_zones(self):
'''You can not send yet, because handshake has not been completed'''
cherrypy.log('Handler created: %r' % cherrypy.request.ws_handler)
if __name__ == '__main__':
cherrypy.config.update({
'server.socket_host' : '127.0.0.1',
'server.socket_port' : 8080,
'server.thread_pool' : 8
})
cherrypy.tools.websocket = WebSocketTool()
WebSocketPlugin(cherrypy.engine).subscribe()
cherrypy.tree.mount(LiveService(), '/live', {
'/ws_zones' : {
'tools.websocket.on' : True,
'tools.websocket.handler_cls' : LoginLiveWebSocketHandler
}
})
cherrypy.engine.signals.subscribe()
cherrypy.engine.start()
cherrypy.engine.block()
关于javascript - ws4py - 发送/接收消息不起作用,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/31548029/
我使用的是 Windows 8.1 和 Python 2.7,我在特定文件路径中设置了所有文件(希望正确),但每当我运行 python manage.py runserver 时,我都会收到此错误。
背景: 我有一个像这样的目录结构: Package/ setup.py src/ __init__.py __main__.py cod
我从这个线程运行了一个示例代码。 How to properly use coverage.py in Python? 但是,当我执行此命令时 py.test test.py --cov=sample
IPython 0.13.1 文档说: $ ipython -h ... Usage ipython [subcommand] [options] [files] If invoked
我写了一个网站,让我困惑的是当我运行这个网站时,首先我需要启动应用程序,所以有 3 种方法: sudo python xxx.py python xxx.py xxx.py 每一个我都不清楚怎么用,目
我不确定为什么它不起作用,这可能是一个您无法解决的问题,但我只是想知道为什么它不起作用。如果我浪费了您的时间,或者没有正确地提出问题,我很抱歉,我 16 岁,对 Python 还算陌生。 在main.
鉴于以下情况:models.py from .managers import PersonManager from django.db import models class Person(model
有没有办法将参数传递给 web.py 处理程序类构造函数? 例如。这些参数可能来自命令行(当主 web.py 脚本运行时),在第一个参数(作为端口号)之后 最佳答案 当然,这取决于你的意思。毕竟都是p
我对 python/django 编程很陌生,因为我没有编程背景。我正在在线上课,我只想确切地知道 manage.py 文件的作用。我试过用谷歌搜索它,但除了在 django-admin.py 周围放
我想将类别及其子类别保存到数据库中,这里每个类别都有多个子类别。您能帮我保存与类别相对应的用户、类别和多个子类别吗?Models.py、Serializers.py、Views .py 并附加传入请求
所以我的机器人开始有很多命令,并且在 main.py 上变得有点困惑。我知道有一种方法可以将命令存储在其他文件中,然后在 discord.js 上触发它们时将它们应用于 main.py。在 disco
我正在尝试制作一个类似于 mee6 的 Discord 机器人,因为它会按特定时间间隔计算用户在我的 Discord 服务器中发送的消息。我已经在网上搜索过,但即使有类似的问题也找不到我要找的东西。例
我正在尝试制作一个机器人,它根据特定 channel 中的消息创建线程。如果有在 discord.py 中的文本 channel 中创建线程的方法,请告诉我。 最佳答案 是的,但有一个问题。 当前版本
我一直在尝试制作一个命令来显示一些信息,然后当我对表情使用react时,它应该会显示另一组信息。 我尝试使用 this 的部分内容,特别是第 335 到 393 行的部分让它工作。但是,它什么也不做。
这是我试过的代码: @client.event async def on_message(message): if client.user.mention in message.content
我试过这段代码,机器人说猜但没有回应我的猜测。 @commands.command() async def game(self, ctx): number = random.randint(0
我决定尝试让我的不和谐机器人播放音乐,但我已经卡住了。主要是因为我找不到任何资源来帮助当前版本,我一直在从文档中获取所有内容。但是,我不知道如何检查机器人是否已连接到语音 channel 。 我试过
我在一个目录中有三个文件: # Untitled-1.py print("UTITLEDPY") if __name__== "__main__": from telegram.ext imp
我对 python 相当陌生,并且一直只使用 Jupyter Notebooks。当我需要运行我已保存在计算机中某处的 .py 文件时,我通常所做的就是使用魔术命令 %run %run '/home/
我有 Django 1.4 和 Python 2.6.6当我使用“django-amdin.py startproject djproject”时,请按照网页中的步骤操作 https://www.ib
我是一名优秀的程序员,十分优秀!