2012-05-07 10:06:09 +04:00
|
|
|
#!/usr/bin/env python
|
2012-04-24 14:22:59 +04:00
|
|
|
# -*- coding: utf-8 -*-
|
2012-05-07 10:06:09 +04:00
|
|
|
# vim: ai ts=4 sts=4 et sw=4
|
2012-04-24 14:22:59 +04:00
|
|
|
|
2012-04-26 23:19:12 +04:00
|
|
|
"""
|
|
|
|
Main script including controller, rooting, dependancy management, and
|
|
|
|
server run.
|
|
|
|
"""
|
|
|
|
|
2012-04-29 13:50:44 +04:00
|
|
|
import sys
|
2012-04-24 14:22:59 +04:00
|
|
|
import os
|
2012-04-24 22:15:38 +04:00
|
|
|
import hashlib
|
2012-04-29 02:07:25 +04:00
|
|
|
import thread
|
2012-04-29 22:10:04 +04:00
|
|
|
import math
|
2012-04-24 14:22:59 +04:00
|
|
|
|
2012-04-27 01:41:20 +04:00
|
|
|
from datetime import datetime, timedelta
|
|
|
|
|
2012-05-06 21:24:46 +04:00
|
|
|
# add project dir and libs dir to the PYTHON PATH to ensure they are
|
|
|
|
# importable
|
2012-05-01 16:21:21 +04:00
|
|
|
import settings
|
|
|
|
sys.path.insert(0, os.path.dirname(settings.ROOT_DIR))
|
|
|
|
sys.path.append(os.path.join(settings.ROOT_DIR, 'libs'))
|
2012-04-29 02:07:25 +04:00
|
|
|
|
2012-05-01 16:21:21 +04:00
|
|
|
import bottle
|
2012-05-02 12:46:51 +04:00
|
|
|
from bottle import (Bottle, route, run, abort, error,
|
2012-04-26 23:19:12 +04:00
|
|
|
static_file, debug, view, request)
|
2012-04-24 14:22:59 +04:00
|
|
|
|
2012-05-01 16:21:21 +04:00
|
|
|
import clize
|
2012-04-24 22:15:38 +04:00
|
|
|
|
2012-05-06 21:24:46 +04:00
|
|
|
from src.paste import Paste
|
|
|
|
from src.utils import drop_privileges
|
2012-04-24 22:15:38 +04:00
|
|
|
|
2012-05-01 16:21:21 +04:00
|
|
|
|
|
|
|
app = Bottle()
|
2012-05-02 13:36:57 +04:00
|
|
|
|
2012-04-24 22:15:38 +04:00
|
|
|
|
|
|
|
@app.route('/')
|
2012-04-24 14:22:59 +04:00
|
|
|
@view('home')
|
|
|
|
def index():
|
2012-05-01 17:46:18 +04:00
|
|
|
return {'max_size': settings.MAX_SIZE, 'max_size_kb': settings.MAX_SIZE_KB}
|
2012-04-24 14:22:59 +04:00
|
|
|
|
|
|
|
|
2012-04-24 22:15:38 +04:00
|
|
|
@app.route('/paste/create', method='POST')
|
|
|
|
def create_paste():
|
|
|
|
|
|
|
|
try:
|
|
|
|
content = unicode(request.forms.get('content', ''), 'utf8')
|
|
|
|
except UnicodeDecodeError:
|
2012-04-27 01:41:20 +04:00
|
|
|
return {'status': 'error',
|
|
|
|
'message': u"Encoding error: the paste couldn't be saved."}
|
2012-04-24 22:15:38 +04:00
|
|
|
|
2012-04-30 16:33:27 +04:00
|
|
|
if '{"iv":' not in content: # reject silently non encrypted content
|
|
|
|
return ''
|
|
|
|
|
2012-04-24 22:15:38 +04:00
|
|
|
if content:
|
2012-05-01 17:20:10 +04:00
|
|
|
# check size of the paste. if more than settings return error without saving paste.
|
|
|
|
# prevent from unusual use of the system.
|
|
|
|
# need to be improved
|
|
|
|
if len(content) < settings.MAX_SIZE:
|
|
|
|
expiration = request.forms.get('expiration', u'burn_after_reading')
|
|
|
|
paste = Paste(expiration=expiration, content=content)
|
|
|
|
paste.save()
|
|
|
|
return {'status': 'ok',
|
|
|
|
'paste': paste.uuid}
|
2012-04-24 22:15:38 +04:00
|
|
|
|
2012-04-27 01:41:20 +04:00
|
|
|
return {'status': 'error',
|
|
|
|
'message': u"Serveur error: the paste couldn't be saved. Please try later."}
|
2012-04-24 22:15:38 +04:00
|
|
|
|
|
|
|
|
2012-04-26 13:56:13 +04:00
|
|
|
@app.route('/paste/:paste_id')
|
|
|
|
@view('paste')
|
2012-04-26 13:38:55 +04:00
|
|
|
def display_paste(paste_id):
|
2012-04-24 22:15:38 +04:00
|
|
|
|
2012-04-27 01:41:20 +04:00
|
|
|
now = datetime.now()
|
|
|
|
keep_alive = False
|
2012-04-26 13:38:55 +04:00
|
|
|
try:
|
|
|
|
paste = Paste.load(paste_id)
|
2012-04-27 01:41:20 +04:00
|
|
|
# Delete the paste if it expired:
|
|
|
|
if 'burn_after_reading' in str(paste.expiration):
|
|
|
|
# burn_after_reading contains the paste creation date
|
|
|
|
# if this read appends 10 seconds after the creation date
|
|
|
|
# we don't delete the paste because it means it's the redirection
|
|
|
|
# to the paste that happens during the paste creation
|
|
|
|
try:
|
|
|
|
keep_alive = paste.expiration.split('#')[1]
|
|
|
|
keep_alive = datetime.strptime(keep_alive,'%Y-%m-%d %H:%M:%S.%f')
|
|
|
|
keep_alive = now < keep_alive + timedelta(seconds=10)
|
|
|
|
except IndexError:
|
|
|
|
keep_alive = False
|
|
|
|
if not keep_alive:
|
|
|
|
paste.delete()
|
|
|
|
|
|
|
|
elif paste.expiration < now:
|
|
|
|
paste.delete()
|
|
|
|
raise ValueError()
|
|
|
|
|
2012-04-26 13:38:55 +04:00
|
|
|
except (TypeError, ValueError):
|
2012-05-02 12:46:51 +04:00
|
|
|
#abort(404, u"This paste doesn't exist or has expired")
|
2012-05-02 13:23:26 +04:00
|
|
|
return error404(ValueError)
|
2012-04-26 13:38:55 +04:00
|
|
|
|
2012-05-02 13:36:57 +04:00
|
|
|
return {'paste': paste, 'keep_alive': keep_alive,
|
|
|
|
'max_size': settings.MAX_SIZE,
|
|
|
|
'max_size_kb': settings.MAX_SIZE_KB}
|
2012-04-24 22:15:38 +04:00
|
|
|
|
|
|
|
|
2012-05-02 13:23:26 +04:00
|
|
|
@app.error(404)
|
2012-05-02 12:46:51 +04:00
|
|
|
@view('404')
|
2012-05-02 13:23:26 +04:00
|
|
|
def error404(code):
|
2012-05-02 12:46:51 +04:00
|
|
|
return {'max_size': settings.MAX_SIZE, 'max_size_kb': settings.MAX_SIZE_KB}
|
|
|
|
|
2012-05-02 13:23:26 +04:00
|
|
|
|
2012-05-01 16:21:21 +04:00
|
|
|
@clize.clize
|
|
|
|
def runserver(host=settings.HOST, port=settings.PORT, debug=settings.DEBUG,
|
|
|
|
serve_static=settings.DEBUG):
|
2012-04-24 14:22:59 +04:00
|
|
|
|
2012-05-01 16:21:21 +04:00
|
|
|
if serve_static:
|
|
|
|
@app.route('/static/<filename:path>')
|
|
|
|
def server_static(filename):
|
|
|
|
return static_file(filename, root=settings.STATIC_FILES_ROOT)
|
2012-04-29 02:07:25 +04:00
|
|
|
|
|
|
|
thread.start_new_thread(drop_privileges, ())
|
|
|
|
|
2012-05-01 16:21:21 +04:00
|
|
|
if debug:
|
|
|
|
bottle.debug(True)
|
|
|
|
run(app, host=host, port=port, reloader=True, server="cherrypy")
|
2012-04-24 14:22:59 +04:00
|
|
|
else:
|
2012-05-01 16:21:21 +04:00
|
|
|
run(app, host=host, port=port, server="cherrypy")
|
|
|
|
|
|
|
|
|
|
|
|
if __name__ == "__main__":
|
|
|
|
clize.run(runserver)
|
2012-04-29 02:07:25 +04:00
|
|
|
|
|
|
|
|