import cgi try: from http.cookies import SimpleCookie except ImportError: from Cookie import SimpleCookie import datetime import db import mimetypes import os import random import settings import jinja2 try: import markdown has_markdown = True except ImportError: has_markdown = False import pygments from pygments import highlight from pygments.lexers import get_all_lexers, get_lexer_by_name from pygments.formatters import HtmlFormatter base62_alphabet = '0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ' class PasteError(Exception): pass class UnknownSyntaxError(PasteError): pass class CustomHtmlFormatter(HtmlFormatter): def wrap(self, source, outfile): yield 0, '
'
line = 1
for i, t in source:
yield i, '%s' % (line, t)
line += 1
yield 0, '
'
class Paste(object):
def __init__(self):
lexers = dict([(x[0], x[1][0]) for x in get_all_lexers()])
self.lexers = []
removed = []
for cat, ls in settings.categories:
for l in ls:
removed.append(l)
self.lexers.append((cat, [(x, lexers[x]) for x in ls]))
for l in removed:
try:
del lexers[l]
except KeyError:
pass
rendered = []
if has_markdown:
rendered.append(('Rendered markdown', 'md-render'))
if len(rendered):
self.lexers.append(('Rendered', rendered))
self.lexers.append(('Others' if settings.categories else 'Syntax', sorted(lexers.items(), key = lambda l: l[0].lower())))
self.formatter = CustomHtmlFormatter(linenos = 'table', lineanchors = 'line', anchorlinenos = True)
random.seed()
self.template_env = jinja2.Environment(loader = jinja2.FileSystemLoader('templates'))
def render_template(self, name, args):
template = self.template_env.get_template(name)
data = template.render(**args).encode('utf-8')
self.start_response('200 OK', [('Content-Type', 'text/html; charset=UTF-8'), ('Content-Length', str(len(data)))])
return [data]
def message(self, msg, title = 'Message'):
return self.render_template('message.html', {
'title': '%s – %s' % (settings.pastebin_name, title),
'header': title,
'text': msg,
})
def paste(self):
c = SimpleCookie(self.environ['HTTP_COOKIE'] if 'HTTP_COOKIE' in self.environ else None)
if self.environ['REQUEST_METHOD'] == 'POST':
mp = cgi.FieldStorage(fp = self.environ['wsgi.input'], environ = self.environ, keep_blank_values = True)
if mp['type'].value == 'Preview':
return self.preview(mp)
elif mp['type'].value == 'Paste':
return self.add_paste(mp)
else:
return self.message('Invalid type "%s".' % mp['type'].value, 'Error')
return self.render_template('paste.html', {
'title': settings.pastebin_name,
'header': settings.pastebin_name,
'lexers': self.lexers,
'nick': c['nick'].value if 'nick' in c else 'Anonymous',
'syntax': c['syntax'].value if 'syntax' in c else settings.default,
'remember_me': 'nick' in c,
'remember_syntax': 'syntax' in c,
})
def preview(self, mp):
try:
lexername, text = self.get_formatted(mp['syntax'].value.decode('utf-8'), mp['text'].value.decode('utf-8'))
except UnknownSyntaxError:
return self.message('Could not find lexer "%s".' % mp['syntax'].value, 'Error')
return self.render_template('view.html', {
'title': settings.pastebin_name,
'header': '%s – Preview' % settings.pastebin_name,
'hash': None,
'date': datetime.datetime.utcnow().ctime(),
'nick': mp['nick'].value.decode('utf-8') or 'Anoynmous',
'syntax': lexername,
'pastetitle': mp['title'].value.decode('utf-8') or 'Untitled',
'text': text,
'rendered': (lexername or '').startswith('Rendered '),
})
def add_paste(self, mp):
nick = mp['nick'].value.decode('utf-8') or None
syntax = mp['syntax'].value.decode('utf-8') or None
title = mp['title'].value.decode('utf-8') or None
text = mp['text'].value.decode('utf-8').replace('\r', '') or None
hash = ''.join(random.choice(base62_alphabet) for x in range(5))
try:
session = db.Session()
paste = db.Paste(hash, nick, datetime.datetime.utcnow(), syntax, title, text, self.environ['REMOTE_ADDR'])
session.add(paste)
session.commit()
finally:
session.close()
headers = [('Location', '/view/%s' % hash)]
c = SimpleCookie()
c['nick'] = nick
dt = (datetime.datetime.utcnow() + datetime.timedelta(days = 30)) if 'remember_me' in mp else datetime.datetime.utcfromtimestamp(0)
c['nick']['expires'] = dt.strftime('%a, %d-%b-%y %H:%M:%S GMT')
c['syntax'] = syntax
dt = (datetime.datetime.utcnow() + datetime.timedelta(days = 30)) if 'remember_syntax' in mp else datetime.datetime.utcfromtimestamp(0)
c['syntax']['expires'] = dt.strftime('%a, %d-%b-%y %H:%M:%S GMT')
headers.append(('Set-Cookie', c.output()))
self.start_response('302 Found', headers)
return []
def get_formatted(self, syntax, text):
if syntax == 'md-render' and has_markdown:
text = markdown.markdown(text, extensions = settings.markdown_extensions)
lexername = 'Rendered markdown'
else:
try:
lex = get_lexer_by_name(syntax or 'text')
except pygments.util.ClassNotFound:
raise UnknownSyntaxError(syntax)
lexername = lex.name
text = highlight(text, lex, self.formatter)
return (lexername, text)
def view(self):
hash = self.path[1]
try:
session = db.Session()
try:
cache = session.query(db.Cache).filter_by(paste_hash = hash).one()
paste = cache.paste
except db.NoResultFound: # No cache found, generate it.
try:
paste = session.query(db.Paste).filter_by(hash = hash).one()
except db.NoResultFound:
self.start_response('404 Not Found', [])
return []
try:
lexername, text = self.get_formatted(paste.syntax, paste.text if type(paste.text) == unicode else paste.text.decode('utf-8'))
except UnknownSyntaxError:
return self.message('Could not find the lexer "%s".' % paste.syntax, 'Error')
cache = db.Cache(hash, lexername, text)
session.add(cache)
session.commit()
# Workaround for attribute refresh.
paste = cache.paste
finally:
session.close()
return self.render_template('view.html', {
'title': '%s – View paste – %s' % (settings.pastebin_name, paste.title or 'Untitled'),
'header': '%s – View paste' % settings.pastebin_name,
'hash': hash,
'date': paste.date.ctime(),
'nick': paste.nick or 'Anonymous',
'syntax': cache.syntax_name,
'pastetitle': paste.title or 'Untitled',
'text': cache.text,
'rendered': (cache.syntax_name or '').startswith('Rendered '),
})
def raw(self):
hash = self.path[1]
try:
session = db.Session()
paste = session.query(db.Paste).filter_by(hash = hash).one()
finally:
session.close()
self.start_response('200 OK', [('Content-Type', 'text/plain; charset=UTF-8')])
return [paste.text.encode('utf-8')]
def highlight_stylesheet(self):
self.start_response('200 OK', [('Content-Type', 'text/css')])
return [self.formatter.get_style_defs()]
def static(self):
filename = settings.static_root + os.path.sep + self.path[1]
if not self.path[1] in ('paste.css', 'edit.js', 'view.js') or not os.path.exists(filename):
self.start_response('404 Not Found', [('Content-Type', 'text/html; charset=UTF-8'), ('Location', '/')])
return ['asdf']
self.start_response('200 OK', [('Content-Type', mimetypes.guess_type(filename)[0] or 'text/plain; charset=UTF-8')])
return open(filename, 'rb')
def __call__(self, environ, start_response):
self.environ = environ
self.start_response = start_response
path = self.environ['PATH_INFO'].split('/')[1:]
module = path[0] or 'paste'
if module in ('list', 'paste', 'view', 'raw', 'static', 'highlight_stylesheet'):
self.path = path
return getattr(self, module)()
else:
return self.message('Invalid module requested.', 'Error')
if __name__ == '__main__':
from wsgiref.simple_server import make_server
httpd = make_server('', 8000, Paste())
httpd.serve_forever()
# vim: noet ts=4