3
# Copyright 2009 Facebook
5
# Licensed under the Apache License, Version 2.0 (the "License"); you may
6
# not use this file except in compliance with the License. You may obtain
7
# a copy of the License at
9
# http://www.apache.org/licenses/LICENSE-2.0
11
# Unless required by applicable law or agreed to in writing, software
12
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
13
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
14
# License for the specific language governing permissions and limitations
24
import wsgiref.handlers
26
from google.appengine.api import users
27
from google.appengine.ext import db
30
class Entry(db.Model):
31
"""A single blog entry."""
32
author = db.UserProperty()
33
title = db.StringProperty(required=True)
34
slug = db.StringProperty(required=True)
35
markdown = db.TextProperty(required=True)
36
html = db.TextProperty(required=True)
37
published = db.DateTimeProperty(auto_now_add=True)
38
updated = db.DateTimeProperty(auto_now=True)
41
def administrator(method):
42
"""Decorate with this method to restrict to site admins."""
43
@functools.wraps(method)
44
def wrapper(self, *args, **kwargs):
45
if not self.current_user:
46
if self.request.method == "GET":
47
self.redirect(self.get_login_url())
49
raise tornado.web.HTTPError(403)
50
elif not self.current_user.administrator:
51
if self.request.method == "GET":
54
raise tornado.web.HTTPError(403)
56
return method(self, *args, **kwargs)
60
class BaseHandler(tornado.web.RequestHandler):
61
"""Implements Google Accounts authentication methods."""
62
def get_current_user(self):
63
user = users.get_current_user()
64
if user: user.administrator = users.is_current_user_admin()
67
def get_login_url(self):
68
return users.create_login_url(self.request.uri)
70
def render_string(self, template_name, **kwargs):
71
# Let the templates access the users module to generate login URLs
72
return tornado.web.RequestHandler.render_string(
73
self, template_name, users=users, **kwargs)
76
class HomeHandler(BaseHandler):
78
entries = db.Query(Entry).order('-published').fetch(limit=5)
80
if not self.current_user or self.current_user.administrator:
81
self.redirect("/compose")
83
self.render("home.html", entries=entries)
86
class EntryHandler(BaseHandler):
88
entry = db.Query(Entry).filter("slug =", slug).get()
89
if not entry: raise tornado.web.HTTPError(404)
90
self.render("entry.html", entry=entry)
93
class ArchiveHandler(BaseHandler):
95
entries = db.Query(Entry).order('-published')
96
self.render("archive.html", entries=entries)
99
class FeedHandler(BaseHandler):
101
entries = db.Query(Entry).order('-published').fetch(limit=10)
102
self.set_header("Content-Type", "application/atom+xml")
103
self.render("feed.xml", entries=entries)
106
class ComposeHandler(BaseHandler):
109
key = self.get_argument("key", None)
110
entry = Entry.get(key) if key else None
111
self.render("compose.html", entry=entry)
115
key = self.get_argument("key", None)
117
entry = Entry.get(key)
118
entry.title = self.get_argument("title")
119
entry.markdown = self.get_argument("markdown")
120
entry.html = markdown.markdown(self.get_argument("markdown"))
122
title = self.get_argument("title")
123
slug = unicodedata.normalize("NFKD", title).encode(
125
slug = re.sub(r"[^\w]+", " ", slug)
126
slug = "-".join(slug.lower().strip().split())
127
if not slug: slug = "entry"
129
existing = db.Query(Entry).filter("slug =", slug).get()
130
if not existing or str(existing.key()) == key:
134
author=self.current_user,
137
markdown=self.get_argument("markdown"),
138
html=markdown.markdown(self.get_argument("markdown")),
141
self.redirect("/entry/" + entry.slug)
144
class EntryModule(tornado.web.UIModule):
145
def render(self, entry):
146
return self.render_string("modules/entry.html", entry=entry)
150
"blog_title": u"Tornado Blog",
151
"template_path": os.path.join(os.path.dirname(__file__), "templates"),
152
"ui_modules": {"Entry": EntryModule},
153
"xsrf_cookies": True,
155
application = tornado.wsgi.WSGIApplication([
157
(r"/archive", ArchiveHandler),
158
(r"/feed", FeedHandler),
159
(r"/entry/([^/]+)", EntryHandler),
160
(r"/compose", ComposeHandler),
165
wsgiref.handlers.CGIHandler().run(application)
168
if __name__ == "__main__":