2016-08-14 10:35:14 +00:00
|
|
|
import cgi
|
|
|
|
import io
|
|
|
|
import json
|
|
|
|
import re
|
|
|
|
from datetime import datetime
|
|
|
|
from szurubooru.func import util
|
|
|
|
from szurubooru.rest import errors, middleware, routes, context
|
|
|
|
|
2016-08-14 12:22:53 +00:00
|
|
|
|
2016-08-14 10:35:14 +00:00
|
|
|
def _json_serializer(obj):
|
|
|
|
''' JSON serializer for objects not serializable by default JSON code '''
|
|
|
|
if isinstance(obj, datetime):
|
|
|
|
serial = obj.isoformat('T') + 'Z'
|
|
|
|
return serial
|
|
|
|
raise TypeError('Type not serializable')
|
|
|
|
|
2016-08-14 12:22:53 +00:00
|
|
|
|
2016-08-14 10:35:14 +00:00
|
|
|
def _dump_json(obj):
|
|
|
|
return json.dumps(obj, default=_json_serializer, indent=2)
|
|
|
|
|
2016-08-14 12:22:53 +00:00
|
|
|
|
2016-08-14 10:35:14 +00:00
|
|
|
def _read(env):
|
|
|
|
length = int(env.get('CONTENT_LENGTH', 0))
|
|
|
|
output = io.BytesIO()
|
|
|
|
while length > 0:
|
2016-08-14 12:22:53 +00:00
|
|
|
part = env['wsgi.input'].read(min(length, 1024 * 200))
|
2016-08-14 10:35:14 +00:00
|
|
|
if not part:
|
|
|
|
break
|
|
|
|
output.write(part)
|
|
|
|
length -= len(part)
|
|
|
|
output.seek(0)
|
|
|
|
return output
|
|
|
|
|
2016-08-14 12:22:53 +00:00
|
|
|
|
2016-08-14 10:35:14 +00:00
|
|
|
def _get_headers(env):
|
|
|
|
headers = {}
|
|
|
|
for key, value in env.items():
|
|
|
|
if key.startswith('HTTP_'):
|
|
|
|
key = util.snake_case_to_upper_train_case(key[5:])
|
|
|
|
headers[key] = value
|
|
|
|
return headers
|
|
|
|
|
2016-08-14 12:22:53 +00:00
|
|
|
|
2016-08-14 10:35:14 +00:00
|
|
|
def _create_context(env):
|
|
|
|
method = env['REQUEST_METHOD']
|
|
|
|
path = '/' + env['PATH_INFO'].lstrip('/')
|
|
|
|
headers = _get_headers(env)
|
|
|
|
|
|
|
|
# obscure, claims to "avoid a bug in cgi.FieldStorage"
|
|
|
|
env.setdefault('QUERY_STRING', '')
|
|
|
|
|
|
|
|
files = {}
|
|
|
|
params = {}
|
|
|
|
|
|
|
|
request_stream = _read(env)
|
|
|
|
form = cgi.FieldStorage(fp=request_stream, environ=env)
|
|
|
|
|
|
|
|
if form.list:
|
|
|
|
for key in form:
|
|
|
|
if key != 'metadata':
|
|
|
|
if isinstance(form[key], cgi.MiniFieldStorage):
|
|
|
|
params[key] = form.getvalue(key)
|
|
|
|
else:
|
2016-08-14 12:22:53 +00:00
|
|
|
# _user_file_name = getattr(form[key], 'filename', None)
|
2016-08-14 10:35:14 +00:00
|
|
|
files[key] = form.getvalue(key)
|
|
|
|
if 'metadata' in form:
|
|
|
|
body = form.getvalue('metadata')
|
|
|
|
else:
|
|
|
|
body = request_stream.read()
|
|
|
|
else:
|
|
|
|
body = None
|
|
|
|
|
|
|
|
if body:
|
|
|
|
try:
|
|
|
|
if isinstance(body, bytes):
|
|
|
|
body = body.decode('utf-8')
|
|
|
|
|
|
|
|
for key, value in json.loads(body).items():
|
|
|
|
params[key] = value
|
|
|
|
except (ValueError, UnicodeDecodeError):
|
|
|
|
raise errors.HttpBadRequest(
|
|
|
|
'Could not decode the request body. The JSON '
|
|
|
|
'was incorrect or was not encoded as UTF-8.')
|
|
|
|
|
|
|
|
return context.Context(method, path, headers, params, files)
|
|
|
|
|
2016-08-14 12:22:53 +00:00
|
|
|
|
2016-08-14 10:35:14 +00:00
|
|
|
def application(env, start_response):
|
|
|
|
try:
|
|
|
|
ctx = _create_context(env)
|
2016-08-14 12:22:53 +00:00
|
|
|
if 'application/json' not in ctx.get_header('Accept'):
|
2016-08-14 10:35:14 +00:00
|
|
|
raise errors.HttpNotAcceptable(
|
|
|
|
'This API only supports JSON responses.')
|
|
|
|
|
|
|
|
for url, allowed_methods in routes.routes.items():
|
|
|
|
match = re.fullmatch(url, ctx.url)
|
|
|
|
if not match:
|
|
|
|
continue
|
|
|
|
if ctx.method not in allowed_methods:
|
|
|
|
raise errors.HttpMethodNotAllowed(
|
|
|
|
'Allowed methods: %r' % allowed_methods)
|
|
|
|
|
|
|
|
for hook in middleware.pre_hooks:
|
|
|
|
hook(ctx)
|
|
|
|
handler = allowed_methods[ctx.method]
|
|
|
|
try:
|
|
|
|
response = handler(ctx, match.groupdict())
|
|
|
|
except Exception as ex:
|
|
|
|
for exception_type, handler in errors.error_handlers.items():
|
|
|
|
if isinstance(ex, exception_type):
|
|
|
|
handler(ex)
|
|
|
|
raise
|
|
|
|
finally:
|
|
|
|
for hook in middleware.post_hooks:
|
|
|
|
hook(ctx)
|
|
|
|
|
|
|
|
start_response('200', [('content-type', 'application/json')])
|
|
|
|
return (_dump_json(response).encode('utf-8'),)
|
|
|
|
|
|
|
|
raise errors.HttpNotFound(
|
|
|
|
'Requested path ' + ctx.url + ' was not found.')
|
|
|
|
|
|
|
|
except errors.BaseHttpError as ex:
|
|
|
|
start_response(
|
|
|
|
'%d %s' % (ex.code, ex.reason),
|
|
|
|
[('content-type', 'application/json')])
|
|
|
|
return (_dump_json({
|
|
|
|
'title': ex.title,
|
|
|
|
'description': ex.description,
|
|
|
|
}).encode('utf-8'),)
|