Skip to content

Commit

Permalink
scuba: Add support for mounting named volumes
Browse files Browse the repository at this point in the history
Named volumes were unintentionally supported prior to #227. This
restores the prior behavior while remaining unambiguiously compatible
with relative bind-mounts added in #227.

This also adds support for explicit named volumes in complex form
configuration.

Fixes #248
  • Loading branch information
JonathonReinhart committed Mar 22, 2024
1 parent aefa26b commit 7a20042
Show file tree
Hide file tree
Showing 7 changed files with 299 additions and 46 deletions.
53 changes: 41 additions & 12 deletions docs/configuration.rst
Original file line number Diff line number Diff line change
Expand Up @@ -147,25 +147,40 @@ style <https://yaml.org/spec/1.2/spec.html#id2788097>`_:

The optional ``volumes`` node *(added in v2.9.0)* allows additional
`bind-mounts <https://docs.docker.com/storage/bind-mounts/>`_ to be specified.
As of v2.13.0, `named volumes <https://docs.docker.com/storage/volumes/>`_
are also supported.

``volumes`` is a mapping (dictionary) where each key is the container-path.
In the simple form, the value is a string, the host-path to be bind-mounted:
In the simple form, the value is a string, which can be:

* An absolute or relative path which results in a bind-mount.
* A Docker volume name.

.. code-block:: yaml
:caption: Example of simple-form volumes
volumes:
/var/lib/foo: /host/foo
/var/lib/bar: ./bar
/var/lib/foo: /host/foo # bind-mount: absolute path
/var/lib/bar: ./bar # bind-mount: path relative to .scuba.yml dir
/var/log: persist-logs # named volume
In the complex form, the value is a mapping with the following supported keys:

* ``hostpath``: An absolute or relative path specifying a host bind-mount.
* ``name``: The name of a named Docker volume.
* ``options``: A comma-separated list of volume options.

In the complex form, the value is a mapping which must contain a ``hostpath``
subkey. It can also contain an ``options`` subkey with a comma-separated list
of volume options:
``hostpath`` and ``name`` are mutually-exclusive and one must be specified.

.. code-block:: yaml
:caption: Example of complex-form volumes
volumes:
/var/lib/foo:
hostpath: /host/foo
hostpath: /host/foo # bind-mount
options: ro,cached
/var/log:
name: persist-logs # named volume
The paths (host or container) used in volume mappings can contain environment
variables **which are expanded in the host environment**. For example, this
Expand All @@ -182,18 +197,32 @@ configuration error.

Volume container paths must be absolute.

Volume host paths can be absolute or relative. If a relative path is used, it
is interpreted as relative to the directory in which ``.scuba.yml`` is found.
To avoid ambiguity, relative paths must start with ``./`` or ``../``.
Bind-mount host paths can be absolute or relative. If a relative path is used,
it is interpreted as relative to the directory in which ``.scuba.yml`` is
found. To avoid ambiguity with a named volume, relative paths must start with
``./`` or ``../``.

Volume host directories which do not already exist are created as the current
user before creating the container.
Bind-mount host directories which do not already exist are created as the
current user before creating the container.

.. note::
Because variable expansion is now applied to all volume paths, if one
desires to use a literal ``$`` character in a path, it must be written as
``$$``.

.. note::
Docker named volumes are created with ``drwxr-xr-x`` (0755) permissions.
If scuba is not run with ``--root``, the scuba user will be unable to write
to this directory. As a workaround, one can use a :ref:`root hook
<conf_hooks>` to change permissions on the directory.

.. code-block:: yaml
volumes:
/foo: foo-volume
hooks:
root: chmod 777 /foo
.. _conf_aliases:

Expand Down
66 changes: 54 additions & 12 deletions scuba/config.py
Original file line number Diff line number Diff line change
Expand Up @@ -19,6 +19,8 @@
Environment = Dict[str, str]
_T = TypeVar("_T")

VOLUME_NAME_PATTERN = re.compile(r"^[a-zA-Z0-9][a-zA-Z0-9_.-]+$")


class ConfigError(Exception):
pass
Expand Down Expand Up @@ -410,9 +412,14 @@ def _absoluteify_path(in_str: str, base_dir: Optional[Path] = None) -> Path:
@dataclasses.dataclass(frozen=True)
class ScubaVolume:
container_path: Path
host_path: Path # TODO: Optional for anonymous volume
host_path: Optional[Path] = None
volume_name: Optional[str] = None
options: List[str] = dataclasses.field(default_factory=list)

