mirror of
https://github.com/mofeng-git/One-KVM.git
synced 2026-01-29 09:01:54 +08:00
pikvm/pikvm#321: server-side uploading counters
This commit is contained in:
@@ -28,10 +28,12 @@ from ....logging import get_logger
|
|||||||
from ....plugins.msd import BaseMsd
|
from ....plugins.msd import BaseMsd
|
||||||
|
|
||||||
from ....validators.basic import valid_bool
|
from ....validators.basic import valid_bool
|
||||||
|
from ....validators.basic import valid_int_f0
|
||||||
from ....validators.kvm import valid_msd_image_name
|
from ....validators.kvm import valid_msd_image_name
|
||||||
|
|
||||||
from ..http import exposed_http
|
from ..http import exposed_http
|
||||||
from ..http import make_json_response
|
from ..http import make_json_response
|
||||||
|
from ..http import get_field_value
|
||||||
from ..http import get_multipart_field
|
from ..http import get_multipart_field
|
||||||
|
|
||||||
|
|
||||||
@@ -71,12 +73,12 @@ class MsdApi:
|
|||||||
name = ""
|
name = ""
|
||||||
written = 0
|
written = 0
|
||||||
try:
|
try:
|
||||||
name_field = await get_multipart_field(reader, "image")
|
name = valid_msd_image_name(await get_field_value(reader, "image"))
|
||||||
name = valid_msd_image_name((await name_field.read()).decode("utf-8"))
|
size = valid_int_f0(await get_field_value(reader, "size"))
|
||||||
|
|
||||||
data_field = await get_multipart_field(reader, "data")
|
data_field = await get_multipart_field(reader, "data")
|
||||||
|
|
||||||
async with self.__msd.write_image(name):
|
async with self.__msd.write_image(name, size):
|
||||||
logger.info("Writing image %r to MSD ...", name)
|
logger.info("Writing image %r to MSD ...", name)
|
||||||
while True:
|
while True:
|
||||||
chunk = await data_field.read_chunk(self.__msd.get_upload_chunk_size())
|
chunk = await data_field.read_chunk(self.__msd.get_upload_chunk_size())
|
||||||
|
|||||||
@@ -171,6 +171,11 @@ def make_json_exception(err: Exception, status: Optional[int]=None) -> aiohttp.w
|
|||||||
|
|
||||||
|
|
||||||
# =====
|
# =====
|
||||||
|
async def get_field_value(reader: aiohttp.MultipartReader, name: str) -> str:
|
||||||
|
field = await get_multipart_field(reader, name)
|
||||||
|
return (await field.read()).decode("utf-8")
|
||||||
|
|
||||||
|
|
||||||
async def get_multipart_field(reader: aiohttp.MultipartReader, name: str) -> aiohttp.BodyPartReader:
|
async def get_multipart_field(reader: aiohttp.MultipartReader, name: str) -> aiohttp.BodyPartReader:
|
||||||
field = await reader.next()
|
field = await reader.next()
|
||||||
if not isinstance(field, aiohttp.BodyPartReader):
|
if not isinstance(field, aiohttp.BodyPartReader):
|
||||||
|
|||||||
@@ -20,6 +20,7 @@
|
|||||||
# ========================================================================== #
|
# ========================================================================== #
|
||||||
|
|
||||||
|
|
||||||
|
import os
|
||||||
import contextlib
|
import contextlib
|
||||||
|
|
||||||
from typing import Dict
|
from typing import Dict
|
||||||
@@ -27,6 +28,11 @@ from typing import Type
|
|||||||
from typing import AsyncGenerator
|
from typing import AsyncGenerator
|
||||||
from typing import Optional
|
from typing import Optional
|
||||||
|
|
||||||
|
import aiofiles
|
||||||
|
import aiofiles.base
|
||||||
|
|
||||||
|
from ... import aiofs
|
||||||
|
|
||||||
from ...errors import OperationError
|
from ...errors import OperationError
|
||||||
from ...errors import IsBusyError
|
from ...errors import IsBusyError
|
||||||
|
|
||||||
@@ -113,7 +119,7 @@ class BaseMsd(BasePlugin):
|
|||||||
raise NotImplementedError()
|
raise NotImplementedError()
|
||||||
|
|
||||||
@contextlib.asynccontextmanager
|
@contextlib.asynccontextmanager
|
||||||
async def write_image(self, name: str) -> AsyncGenerator[None, None]: # pylint: disable=unused-argument
|
async def write_image(self, name: str, size: int) -> AsyncGenerator[None, None]: # pylint: disable=unused-argument
|
||||||
if self is not None: # XXX: Vulture and pylint hack
|
if self is not None: # XXX: Vulture and pylint hack
|
||||||
raise NotImplementedError()
|
raise NotImplementedError()
|
||||||
yield
|
yield
|
||||||
@@ -128,6 +134,52 @@ class BaseMsd(BasePlugin):
|
|||||||
raise NotImplementedError()
|
raise NotImplementedError()
|
||||||
|
|
||||||
|
|
||||||
|
class MsdImageWriter:
|
||||||
|
def __init__(self, path: str, size: int, sync: int) -> None:
|
||||||
|
self.__name = os.path.basename(path)
|
||||||
|
self.__path = path
|
||||||
|
self.__size = size
|
||||||
|
self.__sync = sync
|
||||||
|
|
||||||
|
self.__file: Optional[aiofiles.base.AiofilesContextManager] = None
|
||||||
|
self.__written = 0
|
||||||
|
self.__unsynced = 0
|
||||||
|
|
||||||
|
def get_file(self) -> aiofiles.base.AiofilesContextManager:
|
||||||
|
assert self.__file is not None
|
||||||
|
return self.__file
|
||||||
|
|
||||||
|
def get_state(self) -> Dict:
|
||||||
|
return {
|
||||||
|
"name": self.__name,
|
||||||
|
"size": self.__size,
|
||||||
|
"written": self.__written,
|
||||||
|
}
|
||||||
|
|
||||||
|
async def open(self) -> "MsdImageWriter":
|
||||||
|
assert self.__file is None
|
||||||
|
self.__file = await aiofiles.open(self.__path, mode="w+b", buffering=0) # type: ignore
|
||||||
|
return self
|
||||||
|
|
||||||
|
async def write(self, chunk: bytes) -> int:
|
||||||
|
assert self.__file is not None
|
||||||
|
|
||||||
|
await self.__file.write(chunk) # type: ignore
|
||||||
|
self.__written += len(chunk)
|
||||||
|
|
||||||
|
self.__unsynced += len(chunk)
|
||||||
|
if self.__unsynced >= self.__sync:
|
||||||
|
await aiofs.afile_sync(self.__file)
|
||||||
|
self.__unsynced = 0
|
||||||
|
|
||||||
|
return self.__written
|
||||||
|
|
||||||
|
async def close(self) -> None:
|
||||||
|
assert self.__file is not None
|
||||||
|
await aiofs.afile_sync(self.__file)
|
||||||
|
await self.__file.close() # type: ignore
|
||||||
|
|
||||||
|
|
||||||
# =====
|
# =====
|
||||||
def get_msd_class(name: str) -> Type[BaseMsd]:
|
def get_msd_class(name: str) -> Type[BaseMsd]:
|
||||||
return get_plugin_class("msd", name) # type: ignore
|
return get_plugin_class("msd", name) # type: ignore
|
||||||
|
|||||||
@@ -70,7 +70,7 @@ class Plugin(BaseMsd):
|
|||||||
raise MsdDisabledError()
|
raise MsdDisabledError()
|
||||||
|
|
||||||
@contextlib.asynccontextmanager
|
@contextlib.asynccontextmanager
|
||||||
async def write_image(self, name: str) -> AsyncGenerator[None, None]:
|
async def write_image(self, name: str, size: int) -> AsyncGenerator[None, None]:
|
||||||
if self is not None: # XXX: Vulture and pylint hack
|
if self is not None: # XXX: Vulture and pylint hack
|
||||||
raise MsdDisabledError()
|
raise MsdDisabledError()
|
||||||
yield
|
yield
|
||||||
|
|||||||
@@ -32,9 +32,6 @@ from typing import Dict
|
|||||||
from typing import AsyncGenerator
|
from typing import AsyncGenerator
|
||||||
from typing import Optional
|
from typing import Optional
|
||||||
|
|
||||||
import aiofiles
|
|
||||||
import aiofiles.base
|
|
||||||
|
|
||||||
from ....logging import get_logger
|
from ....logging import get_logger
|
||||||
|
|
||||||
from ....inotify import InotifyMask
|
from ....inotify import InotifyMask
|
||||||
@@ -49,7 +46,6 @@ from ....validators.os import valid_printable_filename
|
|||||||
from ....validators.os import valid_command
|
from ....validators.os import valid_command
|
||||||
|
|
||||||
from .... import aiotools
|
from .... import aiotools
|
||||||
from .... import aiofs
|
|
||||||
|
|
||||||
from .. import MsdError
|
from .. import MsdError
|
||||||
from .. import MsdIsBusyError
|
from .. import MsdIsBusyError
|
||||||
@@ -60,6 +56,7 @@ from .. import MsdImageNotSelected
|
|||||||
from .. import MsdUnknownImageError
|
from .. import MsdUnknownImageError
|
||||||
from .. import MsdImageExistsError
|
from .. import MsdImageExistsError
|
||||||
from .. import BaseMsd
|
from .. import BaseMsd
|
||||||
|
from .. import MsdImageWriter
|
||||||
|
|
||||||
from . import fs
|
from . import fs
|
||||||
from . import helpers
|
from . import helpers
|
||||||
@@ -165,10 +162,8 @@ class Plugin(BaseMsd): # pylint: disable=too-many-instance-attributes
|
|||||||
|
|
||||||
self.__drive = Drive(gadget, instance=0, lun=0)
|
self.__drive = Drive(gadget, instance=0, lun=0)
|
||||||
|
|
||||||
self.__new_file: Optional[aiofiles.base.AiofilesContextManager] = None
|
self.__new_writer: Optional[MsdImageWriter] = None
|
||||||
self.__new_file_written = 0
|
self.__new_writer_tick = 0.0
|
||||||
self.__new_file_unsynced = 0
|
|
||||||
self.__new_file_tick = 0.0
|
|
||||||
|
|
||||||
self.__notifier = aiotools.AioNotifier()
|
self.__notifier = aiotools.AioNotifier()
|
||||||
self.__state = _State(self.__notifier)
|
self.__state = _State(self.__notifier)
|
||||||
@@ -204,11 +199,14 @@ class Plugin(BaseMsd): # pylint: disable=too-many-instance-attributes
|
|||||||
del storage["images"][name]["path"]
|
del storage["images"][name]["path"]
|
||||||
del storage["images"][name]["in_storage"]
|
del storage["images"][name]["in_storage"]
|
||||||
|
|
||||||
storage["uploading"] = bool(self.__new_file)
|
if self.__new_writer:
|
||||||
if self.__new_file: # При загрузке файла показываем размер вручную
|
# При загрузке файла показываем актуальную статистику вручную
|
||||||
|
storage["uploading"] = self.__new_writer.get_state()
|
||||||
space = fs.get_fs_space(self.__storage_path, fatal=False)
|
space = fs.get_fs_space(self.__storage_path, fatal=False)
|
||||||
if space:
|
if space:
|
||||||
storage.update(dataclasses.asdict(space))
|
storage.update(dataclasses.asdict(space))
|
||||||
|
else:
|
||||||
|
storage["uploading"] = None
|
||||||
|
|
||||||
vd: Optional[Dict] = None
|
vd: Optional[Dict] = None
|
||||||
if self.__state.vd:
|
if self.__state.vd:
|
||||||
@@ -253,7 +251,7 @@ class Plugin(BaseMsd): # pylint: disable=too-many-instance-attributes
|
|||||||
|
|
||||||
@aiotools.atomic
|
@aiotools.atomic
|
||||||
async def cleanup(self) -> None:
|
async def cleanup(self) -> None:
|
||||||
await self.__close_new_file()
|
await self.__close_new_writer()
|
||||||
|
|
||||||
# =====
|
# =====
|
||||||
|
|
||||||
@@ -308,7 +306,7 @@ class Plugin(BaseMsd): # pylint: disable=too-many-instance-attributes
|
|||||||
self.__state.vd.connected = connected
|
self.__state.vd.connected = connected
|
||||||
|
|
||||||
@contextlib.asynccontextmanager
|
@contextlib.asynccontextmanager
|
||||||
async def write_image(self, name: str) -> AsyncGenerator[None, None]:
|
async def write_image(self, name: str, size: int) -> AsyncGenerator[None, None]:
|
||||||
try:
|
try:
|
||||||
async with self.__state._region: # pylint: disable=protected-access
|
async with self.__state._region: # pylint: disable=protected-access
|
||||||
try:
|
try:
|
||||||
@@ -326,16 +324,15 @@ class Plugin(BaseMsd): # pylint: disable=too-many-instance-attributes
|
|||||||
|
|
||||||
await self.__remount_storage(rw=True)
|
await self.__remount_storage(rw=True)
|
||||||
self.__set_image_complete(name, False)
|
self.__set_image_complete(name, False)
|
||||||
self.__new_file_written = 0
|
|
||||||
self.__new_file_unsynced = 0
|
self.__new_writer = await MsdImageWriter(path, size, self.__sync_chunk_size).open()
|
||||||
self.__new_file = await aiofiles.open(path, mode="w+b", buffering=0) # type: ignore
|
|
||||||
|
|
||||||
await self.__notifier.notify()
|
await self.__notifier.notify()
|
||||||
yield
|
yield
|
||||||
self.__set_image_complete(name, True)
|
self.__set_image_complete(name, True)
|
||||||
|
|
||||||
finally:
|
finally:
|
||||||
await self.__close_new_file()
|
await self.__close_new_writer()
|
||||||
try:
|
try:
|
||||||
await self.__remount_storage(rw=False)
|
await self.__remount_storage(rw=False)
|
||||||
except Exception:
|
except Exception:
|
||||||
@@ -350,22 +347,14 @@ class Plugin(BaseMsd): # pylint: disable=too-many-instance-attributes
|
|||||||
return self.__upload_chunk_size
|
return self.__upload_chunk_size
|
||||||
|
|
||||||
async def write_image_chunk(self, chunk: bytes) -> int:
|
async def write_image_chunk(self, chunk: bytes) -> int:
|
||||||
assert self.__new_file
|
assert self.__new_writer
|
||||||
|
written = await self.__new_writer.write(chunk)
|
||||||
await self.__new_file.write(chunk) # type: ignore
|
|
||||||
self.__new_file_written += len(chunk)
|
|
||||||
|
|
||||||
self.__new_file_unsynced += len(chunk)
|
|
||||||
if self.__new_file_unsynced >= self.__sync_chunk_size:
|
|
||||||
await aiofs.afile_sync(self.__new_file)
|
|
||||||
self.__new_file_unsynced = 0
|
|
||||||
|
|
||||||
now = time.monotonic()
|
now = time.monotonic()
|
||||||
if self.__new_file_tick + 1 < now:
|
if self.__new_writer_tick + 1 < now:
|
||||||
# Это нужно для ручного оповещения о свободном пространстве на диске, см. get_state()
|
# Это нужно для ручного оповещения о свободном пространстве на диске, см. get_state()
|
||||||
self.__new_file_tick = now
|
self.__new_writer_tick = now
|
||||||
await self.__notifier.notify()
|
await self.__notifier.notify()
|
||||||
return self.__new_file_written
|
return written
|
||||||
|
|
||||||
@aiotools.atomic
|
@aiotools.atomic
|
||||||
async def remove(self, name: str) -> None:
|
async def remove(self, name: str) -> None:
|
||||||
@@ -392,18 +381,15 @@ class Plugin(BaseMsd): # pylint: disable=too-many-instance-attributes
|
|||||||
|
|
||||||
# =====
|
# =====
|
||||||
|
|
||||||
async def __close_new_file(self) -> None:
|
async def __close_new_writer(self) -> None:
|
||||||
try:
|
try:
|
||||||
if self.__new_file:
|
if self.__new_writer:
|
||||||
get_logger().info("Closing new image file ...")
|
get_logger().info("Closing new image file ...")
|
||||||
await aiofs.afile_sync(self.__new_file)
|
await self.__new_writer.close()
|
||||||
await self.__new_file.close() # type: ignore
|
|
||||||
except Exception:
|
except Exception:
|
||||||
get_logger().exception("Can't close image file")
|
get_logger().exception("Can't close image file")
|
||||||
finally:
|
finally:
|
||||||
self.__new_file = None
|
self.__new_writer = None
|
||||||
self.__new_file_written = 0
|
|
||||||
self.__new_file_unsynced = 0
|
|
||||||
|
|
||||||
# =====
|
# =====
|
||||||
|
|
||||||
|
|||||||
@@ -29,13 +29,9 @@ from typing import Dict
|
|||||||
from typing import AsyncGenerator
|
from typing import AsyncGenerator
|
||||||
from typing import Optional
|
from typing import Optional
|
||||||
|
|
||||||
import aiofiles
|
|
||||||
import aiofiles.base
|
|
||||||
|
|
||||||
from ....logging import get_logger
|
from ....logging import get_logger
|
||||||
|
|
||||||
from .... import aiotools
|
from .... import aiotools
|
||||||
from .... import aiofs
|
|
||||||
|
|
||||||
from ....yamlconf import Option
|
from ....yamlconf import Option
|
||||||
|
|
||||||
@@ -54,10 +50,10 @@ from .. import MsdDisconnectedError
|
|||||||
from .. import MsdMultiNotSupported
|
from .. import MsdMultiNotSupported
|
||||||
from .. import MsdCdromNotSupported
|
from .. import MsdCdromNotSupported
|
||||||
from .. import BaseMsd
|
from .. import BaseMsd
|
||||||
|
from .. import MsdImageWriter
|
||||||
|
|
||||||
from .gpio import Gpio
|
from .gpio import Gpio
|
||||||
|
|
||||||
from .drive import ImageInfo
|
|
||||||
from .drive import DeviceInfo
|
from .drive import DeviceInfo
|
||||||
|
|
||||||
|
|
||||||
@@ -91,9 +87,7 @@ class Plugin(BaseMsd): # pylint: disable=too-many-instance-attributes
|
|||||||
self.__device_info: Optional[DeviceInfo] = None
|
self.__device_info: Optional[DeviceInfo] = None
|
||||||
self.__connected = False
|
self.__connected = False
|
||||||
|
|
||||||
self.__device_file: Optional[aiofiles.base.AiofilesContextManager] = None
|
self.__device_writer: Optional[MsdImageWriter] = None
|
||||||
self.__written = 0
|
|
||||||
self.__unsynced = 0
|
|
||||||
|
|
||||||
self.__notifier = aiotools.AioNotifier()
|
self.__notifier = aiotools.AioNotifier()
|
||||||
self.__region = aiotools.AioExclusiveRegion(MsdIsBusyError, self.__notifier)
|
self.__region = aiotools.AioExclusiveRegion(MsdIsBusyError, self.__notifier)
|
||||||
@@ -132,7 +126,7 @@ class Plugin(BaseMsd): # pylint: disable=too-many-instance-attributes
|
|||||||
storage = {
|
storage = {
|
||||||
"size": self.__device_info.size,
|
"size": self.__device_info.size,
|
||||||
"free": self.__device_info.free,
|
"free": self.__device_info.free,
|
||||||
"uploading": bool(self.__device_file)
|
"uploading": (self.__device_writer.get_state() if self.__device_writer else None),
|
||||||
}
|
}
|
||||||
drive = {
|
drive = {
|
||||||
"image": (self.__device_info.image and dataclasses.asdict(self.__device_info.image)),
|
"image": (self.__device_info.image and dataclasses.asdict(self.__device_info.image)),
|
||||||
@@ -177,7 +171,7 @@ class Plugin(BaseMsd): # pylint: disable=too-many-instance-attributes
|
|||||||
@aiotools.atomic
|
@aiotools.atomic
|
||||||
async def cleanup(self) -> None:
|
async def cleanup(self) -> None:
|
||||||
try:
|
try:
|
||||||
await self.__close_device_file()
|
await self.__close_device_writer()
|
||||||
finally:
|
finally:
|
||||||
self.__gpio.close()
|
self.__gpio.close()
|
||||||
|
|
||||||
@@ -214,7 +208,7 @@ class Plugin(BaseMsd): # pylint: disable=too-many-instance-attributes
|
|||||||
self.__connected = connected
|
self.__connected = connected
|
||||||
|
|
||||||
@contextlib.asynccontextmanager
|
@contextlib.asynccontextmanager
|
||||||
async def write_image(self, name: str) -> AsyncGenerator[None, None]:
|
async def write_image(self, name: str, size: int) -> AsyncGenerator[None, None]:
|
||||||
async with self.__working():
|
async with self.__working():
|
||||||
async with self.__region:
|
async with self.__region:
|
||||||
try:
|
try:
|
||||||
@@ -222,30 +216,22 @@ class Plugin(BaseMsd): # pylint: disable=too-many-instance-attributes
|
|||||||
if self.__connected:
|
if self.__connected:
|
||||||
raise MsdConnectedError()
|
raise MsdConnectedError()
|
||||||
|
|
||||||
self.__device_file = await aiofiles.open(self.__device_info.path, mode="w+b", buffering=0) # type: ignore
|
self.__device_writer = await MsdImageWriter(self.__device_info.path, size, self.__sync_chunk_size).open()
|
||||||
self.__written = 0
|
|
||||||
self.__unsynced = 0
|
|
||||||
|
|
||||||
await self.__write_image_info(name, complete=False)
|
await self.__write_image_info(False)
|
||||||
await self.__notifier.notify()
|
await self.__notifier.notify()
|
||||||
yield
|
yield
|
||||||
await self.__write_image_info(name, complete=True)
|
await self.__write_image_info(True)
|
||||||
finally:
|
finally:
|
||||||
await self.__close_device_file()
|
await self.__close_device_writer()
|
||||||
await self.__load_device_info()
|
await self.__load_device_info()
|
||||||
|
|
||||||
def get_upload_chunk_size(self) -> int:
|
def get_upload_chunk_size(self) -> int:
|
||||||
return self.__upload_chunk_size
|
return self.__upload_chunk_size
|
||||||
|
|
||||||
async def write_image_chunk(self, chunk: bytes) -> int:
|
async def write_image_chunk(self, chunk: bytes) -> int:
|
||||||
assert self.__device_file
|
assert self.__device_writer
|
||||||
await self.__device_file.write(chunk) # type: ignore
|
return (await self.__device_writer.write(chunk))
|
||||||
self.__written += len(chunk)
|
|
||||||
self.__unsynced += len(chunk)
|
|
||||||
if self.__unsynced >= self.__sync_chunk_size:
|
|
||||||
await aiofs.afile_sync(self.__device_file)
|
|
||||||
self.__unsynced = 0
|
|
||||||
return self.__written
|
|
||||||
|
|
||||||
@aiotools.atomic
|
@aiotools.atomic
|
||||||
async def remove(self, name: str) -> None:
|
async def remove(self, name: str) -> None:
|
||||||
@@ -262,27 +248,21 @@ class Plugin(BaseMsd): # pylint: disable=too-many-instance-attributes
|
|||||||
|
|
||||||
# =====
|
# =====
|
||||||
|
|
||||||
async def __write_image_info(self, name: str, complete: bool) -> None:
|
async def __write_image_info(self, complete: bool) -> None:
|
||||||
assert self.__device_file
|
assert self.__device_writer
|
||||||
assert self.__device_info
|
assert self.__device_info
|
||||||
if not (await self.__device_info.write_image_info(
|
if not (await self.__device_info.write_image_info(self.__device_writer, complete)):
|
||||||
device_file=self.__device_file,
|
|
||||||
image_info=ImageInfo(name, self.__written, complete),
|
|
||||||
)):
|
|
||||||
get_logger().error("Can't write image info because device is full")
|
get_logger().error("Can't write image info because device is full")
|
||||||
|
|
||||||
async def __close_device_file(self) -> None:
|
async def __close_device_writer(self) -> None:
|
||||||
try:
|
try:
|
||||||
if self.__device_file:
|
if self.__device_writer:
|
||||||
get_logger().info("Closing device file ...")
|
get_logger().info("Closing device file ...")
|
||||||
await aiofs.afile_sync(self.__device_file)
|
await self.__device_writer.close() # type: ignore
|
||||||
await self.__device_file.close() # type: ignore
|
|
||||||
except Exception:
|
except Exception:
|
||||||
get_logger().exception("Can't close device file")
|
get_logger().exception("Can't close device file")
|
||||||
finally:
|
finally:
|
||||||
self.__device_file = None
|
self.__device_writer = None
|
||||||
self.__written = 0
|
|
||||||
self.__unsynced = 0
|
|
||||||
|
|
||||||
async def __load_device_info(self) -> None:
|
async def __load_device_info(self) -> None:
|
||||||
retries = self.__init_retries
|
retries = self.__init_retries
|
||||||
|
|||||||
@@ -29,11 +29,11 @@ import dataclasses
|
|||||||
from typing import IO
|
from typing import IO
|
||||||
from typing import Optional
|
from typing import Optional
|
||||||
|
|
||||||
import aiofiles.base
|
|
||||||
|
|
||||||
from .... import aiotools
|
from .... import aiotools
|
||||||
from .... import aiofs
|
from .... import aiofs
|
||||||
|
|
||||||
|
from .. import MsdImageWriter
|
||||||
|
|
||||||
|
|
||||||
# =====
|
# =====
|
||||||
_IMAGE_INFO_SIZE = 4096
|
_IMAGE_INFO_SIZE = 4096
|
||||||
@@ -121,11 +121,10 @@ class DeviceInfo:
|
|||||||
image=image_info,
|
image=image_info,
|
||||||
)
|
)
|
||||||
|
|
||||||
async def write_image_info(
|
async def write_image_info(self, device_writer: MsdImageWriter, complete: bool) -> bool:
|
||||||
self,
|
device_file = device_writer.get_file()
|
||||||
device_file: aiofiles.base.AiofilesContextManager,
|
state = device_writer.get_state()
|
||||||
image_info: ImageInfo,
|
image_info = ImageInfo(state["name"], state["written"], complete)
|
||||||
) -> bool:
|
|
||||||
|
|
||||||
if self.size - image_info.size > _IMAGE_INFO_SIZE:
|
if self.size - image_info.size > _IMAGE_INFO_SIZE:
|
||||||
await device_file.seek(self.size - _IMAGE_INFO_SIZE) # type: ignore
|
await device_file.seek(self.size - _IMAGE_INFO_SIZE) # type: ignore
|
||||||
|
|||||||
@@ -101,13 +101,13 @@ export function Msd() {
|
|||||||
var __clickUploadNewImageButton = function() {
|
var __clickUploadNewImageButton = function() {
|
||||||
let form_data = new FormData();
|
let form_data = new FormData();
|
||||||
form_data.append("image", __image_file.name);
|
form_data.append("image", __image_file.name);
|
||||||
|
form_data.append("size", __image_file.size);
|
||||||
form_data.append("data", __image_file);
|
form_data.append("data", __image_file);
|
||||||
|
|
||||||
__upload_http = new XMLHttpRequest();
|
__upload_http = new XMLHttpRequest();
|
||||||
__upload_http.open("POST", "/api/msd/write", true);
|
__upload_http.open("POST", "/api/msd/write", true);
|
||||||
__upload_http.upload.timeout = 15000;
|
__upload_http.upload.timeout = 15000;
|
||||||
__upload_http.onreadystatechange = __uploadStateChange;
|
__upload_http.onreadystatechange = __uploadStateChange;
|
||||||
__upload_http.upload.onprogress = __uploadProgress;
|
|
||||||
__upload_http.send(form_data);
|
__upload_http.send(form_data);
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -123,16 +123,8 @@ export function Msd() {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
var __uploadProgress = function(event) {
|
|
||||||
if(event.lengthComputable) {
|
|
||||||
let percent = Math.round((event.loaded * 100) / event.total);
|
|
||||||
tools.progressSetValue($("msd-uploading-progress"), `${percent}%`, percent);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
var __clickAbortUploadingButton = function() {
|
var __clickAbortUploadingButton = function() {
|
||||||
__upload_http.onreadystatechange = null;
|
__upload_http.onreadystatechange = null;
|
||||||
__upload_http.upload.onprogress = null;
|
|
||||||
__upload_http.abort();
|
__upload_http.abort();
|
||||||
__upload_http = null;
|
__upload_http = null;
|
||||||
tools.progressSetValue($("msd-uploading-progress"), "Aborted", 0);
|
tools.progressSetValue($("msd-uploading-progress"), "Aborted", 0);
|
||||||
@@ -238,8 +230,12 @@ export function Msd() {
|
|||||||
tools.hiddenSetVisible($("msd-submenu-new-image"), __image_file);
|
tools.hiddenSetVisible($("msd-submenu-new-image"), __image_file);
|
||||||
$("msd-new-image-name").innerHTML = (__image_file ? __image_file.name : "");
|
$("msd-new-image-name").innerHTML = (__image_file ? __image_file.name : "");
|
||||||
$("msd-new-image-size").innerHTML = (__image_file ? tools.formatSize(__image_file.size) : "");
|
$("msd-new-image-size").innerHTML = (__image_file ? tools.formatSize(__image_file.size) : "");
|
||||||
|
|
||||||
if (!__upload_http) {
|
if (!__upload_http) {
|
||||||
tools.progressSetValue($("msd-uploading-progress"), "Waiting for upload (press UPLOAD button) ...", 0);
|
tools.progressSetValue($("msd-uploading-progress"), "Waiting for upload (press UPLOAD button) ...", 0);
|
||||||
|
} else if (__state.storage.uploading) {
|
||||||
|
let percent = Math.round(__state.storage.uploading.written * 100 / __state.storage.uploading.size);
|
||||||
|
tools.progressSetValue($("msd-uploading-progress"), `${percent}%`, percent);
|
||||||
}
|
}
|
||||||
|
|
||||||
} else {
|
} else {
|
||||||
|
|||||||
Reference in New Issue
Block a user