I always get tripped up by Docker’s different mount types and their syntax, whether I’m
stringing together some CLI commands or writing a docker-compose
file. Docker’s docs cover
these, but for me, the confusion often comes from how “bind” is used in various contexts and
how “volume” and “bind” sometimes get mixed up in the documentation.
Here’s my attempt to disentangle some of my most-used mount commands.
Volume mounts
Volume mounts1 let you store data outside the container in a location managed by Docker. The data persists even after the container stops. On non-Linux systems, volume mounts are faster than bind mounts because data doesn’t need to cross the virtualization boundary.
The -v
option
The -v
flag is the older and more common way to define volume mounts in the Docker CLI.
For example:
docker run -v myvolume:/usr/share/nginx/html:ro nginx
Here’s what each part means:
myvolume
: The name of the Docker-managed volume on the host./usr/share/nginx/html
: The mount point inside the container.:ro
: Mounts the volume as read-only inside the container. The host can still write to the volume, but the container cannot.
The general syntax is:
-v [SOURCE]:[TARGET]:[OPTIONS]
It can be tricky to remember which part is the host and which is the container, especially
since with volumes, the SOURCE
is a volume name, not a host path.
The --mount
option
To make things clearer, Docker introduced the --mount
option, which uses key-value pairs.
The same volume mount using --mount
looks like this:
docker run \
--mount \
type=volume,source=myvolume,target=/usr/share/nginx/html,readonly nginx
Or using shorthands:
docker run --mount type=volume,src=myvolume,dst=/usr/share/nginx/html,ro nginx
I find this syntax more explicit and less error-prone, even if it’s a bit more verbose.
In docker-compose.yml
In docker-compose
, volumes can be defined using both the old and new syntax. Here’s how
they compare:
Old style
services:
app:
image: nginx
volumes:
- myvolume:/usr/share/nginx/html:ro
volumes:
myvolume:
New style
services:
app:
image: nginx
volumes:
- type: volume
source: myvolume
target: /usr/share/nginx/html
read_only: true
volumes:
myvolume:
I prefer the new style because it reduces ambiguity and makes the configuration clearer.
Bind mounts
Bind mounts2 let you directly mount a file or directory from the host into the container. This is especially useful in development when you want the container to have access to your code or data.
The key difference between volume mounts and bind mounts is that volumes are fully managed by Docker and stored in a special location, while bind mounts rely on specific host paths. Volumes are more portable and isolated from the host, whereas bind mounts give you direct access to host files but can introduce permission issues and depend on the exact file structure of the host.
The -v
option
Using the -v
syntax for bind mounts:
docker run -v /path/on/host:/usr/share/nginx/html:ro nginx
Here:
/path/on/host
: The path on the host machine. This must be an absolute path./usr/share/nginx/html
: The mount point inside the container.:ro
: Mounts the directory as read-only inside the container.
The --mount
option
Using --mount
for a bind mount:
docker run \
--mount \
type=bind,source=/path/on/host,target=/usr/share/nginx/html,readonly nginx
This syntax makes it clear that you’re using a bind mount and specifies exactly which paths are involved.
In docker-compose.yml
In docker-compose
, bind mounts can be specified like this:
Old style
services:
app:
image: nginx
volumes:
- ./path/on/host:/usr/share/nginx/html:ro
New style
services:
app:
image: nginx
volumes:
- type: bind
source: ./path/on/host
target: /usr/share/nginx/html
read_only: true
Note: In
docker-compose
, if you specify asource
that doesn’t start with/
(an absolute path) or./
(a relative path), Docker might think you’re referring to a volume. To ensure it’s interpreted as a bind mount, start the path with./
or/
.
Tmpfs mounts
Tmpfs mounts3 store data in the host’s memory, not on disk. This makes them ideal for temporary storage that doesn’t need to persist after the container stops. They’re great for things like caches or scratch space.
The --tmpfs
option
Docker provides a --tmpfs
option to create a tmpfs mount more concisely:
docker run --tmpfs /app/tmp:rw,size=64m nginx
/app/tmp
: The target directory inside the container.rw
: This option allows read and write access to the tmpfs mount.size=64m
: Sets the size of the tmpfs mount to 64 MB.
The --mount
option
Alternatively, using the more flexible --mount
option:
docker run \
--mount type=tmpfs,target=/app/tmp,tmpfs-size=64m,tmpfs-mode=1777 nginx
Here’s what each part means:
type=tmpfs
: Specifies that this is a tmpfs mount, using the host’s memory.target=/app/tmp
: The directory inside the container where the tmpfs mount is mounted.tmpfs-size=64m
: Limits the size of the tmpfs mount to 64 MB.tmpfs-mode=1777
: Sets permissions for the tmpfs mount (1777 grants read, write, and execute permissions to everyone).
In docker-compose.yml
In docker-compose
, tmpfs mounts can be defined using both the old and new syntax.
Old style
services:
app:
image: nginx
tmpfs:
- /app/tmp:size=64m
New style
services:
app:
image: nginx
volumes:
- type: tmpfs
target: /app/tmp
tmpfs:
size: 64m
mode: 1777
Build cache mounts
Build cache mounts4 help speed up Docker image builds by caching intermediate files like package downloads or compiled artifacts. They’re used during the build process and aren’t part of the final container image.
In a Dockerfile
, you might use a build cache like this:
RUN --mount=type=cache,target=/var/cache/apt \
apt-get update && apt-get install -y curl
Here’s what each option does:
--mount=type=cache
: Defines a cache mount that stores the files from theapt-get
commands to speed up future builds by reusing the downloaded packages.target=/var/cache/apt
: Specifies the location inside the container where the cache will be stored during the build process.
In my Python projects, I cache my dependencies and install them with uv
like this:
RUN --mount=type=cache,target=/root/.cache/uv \
--mount=type=bind,source=uv.lock,target=uv.lock \
--mount=type=bind,source=pyproject.toml,target=pyproject.toml \
uv sync --no-install-project --locked --no-dev
Above, the cache mount at /root/.cache/uv
saves dependencies so they don’t need to be
re-downloaded in future builds if nothing changes. The bind mounts provide access to
uv.lock
and pyproject.toml
from the host, allowing the container to read these config
files during the build. Any changes to the files are picked up, while cached dependencies
are reused unless the configurations have been updated.
Recent posts
- The domain knowledge dilemma
- Hierarchical rate limiting with Redis sorted sets
- Dynamic shell variables
- Link blog in a static site
- Running only a single instance of a process
- Function types and single-method interfaces in Go
- SSH saga
- Injecting Pytest fixtures without cluttering test signatures
- Explicit method overriding with @typing.override
- Quicker startup with module-level __getattr__