Docker Documentation: https://docs.docker.com/
This is just Docker basics tutorial. This wiki includes some basic commands how to use docker containers.
-
General info
-
Required programs
-
Installation
3.1 Windows
3.2 Installation on Linux (Ubuntu example)
-
About Docker flow:
4.1 General flow
4.2 Container
4.3 Image
4.4 Container examples
-
Commands
5.1 Basic commands
5.2 Examples
-
Images and containers
6.1 Images
6.2 Containers
6.3 Terminal interaction
6.4 Useful commands
6.5 Useful flags
6.6 Containers states & changes
6.7 Stopping and removing containers and images
6.8 Image - Container relation
6.9 General conslusions
-
Working with containers
7.1 Processes in the container
7.2 Checking history of operations in the container
-
Files and data volumes
8.1 "Persistent" data (shared from host or other container)
8.2 Ephemeral data
8.3 Server Http with persistent data
8.4 Inspect mounts
8.5 Copying files between container and host
8.6 Command "docker volume"
-
World of images
-
Dockerfile
10.1 Example of simple Dockerfile:
10.2 Build an image based on Dockerfile
-
Docker compose
-
Containers networks
-
A - Z example
a) Docker Desktop for Windows (or older Docker Toolbox) https://docs.docker.com/docker-for-windows/
b) Docker CE https://docs.docker.com/install/
c) any SSH client (e.g. Putty) https://www.putty.org/
d) Oracle Virtual Box https://www.virtualbox.org/
a) Install Virtual Box and any SSH client
b) Install Docker for Windows or Docker Toolbox
After installation, open Docker Toolbox and run below command to see if it works and to list possible docker commands
$ docker
Also, you can use below command to see more detailed info about docker
$ docker info
Additionally you can try to run hello-world container as below
$ docker run hello-world
Then you should see some info about pulling hello-world image and status of this operation
c) Additional info
Every Toolbox open, should add (if not added yet) and run new box in Oracle Virtual Box (it's because of emulation and ability to run Linux on Windows).
This Virtual Box, is called "machine", to manage this you can use commands as below:
# should show all possible flags and commands
$ docker-machine
# to log in to the system, inside this virtual box
$ docker-machine ssh
# stop this machine (now on Oracle Virtual Box you should see this machine is stopped)
$ docker-machine stop
# start this machine
$ docker-machine start
Additionally: in Oracle Virtual Machine you can reserve resources you want to allow for docker machine
Machine -> Settings -> System
a) Installation from repository
How to do: https://docs.docker.com/install/linux/docker-ce/ubuntu/
# update packages
$ sudo apt-get update
# install
$ sudo apt-get install apt-transport-https ca-certificates curl gnupg-agent software-properties-common
# install GPG key
$ curl -fsSL https://download.docker.com/linux/ubuntu/gpg | sudo apt-key add -
# verify key
$ sudo apt-key fingerprint 0EBFCD88
# check if it works
$ sudo docker info
b) Install from package
@See: https://docs.docker.com/install/linux/docker-ce/ubuntu/ (section Install from a package)
c) Intall from script
@See: https://docs.docker.com/install/linux/docker-ce/ubuntu/ (section Install using the convenience script)
d) Install in cenOS with SSH client
# install required packages
$ sudo yum install -y yum-utils device-mapper-persistent-data lvm2
# set up repository
$ sudo yum install -y yum-utils device-mapper-persistent-data lvm2
# install docekr CE
$ sudo yum install docker-ce docker-ce-cli containerd.io
# check
$ docker info
@See more: https://docs.docker.com/install/linux/docker-ce/centos/ (section Install using the repository)
-
container is kind of separate machine which is completely independent on master system and also independent on other containers.
-
container is like a separate world, with it's own system, programs, network etc.
-
container can be moved from one system to another without any changes and it will be working properly as container is just "self contained" object
-
container has his own processes, programs and system (container can have completely different system than the host one)
-
container is build based on Image
Docker is not a Virtual Machine. It doesn't virtualise resources but uses some system's namespaces to create containers, it's kind of fragmentation, there is not resources virtualisation etc..
@See more: https://www.docker.com/sites/default/files/d8/2018-11/docker-containerized-and-vm-transparent-bg.png
-
image is kind of "pattern" for building containers
-
images can contain anything, for example complete, installed environment of any kind (e.g. LAMP etc.)
-
image can be also a recorded result of any container, the same as we create Containers from Images, we can also create image from Container
@See https://cdn-images-1.medium.com/max/1600/1*Ef2uxCnIkF0PqLGxYqV2gA.png
# to run any command in container use below syntax
$ docker [ TO DO ] [ OPTIONS ] [ IMAGE ] [ COMMAND ] [ PARAMS ]
[ TO DO ] - what do we want to do, e.g.
# run command
$ run
# execute command
$ exec
# attach container and process
$ attach
[ OPTIONS ] - options, e.g.
# terminal
-t
# interaction
-i
# to set up name of your container
--name name_of_container
# restart
--restart
[ IMAGE ] - image we want to use as a base for container, e.g.
ubuntu:latest
php:7.2-cli
see some more images on Docker Hub (https://hub.docker.com)
[ COMMAND ] - what to do in container, e.g.:
bin/bash
bash
touch super-file.txt
[ PARAMS ] - additional params
Docker cheat sheet:
-
@See: https://jrebel.com/rebellabs/docker-commands-and-best-practices-cheat-sheet/
-
@See: https://www.kdnuggets.com/2018/08/docker-cheat-sheet.html
# create docker container based on nginx image(the latest version), in detached mode (int the background), expose port 80 (map to the same port in host), run nginx server
$ docker run -d -p 80:80 my_image service nginx start
# create docker container based on ubuntu(the latest version), run process /bin/bash inside the container, attach asdin and stdout
$ docker run -a stdin -a stdout -i -t ubuntu /bin/bash
# create docker container with name "my-redis" based on redis image (latest), in detached mode
$ docker run --name my-redis -d redis
# create docker container based on Redis image (latest), run bash process inside, go directly to the container (-ti flag)
$ docker run -it redis bash
@See more: https://docs.docker.com/engine/reference/run/
If you run any process in container and if you don't have this container yet and this image in your local repository, then Docker will pull needed image and save to your local repository.
# to show all images
$ docker images
# show active containers
$ docker ps
# show all containers (check out statuses)
$ docker ps -a
Why some containers are exited instantly? It's because container works as long as any process inside is running so if. you run e.g. docker run image - the container will be instantly created and also will finish his work instantly and exit.
Another example of "quick start" and "quick exit"
# quickly run container, create file and finish work, so exit container
$ docker run image touch file.txt
How to keep container working?
- run any long - term process in container, e.g. "bash"
# run bash in docker container (flag -ti is "terminal interactive" - helps us to go "inside" this container and keep process "bash" active in the container)
# Now container should be "UP" at all time - as long as we keep "bash" inside container working
$ docker run -ti ubuntu bash
How to read docker ps results?
-
container id - ID
-
image: images used to create container
-
command: command ran in the container
-
name: name of container (see flags: --name)
-
status: just status of container
-
created: when created
-
ports: ports exposed from container
How to interact with container and go inside the container's system?
$ docker run -ti [ container_id_or_image ] [ command ]
# [ container_id_or_image ] - container or image we want to use as a base for container's creation
# [ command ] - command we want to run inside the container
example:
# container based on image "maven"(lastest)
$ docker run -ti ubuntu:latest /bin/bash
# we can now check if we are really in this container (with Ubuntu, the latest version)
$ cat /etc/lsb-release
Popular Docker commands.
# start container (exit directly because there is no keep alive process)
$ docker start -i container_id(e.g. "abc123", see "docker ps -a")
# almost the same as docker run -ti bash - but it creates new and independent process (run bash inside container and go inside because of -ti - terminal interaction)
$ docker exec -ti container_id(e.g. "abc123", see "docker ps -a") bash
# go inside the container and connect with currently existing process in the container
$ docker attach container_id(e.g. "abc123", see "docker ps -a")
# leaving the container and go back to the host
$ exit;
# info about the latest closed container
$ docker ps -l
# inspect image - checking what does image contain?
$ docker inspect [ image ]
$ docker inspect 663c1274dff6
# log-in and log-out
$ docker logout [SERVER]
$ docker login [OPTIONS] [SERVER]
[OPTIONS]
--password , -p Password
--password-stdin Take the password from stdin
--username , -u Username
# inspect container - checking container - on working container we can see some useful info such us network info, IP addresses, exposed ports and more
$ docker inspect [ container ]
$ docker inspect
# renaming image
$ docker tag image_name new_name:tag
$ docker tag ubuntu my_new_ubuntu:123
a) --name name_of_my_container - use to force custom name on your container, otherwise random name will be used
$ docker run --name my-redis -d redis
b) --restart - use for managing containers restarts policy. We can use it to keep container alive (even without active process in container). By the default
this flag is settled as --restart=no
, that is why if we don't have any alive process in the container, the container is exited just after finish his work.
we can change this behaviour by using this flag, for example:
$ docker run -ti --restart unless-stopped --name always_working_container ubuntu bash
other possible values:
-
no [default, restart always when container finish work]
-
unless-stopped
-
on-failure
-
always
https://docs.docker.com/engine/reference/run/#restart-policies---restart
c) --rm - remove image just after finish work (use only for a moment and then remove)
$ docker run --rm ubuntu echo "something"
Container automatically saves changes inside. IF we change anything in the container and leave or stop the container, our change will be saved. Test:
# run container and go inside with -ti flag
$ docker run --name my_new_container -ti ubuntu /bin/bash
# create a file inside
$ touch my_new_file.txt
# exit
$ exit;
# go again to container
$ docker start my_new_container
$ docker attach my_new_container
$ ls -la
# we should see our file
a) Containers
# remove container
$ docker rm container_id
in case if container is active (status is "up"), we cannot remove such one, then we have to first stop the container
# stop container
$ docker stop container_id
b) Images
# remove image from local repository
$ docker rmi image_name
# remove image from local repository in case if image is used by any container and cannot be normally removed
$ docker rmi --force
c) Bulk operations:
# stop all containers we have
$ docker stop $(docker ps -a -q)
# delete all containers we have
$ docker rm $(docker ps -a -q)
# delete all images
$ docker rmi $(docker images -q)
Container is always based on image (we need to declare image we want to base, to create any container based on that).
Also we can create Image based on container. For example if we want to share our work (our container) with someone else, then we can create an Images based on our container and share with someone or just publish e.g. on Docker Hub.
So, finally if we create a container and then, do some things inside, such us e.g. install some programs etc, then we can finally "export" this container to an Image and share with anyone, so we can share the whole working container.
To do that, we need to commit our container and tag our newly created image
Example:
# build a container based on any image
$ docker run --name test_container_and_image_test -ti ubuntu /bin/bash
# add a test file and check
$ touch my_test_file.txt && echo "hehehe" > my_test_file.txt
$ cat my_test_file.txt
$ ls -la
# our newly added file file should be in place, now we can commit our container
# we can check the last closed container to get his ID
$ docker ps -l
# and commit our container's changes [ docker commit container_ID ]
$ docker commit d79b218d15df
# commit creates new Image (after commit we should see ID of this image)
# now we should rename this image - set up any better name
Syntax:
$ docker tag _my_newly_generated_image_name_from_docker_commit my_new_image_name_i_want_to_use
Example:
$ docker tag 582d90462 my_own_ubuntu_image
# now, we should see our new image (my_own_ubuntu_image) in docker images
docker images
Now we can share this image with anyone or load to Docker Hub. Also this image is a complete representation of our container with all his changes inside so if we create any new container (or anyone else) based on this image - our changes, so e.g. file my_test_file.txt will be already added inside
test:
# create new container based on our image
$ docker run -ti my_own_ubuntu_image /bin/bash
# list files inside
$ ls -la
# as we can see our file is inside (even in new container based on our image)
$ docker run
... - creates container based on image
$ docker commit
... - creates a new image, based on container
###Processes in the container
Docker container works like a separate system with all his processes inside. Container's life-cycle strongly depend on main process running on that container.
Test:
# create container (in the background)
$ docker run -d -ti --name container_sla1 ubuntu /bin/bash
# go again into this container on separate window and run bash process again
$ docker run -ti --name container_sla1 ubuntu /bin/bash
# now we can check if we have 2 separate bin/bash processes
$ ps aux
# now lets kill the main /bin/bash process (PID 1), our container will be exited because we kill main process
$ docker logs [ container_id_or_name ]
$ docker logs container_sla1
Command docker logs
works on every container, no matter what is the container's state.
Data volumes functionality in kind of mapping between local host and container. We can use this to share some HDD space or directory between container and host system.
How to share volume - map host directory to docker container's one?
Example (mapping my /home directory to /dir_in_container in the container)
$ docker run -ti --name my_container -v "$(PWD)":/dir_in_container ubuntu /bin/bash
Now if we add any file or edit anything in host machine, then it will be available in the container as well. Also, if we remove container, then data on host will still be there of course.
Mounting volume give us possibility to create "data - transparent" images & containers, so mounted data will not be saved in container and image but
will be stored just on host machine. Also mounted data is not committed to an image if we use docker commit
.
Ephemeral data is available only inside the container and is removed once we remove container.
Example:
# create container and add a test file inside
$ docker run -ti --name container1 -v /my_volume1 ubuntu bash
$ touch ./my_volume1/test_file.txt && echo "some info" >> ./my_volume1/test_file.txt
$ cat ./my_volume1/test_file.txt
# create second container using data volum from the first container
$ docker run -ti --name container2 --volumes-from container1 ubuntu bash
# put some data to this file from container 2
$ cd ./my_volume1/
$ echo "new data from container 2" >> test_file.txt
$ cat ./test_file.txt
# exit container 1 and remove
$ exit;
$ docker rm [container 1]
# try to create another container with volume shared with container1
$ docker run -ti --name container5 --volumes-from container1 ubuntu bash
# error
so we lost data from container 1
Server Http with persistent data volume (from host).
# create directory on host and add index.html file inside
/server_files/index.html (<html><h1>This is my index!!!</h1></html>)
# create docker container based on httpd image and mount host's /server_files directory to container's /htdocs directory (public apache server's directory) + expose port 80 to 8080
$ docker run -dti --name my_server -p 8080:80 -v //c/Users/S/server_files:/usr/local/apache2/htdocs httpd
Our host's index.html should be now visible in the container's htdocs directory
Now we can go to host, open the browser and type localhost:8080 or (default docker machine IP) http://192.168.99
and we should see our index.html file's contend so our server in on docker container but data (index file) is on host and it's is Persistent data (will not disappear if we remove container).
# use docker inspect command to check mounts
$ docker inspect container
From host to container
$ docker cp FILE_OR_DIR_TO_COPY containerName:/path_in_container
Example:
$ cd /c/Users/S/
$ docker cp ./server_files/ my_server:/my_files
# now we should see copied directory in the container
From container to host
$ docker cp containerName:/path_in_container where_to_copy_in_the_host
$ docker cp my_server:/my_files C:/
Creating volumes and managing columens.
Create new volume
$ docker volume create my_new_volumen
# now we can mount this volume to any container
$ docker run -ti --name my_volums --mount source=my_new_volumen,target=/my_mounted_data ubuntu bash
# check if we have directory mounted
$ ls
# now we can add or edit any file in the host and it will be visible in container
Other operations on volumes
# inspecting volume
$ docker volume inspect [ volume_name ]
$ docker volume instect my_new_volumen
# removing volumes
$ docker volume rm [ volume_name ]
$ docker volume rm my_new_volumen
# removing unused volumes
$ docker volume prune my_new_volumen
# bulk remove unused
$ docker volume prune $(docker volume ls -q)
Useful flags
$ docker volume create my_new_volumen --label "My new Label" # creating custom labels
There are a lot of images ready to use in the internet, almost with every possible service and configuration already installed.
Syntax:
$ docker search [ image_name_or_part ]
Example:
$ docker search ubuntu
Syntax:
$ docker pull [ image ]
Example:
$ docker pull ubuntu
We need to prepare an image, tag this image and then push to any repository.
# prepare image with tag and repository
$ docker tag [ image ] [our_docker_hub_repository]/[ our_image_name ]:[ tag ]
$ docker tag ubuntu slawekhaa/my_own_ubuntu:1.01
# log in to docker hub
$ docker login
# + authenticate
# push to docker hub
$ docker push slawekhaa/my_own_ubuntu:1.01
# now everybody can use this image and build container based on it
Dockerfile is kind of script or image's description (set of commands) we can use to build an image. The order of commands in Dockerfile is important.
# Comment (e.g. "server with php")
# image we want to use to build an image
FROM ubuntu
# maintainer information
MAINTAINER Slawek <email@example.com>
# analogically we can use (in this case we can see the Label in docker inspection - docker inspect)
# LABEL maintainer="Slawek <email>"
# commands to run in the image after build (here it's PHP installation + packages update)
RUN apt-get update && \
apt-get install -y apache2 vim nano php7.0 libapache2-mod-php7.0 php7.0-mysql php7.0-cli && \
apt-get clean && \
rm -rf /var/lib/apt/lists/*
# [ optional ] - environment variables we want to set
ENV APACHE_RUN_USER www-data
ENV APACHE_RUN_GROUP www-data
ENV APACHE_LOG_DIR /var/log/apache2
# more commands (here we create a phpinfo file)
# better is to write all commands in the single RUN clausule because the Docker will create less
# amount of containers behind and finally our container's size will be smaller
RUN echo "<?php phpinfo() ?>" > /var/www/html/phpinfo.php
# optional (if we want to create container for specific user only instead of default user - root)
# after build we can check user's dir > pwd and it should be our newly added user
RUN useradd -ms /bin/bash slawek
USER slawek
# port we want to expose or map from container to the host system
EXPOSE 80
# [ optional ] command we can use at the moment when container id build
# unlike RUN - CMD is used when container is build, while RUN is used when image is build
# in this example we just run apache in the background
CMD apachectl -D FOREGROUND
# Syntax
$ docker build -t [ image_name_we_want ] [ dockerfile_path ]
# Example
$ cd /dockerfile_path
$ docker build -t serverssh ./
After build we can check our image IP and check if our server works e.g.
$ docker inspect our_image | grep IPAddress
$ elinks our_container_ip
https://miro.medium.com/max/1273/1*p8k1b2DZTQEW_yf0hYniXw.png
Compose is a tool for defining and running multi-container Docker applications. With Compose, you use a YAML file to configure your application’s services. Then, with a single command, you create and start all the services from your configuration. To learn more about all the features of Compose, see the list of features.
Compose works in all environments: production, staging, development, testing, as well as CI workflows. You can learn more about each case in Common Use Cases.
Using Compose is basically a three-step process:
-
Define your app’s environment with a Dockerfile so it can be reproduced anywhere.
-
Define the services that make up your app in docker-compose.yml so they can be run together in an isolated environment.
-
Run docker-compose up and Compose starts and runs your entire app.
Example of Docker compose file:
docker-compose.yml
mysql-docker:
container_name: mysql_docker
image: mysql:5.7
environment:
- MYSQL_ROOT_PASSWORD=root
- MYSQL_DATABASE=realtime_notifer
- MYSQL_USER=sla
- MYSQL_PASSWORD=sla
boot-rethink:
container_name: rethinkdb_docker
image: rethinkdb
ports:
- 8080:8080
- 28015:28015
- 29015:29015
realtime-notifer-docker:
container_name: realtime_notifer
image: slawekhaa/realtime_notifer
links:
- mysql-docker:mysql
- boot-rethink:rethink
ports:
- 8090:8090
Example of Dockerfile to build custome image realtime_notifer
Dockerfile
FROM java:8
ADD realtime_notifer.jar app.jar
RUN bash -c 'touch /app.jar'
ENTRYPOINT ["java","-Djava.security.egd=file:/dev/./urandom","-Dspring.profiles.active=docker","-jar","/app.jar"]
How to run
$ cd docker-compose.yml
$ docker-compose up
Final result: we should have 3 working containers with MySQL, RethikDb database and Java application, linked between each other.
... section in progress ...
a) Server apache + PHP Use Dockerfile from this repository - /apache_php/Dockerfile. Dockerfile contains commands needed to install Apache server with PHP 7.
# go to the Dockerfile directory
cd directory_with_dockerfile
# use our Dockerfile to build an image
$ docker build -t apache_php_slawek ./
# build a container + map port 8080 to 80 and mount host's home directory to server's www/html directory
$ docker run -tid --name MyApachePhpServer -p 8080:80 -v "$PWD":/var/www/html ubuntu:apache_php_slawek
# now we can add a phpinfo.php file and try to reach this server from host e.g. via elinks
# now we should have working Apache server with PHP
elinks http://localhost:8080
See official PHP images on Docker Hub https://hub.docker.com/_/php
b) Server SSH
Use Dockerfile from this repository - /ssh_caontainer/Dockerfile
# go to the Dockerfile directory
cd directory_with_dockerfile
# use our Dockerfile to build an image
$ docker build -t serverssh ./
# build a container based on our image
$ docker run -tid serverssh -p 2022:22
# now weshould have working container with SSH
The end.
- Slawomir Hadas - author - Github