Info: | See github for the latest source. |
---|---|
Author: | Alexandre Fiori <[email protected]> |
Cyclone is a low-level network toolkit, which provides support for HTTP 1.1 in an API very similar to the one implemented by the Tornado web server - which was developed by FriendFeed and later released as open source / free software by Facebook.
- Cyclone is based on Twisted, hence it may be used as a webservice protocol for interconnection with any other protocol implemented in Twisted.
- Localization is based upon the standard Gettext instead of the CSV implementation in the original Tornado. Moreover, it supports pluralization exactly like Tornado does.
- It ships with an asynchronous HTTP client based on TwistedWeb, however, it's fully compatible with one provided by Tornado - which is based on PyCurl. (The HTTP server code is NOT based on TwistedWeb, for several reasons)
- Native support for XMLRPC and JsonRPC. (see the rpc demo)
- WebSocket protocol class is just like any other Twisted Protocol (i.e.: LineReceiver; see the websocket demo)
- Support for sending e-mail based on Twisted Mail, with authentication and TLS, plus an easy way to create plain text or HTML messages, and attachments. (see the e-mail demo)
- Built-in support for Redis, based on txredisapi. We usually need an in-memory caching server like memcache for web applications. However, we prefer redis over memcache because it supports more operations like pubsub, various data types like sets, hashes (python dict), and persistent storage. See the redis demo for details.
- Support for HTTP Authentication. See the authentication demo for details.
- Easy deployment of applications, using twistd.
- RDBM support via: twisted.enterprise.adbapi.
- NoSQL support for MongoDB (TxMongo) and Redis (TxRedisAPI).
- May combine many more functionality within the webserver: sending emails, communicating with message brokers, etc...
Check out the benchmarks page.
As a clone, the API implemented in Cyclone is almost the same of Tornado. Therefore you may use the Tornado Documentation for stuff like templates and so on.
The snippets below will show some tips and tricks regarding the few differences between the two.
#!/usr/bin/env python # coding: utf-8 import sys import cyclone.web from twisted.python import log from twisted.internet import reactor class IndexHandler(cyclone.web.RequestHandler): def get(self): self.write("hello world") class Application(cyclone.web.Application): def __init__(self): handlers = [ (r"/", IndexHandler), ] settings = { "static_path": "./static", "template_path": "./template", } cyclone.web.Application.__init__(self, handlers, **settings) if __name__ == "__main__": log.startLogging(sys.stdout) reactor.listenTCP(8888, Application()) reactor.run()
The advantage of being a Twisted Application is that you don't need to care about basic daemon features like forking, creating pid files, changing application's user and group permissions, and selecting the proper reactor within the code.
Instead, the application may be run by twistd
, as follows:
for testing: /usr/bin/twistd --nodaemon --python=foobar.tac for production: /usr/bin/twistd --pidfile=/var/run/foobar.pid \ --logfile=/var/log/foobar.log \ --uid=nobody --gid=nobody \ --reactor=epoll \ --python=foobar.tac
Following is the Hello World as a twisted application:
# coding: utf-8 # twisted application: foobar.tac import cyclone.web from twisted.application import service, internet class IndexHandler(cyclone.web.RequestHandler): def get(self): self.write("hello world") foobar = cyclone.web.Application([(r"/", IndexHandler)]) application = service.Application("foobar") internet.TCPServer(8888, foobar(), interface="127.0.0.1").setServiceParent(application)
The cyclone.locale
provides an API similar to tornado.locale
, however, instead of using CSV files for translating strings like Tornado does, Cyclone uses the standard Python gettext
module.
Because of that, there is one extra option that may be passed to cyclone.locale.load_translations(path, domain="cyclone")
, which the is the gettext's domain. The default domain is cyclone.
Following is a step-by-step guide to implement localization in any Cyclone application:
Create a python script or twisted application with translatable strings:
# coding: utf-8 # twisted application: foobar.tac import cyclone.web import cyclone.locale from twisted.application import service, internet class BaseHandler(cyclone.web.RequestHandler): def get_user_locale(self): lang = self.get_cookie("lang") return cyclone.locale.get(lang) class IndexHandler(BaseHandler): def get(self): self.render("index.html") def post(self): _ = self.locale.translate name = self.get_argument("name") self.write(_("the name is: %s" % name)) class LangHandler(cyclone.web.RequestHandler): def get(self, lang): if lang in cyclone.locale.get_supported_locales(): self.set_cookie("lang", lang) self.redirect("/") class Application(cyclone.web.Application): def __init__(self): handlers = [ (r"/", IndexHandler), (r"/lang/(.+)", LangHandler), ] settings = { "static_path": "./static", "template_path": "./template", } cyclone.locale.load_translations("./locale", "foobar") cyclone.web.Application.__init__(self, handlers, **settings) application = service.Application("foobar") internet.TCPServer(8888, Application(), interface="127.0.0.1").setServiceParent(application)
Create a file in
./template/index.html
with translatable strings:<html> <body> <form action="/" method="post"> <p>{{ _("write someone's name:") }}</p> <input type="text" name="name"> <input type="submit" value="{{ _('send') }}"> </form> <br> <p>{{ _("change language:") }}</p> <p><a href="/lang/en_US">English (US)</a></p> <p><a href="/lang/pt_BR">Portuguese (BR)</a></p> </body> </html>
Generate PO translatable file from the source code, using
xgettext
:You will notice that
xgettext
cannot parse HTML properly. It was first designed to parse C files, and now it supports many other languages including Python.In order to parse lines like
<input type="submit" value="{{ _('send') }}">
, you'll need an extra script to pre-process the files.Here's what you can use as
fix.py
:#!/usr/bin/env python # coding: utf-8 # fix.py import re, sys if __name__ == "__main__": try: filename = sys.argv[1] assert filename != "-" fd = open(filename) except: fd = sys.stdin line_re = re.compile(r"""['"]{{|}}['"] """) for line in fd: line = line_re.sub(r"", line) sys.stdout.write(line) fd.close()
Then, call
xgettext
to generate the PO translatable file:cat foobar.tac template/index.html | python fix.py | \ xgettext --language=Python --keyword=_:1,2 -d foobar
This will create a file named
foobar.po
, which needs to be translated, then compiled into an MO file:vi foobar.po (translate everything, :wq) mkdir -p ./locale/pt_BR/LC_MESSAGES/ msgfmt foobar.po -o ./locale/pt_BR/LC_MESSAGES/foobar.mo
Finally, test the internationalized application:
twistd -ny foobar.tac
There is also a complete example with pluralization in demos/locale.
Tornado provides decorator functions for asynchronous and authenticated
methods. Obviously, they're also implemented in Cyclone, and yet more
powerful when combined with a famous Twisted decorator: defer.inlineCallbacks
.
The cyclone.web.authenticated
decorator may be combined with defer.inlineCallbacks
,
however, there's a basic rule to use them together. Considering that the authenticated
decorator will check user credentials, and, depending on the result, it will
continue processing the request OR redirect the request to the login page,
it has to be used before the defer.inlineCallbacks
to function properly:
class IndexHandler(cyclone.web.RequestHandler): @cyclone.web.authenticated @defer.inlineCallbacks def get(self): result = yield something() self.write(result)
On the other hand, the cyclone.web.asynchronous
decorator will keep the request open
until you explicitly call self.finish()
later on. Of course, it may also be combined
with defer.inlineCallbacks
, but it MUST be placed after to function properly:
class Indexhandler(cyclone.web.RequestHandler): @defer.inlineCallbacks @cyclone.web.asynchronous def get(self): result = yield something() self.finish(result)
Of course, you may combine the three decorators to have the most powerful and simple code in Cyclone, like this:
class Indexhandler(cyclone.web.RequestHandler): @cyclone.web.authenticated @defer.inlineCallbacks @cyclone.web.asynchronous def get(self): try: result = yield mongo.collection.find_one({"foo":"bar"}) except: self.finish("error or something") defer.returnValue(None) if not result: raise cyclone.web.HTTPError(404, "not found") self.finish(cyclone.escape.json_encode(result))
Keep-Alive
Because of the HTTP 1.1 support, sockets aren't always closed when you call
self.finish()
in a RequestHandler. Cyclone let you enforce that by setting theno_keep_alive
attribute attribute in some of your RequestHandlers:class IndexHandler(cyclone.web.RequestHandler): no_keep_alive = True def get(self): ...
Socket closed notification
One of the great features of TwistedWeb is the
request.notifyFinish()
, which is also available in Cyclone. This method returns a deferred which is fired when the request socket is closed, by eitherself.finish()
, someone closing their browser while receiving data, or closing the connection of a Comet request:class IndexHandler(cyclone.web.RequestHandler): def get(self): ... d = self.notifyFinish() d.addCallback(remove_from_comet_handlers_list)
HTTP X-Headers
When running a Cyclone-based application behind Nginx, it's very important to make it automatically use X-Real-Ip and X-Scheme HTTP headers. In order to make Cyclone recognize those headers, the option
xheaders=True
must be set in the Application settings:class Application(cyclone.web.Application): def __init__(self): handlers = [ (r"/", IndexHandler), ] settings = { "xheaders": True "static_path": "./static", } cyclone.web.Application.__init__(self, handlers, **settings)
Cookie-Secret generation
What I use to generate the "cookie_secrect" key used in cyclone.web.Application's settings is something pretty simple, like this:
>>> import uuid, base64 >>> base64.b64encode(uuid.uuid4().bytes + uuid.uuid4().bytes) 'FoQv5hgLTYCb9aKiBagpJJYtLJInWUcXilg3/vPkUnI='
Where are the request headers?
They are part of the request, dude:
class MyHandler(cyclone.web.RequestHandler): def get(self): # self.request.headers is a dict user_agent = self.request.headers.get("User-Agent")
How do I access raw POST data?
Both raw POST data and GET/DELETE un-parsed query string are available:
class MyHandler(cyclone.web.RequestHandler): def get(self): raw = self.request.query def post(self): raw = self.request.body
Where is the request information, like remote IP address, HTTP method, URI and version?
Everything is available as request attributes:
class MyHandler(cyclone.web.RequestHandler): def get(self): remote_ip = self.request.remote_ip method = self.request.method uri = self.request.uri version = self.request.version
How do I set my own headers for the reply?
Guess what, use self.set_header(name, value):
class MyHandler(cyclone.web.RequestHandler): def get(self): self.set_header("Content-Type", "application/json") self.finish(cyclone.escape.json_encode({"success":True}))
What HTTP methods are supported in RequestHandler?
Well, almost all of them. HEAD, GET, POST, DELETE and PUT are supported. TRACE is disabled by default, because it may get you in trouble. CONNECT has nothing to do with web servers, it's for proxies.
For more information on HTTP 1.1 methods, please refer to the RFC 2612 Fielding, et al.. For information regarding TRACE vulnerabilities, please check the following links: What is HTTP TRACE? and Apache Week, security issues.
Supporting different HTTP methods in the same RequestHandler is easy:
class MyHandler(cyclone.web.RequestHandler): def get(self): pass def head(self): pass ...
We've being using Cyclone for all of our private projects at nuswit.com. Now that it's very stable and responsive, we decided to make it freely available to the public, and hope it become more popular in the Python/Twisted community.
The source code ships with examples and demos.
Also, we've found that some people is already using it:
- RestMQ: a redis based message queue
- Brazilian Ministry of Education: using cyclone on the Digital Library project.
- Free IP Geolocation Web Service: open source geoip web service based on cyclone
Thanks to (in no particular order):
- Nuswit Telephony API
- Granting permission for this code to be published and sponsoring
- Gleicon Moraes
- Testing and using it in the RestMQ web service
- Vanderson Mota
- Patching setup.py and PyPi maintenance
- Andrew Badr
- Fixing auth bugs and adding current Tornado's features
- Jon Oberheide
- Syncing code with Tornado and security features/fixes
- Silas Sewell
- Syncing code and minor mail fix