-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathstreamer.py
90 lines (72 loc) · 3.04 KB
/
streamer.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
from fastapi import FastAPI
from fastapi.responses import StreamingResponse
from remotezip import RemoteZip
import mimetypes
import requests
import logging
import os
import re
ARK_SERVER = os.environ.get("ARK_SERVER", "http://staffaccess.dl.bl.uk")
app = FastAPI()
logger = logging.getLogger(__name__)
def StreamerResponse(url: str, ark_path: str):
# If ark_path is not set, proxy the whole file back with the right Content-Type:
# Needs to return the actual item even if there's a slash after the ARK.
if ark_path == None or ark_path == "":
r = requests.get(url, stream=True)
logger.info(f" Upstream {url} response status code {r.status_code}")
# Interject, if we get a 200 that is text/plain, that's an problem. Assume it's an ePub instead.
if int(r.status_code/100) == 2 and "text/plain" in r.headers.get("content-type", None):
logger.warning(f"HTTP 200 with a text/plain response detected. Assuming it should be application/epub+zip and overriding it.")
r.headers["content-type"] = "application/epub+zip"
# Return the proxied response:
return StreamingResponse(
r.iter_content(chunk_size=10*1024),
status_code=r.status_code,
headers=r.headers)
#
# If we've got this far, we're picking a path out of a ZIP:
#
def iterfile(url):
with RemoteZip(url) as z:
if ark_path:
try:
info = z.getinfo(ark_path)
except Exception as e:
info = None
else:
info = None
# List content?
if info == None or info.is_dir():
for name in z.namelist():
if name.startswith(ark_path):
yield f"{name}\n"
else:
# Return an item:
with z.open(ark_path) as file_like:
yield from file_like
# Guess mimetype:
type, encoding = mimetypes.guess_type(ark_path)
if not type:
type = "text/plain"
return StreamingResponse(iterfile(url), media_type=type)
@app.get("/ark:/81055/{ark_id}")
def get_by_ark(ark_id: str):
backend_url = f"{ARK_SERVER}/ark:/81055/{ark_id}"
return StreamerResponse(backend_url, None)
@app.get("/ark:/81055/{ark_id}/{ark_path:path}")
def get_by_ark_and_path(ark_id: str, ark_path:str):
backend_url = f"{ARK_SERVER}/ark:/81055/{ark_id}"
logger.info(f"ark_and_path {backend_url} / {ark_path}")
return StreamerResponse(backend_url, ark_path)
@app.get("/proxy/{backend_url_and_path:path}")
def get_by_url(backend_url_and_path:str):
# Use Regex to spot the ARK ID and separate the Path off the end.
m = re.search("(.*)/ark:/(\d+)/([^\/]+)(.*)", backend_url_and_path)
backend_url = f"{m.group(1)}/ark:/{m.group(2)}/{m.group(3)}"
ark_path = None
if m.group(4):
ark_path = m.group(4)
ark_path = ark_path.lstrip("/")
logger.info(f"by_url {backend_url} / {ark_path}")
return StreamerResponse(backend_url, ark_path)