Lab 10: An Introduction to Containers

Introduction

In this lab you will perform the following tasks:

  • Install Docker

  • Container Basics with BusyBox

  • Practical Docker: A Simple Webserver

  • Practical Docker: A Webapp with Multiple Containers and Docker Compose

You will be introduced to the following commands:

  • docker

  • docker-compose

Preliminaries

  1. Open an SSH remote terminal session to your Linux server’s IP address

    1. Connect to ITCnet from the computer you will be using as your administrative PC. In most cases this means connecting to the ITC Student VPN (unless you are using the Netlab Windows Administrative PC).

    2. Run the PuTTY software on your computer (or the Windows Administrative PC) and enter in the IP address of your Linux server VM in the "Host Name" box and click the "Open" button.

      Remember that if you do not have a Windows computer to connect from you can either figure out how to SSH from your own computer over the VPN to your Linux server or you can use the Windows Administrative PC that is provided for you in Netlab.
  2. Login with your standard user’s username and password

Install Docker

This lab is based off the excellent Docker Curriculum by Prakhar Srivastav though it is designed to work specifically on the virtual machines we have built thus far in this class and to be a bit more narrowed in focus. When additional explanation would be helpful it’s recommended you refer back to the Docker Curriculum site.
  1. So far in these labs we have been working in a virtual machine environment where your Linux server is completely isolated from other virtual systems running on the same physical hardware. Virtual machines remain a powerful tool used by IT departments; however, there is another technology called containers which provide most of the isolation that virtualization does while sharing some underlying parts of the operating system which saves a lot of computing power.

  2. These containers allow us to run multiple applications (such as one or more web applications, database servers, email systems, etc.) on a single physical or virtual system. They also allow us to quickly push updated versions of the applications out which include all of the dependencies the application requires (and which may conflict with dependencies of other applications) bundled into the container. Probably the most popular architecture for building and running containers today is Docker which we’ll begin to explore in this lab activity.

  3. We will begin by installing Docker which requires that the Docker APT repository be setup on your virtual machine. You should have already set this up in Lab 8 when you were working on adding APT repositories.

  4. Use apt to install this list of packages: docker-ce docker-ce-cli containerd.io docker-buildx-plugin docker-compose-plugin which you should see mostly downloading from the https://download.docker.com site if you have properly configured the Docker APT repository.

  5. Test your installation by running the command docker run hello-world as an administrative user:

    ben@2480-Z:~$ sudo docker run hello-world
    Unable to find image 'hello-world:latest' locally
    latest: Pulling from library/hello-world
    c1ec31eb5944: Pull complete
    Digest: sha256:53cc4d415d839c98be39331c948609b659ed725170ad2ca8eb36951288f81b75
    Status: Downloaded newer image for hello-world:latest
    
    Hello from Docker!
    This message shows that your installation appears to be working correctly.
    
    To generate this message, Docker took the following steps:
     1. The Docker client contacted the Docker daemon.
     2. The Docker daemon pulled the "hello-world" image from the Docker Hub.
        (amd64)
     3. The Docker daemon created a new container from that image which runs the
        executable that produces the output you are currently reading.
     4. The Docker daemon streamed that output to the Docker client, which sent it
        to your terminal.
    
    To try something more ambitious, you can run an Ubuntu container with:
     $ docker run -it ubuntu bash
    
    Share images, automate workflows, and more with a free Docker ID:
     https://hub.docker.com/
    
    For more examples and ideas, visit:
     https://docs.docker.com/get-started/
    
    ben@2480-Z:~$

