2015-06-07 23:07:20 -04:00
|
|
|
<!--[metadata]>
|
|
|
|
+++
|
|
|
|
title = "Working with containers"
|
|
|
|
description = "Learn how to manage and operate Docker containers."
|
|
|
|
keywords = ["docker, the docker guide, documentation, docker.io, monitoring containers, docker top, docker inspect, docker port, ports, docker logs, log, Logs"]
|
|
|
|
[menu.main]
|
|
|
|
parent="smn_containers"
|
|
|
|
+++
|
|
|
|
<![end-metadata]-->
|
2014-05-21 17:05:19 -04:00
|
|
|
|
2015-04-21 11:50:09 -04:00
|
|
|
# Working with containers
|
2014-05-21 17:05:19 -04:00
|
|
|
|
|
|
|
In the [last section of the Docker User Guide](/userguide/dockerizing)
|
|
|
|
we launched our first containers. We launched two containers using the
|
|
|
|
`docker run` command.
|
|
|
|
|
|
|
|
* Containers we ran interactively in the foreground.
|
|
|
|
* One container we ran daemonized in the background.
|
|
|
|
|
|
|
|
In the process we learned about several Docker commands:
|
|
|
|
|
|
|
|
* `docker ps` - Lists containers.
|
|
|
|
* `docker logs` - Shows us the standard output of a container.
|
|
|
|
* `docker stop` - Stops running containers.
|
|
|
|
|
|
|
|
> **Tip:**
|
|
|
|
> Another way to learn about `docker` commands is our
|
2014-07-01 20:30:25 -04:00
|
|
|
> [interactive tutorial](https://www.docker.com/tryit/).
|
2014-05-21 17:05:19 -04:00
|
|
|
|
|
|
|
The `docker` client is pretty simple. Each action you can take
|
|
|
|
with Docker is a command and each command can take a series of
|
|
|
|
flags and arguments.
|
|
|
|
|
2014-10-20 02:17:33 -04:00
|
|
|
# Usage: [sudo] docker [command] [flags] [arguments] ..
|
2014-05-21 17:05:19 -04:00
|
|
|
# Example:
|
2015-03-26 14:12:37 -04:00
|
|
|
$ docker run -i -t ubuntu /bin/bash
|
2014-05-21 17:05:19 -04:00
|
|
|
|
|
|
|
Let's see this in action by using the `docker version` command to return
|
|
|
|
version information on the currently installed Docker client and daemon.
|
|
|
|
|
2015-03-26 14:12:37 -04:00
|
|
|
$ docker version
|
2014-05-21 17:05:19 -04:00
|
|
|
|
|
|
|
This command will not only provide you the version of Docker client and
|
|
|
|
daemon you are using, but also the version of Go (the programming
|
|
|
|
language powering Docker).
|
|
|
|
|
|
|
|
Client version: 0.8.0
|
|
|
|
Go version (client): go1.2
|
|
|
|
|
|
|
|
Git commit (client): cc3a8c8
|
|
|
|
Server version: 0.8.0
|
|
|
|
|
|
|
|
Git commit (server): cc3a8c8
|
|
|
|
Go version (server): go1.2
|
|
|
|
|
|
|
|
Last stable version: 0.8.0
|
|
|
|
|
2015-05-19 01:07:41 -04:00
|
|
|
## Get Docker command help
|
2014-05-21 17:05:19 -04:00
|
|
|
|
2015-05-19 01:07:41 -04:00
|
|
|
You can display the help for specific Docker commands. The help details the
|
|
|
|
options and their usage. To see a list of all the possible commands, use the
|
|
|
|
following:
|
2014-05-21 17:05:19 -04:00
|
|
|
|
2015-05-19 01:07:41 -04:00
|
|
|
$ docker --help
|
2014-05-21 17:05:19 -04:00
|
|
|
|
2015-05-19 01:07:41 -04:00
|
|
|
To see usage for a specific command, specify the command with the `--help` flag:
|
2014-05-21 17:05:19 -04:00
|
|
|
|
2015-03-26 14:12:37 -04:00
|
|
|
$ docker attach --help
|
2014-05-21 17:05:19 -04:00
|
|
|
|
|
|
|
Usage: docker attach [OPTIONS] CONTAINER
|
|
|
|
|
|
|
|
Attach to a running container
|
|
|
|
|
2015-05-19 01:07:41 -04:00
|
|
|
--help=false Print usage
|
|
|
|
--no-stdin=false Do not attach stdin
|
|
|
|
--sig-proxy=true Proxy all received signals to the process
|
2014-05-21 17:05:19 -04:00
|
|
|
|
|
|
|
> **Note:**
|
2015-05-19 01:07:41 -04:00
|
|
|
> For further details and examples of each command, see the
|
|
|
|
> [command reference](/reference/commandline/cli/) in this guide.
|
2014-05-21 17:05:19 -04:00
|
|
|
|
2015-04-21 11:50:09 -04:00
|
|
|
## Running a web application in Docker
|
2014-05-21 17:05:19 -04:00
|
|
|
|
|
|
|
So now we've learnt a bit more about the `docker` client let's move onto
|
|
|
|
the important stuff: running more containers. So far none of the
|
|
|
|
containers we've run did anything particularly useful though. So let's
|
|
|
|
build on that experience by running an example web application in
|
|
|
|
Docker.
|
|
|
|
|
|
|
|
For our web application we're going to run a Python Flask application.
|
|
|
|
Let's start with a `docker run` command.
|
|
|
|
|
2015-03-26 14:12:37 -04:00
|
|
|
$ docker run -d -P training/webapp python app.py
|
2014-05-21 17:05:19 -04:00
|
|
|
|
|
|
|
Let's review what our command did. We've specified two flags: `-d` and
|
|
|
|
`-P`. We've already seen the `-d` flag which tells Docker to run the
|
|
|
|
container in the background. The `-P` flag is new and tells Docker to
|
|
|
|
map any required network ports inside our container to our host. This
|
|
|
|
lets us view our web application.
|
|
|
|
|
|
|
|
We've specified an image: `training/webapp`. This image is a
|
|
|
|
pre-built image we've created that contains a simple Python Flask web
|
|
|
|
application.
|
|
|
|
|
2014-10-08 16:44:21 -04:00
|
|
|
Lastly, we've specified a command for our container to run: `python app.py`. This launches our web application.
|
2014-05-21 17:05:19 -04:00
|
|
|
|
|
|
|
> **Note:**
|
|
|
|
> You can see more detail on the `docker run` command in the [command
|
|
|
|
> reference](/reference/commandline/cli/#run) and the [Docker Run
|
|
|
|
> Reference](/reference/run/).
|
|
|
|
|
2015-04-21 11:50:09 -04:00
|
|
|
## Viewing our web application container
|
2014-05-21 17:05:19 -04:00
|
|
|
|
|
|
|
Now let's see our running container using the `docker ps` command.
|
|
|
|
|
2015-03-26 14:12:37 -04:00
|
|
|
$ docker ps -l
|
2014-05-21 17:05:19 -04:00
|
|
|
CONTAINER ID IMAGE COMMAND CREATED STATUS PORTS NAMES
|
|
|
|
bc533791f3f5 training/webapp:latest python app.py 5 seconds ago Up 2 seconds 0.0.0.0:49155->5000/tcp nostalgic_morse
|
|
|
|
|
|
|
|
You can see we've specified a new flag, `-l`, for the `docker ps`
|
|
|
|
command. This tells the `docker ps` command to return the details of the
|
|
|
|
*last* container started.
|
|
|
|
|
|
|
|
> **Note:**
|
2014-06-26 06:49:29 -04:00
|
|
|
> By default, the `docker ps` command only shows information about running
|
|
|
|
> containers. If you want to see stopped containers too use the `-a` flag.
|
2014-05-21 17:05:19 -04:00
|
|
|
|
|
|
|
We can see the same details we saw [when we first Dockerized a
|
|
|
|
container](/userguide/dockerizing) with one important addition in the `PORTS`
|
|
|
|
column.
|
|
|
|
|
|
|
|
PORTS
|
|
|
|
0.0.0.0:49155->5000/tcp
|
|
|
|
|
|
|
|
When we passed the `-P` flag to the `docker run` command Docker mapped any
|
|
|
|
ports exposed in our image to our host.
|
|
|
|
|
|
|
|
> **Note:**
|
|
|
|
> We'll learn more about how to expose ports in Docker images when
|
|
|
|
> [we learn how to build images](/userguide/dockerimages).
|
|
|
|
|
|
|
|
In this case Docker has exposed port 5000 (the default Python Flask
|
|
|
|
port) on port 49155.
|
|
|
|
|
2015-01-21 07:40:59 -05:00
|
|
|
Network port bindings are very configurable in Docker. In our last example the
|
|
|
|
`-P` flag is a shortcut for `-p 5000` that maps port 5000 inside the container
|
|
|
|
to a high port (from *ephemeral port range* which typically ranges from 32768
|
|
|
|
to 61000) on the local Docker host. We can also bind Docker containers to
|
|
|
|
specific ports using the `-p` flag, for example:
|
2014-05-21 17:05:19 -04:00
|
|
|
|
2015-03-31 06:03:18 -04:00
|
|
|
$ docker run -d -p 80:5000 training/webapp python app.py
|
2014-05-21 17:05:19 -04:00
|
|
|
|
2015-03-31 06:03:18 -04:00
|
|
|
This would map port 5000 inside our container to port 80 on our local
|
2014-05-21 17:05:19 -04:00
|
|
|
host. You might be asking about now: why wouldn't we just want to always
|
|
|
|
use 1:1 port mappings in Docker containers rather than mapping to high
|
|
|
|
ports? Well 1:1 mappings have the constraint of only being able to map
|
|
|
|
one of each port on your local host. Let's say you want to test two
|
2015-01-29 12:09:44 -05:00
|
|
|
Python applications: both bound to port 5000 inside their own containers.
|
|
|
|
Without Docker's port mapping you could only access one at a time on the
|
|
|
|
Docker host.
|
2014-05-21 17:05:19 -04:00
|
|
|
|
|
|
|
So let's now browse to port 49155 in a web browser to
|
|
|
|
see the application.
|
|
|
|
|
|
|
|
![Viewing the web application](/userguide/webapp1.png).
|
|
|
|
|
|
|
|
Our Python application is live!
|
|
|
|
|
2014-06-11 17:29:42 -04:00
|
|
|
> **Note:**
|
2015-04-03 11:53:37 -04:00
|
|
|
> If you have used the `boot2docker` virtual machine on OS X, Windows or Linux,
|
2014-06-30 22:02:39 -04:00
|
|
|
> you'll need to get the IP of the virtual host instead of using localhost.
|
2015-04-03 11:53:37 -04:00
|
|
|
> You can do this by running the following outside of the `boot2docker` shell
|
|
|
|
> (i.e., from your comment line or terminal application).
|
2014-06-11 17:29:42 -04:00
|
|
|
>
|
|
|
|
> $ boot2docker ip
|
|
|
|
> The VM's Host only interface IP address is: 192.168.59.103
|
|
|
|
>
|
2014-06-16 20:38:53 -04:00
|
|
|
> In this case you'd browse to http://192.168.59.103:49155 for the above example.
|
2014-06-11 17:29:42 -04:00
|
|
|
|
2015-04-21 11:50:09 -04:00
|
|
|
## A network port shortcut
|
2014-05-21 17:05:19 -04:00
|
|
|
|
|
|
|
Using the `docker ps` command to return the mapped port is a bit clumsy so
|
|
|
|
Docker has a useful shortcut we can use: `docker port`. To use `docker port` we
|
|
|
|
specify the ID or name of our container and then the port for which we need the
|
|
|
|
corresponding public-facing port.
|
|
|
|
|
2015-03-26 14:12:37 -04:00
|
|
|
$ docker port nostalgic_morse 5000
|
2014-05-21 17:05:19 -04:00
|
|
|
0.0.0.0:49155
|
|
|
|
|
|
|
|
In this case we've looked up what port is mapped externally to port 5000 inside
|
|
|
|
the container.
|
|
|
|
|
2015-04-21 11:50:09 -04:00
|
|
|
## Viewing the web application's logs
|
2014-05-21 17:05:19 -04:00
|
|
|
|
|
|
|
Let's also find out a bit more about what's happening with our application and
|
|
|
|
use another of the commands we've learnt, `docker logs`.
|
|
|
|
|
2015-03-26 14:12:37 -04:00
|
|
|
$ docker logs -f nostalgic_morse
|
2014-05-21 17:05:19 -04:00
|
|
|
* Running on http://0.0.0.0:5000/
|
|
|
|
10.0.2.2 - - [23/May/2014 20:16:31] "GET / HTTP/1.1" 200 -
|
|
|
|
10.0.2.2 - - [23/May/2014 20:16:31] "GET /favicon.ico HTTP/1.1" 404 -
|
|
|
|
|
|
|
|
This time though we've added a new flag, `-f`. This causes the `docker
|
|
|
|
logs` command to act like the `tail -f` command and watch the
|
|
|
|
container's standard out. We can see here the logs from Flask showing
|
|
|
|
the application running on port 5000 and the access log entries for it.
|
|
|
|
|
2015-04-21 11:50:09 -04:00
|
|
|
## Looking at our web application container's processes
|
2014-05-21 17:05:19 -04:00
|
|
|
|
|
|
|
In addition to the container's logs we can also examine the processes
|
|
|
|
running inside it using the `docker top` command.
|
|
|
|
|
2015-03-26 14:12:37 -04:00
|
|
|
$ docker top nostalgic_morse
|
2014-05-21 17:05:19 -04:00
|
|
|
PID USER COMMAND
|
|
|
|
854 root python app.py
|
|
|
|
|
|
|
|
Here we can see our `python app.py` command is the only process running inside
|
|
|
|
the container.
|
|
|
|
|
2015-04-21 11:50:09 -04:00
|
|
|
## Inspecting our web application container
|
2014-05-21 17:05:19 -04:00
|
|
|
|
|
|
|
Lastly, we can take a low-level dive into our Docker container using the
|
|
|
|
`docker inspect` command. It returns a JSON hash of useful configuration
|
|
|
|
and status information about Docker containers.
|
|
|
|
|
2015-03-26 14:12:37 -04:00
|
|
|
$ docker inspect nostalgic_morse
|
2014-05-21 17:05:19 -04:00
|
|
|
|
|
|
|
Let's see a sample of that JSON output.
|
|
|
|
|
|
|
|
[{
|
|
|
|
"ID": "bc533791f3f500b280a9626688bc79e342e3ea0d528efe3a86a51ecb28ea20",
|
|
|
|
"Created": "2014-05-26T05:52:40.808952951Z",
|
|
|
|
"Path": "python",
|
|
|
|
"Args": [
|
|
|
|
"app.py"
|
|
|
|
],
|
|
|
|
"Config": {
|
|
|
|
"Hostname": "bc533791f3f5",
|
|
|
|
"Domainname": "",
|
|
|
|
"User": "",
|
|
|
|
. . .
|
|
|
|
|
|
|
|
We can also narrow down the information we want to return by requesting a
|
|
|
|
specific element, for example to return the container's IP address we would:
|
|
|
|
|
2015-03-26 14:12:37 -04:00
|
|
|
$ docker inspect -f '{{ .NetworkSettings.IPAddress }}' nostalgic_morse
|
2014-05-21 17:05:19 -04:00
|
|
|
172.17.0.5
|
|
|
|
|
2015-04-21 11:50:09 -04:00
|
|
|
## Stopping our web application container
|
2014-05-21 17:05:19 -04:00
|
|
|
|
|
|
|
Okay we've seen web application working. Now let's stop it using the
|
|
|
|
`docker stop` command and the name of our container: `nostalgic_morse`.
|
|
|
|
|
2015-03-26 14:12:37 -04:00
|
|
|
$ docker stop nostalgic_morse
|
2014-05-21 17:05:19 -04:00
|
|
|
nostalgic_morse
|
|
|
|
|
|
|
|
We can now use the `docker ps` command to check if the container has
|
|
|
|
been stopped.
|
|
|
|
|
2015-03-26 14:12:37 -04:00
|
|
|
$ docker ps -l
|
2014-05-21 17:05:19 -04:00
|
|
|
|
2015-04-21 11:50:09 -04:00
|
|
|
## Restarting our web application container
|
2014-05-21 17:05:19 -04:00
|
|
|
|
|
|
|
Oops! Just after you stopped the container you get a call to say another
|
|
|
|
developer needs the container back. From here you have two choices: you
|
|
|
|
can create a new container or restart the old one. Let's look at
|
|
|
|
starting our previous container back up.
|
|
|
|
|
2015-03-26 14:12:37 -04:00
|
|
|
$ docker start nostalgic_morse
|
2014-05-21 17:05:19 -04:00
|
|
|
nostalgic_morse
|
|
|
|
|
|
|
|
Now quickly run `docker ps -l` again to see the running container is
|
|
|
|
back up or browse to the container's URL to see if the application
|
|
|
|
responds.
|
|
|
|
|
|
|
|
> **Note:**
|
|
|
|
> Also available is the `docker restart` command that runs a stop and
|
|
|
|
> then start on the container.
|
|
|
|
|
2015-04-21 11:50:09 -04:00
|
|
|
## Removing our web application container
|
2014-05-21 17:05:19 -04:00
|
|
|
|
|
|
|
Your colleague has let you know that they've now finished with the container
|
|
|
|
and won't need it again. So let's remove it using the `docker rm` command.
|
|
|
|
|
2015-03-26 14:12:37 -04:00
|
|
|
$ docker rm nostalgic_morse
|
2014-05-21 17:05:19 -04:00
|
|
|
Error: Impossible to remove a running container, please stop it first or use -f
|
|
|
|
2014/05/24 08:12:56 Error: failed to remove one or more containers
|
|
|
|
|
2015-03-23 23:42:26 -04:00
|
|
|
What happened? We can't actually remove a running container. This protects
|
2014-05-21 17:05:19 -04:00
|
|
|
you from accidentally removing a running container you might need. Let's try
|
|
|
|
this again by stopping the container first.
|
|
|
|
|
2015-03-26 14:12:37 -04:00
|
|
|
$ docker stop nostalgic_morse
|
2014-05-21 17:05:19 -04:00
|
|
|
nostalgic_morse
|
2015-03-26 14:12:37 -04:00
|
|
|
$ docker rm nostalgic_morse
|
2014-05-21 17:05:19 -04:00
|
|
|
nostalgic_morse
|
|
|
|
|
|
|
|
And now our container is stopped and deleted.
|
|
|
|
|
|
|
|
> **Note:**
|
|
|
|
> Always remember that deleting a container is final!
|
|
|
|
|
|
|
|
# Next steps
|
|
|
|
|
|
|
|
Until now we've only used images that we've downloaded from
|
2014-06-01 16:48:04 -04:00
|
|
|
[Docker Hub](https://hub.docker.com) now let's get introduced to
|
2014-05-21 17:05:19 -04:00
|
|
|
building and sharing our own images.
|
|
|
|
|
|
|
|
Go to [Working with Docker Images](/userguide/dockerimages).
|
|
|
|
|