2022-11-11 16:30:04 +08:00
|
|
|
#
|
|
|
|
# Copyright 2011, 2013 Red Hat, Inc.
|
|
|
|
#
|
|
|
|
# This work is licensed under the GNU GPLv2 or later.
|
|
|
|
# See the COPYING file in the top-level directory.
|
|
|
|
|
|
|
|
import os
|
|
|
|
|
|
|
|
from .device import Device
|
|
|
|
from ..xmlbuilder import XMLProperty
|
|
|
|
|
|
|
|
|
|
|
|
class DeviceFilesystem(Device):
|
|
|
|
XML_NAME = "filesystem"
|
2024-05-17 13:37:53 +08:00
|
|
|
_XML_PROP_ORDER = ["_type_prop", "accessmode", "fmode", "dmode"]
|
2022-11-11 16:30:04 +08:00
|
|
|
|
|
|
|
TYPE_MOUNT = "mount"
|
|
|
|
TYPE_TEMPLATE = "template"
|
|
|
|
TYPE_FILE = "file"
|
|
|
|
TYPE_BLOCK = "block"
|
|
|
|
TYPE_RAM = "ram"
|
|
|
|
|
|
|
|
MODE_MAPPED = "mapped"
|
|
|
|
MODE_SQUASH = "squash"
|
|
|
|
|
|
|
|
DRIVER_LOOP = "loop"
|
|
|
|
DRIVER_NBD = "nbd"
|
|
|
|
|
|
|
|
_type_prop = XMLProperty("./@type")
|
|
|
|
accessmode = XMLProperty("./@accessmode")
|
2024-05-17 13:37:53 +08:00
|
|
|
model = XMLProperty("./@model")
|
|
|
|
multidevs = XMLProperty("./@multidevs")
|
|
|
|
fmode = XMLProperty("./@fmode")
|
|
|
|
dmode = XMLProperty("./@dmode")
|
2022-11-11 16:30:04 +08:00
|
|
|
|
|
|
|
readonly = XMLProperty("./readonly", is_bool=True)
|
2024-05-17 13:37:53 +08:00
|
|
|
space_hard_limit = XMLProperty("./space_hard_limit")
|
|
|
|
space_soft_limit = XMLProperty("./space_soft_limit")
|
|
|
|
|
|
|
|
driver_wrpolicy = XMLProperty("./driver/@wrpolicy")
|
|
|
|
driver_type = XMLProperty("./driver/@type")
|
|
|
|
driver_format = XMLProperty("./driver/@format")
|
|
|
|
driver_queue = XMLProperty("./driver/@queue")
|
|
|
|
driver_name = XMLProperty("./driver/@name")
|
|
|
|
|
|
|
|
target_dir = XMLProperty("./target/@dir")
|
|
|
|
|
|
|
|
source_dir = XMLProperty("./source/@dir")
|
|
|
|
source_name = XMLProperty("./source/@name")
|
|
|
|
source_file = XMLProperty("./source/@file")
|
|
|
|
source_dev = XMLProperty("./source/@dev")
|
|
|
|
source_usage = XMLProperty("./source/@usage")
|
|
|
|
source_units = XMLProperty("./source/@units")
|
|
|
|
source_pool = XMLProperty("./source/@pool")
|
|
|
|
source_volume = XMLProperty("./source/@volume")
|
|
|
|
source_socket = XMLProperty("./source/socket")
|
|
|
|
|
|
|
|
binary_path = XMLProperty("./binary/@path")
|
|
|
|
binary_xattr = XMLProperty("./binary/@xattr", is_onoff=True)
|
|
|
|
binary_cache_mode = XMLProperty("./binary/cache/@mode")
|
|
|
|
binary_lock_posix = XMLProperty("./binary/lock/@posix", is_onoff=True)
|
|
|
|
binary_lock_flock = XMLProperty("./binary/lock/@flock", is_onoff=True)
|
|
|
|
binary_sandbox_mode = XMLProperty("./binary/sandbox/@mode")
|
2022-11-11 16:30:04 +08:00
|
|
|
|
|
|
|
def _type_to_source_prop(self):
|
|
|
|
if self.type == DeviceFilesystem.TYPE_TEMPLATE:
|
2024-05-17 13:37:53 +08:00
|
|
|
return "source_name"
|
2022-11-11 16:30:04 +08:00
|
|
|
elif self.type == DeviceFilesystem.TYPE_FILE:
|
2024-05-17 13:37:53 +08:00
|
|
|
return "source_file"
|
2022-11-11 16:30:04 +08:00
|
|
|
elif self.type == DeviceFilesystem.TYPE_BLOCK:
|
2024-05-17 13:37:53 +08:00
|
|
|
return "source_dev"
|
2022-11-11 16:30:04 +08:00
|
|
|
elif self.type == DeviceFilesystem.TYPE_RAM:
|
2024-05-17 13:37:53 +08:00
|
|
|
return "source_usage"
|
2022-11-11 16:30:04 +08:00
|
|
|
else:
|
2024-05-17 13:37:53 +08:00
|
|
|
return "source_dir"
|
2022-11-11 16:30:04 +08:00
|
|
|
|
|
|
|
def _get_source(self):
|
|
|
|
return getattr(self, self._type_to_source_prop())
|
|
|
|
def _set_source(self, val):
|
|
|
|
return setattr(self, self._type_to_source_prop(), val)
|
|
|
|
source = property(_get_source, _set_source)
|
|
|
|
|
2024-05-17 13:37:53 +08:00
|
|
|
def _get_target(self):
|
|
|
|
return self.target_dir
|
|
|
|
def _set_target(self, val):
|
|
|
|
self.target_dir = val
|
|
|
|
target = property(_get_target, _set_target)
|
|
|
|
|
2022-11-11 16:30:04 +08:00
|
|
|
def _get_type(self):
|
|
|
|
return getattr(self, '_type_prop')
|
|
|
|
def _set_type(self, val):
|
|
|
|
# Get type/value of the attribute of "source" property
|
|
|
|
old_source_type = self._type_to_source_prop()
|
|
|
|
old_source_value = self.source
|
|
|
|
|
|
|
|
# Update "type" property
|
|
|
|
new_type = setattr(self, '_type_prop', val)
|
|
|
|
|
|
|
|
# If the attribute type of 'source' property has changed
|
|
|
|
# restore the value
|
|
|
|
if old_source_type != self._type_to_source_prop():
|
|
|
|
self.source = old_source_value
|
|
|
|
|
|
|
|
return new_type
|
|
|
|
|
|
|
|
type = property(_get_type, _set_type)
|
|
|
|
|
|
|
|
|
|
|
|
##############
|
|
|
|
# Validation #
|
|
|
|
##############
|
|
|
|
|
|
|
|
def validate_target(self, target):
|
|
|
|
# In case of qemu for default fs type (mount) target is not
|
|
|
|
# actually a directory, it is merely a arbitrary string tag
|
|
|
|
# that is exported to the guest as a hint for where to mount
|
|
|
|
if ((self.conn.is_qemu() or self.conn.is_test()) and
|
|
|
|
(self.type is None or
|
|
|
|
self.type == self.TYPE_MOUNT)):
|
|
|
|
return
|
|
|
|
|
|
|
|
if not os.path.isabs(target):
|
|
|
|
raise ValueError(_("Filesystem target '%s' must be an absolute "
|
|
|
|
"path") % target)
|
|
|
|
|
|
|
|
def validate(self):
|
|
|
|
if self.target:
|
|
|
|
self.validate_target(self.target)
|
|
|
|
|
2024-05-17 13:37:53 +08:00
|
|
|
def default_accessmode(self):
|
|
|
|
if self.driver_type == "virtiofs":
|
|
|
|
# let libvirt fill in default accessmode=passthrough
|
|
|
|
return None
|
|
|
|
# libvirt qemu defaults to accessmode=passthrough, but that
|
|
|
|
# really only works well for qemu running as root, which is
|
|
|
|
# not the common case. so use mode=mapped
|
|
|
|
return self.MODE_MAPPED
|
|
|
|
|
2022-11-11 16:30:04 +08:00
|
|
|
|
|
|
|
##################
|
|
|
|
# Default config #
|
|
|
|
##################
|
|
|
|
|
|
|
|
def set_defaults(self, guest):
|
|
|
|
ignore = guest
|
|
|
|
|
2024-05-17 13:37:53 +08:00
|
|
|
if not (self.conn.is_qemu() or
|
|
|
|
self.conn.is_lxc() or
|
|
|
|
self.conn.is_test()):
|
|
|
|
return
|
|
|
|
|
|
|
|
# type=mount is the libvirt default. But hardcode it since other
|
|
|
|
# bits like validation depend on it
|
|
|
|
if self.type is None:
|
|
|
|
self.type = self.TYPE_MOUNT
|
|
|
|
|
|
|
|
if self.accessmode is None:
|
|
|
|
self.accessmode = self.default_accessmode()
|