2023-10-23 02:51:39 +01:00
|
|
|
import asyncio
|
|
|
|
import typing
|
2023-11-08 20:38:40 +00:00
|
|
|
from secrets import token_bytes
|
2023-10-23 02:51:39 +01:00
|
|
|
|
|
|
|
import msgspec
|
2023-10-23 22:57:50 +01:00
|
|
|
from sanic import Blueprint
|
2023-10-23 02:51:39 +01:00
|
|
|
|
2023-11-08 20:38:40 +00:00
|
|
|
from cista import __version__, config, watching
|
2023-10-23 02:51:39 +01:00
|
|
|
from cista.fileio import FileServer
|
2023-11-08 20:38:40 +00:00
|
|
|
from cista.protocol import ControlTypes, FileRange, StatusMsg
|
2023-10-23 02:51:39 +01:00
|
|
|
from cista.util.apphelpers import asend, websocket_wrapper
|
|
|
|
|
|
|
|
bp = Blueprint("api", url_prefix="/api")
|
|
|
|
fileserver = FileServer()
|
|
|
|
|
2023-10-26 15:18:59 +01:00
|
|
|
|
2023-10-23 02:51:39 +01:00
|
|
|
@bp.before_server_start
|
|
|
|
async def start_fileserver(app, _):
|
|
|
|
await fileserver.start()
|
|
|
|
|
2023-10-26 15:18:59 +01:00
|
|
|
|
2023-10-23 02:51:39 +01:00
|
|
|
@bp.after_server_stop
|
|
|
|
async def stop_fileserver(app, _):
|
|
|
|
await fileserver.stop()
|
|
|
|
|
2023-10-26 15:18:59 +01:00
|
|
|
|
|
|
|
@bp.websocket("upload")
|
2023-10-23 02:51:39 +01:00
|
|
|
@websocket_wrapper
|
|
|
|
async def upload(req, ws):
|
|
|
|
alink = fileserver.alink
|
|
|
|
while True:
|
|
|
|
req = None
|
|
|
|
text = await ws.recv()
|
|
|
|
if not isinstance(text, str):
|
2023-10-26 15:18:59 +01:00
|
|
|
raise ValueError(
|
2023-11-08 20:38:40 +00:00
|
|
|
f"Expected JSON control, got binary len(data) = {len(text)}",
|
2023-10-26 15:18:59 +01:00
|
|
|
)
|
2023-10-23 02:51:39 +01:00
|
|
|
req = msgspec.json.decode(text, type=FileRange)
|
|
|
|
pos = req.start
|
|
|
|
data = None
|
|
|
|
while pos < req.end and (data := await ws.recv()) and isinstance(data, bytes):
|
|
|
|
sentsize = await alink(("upload", req.name, pos, data, req.size))
|
|
|
|
pos += typing.cast(int, sentsize)
|
|
|
|
if pos != req.end:
|
|
|
|
d = f"{len(data)} bytes" if isinstance(data, bytes) else data
|
|
|
|
raise ValueError(f"Expected {req.end - pos} more bytes, got {d}")
|
|
|
|
# Report success
|
|
|
|
res = StatusMsg(status="ack", req=req)
|
2023-11-08 20:38:40 +00:00
|
|
|
print("ack", res)
|
2023-10-23 02:51:39 +01:00
|
|
|
await asend(ws, res)
|
|
|
|
|
2023-10-26 15:18:59 +01:00
|
|
|
|
|
|
|
@bp.websocket("download")
|
2023-10-23 02:51:39 +01:00
|
|
|
@websocket_wrapper
|
|
|
|
async def download(req, ws):
|
|
|
|
alink = fileserver.alink
|
|
|
|
while True:
|
|
|
|
req = None
|
|
|
|
text = await ws.recv()
|
|
|
|
if not isinstance(text, str):
|
2023-10-26 15:18:59 +01:00
|
|
|
raise ValueError(
|
2023-11-08 20:38:40 +00:00
|
|
|
f"Expected JSON control, got binary len(data) = {len(text)}",
|
2023-10-26 15:18:59 +01:00
|
|
|
)
|
2023-10-23 02:51:39 +01:00
|
|
|
req = msgspec.json.decode(text, type=FileRange)
|
|
|
|
pos = req.start
|
|
|
|
while pos < req.end:
|
2023-10-26 15:18:59 +01:00
|
|
|
end = min(req.end, pos + (1 << 20))
|
2023-10-23 02:51:39 +01:00
|
|
|
data = typing.cast(bytes, await alink(("download", req.name, pos, end)))
|
|
|
|
await asend(ws, data)
|
|
|
|
pos += len(data)
|
|
|
|
# Report success
|
|
|
|
res = StatusMsg(status="ack", req=req)
|
|
|
|
await asend(ws, res)
|
2023-10-26 15:18:59 +01:00
|
|
|
|
2023-10-23 02:51:39 +01:00
|
|
|
|
|
|
|
@bp.websocket("control")
|
|
|
|
@websocket_wrapper
|
|
|
|
async def control(req, ws):
|
2023-11-08 20:38:40 +00:00
|
|
|
while True:
|
|
|
|
cmd = msgspec.json.decode(await ws.recv(), type=ControlTypes)
|
|
|
|
await asyncio.to_thread(cmd)
|
|
|
|
await asend(ws, StatusMsg(status="ack", req=cmd))
|
|
|
|
|
2023-10-23 02:51:39 +01:00
|
|
|
|
|
|
|
@bp.websocket("watch")
|
|
|
|
@websocket_wrapper
|
|
|
|
async def watch(req, ws):
|
2023-11-08 20:38:40 +00:00
|
|
|
await ws.send(
|
|
|
|
msgspec.json.encode(
|
|
|
|
{
|
|
|
|
"server": {
|
2023-11-08 23:00:07 +00:00
|
|
|
"name": config.config.name or config.config.path.name,
|
2023-11-08 20:38:40 +00:00
|
|
|
"version": __version__,
|
|
|
|
"public": config.config.public,
|
|
|
|
},
|
|
|
|
"user": {
|
|
|
|
"username": req.ctx.username,
|
|
|
|
"privileged": req.ctx.user.privileged,
|
|
|
|
}
|
|
|
|
if req.ctx.user
|
|
|
|
else None,
|
|
|
|
}
|
|
|
|
).decode()
|
|
|
|
)
|
|
|
|
uuid = token_bytes(16)
|
2023-10-23 02:51:39 +01:00
|
|
|
try:
|
2023-11-12 23:20:40 +00:00
|
|
|
with watching.state.lock:
|
2023-11-08 20:38:40 +00:00
|
|
|
q = watching.pubsub[uuid] = asyncio.Queue()
|
2023-10-23 03:24:54 +01:00
|
|
|
# Init with disk usage and full tree
|
2023-11-12 23:20:40 +00:00
|
|
|
await ws.send(watching.format_space(watching.state.space))
|
|
|
|
await ws.send(watching.format_root(watching.state.root))
|
2023-10-23 02:51:39 +01:00
|
|
|
# Send updates
|
|
|
|
while True:
|
|
|
|
await ws.send(await q.get())
|
|
|
|
finally:
|
2023-11-08 20:38:40 +00:00
|
|
|
del watching.pubsub[uuid]
|