tahoe-lafs/src/allmydata/storage/http_server.py

81 lines
2.3 KiB
Python
Raw Normal View History

"""
HTTP server for storage.
"""
from functools import wraps
from klein import Klein
from twisted.web import http
2021-11-12 13:13:19 -05:00
# TODO Make sure to use pure Python versions?
from cbor2 import loads, dumps
from .server import StorageServer
2021-11-16 10:56:21 -05:00
from .http_client import swissnum_auth_header
def _authorization_decorator(f):
"""
Check the ``Authorization`` header, and (TODO: in later revision of code)
extract ``X-Tahoe-Authorization`` headers and pass them in.
"""
@wraps(f)
def route(self, request, *args, **kwargs):
2021-11-16 10:56:21 -05:00
if request.requestHeaders.getRawHeaders("Authorization", [None])[0] != str(
swissnum_auth_header(self._swissnum), "ascii"
2021-11-12 13:13:19 -05:00
):
2021-11-16 10:56:21 -05:00
request.setResponseCode(http.UNAUTHORIZED)
return b""
2021-11-12 13:13:19 -05:00
# authorization = request.requestHeaders.getRawHeaders("X-Tahoe-Authorization", [])
# For now, just a placeholder:
authorization = None
return f(self, request, authorization, *args, **kwargs)
2021-11-12 13:13:19 -05:00
return route
def _route(app, *route_args, **route_kwargs):
"""
Like Klein's @route, but with additional support for checking the
``Authorization`` header as well as ``X-Tahoe-Authorization`` headers. The
latter will (TODO: in later revision of code) get passed in as second
argument to wrapped functions.
"""
def decorator(f):
@app.route(*route_args, **route_kwargs)
@_authorization_decorator
def handle_route(*args, **kwargs):
return f(*args, **kwargs)
return handle_route
return decorator
class HTTPServer(object):
"""
A HTTP interface to the storage server.
"""
_app = Klein()
def __init__(self, storage_server: StorageServer, swissnum):
self._storage_server = storage_server
self._swissnum = swissnum
2021-11-12 13:13:19 -05:00
def get_resource(self):
"""Return twisted.web Resource for this object."""
return self._app.resource()
2021-11-16 11:09:17 -05:00
def _cbor(self, request, data):
"""Return CBOR-encoded data."""
request.setHeader("Content-Type", "application/cbor")
# TODO if data is big, maybe want to use a temporary file eventually...
return dumps(data)
@_route(_app, "/v1/version", methods=["GET"])
def version(self, request, authorization):
2021-11-16 11:09:17 -05:00
return self._cbor(request, self._storage_server.remote_get_version())