def __post_init__(self) -> None:
if sum(bool(x) for x in (self.host_path, self.volume_name)) != 1:
raise ValueError("Exactly one of host_path, volume_name must be set")

@classmethod
def from_dict(
cls, cpath: Path, node: CfgNode, scuba_root: Optional[Path]
Expand All @@ -423,34 +430,69 @@ def from_dict(

# Simple form:
# volumes:
# /foo: /host/foo
# /foo: foo-volume # volume name
# /bar: /host/bar # absolute path
# /snap: ./snap # relative path
if isinstance(node, str):
node = _expand_env_vars(node)

# Absolute or relative path
valid_prefixes = ("/", "./", "../")
if any(node.startswith(pfx) for pfx in valid_prefixes):
return cls(
container_path=cpath,
host_path=_absoluteify_path(node, scuba_root),
)

# Volume name
if not VOLUME_NAME_PATTERN.match(node):
raise ConfigError(f"Invalid volume name: {node!r}")
return cls(
container_path=cpath,
host_path=_absoluteify_path(node, scuba_root),
volume_name=node,
)

# Complex form
# volumes:
# /foo:
# hostpath: /host/foo
# options: ro,z
# /bar:
# name: bar-volume
if isinstance(node, dict):
hpath = node.get("hostpath")
if hpath is None:
raise ConfigError(f"Volume {cpath} must have a 'hostpath' subkey")
hpath = _expand_env_vars(hpath)
return cls(
container_path=cpath,
host_path=_absoluteify_path(hpath, scuba_root),
options=_get_delimited_str_list(node, "options", ","),
)
name = node.get("name")
options = _get_delimited_str_list(node, "options", ",")

if sum(bool(x) for x in (hpath, name)) != 1:
raise ConfigError(
f"Volume {cpath} must have exactly one of"
" 'hostpath' or 'name' subkey"
)

if hpath is not None:
hpath = _expand_env_vars(hpath)
return cls(
container_path=cpath,
host_path=_absoluteify_path(hpath, scuba_root),
options=options,
)

if name is not None:
return cls(
container_path=cpath,
volume_name=_expand_env_vars(name),
options=options,
)

raise ConfigError(f"{cpath}: must be string or dict")

def get_vol_opt(self) -> str:
return make_vol_opt(self.host_path, self.container_path, self.options)
if self.host_path:
return make_vol_opt(self.host_path, self.container_path, self.options)
if self.volume_name:
return make_vol_opt(self.volume_name, self.container_path, self.options)
raise Exception("host_path or volume_name must be set")


@dataclasses.dataclass(frozen=True)
Expand Down
13 changes: 9 additions & 4 deletions scuba/dockerutil.py
Original file line number Diff line number Diff line change
Expand Up @@ -132,17 +132,22 @@ def get_image_entrypoint(image: str) -> Optional[Sequence[str]]:


def make_vol_opt(
hostdir: Path,
hostdir_or_volname: Union[Path, str],
contdir: Path,
options: Optional[Sequence[str]] = None,
) -> str:
"""Generate a docker volume option"""
if not hostdir.is_absolute():
raise ValueError(f"hostdir not absolute: {hostdir}")
if isinstance(hostdir_or_volname, Path):
hostdir: Path = hostdir_or_volname
if not hostdir.is_absolute():
# NOTE: As of Docker Engine version 23, you can use relative paths
# on the host. But we have no minimum Docker version, so we don't
# rely on this.
raise ValueError(f"hostdir not absolute: {hostdir}")
if not contdir.is_absolute():
raise ValueError(f"contdir not absolute: {contdir}")

vol = f"--volume={hostdir}:{contdir}"
vol = f"--volume={hostdir_or_volname}:{contdir}"
if options:
assert not isinstance(options, str)
vol += ":" + ",".join(options)
Expand Down
2 changes: 1 addition & 1 deletion scuba/scuba.py
Original file line number Diff line number Diff line change
Expand Up @@ -165,7 +165,7 @@ def try_create_volumes(self) -> None:
return

for vol in self.context.volumes.values():
if vol.host_path.exists():
if vol.host_path is None or vol.host_path.exists():
continue

try:
Expand Down
Loading

0 comments on commit 7a20042

Please sign in to comment.