Docker is open source software for creating and managing con­tain­ers. With a Docker volume, you can transfer data between con­tain­ers or back up data from a Docker container.

Re­quire­ments

The Docker file system

In order to un­der­stand Docker volumes, it is important to first un­der­stand how the Docker file system works.

A Docker image is a col­lec­tion of read-only layers. When you launch a container from an image, Docker adds a read-write layer to the top of that stack of read-only layers. Docker calls this the Union File System. Any time a file is changed, Docker makes a copy of the file from the read-only layers up into the top read-write layer. This leaves the original (read-only) file unchanged. When a container is deleted, that top read-write layer is lost. This means that any changes made after the container was launched are now gone.

Dedicated Servers
Per­for­mance through in­no­va­tion
  • Dedicated en­ter­prise hardware
  • Con­fig­urable hardware equipment
  • ISO-certified data centers

How a Docker volume can help

A Docker volume allows data to persist, even when a container is deleted. Volumes are also a con­ve­nient way to share data between the host and the container.

Mounting a Docker volume is a good solution if you want to:

  • Push data to a Docker container.
  • Pull data from a Docker container.
  • Share data between Docker con­tain­ers.

Docker volumes exist outside the Union File System of read-only and read-write layers. The volume is a folder which is shared between the container and the host machine. Volumes can also be shared between con­tain­ers.

Tip

On the IONOS Cloud Server you can choose Docker as a pre-installed cloud app. That way you’ll have access to your ap­pli­ca­tions from anywhere.

The basics of Docker volumes

A Docker volume “lives” outside the container, on the host machine. From the container, the volume acts like a folder which you can use to store and retrieve data. It is simply a mount point to a directory on the host. There are several ways to create and manage Docker volumes. Each method has its own ad­van­tages and dis­ad­van­tages.

Using Docker’s “volume create” command

As of version 1.9.0, which was released 11/3/2015, Docker volumes can now be created and managed using the in­te­grat­ed docker volume command.

The docker volume create command will create a named volume. The name allows you to easily locate and assign Docker volumes to con­tain­ers.

Step 1: Create and name a volume

To create a Docker volume, use the command:

sudo docker volume create --name [volume name]

Step 2: Use volume in Docker container

To launch a container which will use a volume that you have created with docker volume, add the following argument to the docker run command:

-v [volume name]:[container directory]

For example, to run a container from the CentOS image named my-volume-test and map the volume data-volume to the container’s /data directory, the command is:

sudo docker run -it --name my-volume-test -v data-volume:/data centos /bin/bash

Step 3: List volumes

To list all Docker volumes on the system, use the command:

sudo docker volume ls

This will return a list of all of the Docker volumes which have been created on the host.

Step 4: Inspect a volume

To inspect a named volume, use the command:

sudo docker volume inspect [volume name]

This will return in­for­ma­tion about the volume, including its mount point (the directory where it “lives”) on the host system.

For example, to get more in­for­ma­tion about data-volume which we created above, the command is:

sudo docker volume inspect data-volume

Step 5: Remove a volume

To remove a named volume, use the command:

sudo docker volume rm [volume name]

You will not be able to remove a volume if it is being used by an existing container. Before removing the volume, you will need to stop and delete the container with the commands:

sudo docker stop [container name or ID]
sudo docker rm [container name or ID]

For example, to delete the volume data-volume, we will first need to delete the container my-volume-test, which is using it:

sudo docker stop my-volume-test
sudo docker rm my-volume-test

The volume data-volume can then be deleted with:

sudo docker volume rm data-volume

Create a Docker volume and specify a host directory

If you want to mount a specific directory on your host machine as a Docker volume on the container, add the following argument to your docker run command:

-v [host directory]:[container directory]

For example, to launch a new container and map the /webfiles folder from the host into the /var/www/html folder in the container, the command is:

sudo docker run -it -v /webfiles:/var/www/html centos /bin/bash

You can test this by first creating a directory to use as a Docker volume with the command:

sudo mkdir /hostvolume

Add a small test file to this directory with the command:

sudo echo "Hello World" >> /hostvolume/host-hello.txt