Container Basics with BusyBox

  1. As opposed to a general purpose installation of an operating system we usually want to keep containers as lightweight as possible meaning they should be small and resource efficient so they can be quickly started and do what they were designed to do. Containers themselves are what run on our system but they are made up of one or more Docker images which contain the application and system executable files. One example of a very basic lightweight image is the BusyBox image which contains many of the common *NIX command line utilities in one tiny executable.

  2. One of the things that makes Docker popular is that lots of people have already created Docker images that are ready to use or can be used as a basis to build upon and create your own custom Docker image. These images are often pubished to Docker Hub which is a registry of images which you can pull (download) to your own system and use. Run the docker pull busybox command as administrator to download the BusyBox image from the Docker Hub.

  3. You can view a list of images which are loaded on your server using the docker images command:

    ben@2480-Z:~$ sudo docker pull busybox
    Using default tag: latest
    latest: Pulling from library/busybox
    ec562eabd705: Pull complete
    Digest: sha256:9ae97d36d26566ff84e8893c64a6dc4fe8ca6d1144bf5b87b2b85a32def253c7
    Status: Downloaded newer image for busybox:latest
    docker.io/library/busybox:latest
    ben@2480-Z:~$ sudo docker images
    REPOSITORY    TAG       IMAGE ID       CREATED         SIZE
    busybox       latest    65ad0d468eb1   15 months ago   4.26MB
    hello-world   latest    d2c94e258dcb   15 months ago   13.3kB
    ben@2480-Z:~$
  4. To run a Docker container based on this image use the docker run busybox command as an administrator:

    ben@2480-Z:~$ sudo docker run busybox
    ben@2480-Z:~$
  5. While it may seem like nothing happened that is far from the truth! In fact, Docker created a container based on the busybox image and then started up that container; however, because we didn’t give any further command to be run inside the container it just immediately exited the the container was shutdown. Can you imagine a virtual machine even just starting up and shutting down that quickly? That is one of the advantages of containers being so lightweight, they can be started and stopped very quickly as different needs arise.

  6. Let’s run our container again but tell it to run the ls -al /home command by using docker run busybox ls -al /home and compare that with what the ls -al /home command looks like on our server system itself:

    ben@2480-Z:~$ sudo docker run busybox ls -al /home
    total 8
    drwxr-xr-x    2 nobody   nobody        4096 May 18  2023 .
    drwxr-xr-x    1 root     root          4096 Aug 19 15:40 ..
    ben@2480-Z:~$ ls -al /home
    total 16
    drwxr-xr-x  4 root   root   4096 Mar 26 15:09 .
    drwxr-xr-x 18 root   root   4096 Aug  8 17:23 ..
    drwx------  9 ben    ben    4096 Aug 17 14:40 ben
    drwx---r-x  3 jsmith jsmith 4096 Apr  2 17:02 jsmith
    ben@2480-Z:~$
  7. You can see that inside the container the /home location is empty but on our server system it has a directory for both of the regular users we have created. Again, compare how quickly the docker run busybox ls -al /home command ran with how long it would take to start up a virtual machine, log in, run that ls command, and then shut down!

  8. While some Docker containers start and stop quickly like this others stay running in the background for a long time (or even all the time) so it’s useful to have a command to check and see what containers are currently running. The command that let’s you see any currently running containers is docker ps, try it now:

    ben@2480-Z:~$ sudo docker ps
    CONTAINER ID   IMAGE     COMMAND   CREATED   STATUS    PORTS     NAMES
    ben@2480-Z:~$
  9. Because all of our containers have actually stopped running you’ll see that the list here is empty. It’s also possible to see a list of previously run containers which still exist on the system with the docker ps -a command which you should also try:

    ben@2480-Z:~$ sudo docker ps -a
    CONTAINER ID   IMAGE         COMMAND            CREATED          STATUS                      PORTS     NAMES
    3d7ff06de75d   busybox       "ls -al /home"     2 minutes ago    Exited (0) 2 minutes ago              thirsty_cannon
    2d55b8b8faa1   busybox       "sh"               2 minutes ago    Exited (0) 2 minutes ago              practical_curran
    a3232151f24e   hello-world   "/hello"           2 minutes ago    Exited (0) 2 minutes ago              hopeful_poincare
    ben@2480-Z:~$
  10. So far we have just been running a single command inside the Docker container and that’s actually a pretty normal thing to do. The idea with containers is to just have them do a single thing (such as run a web server or database server) so they often only need the command to start that thing. However, it is possible to connect to a conatiner in an interactive way so that you get a prompt inside of the container just like you would connecting to a virtual machine. To do this try running the command docker run -it busybox sh which includes the -it option which means to connect it to an interactive console and the sh command which is the executable name for a simple command-line shell inside of busybox (busybox is so small it doesn’t include the full BASH shell).

  11. You should see your prompt change (and indicate you are running as the root user) and can now do things inside the container. Try a few commands like ls and cd to navigate around inside the container and uptime to see how long the container has been running for. Note that because this is a very basic container with only busybox the basic commands will be there but not more advanced commands like apt or lsb_release.

  12. When you are done working in the interactive container you can run the exit command to return back to your Linux server, you should see the prompt change back.

  13. Every time we tell Docker to run it creates a new container from the original image. Even after these containers exit and are stopped the copy of the container sticks around on the system (as you can see in the docker ps -a command). Eventually these start to take up an apprciable amount of space on the system so it makes sense to delete old stopped containers. You can do this by running docker ps -a and getting the container IDs you want to delete and then running docker rm <containerIDs> where you can separate the list of container IDs with spaces. Try deleting a container or two this way:

    ben@2480-Z:~$ sudo docker ps -a
    CONTAINER ID   IMAGE         COMMAND            CREATED          STATUS                      PORTS     NAMES
    8009761decd7   busybox       "sh"               7 minutes ago    Exited (2) 3 minutes ago              serene_buck
    3d7ff06de75d   busybox       "ls -al /home"     16 minutes ago   Exited (0) 16 minutes ago             thirsty_cannon
    2d55b8b8faa1   busybox       "sh"               16 minutes ago   Exited (0) 16 minutes ago             practical_curran
    a3232151f24e   hello-world   "/hello"           16 minutes ago   Exited (0) 16 minutes ago             hopeful_poincare
    b47283d7f7da   busybox       "ls -al /home"     21 minutes ago   Exited (0) 21 minutes ago             magical_chatelet
    82ad967899f0   busybox       "ls"               22 minutes ago   Exited (0) 22 minutes ago             stupefied_pike
    18220ee96832   busybox       "lsb_release"      22 minutes ago   Created                               goofy_morse
    ea7ff9b99903   busybox       "lsb_release -a"   22 minutes ago   Created                               pensive_pare
    553ced9cfb5a   busybox       "sh"               28 minutes ago   Exited (0) 28 minutes ago             serene_beaver
    1c245c444c8e   hello-world   "/hello"           44 hours ago     Exited (0) 44 hours ago               loving_dubinsky
    ben@2480-Z:~$ sudo docker rm 1c245c444c8e 553ced9cfb5a ea7ff9b99903
    1c245c444c8e
    553ced9cfb5a
    ea7ff9b99903
    ben@2480-Z:~$ sudo docker ps -a
    CONTAINER ID   IMAGE         COMMAND          CREATED          STATUS                      PORTS     NAMES
    8009761decd7   busybox       "sh"             7 minutes ago    Exited (2) 4 minutes ago              serene_buck
    3d7ff06de75d   busybox       "ls -al /home"   16 minutes ago   Exited (0) 16 minutes ago             thirsty_cannon
    2d55b8b8faa1   busybox       "sh"             17 minutes ago   Exited (0) 17 minutes ago             practical_curran
    a3232151f24e   hello-world   "/hello"         17 minutes ago   Exited (0) 17 minutes ago             hopeful_poincare
    b47283d7f7da   busybox       "ls -al /home"   22 minutes ago   Exited (0) 22 minutes ago             magical_chatelet
    82ad967899f0   busybox       "ls"             22 minutes ago   Exited (0) 22 minutes ago             stupefied_pike
    18220ee96832   busybox       "lsb_release"    22 minutes ago   Created                               goofy_morse
    ben@2480-Z:~$
  14. However, in newer versions of Docker they added a container prune option which will automatically remove all stopped contianers. Try running it now like docker container prune and remove all your stopped containers:

    ben@2480-Z:~$ sudo docker container prune
    WARNING! This will remove all stopped containers.
    Are you sure you want to continue? [y/N] y
    Deleted Containers:
    8009761decd7af289f723e1f50ea28ccdecb50f25d667a8d22e99f05c5759097
    3d7ff06de75daae7b0283ca7a63bee0324b65087a304075dd003c02e80761043
    2d55b8b8faa1e360cbdca2a6b1937cc60a682712832fa0673560d579928acb51
    a3232151f24e5c5211339fb3973c3e46b088f8f221ca2448c93c39c6d2b17f1e
    b47283d7f7da362663b66033a1533406e430305c7da6149497bf16a372c47020
    82ad967899f0b0add39329d3ead9bf830da31bc0c51649339e5f5a5d9ead9174
    18220ee968329325558b2effc6d09447818aaaeb69028de8ddbc91537e32422c
    
    Total reclaimed space: 14B
    ben@2480-Z:~$ sudo docker ps -a
    CONTAINER ID   IMAGE     COMMAND   CREATED   STATUS    PORTS     NAMES
    ben@2480-Z:~$
  15. You can also remove and prune Docker images which do not have currently running containers in the same way with the docker image rm <imageID> command or docker image prune -a to remove all images which are currently unused. Get a list of the images on your system and then try removing one and pruning the rest:

    ben@2480-Z:~$ sudo docker image ls
    REPOSITORY    TAG       IMAGE ID       CREATED         SIZE
    busybox       latest    65ad0d468eb1   15 months ago   4.26MB
    hello-world   latest    d2c94e258dcb   15 months ago   13.3kB
    ben@2480-Z:~$ sudo docker image rm d2c94e258dcb
    Untagged: hello-world:latest
    Untagged: hello-world@sha256:53cc4d415d839c98be39331c948609b659ed725170ad2ca8eb36951288f81b75
    Deleted: sha256:d2c94e258dcb3c5ac2798d32e1249e42ef01cba4841c2234249495f87264ac5a
    Deleted: sha256:ac28800ec8bb38d5c35b49d45a6ac4777544941199075dff8c4eb63e093aa81e
    ben@2480-Z:~$ sudo docker image ls
    REPOSITORY   TAG       IMAGE ID       CREATED         SIZE
    busybox      latest    65ad0d468eb1   15 months ago   4.26MB
    ben@2480-Z:~$ sudo docker image prune -a
    WARNING! This will remove all images without at least one container associated to them.
    Are you sure you want to continue? [y/N] y
    Deleted Images:
    untagged: busybox:latest
    untagged: busybox@sha256:9ae97d36d26566ff84e8893c64a6dc4fe8ca6d1144bf5b87b2b85a32def253c7
    deleted: sha256:65ad0d468eb1c558bf7f4e64e790f586e9eda649ee9f130cd0e835b292bbc5ac
    deleted: sha256:d51af96cf93e225825efd484ea457f867cb2b967f7415b9a3b7e65a2f803838a
    
    Total reclaimed space: 4.262MB
    ben@2480-Z:~$ sudo docker image ls
    REPOSITORY   TAG       IMAGE ID   CREATED   SIZE
    ben@2480-Z:~$

