2012-05-14 19:17:49 +04:00
|
|
|
"""
|
2015-09-18 18:36:14 +03:00
|
|
|
Script including controller, rooting, and dependency management.
|
2012-05-14 19:17:49 +04:00
|
|
|
"""
|
|
|
|
|
2012-05-18 16:29:52 +04:00
|
|
|
import os
|
|
|
|
import sys
|
2015-05-10 20:19:02 +03:00
|
|
|
|
2020-08-11 17:37:03 +03:00
|
|
|
import _thread as thread
|
2015-05-10 20:19:02 +03:00
|
|
|
|
2020-08-12 16:21:49 +03:00
|
|
|
from urllib.parse import urlparse, parse_qs
|
2015-05-10 20:19:02 +03:00
|
|
|
|
2012-05-14 19:17:49 +04:00
|
|
|
from datetime import datetime, timedelta
|
|
|
|
|
|
|
|
import bottle
|
2020-08-12 16:21:49 +03:00
|
|
|
from bottle import (
|
|
|
|
Bottle,
|
|
|
|
debug,
|
|
|
|
static_file,
|
|
|
|
view,
|
|
|
|
request,
|
|
|
|
HTTPResponse,
|
|
|
|
redirect,
|
|
|
|
abort,
|
|
|
|
)
|
2012-05-14 19:17:49 +04:00
|
|
|
|
2020-08-12 10:19:38 +03:00
|
|
|
from beaker.middleware import SessionMiddleware
|
|
|
|
|
|
|
|
from zerobin import __version__
|
|
|
|
from zerobin.utils import (
|
|
|
|
settings,
|
|
|
|
SettingsValidationError,
|
|
|
|
ensure_var_env,
|
|
|
|
check_password,
|
|
|
|
)
|
2015-05-10 20:19:02 +03:00
|
|
|
from zerobin.paste import Paste
|
2012-05-14 19:17:49 +04:00
|
|
|
|
|
|
|
|
2020-08-12 10:19:38 +03:00
|
|
|
ensure_var_env()
|
|
|
|
|
|
|
|
|
2012-05-14 19:17:49 +04:00
|
|
|
GLOBAL_CONTEXT = {
|
2020-08-11 12:55:29 +03:00
|
|
|
"settings": settings,
|
2020-08-11 17:37:03 +03:00
|
|
|
"VERSION": __version__,
|
2020-08-11 12:55:29 +03:00
|
|
|
"pastes_count": Paste.get_pastes_count(),
|
|
|
|
"refresh_counter": datetime.now(),
|
2012-05-14 19:17:49 +04:00
|
|
|
}
|
2012-05-22 16:39:34 +04:00
|
|
|
|
2012-05-14 19:17:49 +04:00
|
|
|
|
2020-08-12 16:21:49 +03:00
|
|
|
app = Bottle()
|
|
|
|
|
|
|
|
ADMIN_LOGIN_URL = settings.ADMIN_URL + "login/"
|
2020-08-12 10:19:38 +03:00
|
|
|
|
|
|
|
|
2020-08-11 12:55:29 +03:00
|
|
|
@app.route("/")
|
|
|
|
@view("home")
|
2012-05-14 19:17:49 +04:00
|
|
|
def index():
|
|
|
|
return GLOBAL_CONTEXT
|
|
|
|
|
|
|
|
|
2020-08-12 16:21:49 +03:00
|
|
|
@app.get("/faq/")
|
2020-08-11 12:55:29 +03:00
|
|
|
@view("faq")
|
2012-05-21 19:14:01 +04:00
|
|
|
def faq():
|
2012-05-21 14:25:24 +04:00
|
|
|
return GLOBAL_CONTEXT
|
|
|
|
|
|
|
|
|
2020-08-12 16:21:49 +03:00
|
|
|
@app.get(settings.ADMIN_URL)
|
|
|
|
@app.post(settings.ADMIN_URL)
|
2020-08-12 10:19:38 +03:00
|
|
|
@view("admin")
|
|
|
|
def admin():
|
|
|
|
session = request.environ.get("beaker.session")
|
|
|
|
if not session or not session.get("is_authenticated"):
|
2020-08-12 16:21:49 +03:00
|
|
|
redirect(ADMIN_LOGIN_URL)
|
|
|
|
|
|
|
|
paste_id = request.forms.get("paste", "")
|
|
|
|
if paste_id:
|
|
|
|
try:
|
|
|
|
if "/paste/" in paste_id:
|
2020-08-12 16:39:07 +03:00
|
|
|
paste_id = urlparse(paste_id).path.split("/paste/")[-1]
|
2020-08-12 16:21:49 +03:00
|
|
|
paste = Paste.load(paste_id)
|
|
|
|
paste.delete()
|
|
|
|
except (TypeError, ValueError, FileNotFoundError):
|
2020-08-12 16:39:07 +03:00
|
|
|
return {
|
|
|
|
"status": "error",
|
|
|
|
"message": f"Cannot find paste '{paste_id}'",
|
|
|
|
**GLOBAL_CONTEXT,
|
|
|
|
}
|
2020-08-12 10:19:38 +03:00
|
|
|
|
2020-08-12 16:39:07 +03:00
|
|
|
return {"status": "ok", "message": "Paste deleted", **GLOBAL_CONTEXT}
|
2020-08-12 10:19:38 +03:00
|
|
|
|
2020-08-12 16:39:07 +03:00
|
|
|
return {"status": "ok", "message": "" ** GLOBAL_CONTEXT}
|
2020-08-12 10:19:38 +03:00
|
|
|
|
|
|
|
|
2020-08-12 16:21:49 +03:00
|
|
|
@app.get(ADMIN_LOGIN_URL)
|
|
|
|
@app.post(ADMIN_LOGIN_URL)
|
|
|
|
@view("login")
|
2020-08-12 10:19:38 +03:00
|
|
|
def login():
|
|
|
|
|
|
|
|
password = request.forms.get("password")
|
|
|
|
if password:
|
|
|
|
if not check_password(password):
|
2020-08-12 16:21:49 +03:00
|
|
|
return {"status": "error", "message": "Wrong password", **GLOBAL_CONTEXT}
|
2020-08-12 10:19:38 +03:00
|
|
|
|
|
|
|
session = request.environ.get("beaker.session")
|
|
|
|
session["is_authenticated"] = True
|
|
|
|
session.save()
|
|
|
|
|
|
|
|
redirect(settings.ADMIN_URL)
|
2020-08-12 16:21:49 +03:00
|
|
|
|
|
|
|
return {"status": "ok", **GLOBAL_CONTEXT}
|
2020-08-12 10:19:38 +03:00
|
|
|
|
|
|
|
|
2020-08-12 16:21:49 +03:00
|
|
|
@app.post(settings.ADMIN_URL + "logout/")
|
|
|
|
@view("logout")
|
|
|
|
def logout():
|
|
|
|
session = request.environ.get("beaker.session")
|
|
|
|
session["is_authenticated"] = False
|
|
|
|
session.save()
|
|
|
|
redirect("/")
|
|
|
|
|
|
|
|
|
|
|
|
@app.post("/paste/create")
|
2012-05-14 19:17:49 +04:00
|
|
|
def create_paste():
|
2020-08-11 12:55:29 +03:00
|
|
|
|
2013-01-19 21:50:07 +04:00
|
|
|
try:
|
2020-08-12 16:21:49 +03:00
|
|
|
body = parse_qs(request.body.read(int(settings.MAX_SIZE * 1.1)))
|
2013-01-19 21:50:07 +04:00
|
|
|
except ValueError:
|
2020-08-11 12:55:29 +03:00
|
|
|
return {"status": "error", "message": "Wrong data payload."}
|
2012-05-14 19:17:49 +04:00
|
|
|
|
|
|
|
try:
|
2020-08-11 12:55:29 +03:00
|
|
|
content = "".join(x.decode("utf8") for x in body[b"content"])
|
2013-01-19 21:50:07 +04:00
|
|
|
except (UnicodeDecodeError, KeyError):
|
2020-08-11 12:55:29 +03:00
|
|
|
return {
|
|
|
|
"status": "error",
|
|
|
|
"message": "Encoding error: the paste couldn't be saved.",
|
|
|
|
}
|
2012-05-14 19:17:49 +04:00
|
|
|
|
2013-01-19 20:05:37 +04:00
|
|
|
if '{"iv":' not in content: # reject silently non encrypted content
|
2020-08-11 12:55:29 +03:00
|
|
|
return {"status": "error", "message": "Wrong data payload."}
|
2015-05-10 20:19:02 +03: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 0 < len(content) < settings.MAX_SIZE:
|
2020-08-11 12:55:29 +03:00
|
|
|
expiration = body.get(b"expiration", [b"burn_after_reading"])[0]
|
|
|
|
paste = Paste(
|
|
|
|
expiration=expiration.decode("utf8"),
|
|
|
|
content=content,
|
|
|
|
uuid_length=settings.PASTE_ID_LENGTH,
|
|
|
|
)
|
2015-05-10 20:19:02 +03:00
|
|
|
paste.save()
|
|
|
|
|
|
|
|
# display counter
|
|
|
|
if settings.DISPLAY_COUNTER:
|
|
|
|
|
2020-08-11 12:55:29 +03:00
|
|
|
# increment paste counter
|
2015-05-10 20:19:02 +03:00
|
|
|
paste.increment_counter()
|
|
|
|
|
|
|
|
# if refresh time elapsed pick up new counter value
|
|
|
|
now = datetime.now()
|
2020-08-11 12:55:29 +03:00
|
|
|
timeout = GLOBAL_CONTEXT["refresh_counter"] + timedelta(
|
|
|
|
seconds=settings.REFRESH_COUNTER
|
|
|
|
)
|
2015-05-10 20:19:02 +03:00
|
|
|
if timeout < now:
|
2020-08-11 12:55:29 +03:00
|
|
|
GLOBAL_CONTEXT["pastes_count"] = Paste.get_pastes_count()
|
|
|
|
GLOBAL_CONTEXT["refresh_counter"] = now
|
2015-05-10 20:19:02 +03:00
|
|
|
|
2020-08-11 17:37:03 +03:00
|
|
|
return {"status": "ok", "paste": paste.uuid, "owner_key": paste.owner_key}
|
2012-05-14 19:17:49 +04:00
|
|
|
|
2020-08-11 12:55:29 +03:00
|
|
|
return {
|
|
|
|
"status": "error",
|
|
|
|
"message": "Serveur error: the paste couldn't be saved. " "Please try later.",
|
|
|
|
}
|
2012-05-14 19:17:49 +04:00
|
|
|
|
|
|
|
|
2020-08-12 16:21:49 +03:00
|
|
|
@app.get("/paste/:paste_id")
|
2020-08-11 12:55:29 +03:00
|
|
|
@view("paste")
|
2012-05-14 19:17:49 +04:00
|
|
|
def display_paste(paste_id):
|
|
|
|
|
|
|
|
now = datetime.now()
|
|
|
|
keep_alive = False
|
|
|
|
try:
|
|
|
|
paste = Paste.load(paste_id)
|
|
|
|
# Delete the paste if it expired:
|
2015-05-10 20:19:02 +03:00
|
|
|
if not isinstance(paste.expiration, datetime):
|
2012-05-14 19:17:49 +04:00
|
|
|
# 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:
|
2020-08-11 12:55:29 +03:00
|
|
|
keep_alive = paste.expiration.split("#")[1]
|
|
|
|
keep_alive = datetime.strptime(keep_alive, "%Y-%m-%d %H:%M:%S.%f")
|
2012-05-14 19:17:49 +04:00
|
|
|
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()
|
|
|
|
|
|
|
|
except (TypeError, ValueError):
|
|
|
|
return error404(ValueError)
|
|
|
|
|
2020-08-12 10:19:38 +03:00
|
|
|
return {"paste": paste, "keep_alive": keep_alive, **GLOBAL_CONTEXT}
|
2012-05-14 19:17:49 +04:00
|
|
|
|
|
|
|
|
2020-08-12 16:21:49 +03:00
|
|
|
@app.delete("/paste/:paste_id")
|
2020-08-11 17:37:03 +03:00
|
|
|
def delete_paste(paste_id):
|
|
|
|
|
|
|
|
try:
|
|
|
|
paste = Paste.load(paste_id)
|
|
|
|
except (TypeError, ValueError):
|
|
|
|
return error404(ValueError)
|
|
|
|
|
|
|
|
if paste.owner_key != request.forms.get("owner_key", None):
|
|
|
|
return HTTPResponse(status=403, body="Wrong owner key")
|
|
|
|
|
|
|
|
paste.delete()
|
|
|
|
|
|
|
|
return {
|
|
|
|
"status": "ok",
|
|
|
|
"message": "Paste deleted",
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-05-14 19:17:49 +04:00
|
|
|
@app.error(404)
|
2020-08-11 12:55:29 +03:00
|
|
|
@view("404")
|
2012-05-14 19:17:49 +04:00
|
|
|
def error404(code):
|
|
|
|
return GLOBAL_CONTEXT
|
|
|
|
|
|
|
|
|
2020-08-12 16:21:49 +03:00
|
|
|
@app.get("/static/<filename:path>")
|
2012-05-18 16:29:52 +04:00
|
|
|
def server_static(filename):
|
|
|
|
return static_file(filename, root=settings.STATIC_FILES_ROOT)
|
|
|
|
|
|
|
|
|
2020-08-11 12:55:29 +03:00
|
|
|
def get_app(debug=None, settings_file="", compressed_static=None, settings=settings):
|
2012-05-19 00:18:40 +04:00
|
|
|
"""
|
2013-01-19 20:05:37 +04:00
|
|
|
Return a tuple (settings, app) configured using passed
|
|
|
|
parameters and/or a setting file.
|
2012-05-19 00:18:40 +04:00
|
|
|
"""
|
2013-07-03 13:12:12 +04:00
|
|
|
|
2020-08-11 12:55:29 +03:00
|
|
|
settings_file = settings_file or os.environ.get("ZEROBIN_SETTINGS_FILE")
|
2013-07-03 13:12:12 +04:00
|
|
|
|
2012-05-14 19:17:49 +04:00
|
|
|
if settings_file:
|
2013-07-03 13:12:12 +04:00
|
|
|
settings.update_with_file(os.path.realpath(settings_file))
|
2012-05-14 19:17:49 +04:00
|
|
|
|
2013-04-29 22:17:37 +04:00
|
|
|
if settings.PASTE_ID_LENGTH < 4:
|
2020-08-11 12:55:29 +03:00
|
|
|
raise SettingsValidationError("PASTE_ID_LENGTH cannot be lower than 4")
|
2013-04-29 22:17:37 +04:00
|
|
|
|
2012-05-17 13:13:40 +04:00
|
|
|
if compressed_static is not None:
|
|
|
|
settings.COMPRESSED_STATIC_FILES = compressed_static
|
|
|
|
|
|
|
|
if debug is not None:
|
|
|
|
settings.DEBUG = debug
|
|
|
|
|
2012-05-14 19:17:49 +04:00
|
|
|
# make sure the templates can be loaded
|
|
|
|
for d in reversed(settings.TEMPLATE_DIRS):
|
|
|
|
bottle.TEMPLATE_PATH.insert(0, d)
|
|
|
|
|
2012-05-19 00:18:40 +04:00
|
|
|
if settings.DEBUG:
|
|
|
|
bottle.debug(True)
|
|
|
|
|
|
|
|
return settings, app
|
2020-08-12 16:21:49 +03:00
|
|
|
|
|
|
|
|
|
|
|
app = SessionMiddleware(
|
|
|
|
app,
|
|
|
|
{
|
|
|
|
"session.type": "file",
|
|
|
|
"session.cookie_expires": 300,
|
|
|
|
"session.data_dir": settings.SESSIONS_DIR,
|
|
|
|
"session.auto": True,
|
|
|
|
},
|
|
|
|
)
|