Add simple test application for websockets
This commit is contained in:
parent
c5b5a6ad1b
commit
cad4a30879
|
@ -0,0 +1,101 @@
|
||||||
|
#!/usr/bin/env python
|
||||||
|
#
|
||||||
|
# Copyright 2009 Facebook
|
||||||
|
#
|
||||||
|
# Licensed under the Apache License, Version 2.0 (the "License"); you may
|
||||||
|
# not use this file except in compliance with the License. You may obtain
|
||||||
|
# a copy of the License at
|
||||||
|
#
|
||||||
|
# http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
#
|
||||||
|
# Unless required by applicable law or agreed to in writing, software
|
||||||
|
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
||||||
|
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
||||||
|
# License for the specific language governing permissions and limitations
|
||||||
|
# under the License.
|
||||||
|
"""Simplified chat demo for websockets.
|
||||||
|
|
||||||
|
Authentication, error handling, etc are left as an exercise for the reader :)
|
||||||
|
"""
|
||||||
|
|
||||||
|
import logging
|
||||||
|
import tornado.escape
|
||||||
|
import tornado.ioloop
|
||||||
|
import tornado.options
|
||||||
|
import tornado.web
|
||||||
|
import tornado.websocket
|
||||||
|
import os.path
|
||||||
|
import uuid
|
||||||
|
|
||||||
|
from tornado.options import define, options
|
||||||
|
|
||||||
|
define("port", default=8888, help="run on the given port", type=int)
|
||||||
|
|
||||||
|
|
||||||
|
class Application(tornado.web.Application):
|
||||||
|
def __init__(self):
|
||||||
|
handlers = [
|
||||||
|
(r"/", MainHandler),
|
||||||
|
(r"/chatsocket", ChatSocketHandler),
|
||||||
|
]
|
||||||
|
settings = dict(
|
||||||
|
cookie_secret="43oETzKXQAGaYdkL5gEmGeJJFuYh7EQnp2XdTP1o/Vo=",
|
||||||
|
template_path=os.path.join(os.path.dirname(__file__), "templates"),
|
||||||
|
static_path=os.path.join(os.path.dirname(__file__), "static"),
|
||||||
|
xsrf_cookies=True,
|
||||||
|
)
|
||||||
|
tornado.web.Application.__init__(self, handlers, **settings)
|
||||||
|
|
||||||
|
|
||||||
|
class MainHandler(tornado.web.RequestHandler):
|
||||||
|
def get(self):
|
||||||
|
self.render("index.html", messages=ChatSocketHandler.cache)
|
||||||
|
|
||||||
|
class ChatSocketHandler(tornado.websocket.WebSocketHandler):
|
||||||
|
waiters = set()
|
||||||
|
cache = []
|
||||||
|
cache_size = 200
|
||||||
|
|
||||||
|
def open(self):
|
||||||
|
ChatSocketHandler.waiters.add(self)
|
||||||
|
|
||||||
|
def on_close(self):
|
||||||
|
ChatSocketHandler.waiters.remove(self)
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def update_cache(cls, chat):
|
||||||
|
cls.cache.append(chat)
|
||||||
|
if len(cls.cache) > cls.cache_size:
|
||||||
|
cls.cache = cls.cache[-cls.cache_size:]
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def send_updates(cls, chat):
|
||||||
|
logging.info("sending message to %d waiters", len(cls.waiters))
|
||||||
|
for waiter in cls.waiters:
|
||||||
|
try:
|
||||||
|
waiter.write_message(chat)
|
||||||
|
except:
|
||||||
|
logging.error("Error sending message", exc_info=True)
|
||||||
|
|
||||||
|
def on_message(self, message):
|
||||||
|
logging.info("got message %r", message)
|
||||||
|
parsed = tornado.escape.json_decode(message)
|
||||||
|
chat = {
|
||||||
|
"id": str(uuid.uuid4()),
|
||||||
|
"body": parsed["body"],
|
||||||
|
}
|
||||||
|
chat["html"] = self.render_string("message.html", message=chat)
|
||||||
|
|
||||||
|
ChatSocketHandler.update_cache(chat)
|
||||||
|
ChatSocketHandler.send_updates(chat)
|
||||||
|
|
||||||
|
|
||||||
|
def main():
|
||||||
|
tornado.options.parse_command_line()
|
||||||
|
app = Application()
|
||||||
|
app.listen(options.port)
|
||||||
|
tornado.ioloop.IOLoop.instance().start()
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
main()
|
|
@ -0,0 +1,56 @@
|
||||||
|
/*
|
||||||
|
* Copyright 2009 FriendFeed
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License"); you may
|
||||||
|
* not use this file except in compliance with the License. You may obtain
|
||||||
|
* a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
||||||
|
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
||||||
|
* License for the specific language governing permissions and limitations
|
||||||
|
* under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
body {
|
||||||
|
background: white;
|
||||||
|
margin: 10px;
|
||||||
|
}
|
||||||
|
|
||||||
|
body,
|
||||||
|
input {
|
||||||
|
font-family: sans-serif;
|
||||||
|
font-size: 10pt;
|
||||||
|
color: black;
|
||||||
|
}
|
||||||
|
|
||||||
|
table {
|
||||||
|
border-collapse: collapse;
|
||||||
|
border: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
td {
|
||||||
|
border: 0;
|
||||||
|
padding: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
#body {
|
||||||
|
position: absolute;
|
||||||
|
bottom: 10px;
|
||||||
|
left: 10px;
|
||||||
|
}
|
||||||
|
|
||||||
|
#input {
|
||||||
|
margin-top: 0.5em;
|
||||||
|
}
|
||||||
|
|
||||||
|
#inbox .message {
|
||||||
|
padding-top: 0.25em;
|
||||||
|
}
|
||||||
|
|
||||||
|
#nav {
|
||||||
|
float: right;
|
||||||
|
z-index: 99;
|
||||||
|
}
|
|
@ -0,0 +1,67 @@
|
||||||
|
// Copyright 2009 FriendFeed
|
||||||
|
//
|
||||||
|
// Licensed under the Apache License, Version 2.0 (the "License"); you may
|
||||||
|
// not use this file except in compliance with the License. You may obtain
|
||||||
|
// a copy of the License at
|
||||||
|
//
|
||||||
|
// http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
//
|
||||||
|
// Unless required by applicable law or agreed to in writing, software
|
||||||
|
// distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
||||||
|
// WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
||||||
|
// License for the specific language governing permissions and limitations
|
||||||
|
// under the License.
|
||||||
|
|
||||||
|
$(document).ready(function() {
|
||||||
|
if (!window.console) window.console = {};
|
||||||
|
if (!window.console.log) window.console.log = function() {};
|
||||||
|
|
||||||
|
$("#messageform").live("submit", function() {
|
||||||
|
newMessage($(this));
|
||||||
|
return false;
|
||||||
|
});
|
||||||
|
$("#messageform").live("keypress", function(e) {
|
||||||
|
if (e.keyCode == 13) {
|
||||||
|
newMessage($(this));
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
$("#message").select();
|
||||||
|
updater.start();
|
||||||
|
});
|
||||||
|
|
||||||
|
function newMessage(form) {
|
||||||
|
var message = form.formToDict();
|
||||||
|
updater.socket.send(JSON.stringify(message));
|
||||||
|
form.find("input[type=text]").val("").select();
|
||||||
|
}
|
||||||
|
|
||||||
|
jQuery.fn.formToDict = function() {
|
||||||
|
var fields = this.serializeArray();
|
||||||
|
var json = {}
|
||||||
|
for (var i = 0; i < fields.length; i++) {
|
||||||
|
json[fields[i].name] = fields[i].value;
|
||||||
|
}
|
||||||
|
if (json.next) delete json.next;
|
||||||
|
return json;
|
||||||
|
};
|
||||||
|
|
||||||
|
var updater = {
|
||||||
|
socket: null,
|
||||||
|
|
||||||
|
start: function() {
|
||||||
|
updater.socket = new WebSocket("ws://localhost:8888/chatsocket");
|
||||||
|
updater.socket.onmessage = function(event) {
|
||||||
|
updater.showMessage(JSON.parse(event.data));
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
showMessage: function(message) {
|
||||||
|
var existing = $("#m" + message.id);
|
||||||
|
if (existing.length > 0) return;
|
||||||
|
var node = $(message.html);
|
||||||
|
node.hide();
|
||||||
|
$("#inbox").append(node);
|
||||||
|
node.slideDown();
|
||||||
|
}
|
||||||
|
};
|
|
@ -0,0 +1,33 @@
|
||||||
|
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Strict//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-strict.dtd">
|
||||||
|
<html xmlns="http://www.w3.org/1999/xhtml">
|
||||||
|
<head>
|
||||||
|
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8"/>
|
||||||
|
<title>Tornado Chat Demo</title>
|
||||||
|
<link rel="stylesheet" href="{{ static_url("chat.css") }}" type="text/css"/>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<div id="body">
|
||||||
|
<div id="inbox">
|
||||||
|
{% for message in messages %}
|
||||||
|
{% include "message.html" %}
|
||||||
|
{% end %}
|
||||||
|
</div>
|
||||||
|
<div id="input">
|
||||||
|
<form action="/a/message/new" method="post" id="messageform">
|
||||||
|
<table>
|
||||||
|
<tr>
|
||||||
|
<td><input name="body" id="message" style="width:500px"/></td>
|
||||||
|
<td style="padding-left:5px">
|
||||||
|
<input type="submit" value="{{ _("Post") }}"/>
|
||||||
|
<input type="hidden" name="next" value="{{ request.path }}"/>
|
||||||
|
{{ xsrf_form_html() }}
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
</table>
|
||||||
|
</form>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<script src="http://ajax.googleapis.com/ajax/libs/jquery/1.3/jquery.min.js" type="text/javascript"></script>
|
||||||
|
<script src="{{ static_url("chat.js") }}" type="text/javascript"></script>
|
||||||
|
</body>
|
||||||
|
</html>
|
|
@ -0,0 +1,2 @@
|
||||||
|
{% import tornado.escape %}
|
||||||
|
<div class="message" id="m{{ message["id"] }}">{{ tornado.escape.linkify(message["body"]) }}</div>
|
Loading…
Reference in New Issue