Next, launch a container named my-directory-test and map /hostvol­ume on the host to /con­tain­er­vol­ume on the container with the command:

sudo docker run -it --name my-directory-test -v /hostvolume:/containervolume centos /bin/bash

Once you are at the new con­tain­er's command prompt, list the files in the shared volume with the command:

ls /containervolume

You will see the host-hello.txt file which we created on the host.

This works in the opposite direction, as well. Files you put into this directory will appear on the host. You can test this from the container by adding another file to the shared volume with the command:

echo "Hello from the container." >> /containervolume/container-hello.txt

Detach from the container with [Ctrl] + [P] and [Ctrl] + [Q] and return to the host machine’s command prompt. Once there, list the files in the shared volume on with the command:

sudo ls /hostvolume

You will see the two test files we created from the host and from the container.

Create a Docker volume using a Dock­er­file

Use the following command in a Dock­er­file to create a shared storage volume in the container:

VOLUME [volume path]

For example, to create a volume /myvolume in the container to be launched from the Dock­er­file, the command is:

VOLUME /myvolume

To test this, begin by creating a file called Dock­er­file with the command:

sudo nano Dockerfile

Put the following content into this file:

# The source image to start with
FROM centos
# Create a volume
VOLUME /dockerfilevolume

Save and exit the file.

Next, build an image named dock­er­file-vol­ume­test from this Dock­er­file with the command:

sudo docker build -t dockerfile-volumetest .

Then launch a container named my-dock­er­file-test from this image with the command:

sudo docker run --name my-dockerfile-test -it dockerfile-volumetest /bin/bash

Once you are at the new con­tain­er's command prompt, create a small test file in the shared volume with the command:

echo "Hello World" >> /dockerfilevolume/dockerfile-container-hello.txt

Detach from the container with [Ctrl] + [P] and [Ctrl] + [Q] and return to the host machine’s command prompt.

Next, let’s find the mount point. To do this, use the command:

sudo docker inspect my-dockerfile-test

Scroll through the output until you find a section titled "Mounts" which will look something like this:

  • Source is the directory on the host machine.
  • Des­ti­na­tion is the folder on the container.

Check the source directory on your host machine. In this example, the command is:

sudo ls  /var/lib/docker/volumes/30275034a424251a771c91b65ba44261a27f91e3f6af31097b5226b1f46bfe20/_data/test

Here you will find the dock­er­file-container-hello.txt file which you created on the container.

Overview of the ad­van­tages and dis­ad­van­tages of the various methods

Method Ad­van­tages Dis­ad­van­tages
Command “volume create” Quick and easy to use Volume on the host is created au­to­mat­i­cal­ly by Docker, and can be difficult to locate and use
Create Docker volume with directory on the host Allows you to map a specific host folder to a container Cannot create a named volume as with docker volume create. Cannot be automated with a Dock­er­file
Create with Dock­er­file Allows you to automate the process Cannot create a named volume. Cannot specify a directory on the host

Sharing Docker volumes between con­tain­ers

There are many sit­u­a­tions where it is useful to share a Docker volume between con­tain­ers, and several ways to ac­com­plish this goal.

Sharing a Volume on the Host

If you create a volume on the host machine, it can be used by multiple different con­tain­ers at once. This allows you to share data between con­tain­ers and the host. For this example we will create a directory on the host, and use that directory as a shared volume between two con­tain­ers.

Begin by creating a directory to use as a Docker volume with the command:

sudo mkdir /webdata

Create a small test file in this directory with the command:

sudo echo "Hello from the host." >> /webdata/host-hello.txt

Next, launch a container named sql-database from the official Post­greSQL image, and map /webdata on the host to /data on the container with the command:

sudo docker run -it --name sql-database -v /webdata:/data postgres /bin/bash

Once you are at the new container’s command prompt, verify that the shared volume is set up correctly with the command:

ls /data

You will see the host-hello.txt file which we created on the host. Let's add a file to this shared volume with the command:

echo "Hello from the sql-database container." >> /data/sql-hello.txt

Detach from the container with [Ctrl] + [P] and [Ctrl] + [Q] and return to the host machine’s command prompt.

