diff options
author | Maxim Devaev <[email protected]> | 2022-08-14 21:03:45 +0300 |
---|---|---|
committer | Maxim Devaev <[email protected]> | 2022-08-14 21:03:45 +0300 |
commit | 4f9501a8066f72dea235754ccf11bd83b87c3340 (patch) | |
tree | b84e9eec4b904322e5b3750228e232fa80a9fabc /kvmd | |
parent | de6fd9434993b87ff6e5569c3984f76b5ad963a9 (diff) |
/api/msd/read: zstd supported
Diffstat (limited to 'kvmd')
-rw-r--r-- | kvmd/apps/kvmd/api/msd.py | 61 |
1 files changed, 39 insertions, 22 deletions
diff --git a/kvmd/apps/kvmd/api/msd.py b/kvmd/apps/kvmd/api/msd.py index 74290d89..18d3ba2c 100644 --- a/kvmd/apps/kvmd/api/msd.py +++ b/kvmd/apps/kvmd/api/msd.py @@ -29,6 +29,7 @@ from typing import Optional from typing import Union import aiohttp +import zstandard from aiohttp.web import Request from aiohttp.web import Response @@ -48,6 +49,7 @@ from ....htserver import stream_json_exception from ....plugins.msd import BaseMsd +from ....validators import check_string_in_list from ....validators.basic import valid_bool from ....validators.basic import valid_int_f0 from ....validators.basic import valid_float_f01 @@ -90,34 +92,49 @@ class MsdApi: @exposed_http("GET", "/msd/read") async def __read_handler(self, request: Request) -> StreamResponse: name = valid_msd_image_name(request.query.get("image")) - compress = valid_bool(request.query.get("compress", False)) + compressors = { + "": ("", None), + "none": ("", None), + "lzma": (".xz", (lambda: lzma.LZMACompressor())), # pylint: disable=unnecessary-lambda + "zstd": (".zst", (lambda: zstandard.ZstdCompressor().compressobj())), # pylint: disable=unnecessary-lambda + } + (suffix, make_compressor) = compressors[check_string_in_list( + arg=request.query.get("compress", ""), + name="Compression mode", + variants=set(compressors), + )] + async with self.__msd.read_image(name) as reader: - size = reader.get_total_size() - src = reader.read_chunked() - if compress: - name += ".xz" + if make_compressor is None: + src = reader.read_chunked() + size = reader.get_total_size() + + else: + async def compressed() -> AsyncGenerator[bytes, None]: + assert make_compressor is not None + compressor = make_compressor() # pylint: disable=not-callable + limit = reader.get_chunk_size() + buf = b"" + try: + async for chunk in reader.read_chunked(): + buf += await aiotools.run_async(compressor.compress, chunk) + if len(buf) >= limit: + yield buf + buf = b"" + finally: + # Закрыть в любом случае + buf += await aiotools.run_async(compressor.flush) + if len(buf) > 0: + yield buf + + src = compressed() size = -1 - src = self.__compressed(reader.get_chunk_size(), src) - response = await start_streaming(request, "application/octet-stream", size, name) + + response = await start_streaming(request, "application/octet-stream", size, name + suffix) async for chunk in src: await response.write(chunk) return response - async def __compressed(self, limit: int, src: AsyncGenerator[bytes, None]) -> AsyncGenerator[bytes, None]: - buf = b"" - xz = lzma.LZMACompressor() - try: - async for chunk in src: - buf += await aiotools.run_async(xz.compress, chunk) - if len(buf) >= limit: - yield buf - buf = b"" - finally: - # Закрыть в любом случае - buf += await aiotools.run_async(xz.flush) - if len(buf) > 0: - yield buf - # ===== @exposed_http("POST", "/msd/write") |