- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
我最近开始学习 websockets,我决定尝试学习和使用 Python 的 framweork Tornado 来创建我的简单测试项目(没什么特别的,只是基本项目,可以帮助我了解 Tornado 和 websockets 的一般知识) .
所以,这是我的想法(工作流程):
1)我从其他应用程序向我的服务器发送 http post 请求(例如有关某人的姓名和电子邮件的信息)
2) 我将收到的数据保存到我的 postgresql 数据库并通知监听器(发布/订阅)新数据已添加到数据库
3)服务器收到通知后应该向客户端发送消息(write_message方法)
这是我现在的代码
simple_server.py
import tornado.httpserver
import tornado.ioloop
import tornado.options
import tornado.web
import tornado.websocket
import psycopg2
import psycopg2.extensions
import os
from tornado.options import define, options
define("port", default=8000, help="run on the given port", type=int)
io_loop = tornado.ioloop.IOLoop.instance()
connection = psycopg2.connect('dbname=mydb user=myusername password=mypassword')
connection.set_isolation_level(psycopg2.extensions.ISOLATION_LEVEL_AUTOCOMMIT)
class IndexHandler(tornado.web.RequestHandler):
def get(self):
self.render('index.html')
class ReceivedDataHandler(tornado.web.RequestHandler):
def post(self):
cursor = connection.cursor()
name=self.get_argument('name', 'No name info received')
email = self.get_argument('email', 'No email info received')
self.write("New person with name %s and email %s" %(name, email))
cursor.execute("INSERT INTO mydata VALUES (%s, %s)" %(name, email))
cursor.execute("NOTIFY test_channel;")
class EchoHandler(tornado.websocket.WebSocketHandler):
def open(self):
self.write_message('connected!')
def on_message(self, message):
self.write_message("Received info about new person: "+message)
def on_close(self):
print 'connection closed'
def listen():
cursor = connection.cursor()
cursor.execute("LISTEN test_channel;")
def receive(fd, events):
"""Receive a notify message from channel I listen."""
state = connection.poll()
if state == psycopg2.extensions.POLL_OK:
if connection.notifies:
notify = connection.notifies.pop()
print "New notify message"
io_loop.add_handler(connection.fileno(), receive, io_loop.READ)
if __name__=="__main__":
tornado.options.parse_command_line()
app = tornado.web.Application(
handlers=[
(r'/', IndexHandler),
(r'/person-info', ReceivedDataHandler),
(r'/websocket', EchoHandler)
],
template_path=os.path.join(os.path.dirname(__file__), "templates"),
static_path=os.path.join(os.path.dirname(__file__), "static"),
debug=True
)
http_server = tornado.httpserver.HTTPServer(app)
http_server.listen(options.port)
listen()
io_loop.start()
当我测试发送发布请求(通过 Postman REST 客户端)到定义的 url 时,一切正常。数据确实保存到我的数据库中,它确实通知监听器有新通知,但我只是不知道之后如何将该消息发送给客户端。如果我能做到这一点,那么它就会在浏览器中显示该消息,这就是我这次想做的。
所以,我想做的实际上是在收到有关数据库中新条目的通知后调用 write_message 函数(而不是仅仅打印“新通知消息”),但我只是不知道如何在 Tornado 中执行此操作。我相信实际上应该很容易,但由于我显然缺乏 Tornado(和异步编程)的经验,所以我有点卡住了。
感谢您的帮助
最佳答案
最终我找到了解决这个问题的方法。我刚刚添加了全局变量,在其中添加所有连接的客户端,然后在收到通知时向每个连接的客户端发送消息。 (这对我来说没问题,因为我实际上想向所有连接的客户端发送消息)
这就是现在的样子
simple_server.py
import tornado.httpserver
import tornado.ioloop
import tornado.options
import tornado.web
import tornado.websocket
import psycopg2
import psycopg2.extensions
import os
from tornado.options import define, options
define("port", default=8000, help="run on the given port", type=int)
io_loop = tornado.ioloop.IOLoop.instance()
connection = psycopg2.connect('dbname=mydb user=myusername password=mypassword')
connection.set_isolation_level(psycopg2.extensions.ISOLATION_LEVEL_AUTOCOMMIT)
# This is a global variable to store all connected clients
websockets = []
class IndexHandler(tornado.web.RequestHandler):
def get(self):
self.render('index.html')
class ReceivedDataHandler(tornado.web.RequestHandler):
def post(self):
cursor = connection.cursor()
name=self.get_argument('name', 'No name info received')
email = self.get_argument('email', 'No email info received')
self.write("New person with name %s and email %s" %(name, email))
cursor.execute("INSERT INTO mydata VALUES (%s, %s)" %(name, email))
cursor.execute("NOTIFY test_channel;")
class EchoHandler(tornado.websocket.WebSocketHandler):
def open(self):
self.write_message('connected!')
def on_message(self, message):
self.write_message("Received info about new person: "+message)
def on_close(self):
print 'connection closed'
def listen():
cursor = connection.cursor()
cursor.execute("LISTEN test_channel;")
def receive(fd, events):
"""Receive a notify message from channel I listen."""
state = connection.poll()
if state == psycopg2.extensions.POLL_OK:
if connection.notifies:
notify = connection.notifies.pop()
for ws in websockets:
ws.write_message("my message")
io_loop.add_handler(connection.fileno(), receive, io_loop.WRITE)
if __name__=="__main__":
tornado.options.parse_command_line()
app = tornado.web.Application(
handlers=[
(r'/', IndexHandler),
(r'/person-info', ReceivedDataHandler),
(r'/websocket', EchoHandler)
],
template_path=os.path.join(os.path.dirname(__file__), "templates"),
static_path=os.path.join(os.path.dirname(__file__), "static"),
debug=True
)
http_server = tornado.httpserver.HTTPServer(app)
http_server.listen(options.port)
listen()
io_loop.start()
关于python - 收到通知后从服务器向客户端发送消息(Tornado+websockets),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/21023104/
根据一些谷歌搜索,我安装了以下错误处理程序。然而,似乎返回 http 500 的 python 异常并没有被这些东西捕获,尽管 404 是这样。通过我在下面的代码中留下的打印语句,我可以看到它没有命中
我刚刚意识到 WebSocketHandler.write_message() 返回一个 Future。我以前没有在我的函数中产生过这个函数: @tornado.gen.coroutine
这是我的 Tornado 文件:: from tornado.wsgi import WSGIContainer from tornado.ioloop import IOLoop from torn
class MainHandler(BaseHandler): @tornado.web.authenticated def get(self): self.rende
我正在尝试使用 AsyncHTTPTestCase 测试 Tornado .我想测试标有 @tornado.web.authenticated 注释的处理程序。因为此处理程序需要身份验证,所以我们必须
我正在使用 Tornado Web Server (版本 4.1)使用 Python 2.7 创建 REST Web 应用程序。我的请求处理程序之一 (web.RequestHandler) 使用多部
我想知道tornado 的内部工作流程,并且看过this article ,很好,但我就是想不通 ioloop.py里面有这样一个函数 def add_handler(self, fd, handle
如何遍历从 Python/Tornado 处理程序传递到 Tornado 模板的字典? 我试过 {% for key, value in statistics %}
我有一个 Tornado 后端,为 Angular 前端提供服务。更新数据库时,tornado api 不会获取更新的数据。它仅在我重新启动服务器后出现。有人可以帮我解决这个问题吗?我希望获取的数据能
我尝试使用自定义的 WSGIContainer 来处理异步操作: from tornado import httpserver, httpclient, ioloop, wsgi, gen @gen.
from tornado.web import RequestHandler class HelloWorldHandler(RequestHandler): def get(self):
Pylint 遇到 @tornado.web.authenticated 时崩溃 class Handler1(tornado.web.RequestHandler): def get(sel
经过 tornado.gen documentation有人可以帮我理解 tornado.gen.coroutine 和 tornado.gen.engine 之间的确切区别 最佳答案 正如 gen.
代码如下: from tornadoredis import Client from tornado.ioloop import IOLoop from tornado.gen import coro
我有一个 tornado.websocket.WebSocketHandler 的子类。在该类中,我有一个方法使用 Django ORM 从子类模型中获取用户:django.contrib.auth.
我是 ssl 之类的新手,我已经使用 openssl 生成了自签名证书。 openssl req -newkey rsa:2048 -nodes -keyout key.pem -x509 -days
我已经从 tornado 4.2 移动到 tornado 6.0.3,我得到了错误 AttributeError:模块“tornado.web”没有属性“异步” 根据 tornado v6 seems
我一直在关注此 ( https://developer.ibm.com/tutorials/se-distributed-apps-zeromq-part2/) 教程,以设置使用 CurveZMQ 加
我在使用tornado-celery整合tornado和celery时,出现错误:``` traceback (most recent call last): File "/usr/local/l
我正在使用 Tornado 与 twitter 等第三方进行身份验证。 我的登录处理程序看起来像这样 class AuthLoginHandler(BaseHandler, tornado.auth.
我是一名优秀的程序员,十分优秀!