2009-09-10 07:50:51 +00:00
|
|
|
#!/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.
|
|
|
|
|
2015-03-29 15:45:50 +00:00
|
|
|
import bcrypt
|
|
|
|
import concurrent.futures
|
2015-03-28 19:27:56 +00:00
|
|
|
import MySQLdb
|
2009-09-10 07:50:51 +00:00
|
|
|
import markdown
|
|
|
|
import os.path
|
|
|
|
import re
|
2015-03-28 19:27:56 +00:00
|
|
|
import subprocess
|
2012-09-16 21:02:49 +00:00
|
|
|
import torndb
|
2015-03-29 15:45:50 +00:00
|
|
|
import tornado.escape
|
|
|
|
from tornado import gen
|
2009-09-10 07:50:51 +00:00
|
|
|
import tornado.httpserver
|
|
|
|
import tornado.ioloop
|
|
|
|
import tornado.options
|
|
|
|
import tornado.web
|
|
|
|
import unicodedata
|
|
|
|
|
|
|
|
from tornado.options import define, options
|
|
|
|
|
|
|
|
define("port", default=8888, help="run on the given port", type=int)
|
|
|
|
define("mysql_host", default="127.0.0.1:3306", help="blog database host")
|
|
|
|
define("mysql_database", default="blog", help="blog database name")
|
|
|
|
define("mysql_user", default="blog", help="blog database user")
|
|
|
|
define("mysql_password", default="blog", help="blog database password")
|
|
|
|
|
|
|
|
|
2015-03-29 15:45:50 +00:00
|
|
|
# A thread pool to be used for password hashing with bcrypt.
|
|
|
|
executor = concurrent.futures.ThreadPoolExecutor(2)
|
|
|
|
|
|
|
|
|
2009-09-10 07:50:51 +00:00
|
|
|
class Application(tornado.web.Application):
|
|
|
|
def __init__(self):
|
|
|
|
handlers = [
|
|
|
|
(r"/", HomeHandler),
|
|
|
|
(r"/archive", ArchiveHandler),
|
|
|
|
(r"/feed", FeedHandler),
|
|
|
|
(r"/entry/([^/]+)", EntryHandler),
|
|
|
|
(r"/compose", ComposeHandler),
|
2015-03-29 15:45:50 +00:00
|
|
|
(r"/auth/create", AuthCreateHandler),
|
2009-09-10 07:50:51 +00:00
|
|
|
(r"/auth/login", AuthLoginHandler),
|
|
|
|
(r"/auth/logout", AuthLogoutHandler),
|
|
|
|
]
|
|
|
|
settings = dict(
|
|
|
|
blog_title=u"Tornado Blog",
|
|
|
|
template_path=os.path.join(os.path.dirname(__file__), "templates"),
|
|
|
|
static_path=os.path.join(os.path.dirname(__file__), "static"),
|
|
|
|
ui_modules={"Entry": EntryModule},
|
|
|
|
xsrf_cookies=True,
|
2012-09-01 06:47:11 +00:00
|
|
|
cookie_secret="__TODO:_GENERATE_YOUR_OWN_RANDOM_VALUE_HERE__",
|
2009-09-10 07:50:51 +00:00
|
|
|
login_url="/auth/login",
|
2013-03-03 17:09:29 +00:00
|
|
|
debug=True,
|
2009-09-10 07:50:51 +00:00
|
|
|
)
|
|
|
|
tornado.web.Application.__init__(self, handlers, **settings)
|
|
|
|
|
|
|
|
# Have one global connection to the blog DB across all handlers
|
2012-09-16 21:02:49 +00:00
|
|
|
self.db = torndb.Connection(
|
2009-09-10 07:50:51 +00:00
|
|
|
host=options.mysql_host, database=options.mysql_database,
|
|
|
|
user=options.mysql_user, password=options.mysql_password)
|
|
|
|
|
2015-03-28 19:27:56 +00:00
|
|
|
self.maybe_create_tables()
|
|
|
|
|
|
|
|
def maybe_create_tables(self):
|
|
|
|
try:
|
|
|
|
self.db.get("SELECT COUNT(*) from entries;")
|
|
|
|
except MySQLdb.ProgrammingError:
|
|
|
|
subprocess.check_call(['mysql',
|
|
|
|
'--host=' + options.mysql_host,
|
|
|
|
'--database=' + options.mysql_database,
|
|
|
|
'--user=' + options.mysql_user,
|
|
|
|
'--password=' + options.mysql_password],
|
|
|
|
stdin=open('schema.sql'))
|
|
|
|
|
2009-09-10 07:50:51 +00:00
|
|
|
|
|
|
|
class BaseHandler(tornado.web.RequestHandler):
|
|
|
|
@property
|
|
|
|
def db(self):
|
|
|
|
return self.application.db
|
|
|
|
|
|
|
|
def get_current_user(self):
|
2013-03-03 17:16:03 +00:00
|
|
|
user_id = self.get_secure_cookie("blogdemo_user")
|
2009-09-10 07:50:51 +00:00
|
|
|
if not user_id: return None
|
|
|
|
return self.db.get("SELECT * FROM authors WHERE id = %s", int(user_id))
|
|
|
|
|
2015-03-29 15:45:50 +00:00
|
|
|
def any_author_exists(self):
|
|
|
|
return bool(self.db.get("SELECT * FROM authors LIMIT 1"))
|
|
|
|
|
2009-09-10 07:50:51 +00:00
|
|
|
|
|
|
|
class HomeHandler(BaseHandler):
|
|
|
|
def get(self):
|
|
|
|
entries = self.db.query("SELECT * FROM entries ORDER BY published "
|
|
|
|
"DESC LIMIT 5")
|
|
|
|
if not entries:
|
|
|
|
self.redirect("/compose")
|
|
|
|
return
|
|
|
|
self.render("home.html", entries=entries)
|
|
|
|
|
|
|
|
|
|
|
|
class EntryHandler(BaseHandler):
|
|
|
|
def get(self, slug):
|
|
|
|
entry = self.db.get("SELECT * FROM entries WHERE slug = %s", slug)
|
|
|
|
if not entry: raise tornado.web.HTTPError(404)
|
|
|
|
self.render("entry.html", entry=entry)
|
|
|
|
|
|
|
|
|
|
|
|
class ArchiveHandler(BaseHandler):
|
|
|
|
def get(self):
|
|
|
|
entries = self.db.query("SELECT * FROM entries ORDER BY published "
|
|
|
|
"DESC")
|
|
|
|
self.render("archive.html", entries=entries)
|
|
|
|
|
|
|
|
|
|
|
|
class FeedHandler(BaseHandler):
|
|
|
|
def get(self):
|
|
|
|
entries = self.db.query("SELECT * FROM entries ORDER BY published "
|
|
|
|
"DESC LIMIT 10")
|
|
|
|
self.set_header("Content-Type", "application/atom+xml")
|
|
|
|
self.render("feed.xml", entries=entries)
|
|
|
|
|
|
|
|
|
|
|
|
class ComposeHandler(BaseHandler):
|
|
|
|
@tornado.web.authenticated
|
|
|
|
def get(self):
|
|
|
|
id = self.get_argument("id", None)
|
|
|
|
entry = None
|
|
|
|
if id:
|
|
|
|
entry = self.db.get("SELECT * FROM entries WHERE id = %s", int(id))
|
|
|
|
self.render("compose.html", entry=entry)
|
|
|
|
|
|
|
|
@tornado.web.authenticated
|
|
|
|
def post(self):
|
|
|
|
id = self.get_argument("id", None)
|
|
|
|
title = self.get_argument("title")
|
|
|
|
text = self.get_argument("markdown")
|
|
|
|
html = markdown.markdown(text)
|
|
|
|
if id:
|
|
|
|
entry = self.db.get("SELECT * FROM entries WHERE id = %s", int(id))
|
|
|
|
if not entry: raise tornado.web.HTTPError(404)
|
|
|
|
slug = entry.slug
|
|
|
|
self.db.execute(
|
|
|
|
"UPDATE entries SET title = %s, markdown = %s, html = %s "
|
|
|
|
"WHERE id = %s", title, text, html, int(id))
|
|
|
|
else:
|
|
|
|
slug = unicodedata.normalize("NFKD", title).encode(
|
|
|
|
"ascii", "ignore")
|
|
|
|
slug = re.sub(r"[^\w]+", " ", slug)
|
|
|
|
slug = "-".join(slug.lower().strip().split())
|
|
|
|
if not slug: slug = "entry"
|
|
|
|
while True:
|
|
|
|
e = self.db.get("SELECT * FROM entries WHERE slug = %s", slug)
|
|
|
|
if not e: break
|
|
|
|
slug += "-2"
|
|
|
|
self.db.execute(
|
|
|
|
"INSERT INTO entries (author_id,title,slug,markdown,html,"
|
|
|
|
"published) VALUES (%s,%s,%s,%s,%s,UTC_TIMESTAMP())",
|
|
|
|
self.current_user.id, title, slug, text, html)
|
|
|
|
self.redirect("/entry/" + slug)
|
|
|
|
|
|
|
|
|
2015-03-29 15:45:50 +00:00
|
|
|
class AuthCreateHandler(BaseHandler):
|
2009-09-10 07:50:51 +00:00
|
|
|
def get(self):
|
2015-03-29 15:45:50 +00:00
|
|
|
self.render("create_author.html")
|
|
|
|
|
|
|
|
@gen.coroutine
|
|
|
|
def post(self):
|
|
|
|
if self.any_author_exists():
|
|
|
|
raise tornado.web.HTTPError(400, "author already created")
|
|
|
|
hashed_password = yield executor.submit(
|
|
|
|
bcrypt.hashpw, tornado.escape.utf8(self.get_argument("password")),
|
|
|
|
bcrypt.gensalt())
|
|
|
|
author_id = self.db.execute(
|
|
|
|
"INSERT INTO authors (email, name, hashed_password) "
|
|
|
|
"VALUES (%s, %s, %s)",
|
|
|
|
self.get_argument("email"), self.get_argument("name"),
|
|
|
|
hashed_password)
|
|
|
|
self.set_secure_cookie("blogdemo_user", str(author_id))
|
|
|
|
self.redirect(self.get_argument("next", "/"))
|
2012-09-16 21:02:49 +00:00
|
|
|
|
2015-03-29 15:45:50 +00:00
|
|
|
|
|
|
|
class AuthLoginHandler(BaseHandler):
|
|
|
|
def get(self):
|
|
|
|
# If there are no authors, redirect to the account creation page.
|
|
|
|
if not self.any_author_exists():
|
|
|
|
self.redirect("/auth/create")
|
|
|
|
else:
|
|
|
|
self.render("login.html", error=None)
|
|
|
|
|
|
|
|
@gen.coroutine
|
|
|
|
def post(self):
|
2009-09-10 07:50:51 +00:00
|
|
|
author = self.db.get("SELECT * FROM authors WHERE email = %s",
|
2015-03-29 15:45:50 +00:00
|
|
|
self.get_argument("email"))
|
2009-09-10 07:50:51 +00:00
|
|
|
if not author:
|
2015-03-29 15:45:50 +00:00
|
|
|
self.render("login.html", error="email not found")
|
|
|
|
return
|
|
|
|
hashed_password = yield executor.submit(
|
|
|
|
bcrypt.hashpw, tornado.escape.utf8(self.get_argument("password")),
|
|
|
|
tornado.escape.utf8(author.hashed_password))
|
|
|
|
if hashed_password == author.hashed_password:
|
|
|
|
self.set_secure_cookie("blogdemo_user", str(author.id))
|
|
|
|
self.redirect(self.get_argument("next", "/"))
|
2009-09-10 07:50:51 +00:00
|
|
|
else:
|
2015-03-29 15:45:50 +00:00
|
|
|
self.render("login.html", error="incorrect password")
|
2009-09-10 07:50:51 +00:00
|
|
|
|
|
|
|
|
|
|
|
class AuthLogoutHandler(BaseHandler):
|
|
|
|
def get(self):
|
2013-03-03 17:16:03 +00:00
|
|
|
self.clear_cookie("blogdemo_user")
|
2009-09-10 07:50:51 +00:00
|
|
|
self.redirect(self.get_argument("next", "/"))
|
|
|
|
|
|
|
|
|
|
|
|
class EntryModule(tornado.web.UIModule):
|
|
|
|
def render(self, entry):
|
|
|
|
return self.render_string("modules/entry.html", entry=entry)
|
|
|
|
|
|
|
|
|
|
|
|
def main():
|
|
|
|
tornado.options.parse_command_line()
|
|
|
|
http_server = tornado.httpserver.HTTPServer(Application())
|
|
|
|
http_server.listen(options.port)
|
2015-03-29 20:22:26 +00:00
|
|
|
tornado.ioloop.IOLoop.current().start()
|
2009-09-10 07:50:51 +00:00
|
|
|
|
|
|
|
|
|
|
|
if __name__ == "__main__":
|
|
|
|
main()
|