summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--kvmd/kvmd/msd.py64
-rw-r--r--kvmd/kvmd/server.py7
2 files changed, 40 insertions, 31 deletions
diff --git a/kvmd/kvmd/msd.py b/kvmd/kvmd/msd.py
index 84cd7477..8dac6f50 100644
--- a/kvmd/kvmd/msd.py
+++ b/kvmd/kvmd/msd.py
@@ -53,6 +53,7 @@ class MassStorageDeviceInfo(NamedTuple):
real: str
size: int
image_name: str
+ image_complete: bool
manufacturer: str = ""
product: str = ""
serial: str = ""
@@ -61,8 +62,8 @@ class MassStorageDeviceInfo(NamedTuple):
_DISK_META_SIZE = 4096
_DISK_META_MAGIC_SIZE = 16
_DISK_META_IMAGE_NAME_SIZE = 256
-_DISK_META_PADS_SIZE = _DISK_META_SIZE - _DISK_META_IMAGE_NAME_SIZE - _DISK_META_MAGIC_SIZE * 8
-_DISK_META_FORMAT = ">%dL%dc%dx%dL" % (
+_DISK_META_PADS_SIZE = _DISK_META_SIZE - _DISK_META_IMAGE_NAME_SIZE - 1 - _DISK_META_MAGIC_SIZE * 8
+_DISK_META_FORMAT = ">%dL%dc?%dx%dL" % (
_DISK_META_MAGIC_SIZE,
_DISK_META_IMAGE_NAME_SIZE,
_DISK_META_PADS_SIZE,
@@ -71,7 +72,7 @@ _DISK_META_FORMAT = ">%dL%dc%dx%dL" % (
_DISK_META_MAGIC = [0x1ACE1ACE] * _DISK_META_MAGIC_SIZE
-def _make_disk_meta(image_name: str) -> bytes:
+def _make_disk_meta(image_name: str, image_complete: bool) -> bytes:
return struct.pack(
_DISK_META_FORMAT,
*_DISK_META_MAGIC,
@@ -79,12 +80,16 @@ def _make_disk_meta(image_name: str) -> bytes:
image_name.encode("utf-8")
+ b"\x00" * _DISK_META_IMAGE_NAME_SIZE
)[:_DISK_META_IMAGE_NAME_SIZE]).cast("c"),
+ image_complete,
*_DISK_META_MAGIC,
)
def _parse_disk_meta(data: bytes) -> Dict:
- disk_meta = {"image_name": ""}
+ disk_meta = {
+ "image_name": "",
+ "image_complete": False,
+ }
try:
parsed = list(struct.unpack(_DISK_META_FORMAT, data))
except struct.error:
@@ -95,32 +100,32 @@ def _parse_disk_meta(data: bytes) -> Dict:
if magic_begin == magic_end == _DISK_META_MAGIC:
image_name_bytes = b"".join(parsed[_DISK_META_MAGIC_SIZE:_DISK_META_MAGIC_SIZE + _DISK_META_IMAGE_NAME_SIZE])
disk_meta["image_name"] = image_name_bytes.decode("utf-8", errors="ignore").strip("\x00").strip()
+ disk_meta["image_complete"] = parsed[_DISK_META_MAGIC_SIZE + _DISK_META_IMAGE_NAME_SIZE]
return disk_meta
-def explore_device(path: str) -> Optional[MassStorageDeviceInfo]:
+def explore_device(device_path: str) -> Optional[MassStorageDeviceInfo]:
# udevadm info -a -p $(udevadm info -q path -n /dev/sda)
ctx = pyudev.Context()
- device = pyudev.Devices.from_device_file(ctx, path)
+ device = pyudev.Devices.from_device_file(ctx, device_path)
if device.subsystem != "block":
return None
try:
size = device.attributes.asint("size") * 512
except KeyError:
return None
-
usb_device = device.find_parent("usb", "usb_device")
- with open(path, "rb") as device_file:
+ with open(device_path, "rb") as device_file:
device_file.seek(size - _DISK_META_SIZE)
disk_meta = _parse_disk_meta(device_file.read())
- return MassStorageDeviceInfo(
- path=path,
- real=(os.readlink(path) if os.path.islink(path) else ""),
+ return MassStorageDeviceInfo( # type: ignore
+ path=device_path,
+ real=(os.readlink(device_path) if os.path.islink(device_path) else device_path),
size=size,
- image_name=disk_meta["image_name"],
+ **disk_meta,
**{
attr: usb_device.attributes.asstring(attr).strip()
for attr in ["manufacturer", "product", "serial"]
@@ -182,7 +187,7 @@ class MassStorageDevice: # pylint: disable=too-many-instance-attributes
# TODO: disable gpio
if not no_delay:
await asyncio.sleep(self.__init_delay)
- await self.__reread_device_info()
+ await self.__load_device_info()
get_logger().info("Mass-storage device switched to KVM: %s", self.__device_info)
@_operated_and_locked
@@ -211,30 +216,27 @@ class MassStorageDevice: # pylint: disable=too-many-instance-attributes
async def __aenter__(self) -> "MassStorageDevice":
if not self.__device_info:
raise IsNotConnectedToKvmError()
- self._device_file = await aiofiles.open(self.__device_info.path, mode="wb", buffering=0)
+ self._device_file = await aiofiles.open(self.__device_info.path, mode="w+b", buffering=0)
self.__writed = 0
return self
- async def write_image_name(self, image_name: str) -> None:
+ async def write_image_meta(self, image_name: str, image_complete: bool) -> None:
async with self._lock:
assert self._device_file
assert self.__device_info
if self.__write_meta:
- await self._device_file.seek(self.__device_info.size - _DISK_META_SIZE)
- await self._device_file.write(_make_disk_meta(image_name))
- await self._device_file.flush()
- await self.__loop.run_in_executor(None, os.fsync, self._device_file.fileno())
- await self._device_file.seek(0)
- await self.__reread_device_info()
+ if self.__device_info.size - self.__writed > _DISK_META_SIZE:
+ await self._device_file.seek(self.__device_info.size - _DISK_META_SIZE)
+ await self.__write_to_device_file(_make_disk_meta(image_name, image_complete))
+ await self._device_file.seek(0)
+ await self.__load_device_info()
+ else:
+ get_logger().error("Can't write image meta because device is full")
async def write_image_chunk(self, chunk: bytes) -> int:
async with self._lock:
- assert self._device_file
- size = len(chunk)
- await self._device_file.write(chunk)
- await self._device_file.flush()
- await self.__loop.run_in_executor(None, os.fsync, self._device_file.fileno())
- self.__writed += size
+ await self.__write_to_device_file(chunk)
+ self.__writed += len(chunk)
return self.__writed
async def __aexit__(
@@ -246,7 +248,13 @@ class MassStorageDevice: # pylint: disable=too-many-instance-attributes
async with self._lock:
await self.__close_device_file()
- async def __reread_device_info(self) -> None:
+ async def __write_to_device_file(self, data: bytes) -> None:
+ assert self._device_file
+ await self._device_file.write(data)
+ await self._device_file.flush()
+ await self.__loop.run_in_executor(None, os.fsync, self._device_file.fileno())
+
+ async def __load_device_info(self) -> None:
device_info = await self.__loop.run_in_executor(None, explore_device, self._device_path)
if not device_info:
raise MassStorageError("Can't explore device %r" % (self._device_path))
diff --git a/kvmd/kvmd/server.py b/kvmd/kvmd/server.py
index fedc1b07..62a13560 100644
--- a/kvmd/kvmd/server.py
+++ b/kvmd/kvmd/server.py
@@ -173,23 +173,24 @@ class Server: # pylint: disable=too-many-instance-attributes
writed = 0
try:
field = await reader.next()
- if field.name != "image_name":
+ if not field or field.name != "image_name":
raise RuntimeError("Missing 'image_name' field")
image_name = (await field.read()).decode("utf-8")[:256]
field = await reader.next()
- if field.name != "image_data":
+ if not field or field.name != "image_data":
raise RuntimeError("Missing 'image_data' field")
async with self.__msd:
await self.__broadcast_event("msd_state", state="busy") # type: ignore
logger.info("Writing image %r to mass-storage device ...", image_name)
- await self.__msd.write_image_name(image_name)
+ await self.__msd.write_image_meta(image_name, False)
while True:
chunk = await field.read_chunk(self.__msd_chunk_size)
if not chunk:
break
writed = await self.__msd.write_image_chunk(chunk)
+ await self.__msd.write_image_meta(image_name, True)
await self.__broadcast_event("msd_state", state="free") # type: ignore
finally:
if writed != 0: