Getting Started =============== (If you haven't yet, make sure you :doc:`install Channels `) Now, let's get to writing some consumers. If you've not read it already, you should read :doc:`concepts`, as it covers the basic description of what channels and groups are, and lays out some of the important implementation patterns and caveats. First Consumers --------------- Now, by default, Django will run things through Channels but it will also tie in the URL router and view subsystem to the default ``http.request`` channel if you don't provide another consumer that listens to it - remember, only one consumer can listen to any given channel. As a very basic example, let's write a consumer that overrides the built-in handling and handles every HTTP request directly. This isn't something you'd usually do in a project, but it's a good illustration of how channels now underlie every part of Django. Make a new project, a new app, and put this in a ``consumers.py`` file in the app:: from django.http import HttpResponse def http_consumer(message): response = HttpResponse("Hello world! You asked for %s" % message.content['path']) message.reply_channel.send(response.channel_encode()) The most important thing to note here is that, because things we send in messages must be JSON serializable, the request and response messages are in a key-value format. There are ``channel_decode()`` and ``channel_encode()`` methods on both Django's request and response classes, but here we just use the message's ``content`` attribute directly for simplicity (message content is always a dict). Now, go into your ``settings.py`` file, and set up a channel backend; by default, Django will just use a local backend and route HTTP requests to the normal URL resolver (we'll come back to backends in a minute). For now, we want to override the *channel routing* so that, rather than going to the URL resolver and our normal view stack, all HTTP requests go to our custom consumer we wrote above. Here's what that looks like:: # In settings.py CHANNEL_BACKENDS = { "default": { "BACKEND": "channels.database_layer.DatabaseChannelLayer", "ROUTING": "myproject.routing.channel_routing", }, } # In routing.py channel_routing = { "http.request": "myproject.myapp.consumers.http_consumer", } As you can see, this is a little like Django's ``DATABASES`` setting; there are named channel backends, with a default one called ``default``. Each backend needs a class specified which powers it - we'll come to the options there later - and a routing scheme, which points to a dict containing the routing settings. It's recommended you call this ``routing.py`` and put it alongside ``urls.py`` in your project. If you start up ``python manage.py runserver`` and go to ``http://localhost:8000``, you'll see that, rather than a default Django page, you get the Hello World response, so things are working. If you don't see a response, check you :doc:`installed Channels correctly `. Now, that's not very exciting - raw HTTP responses are something Django has been able to do for a long time. Let's try some WebSockets, and make a basic chat server! Delete that consumer and its routing - we'll want the normal Django view layer to serve HTTP requests from now on - and make this WebSocket consumer instead:: # In consumers.py from channels import Group def ws_add(message): Group("chat").add(message.reply_channel) Hook it up to the ``websocket.connect`` channel like this:: # In routing.py channel_routing = { "websocket.connect": "myproject.myapp.consumers.ws_add", } Now, let's look at what this is doing. It's tied to the ``websocket.connect`` channel, which means that it'll get a message whenever a new WebSocket connection is opened by a client. When it gets that message, it takes the ``reply_channel`` attribute from it, which is the unique response channel for that client, and adds it to the ``chat`` group, which means we can send messages to all connected chat clients. Of course, if you've read through :doc:`concepts`, you'll know that channels added to groups expire out after a while unless you keep renewing their membership. This is because Channels is stateless; the worker processes don't keep track of the open/close states of the potentially thousands of connections you have open at any one time. The solution to this is that the WebSocket interface servers will send periodic "keepalive" messages on the ``websocket.keepalive`` channel, so we can hook that up to re-add the channel:: # In consumers.py from channels import Group # Connected to websocket.keepalive def ws_keepalive(message): Group("chat").add(message.reply_channel) It's safe to add the channel to a group it's already in - similarly, it's safe to discard a channel from a group it's not in. Of course, this is exactly the same code as the ``connect`` handler, so let's just route both channels to the same consumer:: # In routing.py channel_routing = { "websocket.connect": "myproject.myapp.consumers.ws_add", "websocket.keepalive": "myproject.myapp.consumers.ws_add", } And, even though channels will expire out, let's add an explicit ``disconnect`` handler to clean up as people disconnect (most channels will cleanly disconnect and get this called):: # In consumers.py from channels import Group # Connected to websocket.disconnect def ws_disconnect(message): Group("chat").discard(message.reply_channel) Now, that's taken care of adding and removing WebSocket send channels for the ``chat`` group; all we need to do now is take care of message sending. For now, we're not going to store a history of messages or anything and just replay any message sent in to all connected clients. Here's all the code:: # In consumers.py from channels import Group # Connected to websocket.connect and websocket.keepalive def ws_add(message): Group("chat").add(message.reply_channel) # Connected to websocket.receive def ws_message(message): Group("chat").send(message.content) # Connected to websocket.disconnect def ws_disconnect(message): Group("chat").discard(message.reply_channel) And what our routing should look like in ``routing.py``:: channel_routing = { "websocket.connect": "myproject.myapp.consumers.ws_add", "websocket.keepalive": "myproject.myapp.consumers.ws_add", "websocket.receive": "myproject.myapp.consumers.ws_message", "websocket.disconnect": "myproject.myapp.consumers.ws_disconnect", } With all that code, you now have a working set of a logic for a chat server. All you need to do now is get it deployed, and as we'll see, that's not too hard. Running with Channels --------------------- Because Channels takes Django into a multi-process model, you can no longer just run one process if you want to serve more than one protocol type. There are multiple kinds of "interface servers", and each one will service a different type of request - one might do WSGI requests, one might handle WebSockets, or you might have one that handles both. These are separate from the "worker servers" where Django will run actual logic, though, and so you'll need to configure a channel backend to allow the channels to run over the network. By default, when you're using Django out of the box, the channel backend is set to an in-memory one that only works in process; this is enough to serve normal WSGI style requests (``runserver`` is just running a WSGI interface and a worker in two separate threads), but now we want WebSocket support we'll need a separate process to keep things clean. If you notice, in the example above we switched our default backend to the database channel backend. This uses two tables in the database to do message handling, and isn't particularly fast but requires no extra dependencies. When you deploy to production, you'll want to use a backend like the Redis backend that has much better throughput. The second thing, once we have a networked channel backend set up, is to make sure we're running the WebSocket interface server. Even in development, we need to do this; ``runserver`` will take care of normal Web requests and running a worker for us, but WebSockets isn't compatible with WSGI and needs to run separately. The easiest way to do this is to use the ``runwsserver`` management command that ships with Django; just make sure you've installed the latest release of ``autobahn`` first:: pip install -U autobahn[twisted] python manage.py runwsserver Run that alongside ``runserver`` and you'll have two interface servers, a worker thread, and the channel backend all connected and running. You can even launch separate worker processes with ``runworker`` if you like (you'll need at least one of those if you're not also running ``runserver``). Now, just open a browser and put the following into the JavaScript console to test your new code:: socket = new WebSocket("ws://127.0.0.1:9000"); socket.onmessage = function(e) { alert(e.data); } socket.onopen = function() { socket.send("hello world"); } You should see an alert come back immediately saying "hello world" - your message has round-tripped through the server and come back to trigger the alert. You can open another tab and do the same there if you like, and both tabs will receive the message and show an alert, as any incoming message is sent to the ``chat`` group by the ``ws_message`` consumer, and both your tabs will have been put into the ``chat`` group when they connected. Feel free to put some calls to ``print`` in your handler functions too, if you like, so you can understand when they're called. If you run three or four copies of ``runworker`` you'll probably be able to see the tasks running on different workers. Persisting Data --------------- Echoing messages is a nice simple example, but it's skirting around the real design pattern - persistent state for connections. Let's consider a basic chat site where a user requests a chat room upon initial connection, as part of the query string (e.g. ``wss://host/websocket?room=abc``). The ``reply_channel`` attribute you've seen before is our unique pointer to the open WebSocket - because it varies between different clients, it's how we can keep track of "who" a message is from. Remember, Channels is network-transparent and can run on multiple workers, so you can't just store things locally in global variables or similar. Instead, the solution is to persist information keyed by the ``reply_channel`` in some other data store - sound familiar? This is what Django's session framework does for HTTP requests, only there it uses cookies as the lookup key rather than the ``reply_channel``. Channels provides a ``channel_session`` decorator for this purpose - it provides you with an attribute called ``message.channel_session`` that acts just like a normal Django session. Let's use it now to build a chat server that expects you to pass a chatroom name in the path of your WebSocket request (we'll ignore auth for now - that's next):: # In consumers.py from channels import Group from channels.decorators import channel_session # Connected to websocket.connect @channel_session def ws_connect(message): # Work out room name from path (ignore slashes) room = message.content['path'].strip("/") # Save room in session and add us to the group message.channel_session['room'] = room Group("chat-%s" % room).add(message.reply_channel) # Connected to websocket.keepalive @channel_session def ws_keepalive(message): Group("chat-%s" % message.channel_session['room']).add(message.reply_channel) # Connected to websocket.receive @channel_session def ws_message(message): Group("chat-%s" % message.channel_session['room']).send(message.content) # Connected to websocket.disconnect @channel_session def ws_disconnect(message): Group("chat-%s" % message.channel_session['room']).discard(message.reply_channel) If you play around with it from the console (or start building a simple JavaScript chat client that appends received messages to a div), you'll see that you can set a chat room with the initial request. Authentication -------------- Now, of course, a WebSocket solution is somewhat limited in scope without the ability to live with the rest of your website - in particular, we want to make sure we know what user we're talking to, in case we have things like private chat channels (we don't want a solution where clients just ask for the right channels, as anyone could change the code and just put in private channel names) It can also save you having to manually make clients ask for what they want to see; if I see you open a WebSocket to my "updates" endpoint, and I know which user you are, I can just auto-add that channel to all the relevant groups (mentions of that user, for example). Handily, as WebSockets start off using the HTTP protocol, they have a lot of familiar features, including a path, GET parameters, and cookies. We'd like to use these to hook into the familiar Django session and authentication systems; after all, WebSockets are no good unless we can identify who they belong to and do things securely. In addition, we don't want the interface servers storing data or trying to run authentication; they're meant to be simple, lean, fast processes without much state, and so we'll need to do our authentication inside our consumer functions. Fortunately, because Channels has standardised WebSocket event :doc:`message-standards`, it ships with decorators that help you with both authentication and getting the underlying Django session (which is what Django authentication relies on). Channels can use Django sessions either from cookies (if you're running your websocket server on the same port as your main site, which requires a reverse proxy that understands WebSockets), or from a ``session_key`` GET parameter, which is much more portable, and works in development where you need to run a separate WebSocket server (by default, on port 9000). You get access to a user's normal Django session using the ``http_session`` decorator - that gives you a ``message.http_session`` attribute that behaves just like ``request.session``. You can go one further and use ``http_session_user`` which will provide a ``message.user`` attribute as well as the session attribute. Now, one thing to note is that you only get the detailed HTTP information during the ``connect`` message of a WebSocket connection (you can read more about what you get when in :doc:`message-standards`) - this means we're not wasting bandwidth sending the same information over the wire needlessly. This also means we'll have to grab the user in the connection handler and then store it in the session; thankfully, Channels ships with both a ``channel_session_user`` decorator that works like the ``http_session_user`` decorator you saw above but loads the user from the *channel* session rather than the *HTTP* session, and a function called ``transfer_user`` which replicates a user from one session to another. Bringing that all together, let's make a chat server where users can only chat to people with the same first letter of their username:: # In consumers.py from channels import Channel, Group from channels.decorators import channel_session from channels.auth import http_session_user, channel_session_user, transfer_user # Connected to websocket.connect @channel_session @http_session_user def ws_add(message): # Copy user from HTTP to channel session transfer_user(message.http_session, message.channel_session) # Add them to the right group Group("chat-%s" % message.user.username[0]).add(message.reply_channel) # Connected to websocket.keepalive @channel_session_user def ws_keepalive(message): # Keep them in the right group Group("chat-%s" % message.user.username[0]).add(message.reply_channel) # Connected to websocket.receive @channel_session_user def ws_message(message): Group("chat-%s" % message.user.username[0]).send(message.content) # Connected to websocket.disconnect @channel_session_user def ws_disconnect(message): Group("chat-%s" % message.user.username[0]).discard(message.reply_channel) Now, when we connect to the WebSocket we'll have to remember to provide the Django session ID as part of the URL, like this:: socket = new WebSocket("ws://127.0.0.1:9000/?session_key=abcdefg"); You can get the current session key in a template with ``{{ request.session.session_key }}``. Note that Channels can't work with signed cookie sessions - since only HTTP responses can set cookies, it needs a backend it can write to to separately store state. Models ------ So far, we've just been taking incoming messages and rebroadcasting them to other clients connected to the same group, but this isn't that great; really, we want to persist messages to a datastore, and we'd probably like to be able to inject messages into chatrooms from things other than WebSocket client connections (perhaps a built-in bot, or server status messages). Thankfully, we can just use Django's ORM to handle persistence of messages and easily integrate the send into the save flow of the model, rather than the message receive - that way, any new message saved will be broadcast to all the appropriate clients, no matter where it's saved from. We'll even take some performance considerations into account: We'll make our own custom channel for new chat messages and move the model save and the chat broadcast into that, meaning the sending process/consumer can move on immediately and not spend time waiting for the database save and the (slow on some backends) ``Group.send()`` call. Let's see what that looks like, assuming we have a ChatMessage model with ``message`` and ``room`` fields:: # In consumers.py from channels import Channel from channels.decorators import channel_session from .models import ChatMessage # Connected to chat-messages def msg_consumer(message): # Save to model ChatMessage.objects.create( room=message.content['room'], message=message.content['message'], ) # Broadcast to listening sockets Group("chat-%s" % room).send({ "content": message.content['message'], }) # Connected to websocket.connect @channel_session def ws_connect(message): # Work out room name from path (ignore slashes) room = message.content['path'].strip("/") # Save room in session and add us to the group message.channel_session['room'] = room Group("chat-%s" % room).add(message.reply_channel) # Connected to websocket.keepalive @channel_session def ws_add(message): Group("chat-%s" % message.channel_session['room']).add(message.reply_channel) # Connected to websocket.receive @channel_session def ws_message(message): # Stick the message onto the processing queue Channel("chat-messages").send({ "room": channel_session['room'], "message": content, }) # Connected to websocket.disconnect @channel_session def ws_disconnect(message): Group("chat-%s" % message.channel_session['room']).discard(message.reply_channel) Note that we could add messages onto the ``chat-messages`` channel from anywhere; inside a View, inside another model's ``post_save`` signal, inside a management command run via ``cron``. If we wanted to write a bot, too, we could put its listening logic inside the ``chat-messages`` consumer, as every message would pass through it. Linearization ------------- There's one final concept we want to introduce you to before you go on to build sites with Channels - linearizing consumers. Because Channels is a distributed system that can have many workers, by default it's entirely feasible for a WebSocket interface server to send out a ``connect`` and a ``receive`` message close enough together that a second worker will pick up and start processing the ``receive`` message before the first worker has finished processing the ``connect`` worker. This is particularly annoying if you're storing things in the session in the ``connect`` consumer and trying to get them in the ``receive`` consumer - because the ``connect`` consumer hasn't exited, its session hasn't saved. You'd get the same effect if someone tried to request a view before the login view had finished processing, but there you're not expecting that page to run after the login, whereas you'd naturally expect ``receive`` to run after ``connect``. But, of course, Channels has a solution - the ``linearize`` decorator. Any handler decorated with this will use locking to ensure it does not run at the same time as any other view with ``linearize`` **on messages with the same reply channel**. That means your site will happily multitask with lots of different people's messages, but if two happen to try to run at the same time for the same client, they'll be deconflicted. There's a small cost to using ``linearize``, which is why it's an optional decorator, but generally you'll want to use it for most session-based WebSocket and other "continuous protocol" things. Here's an example, improving our first-letter-of-username chat from earlier:: # In consumers.py from channels import Channel, Group from channels.decorators import channel_session, linearize from channels.auth import http_session_user, channel_session_user, transfer_user # Connected to websocket.connect @linearize @channel_session @http_session_user def ws_add(message): # Copy user from HTTP to channel session transfer_user(message.http_session, message.channel_session) # Add them to the right group Group("chat-%s" % message.user.username[0]).add(message.reply_channel) # Connected to websocket.keepalive # We don't linearize as we know this will happen a decent time after add @channel_session_user def ws_keepalive(message): # Keep them in the right group Group("chat-%s" % message.user.username[0]).add(message.reply_channel) # Connected to websocket.receive @linearize @channel_session_user def ws_message(message): Group("chat-%s" % message.user.username[0]).send(message.content) # Connected to websocket.disconnect # We don't linearize as even if this gets an empty session, the group # will auto-discard after the expiry anyway. @channel_session_user def ws_disconnect(message): Group("chat-%s" % message.user.username[0]).discard(message.reply_channel) Next Steps ---------- That covers the basics of using Channels; you've seen not only how to use basic channels, but also seen how they integrate with WebSockets, how to use groups to manage logical sets of channels, and how Django's session and authentication systems easily integrate with WebSockets. We recommend you read through the rest of the reference documentation to see all of what Channels has to offer; in particular, you may want to look at our :doc:`deploying` and :doc:`scaling` resources to get an idea of how to design and run apps in production environments.