Use template responses
This commit is contained in:
parent
5db62c2dbf
commit
d815d6855f
121
fhost.py
121
fhost.py
|
@ -19,12 +19,12 @@
|
||||||
and limitations under the License.
|
and limitations under the License.
|
||||||
"""
|
"""
|
||||||
|
|
||||||
from flask import Flask, abort, escape, make_response, redirect, request, send_from_directory, url_for, Response
|
from flask import Flask, abort, make_response, redirect, request, send_from_directory, url_for, Response, render_template
|
||||||
from flask_sqlalchemy import SQLAlchemy
|
from flask_sqlalchemy import SQLAlchemy
|
||||||
from flask_script import Manager
|
from flask_script import Manager
|
||||||
from flask_migrate import Migrate, MigrateCommand
|
from flask_migrate import Migrate, MigrateCommand
|
||||||
|
from jinja2.exceptions import *
|
||||||
from hashlib import sha256
|
from hashlib import sha256
|
||||||
from humanize import naturalsize
|
|
||||||
from magic import Magic
|
from magic import Magic
|
||||||
from mimetypes import guess_extension
|
from mimetypes import guess_extension
|
||||||
import os, sys
|
import os, sys
|
||||||
|
@ -192,7 +192,7 @@ def store_file(f, addr):
|
||||||
|
|
||||||
if existing:
|
if existing:
|
||||||
if existing.removed:
|
if existing.removed:
|
||||||
return legal()
|
abort(451)
|
||||||
|
|
||||||
epath = getpath(existing.sha256)
|
epath = getpath(existing.sha256)
|
||||||
|
|
||||||
|
@ -256,7 +256,7 @@ def store_file(f, addr):
|
||||||
|
|
||||||
def store_url(url, addr):
|
def store_url(url, addr):
|
||||||
if is_fhost_url(url):
|
if is_fhost_url(url):
|
||||||
return segfault(508)
|
abort(400)
|
||||||
|
|
||||||
h = { "Accept-Encoding" : "identity" }
|
h = { "Accept-Encoding" : "identity" }
|
||||||
r = requests.get(url, stream=True, verify=False, headers=h)
|
r = requests.get(url, stream=True, verify=False, headers=h)
|
||||||
|
@ -277,12 +277,9 @@ def store_url(url, addr):
|
||||||
|
|
||||||
return store_file(f, addr)
|
return store_file(f, addr)
|
||||||
else:
|
else:
|
||||||
hl = naturalsize(l, binary = True)
|
abort(413)
|
||||||
hml = naturalsize(app.config["MAX_CONTENT_LENGTH"], binary=True)
|
|
||||||
|
|
||||||
return "Remote file too large ({0} > {1}).\n".format(hl, hml), 413
|
|
||||||
else:
|
else:
|
||||||
return "Could not determine remote file size (no Content-Length in response header; shoot admin).\n", 411
|
abort(411)
|
||||||
|
|
||||||
@app.route("/<path:path>")
|
@app.route("/<path:path>")
|
||||||
def get(path):
|
def get(path):
|
||||||
|
@ -294,7 +291,7 @@ def get(path):
|
||||||
|
|
||||||
if f and f.ext == p[1]:
|
if f and f.ext == p[1]:
|
||||||
if f.removed:
|
if f.removed:
|
||||||
return legal()
|
abort(451)
|
||||||
|
|
||||||
fpath = getpath(f.sha256)
|
fpath = getpath(f.sha256)
|
||||||
|
|
||||||
|
@ -351,76 +348,7 @@ def fhost():
|
||||||
|
|
||||||
abort(400)
|
abort(400)
|
||||||
else:
|
else:
|
||||||
fmts = list(app.config["FHOST_EXT_OVERRIDE"])
|
return render_template("index.html")
|
||||||
fmts.sort()
|
|
||||||
maxsize = naturalsize(app.config["MAX_CONTENT_LENGTH"], binary=True)
|
|
||||||
maxsizenum, maxsizeunit = maxsize.split(" ")
|
|
||||||
maxsizenum = float(maxsizenum)
|
|
||||||
maxsizehalf = maxsizenum / 2
|
|
||||||
|
|
||||||
if maxsizenum.is_integer():
|
|
||||||
maxsizenum = int(maxsizenum)
|
|
||||||
if maxsizehalf.is_integer():
|
|
||||||
maxsizehalf = int(maxsizehalf)
|
|
||||||
|
|
||||||
return """<pre>
|
|
||||||
THE NULL POINTER
|
|
||||||
================
|
|
||||||
|
|
||||||
HTTP POST files here:
|
|
||||||
curl -F'file=@yourfile.png' {0}
|
|
||||||
You can also POST remote URLs:
|
|
||||||
curl -F'url=http://example.com/image.jpg' {0}
|
|
||||||
Or you can shorten URLs:
|
|
||||||
curl -F'shorten=http://example.com/some/long/url' {0}
|
|
||||||
|
|
||||||
File URLs are valid for at least 30 days and up to a year (see below).
|
|
||||||
Shortened URLs do not expire.
|
|
||||||
|
|
||||||
Maximum file size: {1}
|
|
||||||
Not allowed: {5}
|
|
||||||
|
|
||||||
|
|
||||||
FILE RETENTION PERIOD
|
|
||||||
---------------------
|
|
||||||
|
|
||||||
retention = min_age + (-max_age + min_age) * pow((file_size / max_size - 1), 3)
|
|
||||||
|
|
||||||
days
|
|
||||||
365 | \\
|
|
||||||
| \\
|
|
||||||
| \\
|
|
||||||
| \\
|
|
||||||
| \\
|
|
||||||
| \\
|
|
||||||
| ..
|
|
||||||
| \\
|
|
||||||
197.5 | ----------..-------------------------------------------
|
|
||||||
| ..
|
|
||||||
| \\
|
|
||||||
| ..
|
|
||||||
| ...
|
|
||||||
| ..
|
|
||||||
| ...
|
|
||||||
| ....
|
|
||||||
| ......
|
|
||||||
30 | ....................
|
|
||||||
0{2}{3}
|
|
||||||
{4}
|
|
||||||
|
|
||||||
|
|
||||||
ABUSE
|
|
||||||
-----
|
|
||||||
|
|
||||||
If you would like to request permanent deletion, please contact lachs0r via
|
|
||||||
IRC on Freenode, or send an email to lachs0r@(this domain).
|
|
||||||
|
|
||||||
Please allow up to 24 hours for a response.
|
|
||||||
</pre>
|
|
||||||
""".format(fhost_url(),
|
|
||||||
maxsize, str(maxsizehalf).rjust(27), str(maxsizenum).rjust(27),
|
|
||||||
maxsizeunit.rjust(54),
|
|
||||||
", ".join(app.config["FHOST_MIME_BLACKLIST"]))
|
|
||||||
|
|
||||||
@app.route("/robots.txt")
|
@app.route("/robots.txt")
|
||||||
def robots():
|
def robots():
|
||||||
|
@ -428,35 +356,18 @@ def robots():
|
||||||
Disallow: /
|
Disallow: /
|
||||||
"""
|
"""
|
||||||
|
|
||||||
def legal():
|
|
||||||
return "451 Unavailable For Legal Reasons\n", 451
|
|
||||||
|
|
||||||
@app.errorhandler(400)
|
@app.errorhandler(400)
|
||||||
@app.errorhandler(404)
|
@app.errorhandler(404)
|
||||||
|
@app.errorhandler(411)
|
||||||
|
@app.errorhandler(413)
|
||||||
@app.errorhandler(414)
|
@app.errorhandler(414)
|
||||||
@app.errorhandler(415)
|
@app.errorhandler(415)
|
||||||
def segfault(e):
|
@app.errorhandler(451)
|
||||||
return "Segmentation fault\n", e.code
|
def ehandler(e):
|
||||||
|
try:
|
||||||
@app.errorhandler(404)
|
return render_template(f"{e.code}.html", id=id), e.code
|
||||||
def notfound(e):
|
except TemplateNotFound:
|
||||||
return u"""<pre>Process {0} stopped
|
return "Segmentation fault\n", e.code
|
||||||
* thread #1: tid = {0}, {1:#018x}, name = '{2}'
|
|
||||||
frame #0:
|
|
||||||
Process {0} stopped
|
|
||||||
* thread #8: tid = {0}, {3:#018x} fhost`get(path='{4}') + 27 at fhost.c:139, name = 'fhost/responder', stop reason = invalid address (fault address: 0x30)
|
|
||||||
frame #0: {3:#018x} fhost`get(path='{4}') + 27 at fhost.c:139
|
|
||||||
136 get(SrvContext *ctx, const char *path)
|
|
||||||
137 {{
|
|
||||||
138 StoredObj *obj = ctx->store->query(shurl_debase(path));
|
|
||||||
-> 139 switch (obj->type) {{
|
|
||||||
140 case ObjTypeFile:
|
|
||||||
141 ctx->serve_file_id(obj->id);
|
|
||||||
142 break;
|
|
||||||
(lldb) q</pre>
|
|
||||||
""".format(os.getpid(), id(app), "fhost", id(get), escape(request.path)), e.code
|
|
||||||
|
|
||||||
@manager.command
|
|
||||||
def debug():
|
def debug():
|
||||||
app.config["FHOST_USE_X_ACCEL_REDIRECT"] = False
|
app.config["FHOST_USE_X_ACCEL_REDIRECT"] = False
|
||||||
app.run(debug=True, port=4562,host="0.0.0.0")
|
app.run(debug=True, port=4562,host="0.0.0.0")
|
||||||
|
|
|
@ -0,0 +1,15 @@
|
||||||
|
{% set pid = range(20,100)|random %}
|
||||||
|
<pre>Process {{ pid }} stopped
|
||||||
|
* thread #1: tid = {{ pid }}, {{ "{:#018x}".format(id(g)) }}, name = 'fhost'
|
||||||
|
frame #0:
|
||||||
|
Process {{ pid }} stopped
|
||||||
|
* thread #8: tid = {{ pid }}, {{ "{:#018x}".format(id(request)) }} fhost`get(path='{{ request.path }}') + 27 at fhost.c:139, name = 'fhost/responder', stop reason = invalid address (fault address: 0x30)
|
||||||
|
frame #0: {3:#018x} fhost`get(path='{{ request.path }}') + 27 at fhost.c:139
|
||||||
|
136 get(SrvContext *ctx, const char *path)
|
||||||
|
137 {
|
||||||
|
138 StoredObj *obj = ctx->store->query(shurl_debase(path));
|
||||||
|
-> 139 switch (obj->type) {
|
||||||
|
140 case ObjTypeFile:
|
||||||
|
141 ctx->serve_file_id(obj->id);
|
||||||
|
142 break;
|
||||||
|
(lldb) q</pre>
|
|
@ -0,0 +1 @@
|
||||||
|
Could not determine remote file size (no Content-Length in response header; shoot admin).
|
|
@ -0,0 +1 @@
|
||||||
|
Remote file too large ({{ request.headers["content-length"]|filesizeformat(True) }} > {{ config["MAX_CONTENT_LENGTH"]|filesizeformat(True) }}).
|
|
@ -0,0 +1 @@
|
||||||
|
451 Unavailable For Legal Reasons
|
|
@ -0,0 +1,45 @@
|
||||||
|
<pre>
|
||||||
|
THE NULL POINTER
|
||||||
|
================
|
||||||
|
{% set fhost_url = url_for("fhost", _external=True).rstrip("/") %}
|
||||||
|
HTTP POST files here:
|
||||||
|
curl -F'file=@yourfile.png' {{ fhost_url }}
|
||||||
|
You can also POST remote URLs:
|
||||||
|
curl -F'url=http://example.com/image.jpg' {{ fhost_url }}
|
||||||
|
Or you can shorten URLs:
|
||||||
|
curl -F'shorten=http://example.com/some/long/url' {{ fhost_url }}
|
||||||
|
|
||||||
|
File URLs are valid for at least 30 days and up to a year (see below).
|
||||||
|
Shortened URLs do not expire.
|
||||||
|
{% set max_size = config["MAX_CONTENT_LENGTH"]|filesizeformat(True) %}
|
||||||
|
Maximum file size: {{ max_size }}
|
||||||
|
Not allowed: {{ config["FHOST_MIME_BLACKLIST"]|join(", ") }}
|
||||||
|
|
||||||
|
|
||||||
|
FILE RETENTION PERIOD
|
||||||
|
---------------------
|
||||||
|
|
||||||
|
retention = min_age + (-max_age + min_age) * pow((file_size / max_size - 1), 3)
|
||||||
|
|
||||||
|
days
|
||||||
|
365 | \\
|
||||||
|
| \\
|
||||||
|
| \\
|
||||||
|
| \\
|
||||||
|
| \\
|
||||||
|
| \\
|
||||||
|
| ..
|
||||||
|
| \\
|
||||||
|
197.5 | ----------..-------------------------------------------
|
||||||
|
| ..
|
||||||
|
| \\
|
||||||
|
| ..
|
||||||
|
| ...
|
||||||
|
| ..
|
||||||
|
| ...
|
||||||
|
| ....
|
||||||
|
| ......
|
||||||
|
30 | ....................
|
||||||
|
0{{ ((config["MAX_CONTENT_LENGTH"]/2)|filesizeformat(True)).split(" ")[0].rjust(27) }}{{ max_size.split(" ")[0].rjust(27) }}
|
||||||
|
{{ max_size.split(" ")[1].rjust(54) }}
|
||||||
|
</pre>
|
Loading…
Reference in New Issue