Extract httputils.serve_folder
This commit is contained in:
parent
555e4ccc51
commit
33fcda7c32
@ -23,10 +23,12 @@ Helper functions for HTTP.
|
|||||||
"""
|
"""
|
||||||
|
|
||||||
import contextlib
|
import contextlib
|
||||||
|
import os
|
||||||
|
import time
|
||||||
from http import client
|
from http import client
|
||||||
from typing import List, cast
|
from typing import List, Mapping, cast
|
||||||
|
|
||||||
from radicale import config, types
|
from radicale import config, pathutils, types
|
||||||
from radicale.log import logger
|
from radicale.log import logger
|
||||||
|
|
||||||
NOT_ALLOWED: types.WSGIResponse = (
|
NOT_ALLOWED: types.WSGIResponse = (
|
||||||
@ -67,6 +69,22 @@ INTERNAL_SERVER_ERROR: types.WSGIResponse = (
|
|||||||
|
|
||||||
DAV_HEADERS: str = "1, 2, 3, calendar-access, addressbook, extended-mkcol"
|
DAV_HEADERS: str = "1, 2, 3, calendar-access, addressbook, extended-mkcol"
|
||||||
|
|
||||||
|
MIMETYPES: Mapping[str, str] = {
|
||||||
|
".css": "text/css",
|
||||||
|
".eot": "application/vnd.ms-fontobject",
|
||||||
|
".gif": "image/gif",
|
||||||
|
".html": "text/html",
|
||||||
|
".js": "application/javascript",
|
||||||
|
".manifest": "text/cache-manifest",
|
||||||
|
".png": "image/png",
|
||||||
|
".svg": "image/svg+xml",
|
||||||
|
".ttf": "application/font-sfnt",
|
||||||
|
".txt": "text/plain",
|
||||||
|
".woff": "application/font-woff",
|
||||||
|
".woff2": "font/woff2",
|
||||||
|
".xml": "text/xml"}
|
||||||
|
FALLBACK_MIMETYPE: str = "application/octet-stream"
|
||||||
|
|
||||||
|
|
||||||
def decode_request(configuration: "config.Configuration",
|
def decode_request(configuration: "config.Configuration",
|
||||||
environ: types.WSGIEnviron, text: bytes) -> str:
|
environ: types.WSGIEnviron, text: bytes) -> str:
|
||||||
@ -120,3 +138,38 @@ def redirect(location: str, status: int = client.FOUND) -> types.WSGIResponse:
|
|||||||
return (status,
|
return (status,
|
||||||
{"Location": location, "Content-Type": "text/plain"},
|
{"Location": location, "Content-Type": "text/plain"},
|
||||||
"Redirected to %s" % location)
|
"Redirected to %s" % location)
|
||||||
|
|
||||||
|
|
||||||
|
def serve_folder(folder: str, base_prefix: str, path: str,
|
||||||
|
path_prefix: str = "/.web", index_file: str = "index.html",
|
||||||
|
mimetypes: Mapping[str, str] = MIMETYPES,
|
||||||
|
fallback_mimetype: str = FALLBACK_MIMETYPE,
|
||||||
|
) -> types.WSGIResponse:
|
||||||
|
if path != path_prefix and not path.startswith(path_prefix):
|
||||||
|
raise ValueError("path must start with path_prefix: %r --> %r" %
|
||||||
|
(path_prefix, path))
|
||||||
|
assert pathutils.sanitize_path(path) == path
|
||||||
|
try:
|
||||||
|
filesystem_path = pathutils.path_to_filesystem(
|
||||||
|
folder, path[len(path_prefix):].strip("/"))
|
||||||
|
except ValueError as e:
|
||||||
|
logger.debug("Web content with unsafe path %r requested: %s",
|
||||||
|
path, e, exc_info=True)
|
||||||
|
return NOT_FOUND
|
||||||
|
if os.path.isdir(filesystem_path) and not path.endswith("/"):
|
||||||
|
return redirect(base_prefix + path + "/")
|
||||||
|
if os.path.isdir(filesystem_path) and index_file:
|
||||||
|
filesystem_path = os.path.join(filesystem_path, index_file)
|
||||||
|
if not os.path.isfile(filesystem_path):
|
||||||
|
return NOT_FOUND
|
||||||
|
content_type = MIMETYPES.get(
|
||||||
|
os.path.splitext(filesystem_path)[1].lower(), FALLBACK_MIMETYPE)
|
||||||
|
with open(filesystem_path, "rb") as f:
|
||||||
|
answer = f.read()
|
||||||
|
last_modified = time.strftime(
|
||||||
|
"%a, %d %b %Y %H:%M:%S GMT",
|
||||||
|
time.gmtime(os.fstat(f.fileno()).st_mtime))
|
||||||
|
headers = {
|
||||||
|
"Content-Type": content_type,
|
||||||
|
"Last-Modified": last_modified}
|
||||||
|
return client.OK, headers, answer
|
||||||
|
@ -25,32 +25,10 @@ Features:
|
|||||||
|
|
||||||
"""
|
"""
|
||||||
|
|
||||||
|
|
||||||
import os
|
|
||||||
import time
|
|
||||||
from http import client
|
|
||||||
from typing import Mapping
|
|
||||||
|
|
||||||
import pkg_resources
|
import pkg_resources
|
||||||
|
|
||||||
from radicale import config, httputils, pathutils, types, web
|
from radicale import config, httputils, types, web
|
||||||
from radicale.log import logger
|
from radicale.httputils import FALLBACK_MIMETYPE, MIMETYPES # noqa:F401
|
||||||
|
|
||||||
MIMETYPES: Mapping[str, str] = {
|
|
||||||
".css": "text/css",
|
|
||||||
".eot": "application/vnd.ms-fontobject",
|
|
||||||
".gif": "image/gif",
|
|
||||||
".html": "text/html",
|
|
||||||
".js": "application/javascript",
|
|
||||||
".manifest": "text/cache-manifest",
|
|
||||||
".png": "image/png",
|
|
||||||
".svg": "image/svg+xml",
|
|
||||||
".ttf": "application/font-sfnt",
|
|
||||||
".txt": "text/plain",
|
|
||||||
".woff": "application/font-woff",
|
|
||||||
".woff2": "font/woff2",
|
|
||||||
".xml": "text/xml"}
|
|
||||||
FALLBACK_MIMETYPE: str = "application/octet-stream"
|
|
||||||
|
|
||||||
|
|
||||||
class Web(web.BaseWeb):
|
class Web(web.BaseWeb):
|
||||||
@ -59,34 +37,9 @@ class Web(web.BaseWeb):
|
|||||||
|
|
||||||
def __init__(self, configuration: config.Configuration) -> None:
|
def __init__(self, configuration: config.Configuration) -> None:
|
||||||
super().__init__(configuration)
|
super().__init__(configuration)
|
||||||
self.folder = pkg_resources.resource_filename(__name__,
|
self.folder = pkg_resources.resource_filename(
|
||||||
"internal_data")
|
__name__, "internal_data")
|
||||||
|
|
||||||
def get(self, environ: types.WSGIEnviron, base_prefix: str, path: str,
|
def get(self, environ: types.WSGIEnviron, base_prefix: str, path: str,
|
||||||
user: str) -> types.WSGIResponse:
|
user: str) -> types.WSGIResponse:
|
||||||
assert path == "/.web" or path.startswith("/.web/")
|
return httputils.serve_folder(self.folder, base_prefix, path)
|
||||||
assert pathutils.sanitize_path(path) == path
|
|
||||||
try:
|
|
||||||
filesystem_path = pathutils.path_to_filesystem(
|
|
||||||
self.folder, path[len("/.web"):].strip("/"))
|
|
||||||
except ValueError as e:
|
|
||||||
logger.debug("Web content with unsafe path %r requested: %s",
|
|
||||||
path, e, exc_info=True)
|
|
||||||
return httputils.NOT_FOUND
|
|
||||||
if os.path.isdir(filesystem_path) and not path.endswith("/"):
|
|
||||||
return httputils.redirect(base_prefix + path + "/")
|
|
||||||
if os.path.isdir(filesystem_path):
|
|
||||||
filesystem_path = os.path.join(filesystem_path, "index.html")
|
|
||||||
if not os.path.isfile(filesystem_path):
|
|
||||||
return httputils.NOT_FOUND
|
|
||||||
content_type = MIMETYPES.get(
|
|
||||||
os.path.splitext(filesystem_path)[1].lower(), FALLBACK_MIMETYPE)
|
|
||||||
with open(filesystem_path, "rb") as f:
|
|
||||||
answer = f.read()
|
|
||||||
last_modified = time.strftime(
|
|
||||||
"%a, %d %b %Y %H:%M:%S GMT",
|
|
||||||
time.gmtime(os.fstat(f.fileno()).st_mtime))
|
|
||||||
headers = {
|
|
||||||
"Content-Type": content_type,
|
|
||||||
"Last-Modified": last_modified}
|
|
||||||
return client.OK, headers, answer
|
|
||||||
|
Loading…
Reference in New Issue
Block a user