import django import time from autobahn.twisted.websocket import WebSocketServerProtocol, WebSocketServerFactory from collections import deque from twisted.internet import reactor from channels import Channel, channel_backends, DEFAULT_CHANNEL_BACKEND class InterfaceProtocol(WebSocketServerProtocol): """ Protocol which supports WebSockets and forwards incoming messages to the django.websocket channels. """ def onConnect(self, request): self.channel_backend = channel_backends[DEFAULT_CHANNEL_BACKEND] self.request_info = { "path": request.path, "GET": request.params, } def onOpen(self): # Make sending channel self.reply_channel = Channel.new_name("!django.websocket.send") self.request_info["reply_channel"] = self.reply_channel self.last_keepalive = time.time() self.factory.protocols[self.reply_channel] = self # Send news that this channel is open Channel("django.websocket.connect").send(self.request_info) def onMessage(self, payload, isBinary): if isBinary: Channel("django.websocket.receive").send(dict( self.request_info, content = payload, binary = True, )) else: Channel("django.websocket.receive").send(dict( self.request_info, content = payload.decode("utf8"), binary = False, )) def serverSend(self, content, binary=False, **kwargs): """ Server-side channel message to send a message. """ if binary: self.sendMessage(content, binary) else: self.sendMessage(content.encode("utf8"), binary) def serverClose(self): """ Server-side channel message to close the socket """ self.sendClose() def onClose(self, wasClean, code, reason): if hasattr(self, "reply_channel"): del self.factory.protocols[self.reply_channel] Channel("django.websocket.disconnect").send(self.request_info) def sendKeepalive(self): """ Sends a keepalive packet on the keepalive channel. """ Channel("django.websocket.keepalive").send(self.request_info) self.last_keepalive = time.time() class InterfaceFactory(WebSocketServerFactory): """ Factory which keeps track of its open protocols' receive channels and can dispatch to them. """ # TODO: Clean up dead protocols if needed? def __init__(self, *args, **kwargs): super(InterfaceFactory, self).__init__(*args, **kwargs) self.protocols = {} def reply_channels(self): return self.protocols.keys() def dispatch_send(self, channel, message): if message.get("close", False): self.protocols[channel].serverClose() else: self.protocols[channel].serverSend(**message) class WebsocketTwistedInterface(object): """ Easy API to run a WebSocket interface server using Twisted. Integrates the channel backend by running it in a separate thread, using the always-compatible polling style. """ def __init__(self, channel_backend, port=9000): self.channel_backend = channel_backend self.port = port def run(self): self.factory = InterfaceFactory("ws://0.0.0.0:%i" % self.port, debug=False) self.factory.protocol = InterfaceProtocol reactor.listenTCP(self.port, self.factory) reactor.callInThread(self.backend_reader) reactor.callLater(1, self.keepalive_sender) reactor.run() def backend_reader(self): """ Run in a separate thread; reads messages from the backend. """ while True: channels = self.factory.reply_channels() # Quit if reactor is stopping if not reactor.running: return # Don't do anything if there's no channels to listen on if channels: channel, message = self.channel_backend.receive_many(channels) else: time.sleep(0.1) continue # Wait around if there's nothing received if channel is None: time.sleep(0.05) continue # Deal with the message self.factory.dispatch_send(channel, message) def keepalive_sender(self): """ Sends keepalive messages for open WebSockets every (channel_backend expiry / 2) seconds. """ expiry_window = int(self.channel_backend.expiry / 2) for protocol in self.factory.protocols.values(): if time.time() - protocol.last_keepalive > expiry_window: protocol.sendKeepalive() reactor.callLater(1, self.keepalive_sender)