Practical Docker: A Simple Webserver

  1. Now that we have a basic understanding of how Docker works let’s try a simple practical example, running an Nginx webserver in Docker. One advantage of doing this in a Docker container is that you could more easily run a newer version of something like Nginx than what is packaged with your Linux distribution; however, this would make you responsible for doing upgrades/security updates outside of the package management system of your distribution as well. Because we already have a webserver on our Linux server running on port 80 we’ll run this new webserver on port 8888.

  2. Many popular open source server applications today provide official Docker images for just this purpose and Nginx is no different. If you search hub.docker.com for Nginx you’ll find this official image.

    In addition to official Docker images there are often times unofficial images which have been created by users. You should think carefully about how much you trust they person who created these unofficial images before using them as the image creator could have included just about any code they wanted including deliberately insecure or malicious code!
  3. We’ll use the command docker run --rm -d -p 8888:80 --name webcontainer nginx to run the server but don’t do it just yet, let’s explain what’s going on with this command first.

    1. Because we don’t already have the nginx image the first thing to note is that Docker will automatically try and download the image from the Docker Hub.

    2. There are also a few new options to docker run to talk about. First the --rm option tells Docker to automatically delete the container after it exits so we don’t have to worry about doing a docker rm or docker container prune later on.

    3. Next, the -d option tells Docker we want to run the container in detatched mode so that it won’t take over our console, will run in the background, and will keep running even if we close our SSH session.

    4. The -p 8888:80 option is our introduction to networking with Docker. This option says that we want to forward port 8888 on our Linux server (the Docker host) to port 80 in the container which is what the Nginx configuration in the container is set to use as it’s default port. Without this option we would not have access to the port the webserver is running on from outside the container.

      There are much more advanced ways to do Docker networking too including dedicating an IP address for the container and just passing all ports on that IP to the container or creating private Docker networks where multiple containers can talk amongst themselves (such as a database container and a web server container) but have no access from the outside.
    5. Lastly the --name webcontainer option gives a name to the container we can use to issue commands to Docker without needing to use the container ID all the time, this just makes things a bit more user friendly.

  4. Now that you have an idea what each part of the command will do try running the command docker run --rm -d -p 8888:80 --name webcontainer nginx and then run the docker ps command to make sure the container is running:

    ben@2480-Z:~$ sudo docker run --rm -d -p 8888:80 --name webcontainer nginx
    Unable to find image 'nginx:latest' locally
    latest: Pulling from library/nginx
    e4fff0779e6d: Pull complete
    2a0cb278fd9f: Pull complete
    7045d6c32ae2: Pull complete
    03de31afb035: Pull complete
    0f17be8dcff2: Pull complete
    14b7e5e8f394: Pull complete
    23fa5a7b99a6: Pull complete
    Digest: sha256:447a8665cc1dab95b1ca778e162215839ccbb9189104c79d7ec3a81e14577add
    Status: Downloaded newer image for nginx:latest
    33d17f1de3048110f5e2b4947e2e96561fcdab37e050d734f6e49e8e58108547
    ben@2480-Z:~$ sudo docker ps
    CONTAINER ID   IMAGE     COMMAND                  CREATED          STATUS          PORTS                                   NAMES
    33d17f1de304   nginx     "/docker-entrypoint.…"   13 seconds ago   Up 12 seconds   0.0.0.0:8888->80/tcp, :::8888->80/tcp   webcontainer
    ben@2480-Z:~$
  5. Since the container is running we should be able to just access the server now by browsing to http://z.itc2480.campus.ihitc.net:8888 (with your Pod ID letter replacing z) from our management computer.

    You may recall that we have a firewall setup on our system and are wondering why we don’t need to add a rule allowing port 8888 through that firewall. As it turns out Docker, in an effort to make things simple to use, will automatically install rules directly in nftables allowing the traffic. This may seem convenient but now we have a hole in our security which won’t show up on a firewall-cmd --zone=external --list-all list becuase it is happening outside of firewalld. This is not a very good idea in production. There are a number of ways to solve this including disabling Docker from modifying nftables and manually creating rules in firewalld for our containers or using another runtime to manage the Docker containers like Podman or other ways. Each of these has advantages and disadvantges which should be weighed carefully in a production environment but are outside the scope of this Docker introduction!
  6. You should now stop the container with the docker stop webcontainer command because we named our container webcontainer. You should also notice that the container is automatically removed because we launched it with the --rm option:

    ben@2480-Z:~$ sudo docker stop webcontainer
    webcontainer
    ben@2480-Z:~$ sudo docker ps
    CONTAINER ID   IMAGE     COMMAND   CREATED   STATUS    PORTS     NAMES
    ben@2480-Z:~$ sudo docker ps -a
    CONTAINER ID   IMAGE     COMMAND   CREATED   STATUS    PORTS     NAMES
    ben@2480-Z:~$
  7. So far so good, but how can we change the contents of the web site which is currently in the /usr/share/nginx/html location inside the container? There are a few ways to do this but the first one we’ll explore is to use bind mounts which allow us to connect a location on our Docker host system onto a location inside a container.

    1. First, create a new directory on your system in your user’s home directory like ~/docker-web and another new directory inside that one named content.

    2. Create a file inside the content directory named index.html with this inside it:

      <!doctype html>
      <html lang="en">
      <head>
        <meta charset="utf-8">
        <title>Docker Nginx</title>
      </head>
      <body>
        <h2>Hello from my ITC-2480 Nginx Docker container!</h2>
      </body>
      </html>
    3. We’re now going to run the command docker run --rm -d -p 8888:80 --name webcontainer -v ~/docker-web/content:/usr/share/nginx/html nginx. You’ll notice this command has one extra option compared with what we ran previously. The -v ~/docker-web/content:/usr/share/nginx/html option tells Docker to map the host directory ~/docker-web/content to the /usr/share/nginx/html directory inside the container.

    4. Check on your administrative PC that the contents of the website are now updated to reflect it is running in Docker. You may need to manually refresh the page in the browser of your administrative PC to see the latest content.

    5. It’s also possible to map your own custom Nginx configuration file over the default file in much the same way if you want to customize the configuration.

  8. There are many other ways to get data into and out of a Docker container including building your own custom image based on another image and useing Docker volumes.

  9. You should now stop the container with the docker stop webcontainer command to clean things up for the next section.

Practical Docker: A Webapp with Multiple Containers and Docker Compose

  1. While you can already see some of the advantages in working with Docker you also can start to see the complexities of working with it as command lines to start containers are growing in length. Because a containerized approach to running applications usually tries to limit each container to doing just one thing (webserver, database server, etc.) it is also often the case that to run a single "application" we need to run multiple containers which depend on each other and need to share information with each other. Obviously starting and stopping one application then could involve multiple long commands it makes sense to have a way to organize containers into application "groups" and start/stop and configure them together. This is where the Docker Compose tool comes into play.

  2. Docker Compose is based around Compose files (by default a compose.yaml in a separate directory for each application) which define what image(s) should be used to start or stop an application and what the configuration settings for the container(s) should be. Let’s start by setting up a very basic Compose file for the web server container we were working with in the last section.

    1. Open a new file ~/docker-web/compose.yaml in a text editor and paste this in (modifying the volume path as needed for your own home directory name):

      name: webdemo
      
      services:
          webserver:
              image: nginx
              ports:
                  - "8888:80"
              volumes:
                  - /home/ben/docker-web/content:/usr/share/nginx/html
    2. You can see how the command line options from our old docker run command got transposed into this Compose file. To run this Compose file all we need to do is be in the ~/docker-web directory with the compose.yaml file and run docker compose up -d (again the -d option on the end indicates this should be run in detatched mode in the background). Try starting the container through Docker Compose like this now.

    3. You should find that the web page is displaying and working just as it did when we ran the docker run command before.

    4. If you check what containers are running with docker ps you should see:

      ben@2480-Z:~/docker-web$ sudo docker compose up -d
      [+] Running 2/2
       ✔ Network webdemo_default        Created                                                                                                                                               0.3s
       ✔ Container webdemo-webserver-1  Started                                                                                                                                               0.4s
      ben@2480-Z:~/docker-web$ sudo docker ps
      CONTAINER ID   IMAGE     COMMAND                  CREATED         STATUS         PORTS                                   NAMES
      1ff16fe58a83   nginx     "/docker-entrypoint.…"   5 seconds ago   Up 4 seconds   0.0.0.0:8888->80/tcp, :::8888->80/tcp   webdemo-webserver-1
      ben@2480-Z:~/docker-web$
    5. Notice that the name of the Docker container this time is set to webdemo-webserver-1 which is a combination of the project name we set at the beginning of the compose.yaml file (webdemo) and the service name we set in the file (webserver) for the specific image we’re running. Both of these are needed because we may have multiple containers that start from the compose.yaml file.

    6. As long as you’re in the same directory as the compose.yaml file you can also use the docker compose ps command to just see the continers associated with the current project:

      ben@2480-Z:~/docker-web$ sudo docker compose ps
      NAME                  IMAGE     COMMAND                  SERVICE     CREATED         STATUS         PORTS
      webdemo-webserver-1   nginx     "/docker-entrypoint.…"   webserver   4 minutes ago   Up 4 minutes   0.0.0.0:8888->80/tcp, :::8888->80/tcp
      ben@2480-Z:~/docker-web$
    7. To stop the container without removing it (so the same container could be started again without rebuilding it from an image) you can run the docker compose stop command. Or, if you want to remove the container as well you can run the docker compose down command, which you should do now.

  3. Now that we have a basic idea of how Docker Compose works let’s try running an actual web application consisting of multiple containers. In this example we’ll run another copy of Wordpress as our sample application, which conveniently has an official Docker image. In addition to the Wordpress image we’ll also need a MariaDB database server and a Nginx web server. We’ll be able to see how all three of these containers communicate with one another and how we store our data in this containerized environment.

  4. As the heart of this setup is really the Docker Compose file we’ll start by setting up that.

    1. Create a new directory inside your home directory ~/wpdockerdemo

    2. Create a new compose.yaml file inside that directory and open it for editing with a text editor. Paste this in:

      name: wpdockerdemo
      
      services:
         db:
            image: mariadb
            environment:
               MYSQL_ROOT_PASSWORD: ${MYSQL_ROOT_PASSWORD}
               MYSQL_DATABASE: ${MYSQL_DATABASE}
               MYSQL_USER: ${MYSQL_USER}
               MYSQL_PASSWORD: ${MYSQL_PASSWORD}
            volumes:
               - mysql-datavolume:/var/lib/mysql
            restart: unless-stopped
      
         wp:
            image: wordpress:fpm-alpine
            environment:
               WORDPRESS_DB_HOST: db:3306
               WORDPRESS_DB_NAME: ${MYSQL_DATABASE}
               WORDPRESS_DB_USER: ${MYSQL_USER}
               WORDPRESS_DB_PASSWORD: ${MYSQL_PASSWORD}
               WORDPRESS_TABLE_PREFIX: ${WORDPRESS_TABLE_PREFIX}
            depends_on:
               - db
            volumes:
               - ${WP_CONTENT}:/var/www/html/wp-content
               - wordpress:/var/www/html
            restart: unless-stopped
      
         webserver:
            image: nginx
            volumes:
               - ./nginx:/etc/nginx/conf.d
               - wordpress:/var/www/html
               - ${WP_CONTENT}:/var/www/html/wp-content
            ports:
               - "8888:80"
            depends_on:
               - wp
            restart: unless-stopped
      
      volumes:
         mysql-datavolume:
         wordpress:
    3. Obviously we have the three containers configured in that Compose file but there are a few new things to explain.

      1. First, the environment section of the db and wp containers is another way to pass configuration information into a container that has been specifically setup to recieve configuration using environment variables. This is commonly used to pass basic one line configuration things into containers, especially secrets like passwords which we want to be careful where we store in case we share our Compose file. In this case we’ll soon create a .env file where these settings are stored. This also lets us set some things like the database name once which are used in multiple containers.

      2. Second, we have created two volumes named mysql-datavolume and wordpress which can store information from the containers which will persist even if the containers are removed and re-created (such as when they might be upgraded to new versions) but which are not bind mounts tied to a directory on our host system. These will hold our actual database files and our Wordpress installation files respectively.

      3. Lastly, we have added the restart: unless-stopped option to each container telling Docker that in the case the container somehow exits in any way other than us intentionally stopping it Docer should automatiacally restart it.

    4. Save and exit the compose.yaml file.

  5. Before we forget we should create that .env file with the settings it should contain.

    1. Create a new file at ~/wpdockerdemo/.env and open it in a text editor. Paste this in:

      # Root password for your database
      MYSQL_ROOT_PASSWORD=secret-db-pass
      
      # Database name, user and password for your wordpress
      MYSQL_DATABASE=wordpress
      MYSQL_USER=wordpress
      MYSQL_PASSWORD=wp-secret-db-pass
      
      # Path to store your wordpress content files such as image uploads, etc.
      WP_CONTENT=./wp-content
      
      # Wordpress database table prefix
      WORDPRESS_TABLE_PREFIX=wp_
    2. For our demonstartion site it’s fine to keep all the passwords as they are, but in a production setup you would want to pick secure passwords.

    3. Save and close the .env file

    4. All of the settings from this file will essentially be copied and pasted into the Docker Compose file on the fly when you bring up the application. You can actually see what this looks like (along with other default settings) with the docker compose config command.

    5. Create a new directory ~/wpdockerdemo/wp-content since we have set that up as the mounted location for the wordpress container to store content files for our site.

  6. You’ll also notice that the webserver container is set to mount the ~/wpdockerdemo/nginx directory into the Nginx configuration location so we need to set up our Nginx configuration there.

    1. Create a ~/wpdockerdemo/nginx directory

    2. Create and open a wp.conf file inside that directory for editing. Paste this in:

      server {
          listen 80 default_server;
          server_name _;
      
          root /var/www/html;
          index index.php;
      
          access_log /var/log/nginx/access.log;
          error_log /var/log/nginx/error.log;
      
          location / {
              try_files $uri $uri/ /index.php?$args;
          }
      
          location ~ \.php$ {
              try_files $uri =404;
              fastcgi_split_path_info ^(.+\.php)(/.+)$;
              fastcgi_pass wp:9000;
              fastcgi_index index.php;
              include fastcgi_params;
              fastcgi_param SCRIPT_FILENAME $document_root$fastcgi_script_name;
              fastcgi_param PATH_INFO $fastcgi_path_info;
          }
      }
    3. This should look like a pretty standard Nginx configuration. The one thing of note here is how Nginx is connecting to the Wordpress PHP-FPM application. Notice the fastcgi_pass wp:9000; line. This directs Nginx to make a FastCGI connection to a different system, in this case the Wordpress container which is running PHP-FPM. Normally we could put a local socket (with PHP-FPM running on the same machine as we did in a previous lab) here or an IP address or DNS name of a different system. When we use Docker Compose the containers are all accessible to one another using the name of the container in the compose.yaml file so because our Wordpress FPM container is named wp we can use the line fastcgi_pass wp:9000; to connect to it on port 9000.

    4. Save and close the file

  7. At this point you should be able to run the docker compose up -d command which will begin pulling the required images and creating containers.

  8. Once all the containers have started (which you can confirm with docker compose ps) you should be able to connect from your administrative PC’s browser to http://z.itc2480.campus.ihitc.net:8888 again (with your own Pod ID letter instead of z) and you should be launched into the Wordpress installation wizard.

  9. Go ahead and complete the installation wizard and make sure your Wordpress site is running as expected.

  10. Stop and remove your containers with the docker compose down command.

  11. Of course all your data is still intact even though we removed the containers because we are storing the database and Wordpress files in some Docker volumes and any uploaded content in the mounted wp-content directory.

    1. You can view the wp-content directory like normal:

      ben@2480-Z:~/wpdockerdemo$ ls -al wp-content/
      total 24
      drwxr-xr-x 5  82  82 4096 Aug 19 16:37 .
      drwxr-xr-x 4 ben ben 4096 Aug 19 16:34 ..
      -rw-r--r-- 1  82  82   28 Jan  8  2012 index.php
      drwxr-xr-x 3  82  82 4096 Jul 23 10:15 plugins
      drwxr-xr-x 5  82  82 4096 Jul 23 10:15 themes
      drwxr-xr-x 3  82  82 4096 Aug 19 16:37 uploads
      ben@2480-Z:~/wpdockerdemo$
    2. To see the Docker volumes we can use the docker volume ls command:

      ben@2480-Z:~/wpdockerdemo$ sudo docker volume ls
      DRIVER    VOLUME NAME
      local     wpdockerdemo_mysql-datavolume
      local     wpdockerdemo_wordpress
      ben@2480-Z:~/wpdockerdemo$
    3. Notice that these are still in existance even though the containers have been removed. In many cases this is a good thing, if we were to run docker compose up -d new containers would be created from the images but everything would go back to just the way it was (Wordpress already installed, all the same users and blog posts, etc) because the underlying database and files are all still the same.

    4. If we want to actually see how much space all these Docker components are taking we can use the docker system df -v command:

      ben@2480-Z:~/wpdockerdemo$ sudo docker system df -v
      Images space usage:
      
      REPOSITORY   TAG          IMAGE ID       CREATED       SIZE      SHARED SIZE   UNIQUE SIZE   CONTAINERS
      mariadb      latest       92520f86618b   4 days ago    407MB     0B            406.9MB       0
      nginx        latest       5ef79149e0ec   5 days ago    188MB     0B            187.7MB       0
      wordpress    fpm-alpine   b72041955fbc   3 weeks ago   253MB     0B            252.9MB       0
      
      Containers space usage:
      
      CONTAINER ID   IMAGE     COMMAND   LOCAL VOLUMES   SIZE      CREATED   STATUS    NAMES
      
      Local Volumes space usage:
      
      VOLUME NAME                     LINKS     SIZE
      wpdockerdemo_mysql-datavolume   0         162.4MB
      wpdockerdemo_wordpress          0         58.06MB
      
      Build cache usage: 0B
      
      CACHE ID   CACHE TYPE   SIZE      CREATED   LAST USED   USAGE     SHARED
      ben@2480-Z:~/wpdockerdemo$
    5. If we want to actually remove the local volumes we can do so either one volume at a time with the docker volume rm <volumeName> command or remove all volumes for a Compose project (as well as any containers) with the docker compose down -v command. Run this now to clear out your Docker volumes.

  12. As you can see running containerized applications can add a lot of flexibility to quickly setting up identical or similar services on servers if you have all the required configuration saved somewhere. It can also make it faster to update software. However, it also comes with a cost of added complexity compared with running the services directly on a server. Whether this makes sense in your environment will depend on the specifics of your needs/environment, how the services were built, and your own experience but it is quite common to have containerized services in the enterprise today so it’s certainly worth knowing something about.

Wrapping Up

  1. Close the SSH session

    1. Type exit to close the connection while leaving your Linux server VM running.

  2. If you are using the Administrative PC in Netlab instead of your own computer as the administrative computer you should also shut down that system in the usual way each time you are done with the Netlab system and then end your Netlab Reservation. You should do these steps each time you finish using the adminsitrative PC in future labs as well.

You can keep your Linux Server running, you do not need to shut it down.

Document Build Time: 2025-01-15 23:51:52 UTC
Page Version: 2024.08
Creative Commons License
This work is licensed under a Creative Commons Attribution-ShareAlike 4.0 International License