one tasks queue instead of one per websocket
This commit is contained in:
parent
e210d9329c
commit
5b33721c87
6 changed files with 71 additions and 51 deletions
|
|
@ -4,9 +4,7 @@ from __future__ import division
|
|||
|
||||
from tornado.websocket import WebSocketHandler
|
||||
from tornado.ioloop import IOLoop
|
||||
from Queue import Queue
|
||||
import json
|
||||
from threading import Thread
|
||||
|
||||
from oxtornado import json_dumps
|
||||
|
||||
|
|
@ -15,70 +13,37 @@ import state
|
|||
import logging
|
||||
logger = logging.getLogger('oml.websocket')
|
||||
|
||||
class Background:
|
||||
|
||||
def __init__(self, handler):
|
||||
self.handler = handler
|
||||
self.q = Queue()
|
||||
self.connected = True
|
||||
self.main = IOLoop.instance()
|
||||
|
||||
def worker(self):
|
||||
while self.connected:
|
||||
message = self.q.get()
|
||||
action, data = json.loads(message)
|
||||
import item.scan
|
||||
if action == 'ping':
|
||||
self.post(['pong', data])
|
||||
elif action == 'import':
|
||||
item.scan.run_import(data)
|
||||
elif action == 'scan':
|
||||
item.scan.run_scan()
|
||||
elif action == 'update':
|
||||
self.post(['error', {'error': 'not implemented'}])
|
||||
else:
|
||||
self.post(['error', {'error': 'unknown action'}])
|
||||
self.q.task_done()
|
||||
|
||||
def join(self):
|
||||
self.q.join()
|
||||
|
||||
def put(self, data):
|
||||
self.q.put(data)
|
||||
|
||||
def post(self, data):
|
||||
if not isinstance(data, basestring):
|
||||
data = json_dumps(data)
|
||||
self.main.add_callback(lambda: self.handler.write_message(data))
|
||||
|
||||
|
||||
class Handler(WebSocketHandler):
|
||||
background = None
|
||||
|
||||
def open(self):
|
||||
if self.request.host not in self.request.headers['origin']:
|
||||
logger.debug('reject cross site attempt to open websocket %s', self.request)
|
||||
self.close()
|
||||
self.background = Background(self)
|
||||
state.websockets.append(self.background)
|
||||
self.t = Thread(target=self.background.worker)
|
||||
self.t.daemon = True
|
||||
self.t.start()
|
||||
if self not in state.websockets:
|
||||
state.websockets.append(self)
|
||||
|
||||
|
||||
#websocket calls
|
||||
def on_message(self, message):
|
||||
self.background.put(message)
|
||||
action, data = json.load(message)
|
||||
if state.tasks:
|
||||
state.tasks.queue(action, data)
|
||||
|
||||
def on_close(self):
|
||||
if self.background:
|
||||
state.websockets.remove(self.background)
|
||||
self.background.connected = False
|
||||
if self in state.websockets:
|
||||
state.websockets.remove(self)
|
||||
|
||||
def post(self, event, data):
|
||||
message = json_dumps([event, data])
|
||||
main = IOLoop.instance()
|
||||
main.add_callback(lambda: self.write_message(message))
|
||||
|
||||
def trigger_event(event, data):
|
||||
if len(state.websockets):
|
||||
logger.debug('trigger event %s %s %s', event, data, len(state.websockets))
|
||||
for ws in state.websockets:
|
||||
try:
|
||||
ws.post([event, data])
|
||||
ws.post(event, data)
|
||||
except:
|
||||
logger.debug('failed to send to ws %s %s %s', ws, event, data, exc_info=1)
|
||||
|
|
|
|||
Loading…
Add table
Add a link
Reference in a new issue