-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathapp_websocket.py
104 lines (86 loc) · 3.37 KB
/
app_websocket.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
#!/usr/bin/python
# -*- coding: utf-8 -*-
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=8889, 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='insert_good_cookie',
template_path=os.path.join(os.path.dirname(__file__), "templates"),
static_path=os.path.join(os.path.dirname(__file__), "static"),
xsrf_cookies=True,
debug=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 = 100
# def get_compression_options(self):
# return {}
def open(self):
"""
Выполняется в момент соединения клиента и сервера
в данном случае в момент загрузки web страницы.
"""
logging.info('websocket client +1')
ChatSocketHandler.waiters.add(self)
def on_close(self):
""" Выполняется в момент закрытия или обновления web страницы. """
logging.info('websocket client -1')
ChatSocketHandler.waiters.remove(self)
@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)
@classmethod
def update_cache(cls, chat):
"""
Псевдо хранение сообщений в оперативной памяти
с пулом в 100 последних сообщений
"""
cls.cache.append(chat)
if len(cls.cache) > cls.cache_size:
cls.cache = cls.cache[-cls.cache_size:]
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'] = tornado.escape.to_basestring(
self.render_string('message.html', message=chat)
)
# put it in cache and give to listeners
ChatSocketHandler.update_cache(chat)
ChatSocketHandler.send_updates(chat)
def main():
logging.info("Server started")
tornado.options.parse_command_line()
app = Application()
app.listen(options.port)
tornado.ioloop.IOLoop.instance().start()
if __name__ == '__main__':
main()