Now launch a container named webapp from the official PHP+Apache image, and map /webdata on the host to /var/www/html on the container.

sudo docker run -it --name webapp -v /webdata:/var/www/html php:5.6-apache /bin/bash

Once you are at the new con­tain­er's command prompt, verify that the shared volume is set up correctly with the command:

ls /var/www/html

You will see both the host-hello.txt file which we created on the host, and the sql-hello.txt file we created on the sql-database container.

Let's add a file from this container as well:

echo "Hello from the webapp container." >> /var/www/html/webapp-hello.txt

Detach from the container with [Ctrl] + [P] and [Ctrl] + [Q] and return to the host machine’s command prompt. Once on the host machine, you will see all three files listed with the command:

sudo ls /webdata

Now that the two con­tain­ers are sharing a directory which “lives” on the host, data can be trans­ferred instantly between all three locations simply by moving it to that directory.

Using a container as a shared data volume

You can also set up a separate container as a shared data volume. To do this, first create the data container. Then, when you create the container that will be using that data container, add the following argument to the docker run command:

--volumes-from [name or ID of data container]
Note

This will work whether or not the target container is running. Docker volumes are never deleted, and persist even after the container has been stopped.

For this example, we will create a data container called data-storage which will serve as the data volume, and two other con­tain­ers that share it as a storage volume.

First, launch the data-storage container from the official CentOS 7 image:

sudo docker run -it -v /shared-data --name data-storage centos /bin/bash

Then add a small file to the /shared-data folder:

echo "Hello from the data-storage container." >> /shared-data/data-storage-hello.txt

Detach from the container with [Ctrl] + [P] and [Ctrl] + [Q] and return to the host machine’s command prompt.

Now launch the app container from the official Python image and mount the data-storage container as a volume:

sudo docker run -it --name app --volumes-from data-storage python /bin/bash

List the files in the shared volume with the command:

ls /shared-data

As you can see, the /shared-data folder has been mounted from the /shared-data folder on the data-storage container, and contains the data-storage-hello.txt file.

Let’s add one from this container:

echo "Hello from the app container." >> /shared-data/app-hello.txt

Detach from the container with [Ctrl] + [P] and [Ctrl] + [Q] and return to the host machine’s command prompt.

Finally, launch the web container from the official Apache image and mount the data-storage container as a volume:

sudo docker run -it --name web --volumes-from data-storage httpd /bin/bash

List the files in the shared volume with the command:

ls /shared-data

You will see the files we created on the data-storage and app con­tain­ers listed here.

Dedicated Servers
Per­for­mance through in­no­va­tion
  • Dedicated en­ter­prise hardware
  • Con­fig­urable hardware equipment
  • ISO-certified data centers

Mounting a volume as read-only

Through­out this guide we have been mounting volumes with the default read-write access. If you want to restrict a container to having read-only access to a volume, simply add :ro to the container volume specified in the -v statement:

docker run -v /directory:/path:ro

This can be useful for security purposes, or when you want to ensure that the data in a par­tic­u­lar volume is safe­guard­ed against being ac­ci­den­tal­ly over­writ­ten or deleted by another container.

For example, begin by creating a volume on the host named limited-access with the command:

sudo docker volume create --name limited-access

Then run a container from the CentOS image named allowed-to-write and map the volume limited-access as a normal (read-write) volume:

sudo docker run -it --name allowed-to-write -v limited-access:/data centos /bin/bash

Once you are at this con­tain­er's command prompt, create a test file with the command:

echo "Hello from the container that is allowed to write." >> /data/hello.txt

Detach from the container with [Ctrl] + [P] and [Ctrl] + [Q] and return to the host machine’s command prompt.

Next, run a container from the CentOS image named not-allowed-to-write and map the volume limited-access as a read-only volume:

sudo docker run -it --name not-allowed-to-write -v limited-access:/data:ro centos /bin/bash

If you attempt to create a test file in the shared volume with a command such as this

echo "Hello from the container that is not allowed to write." >> /data/no-access.txt

you will receive an error which explains that this container does not have write access to that directory:

bash: /data/no-access.txt: Read-only file system
Go to Main Menu