## Установка Docker Engine на Debian/Ubuntu
Tиповой путь через официальный репозиторий Docker.
Лучше проверить актуальную версию в [официальном гайде](https://docs.docker.com/engine/install/ubuntu/)
```
sudo apt-get update
sudo apt-get install ca-certificates curl
sudo install -m 0755 -d /etc/apt/keyrings
sudo curl -fsSL https://download.docker.com/linux/ubuntu/gpg -o /etc/apt/keyrings/docker.asc
sudo chmod a+r /etc/apt/keyrings/docker.asc
echo \
"deb [arch=$(dpkg --print-architecture) signed-by=/etc/apt/keyrings/docker.asc] https://download.docker.com/linux/ubuntu \
$(. /etc/os-release && echo "${VERSION_CODENAME:-$VERSION_CODENAME}") stable" | \
sudo tee /etc/apt/sources.list.d/docker.list > /dev/null
sudo apt-get update
sudo apt-get install docker-ce docker-ce-cli containerd.io docker-buildx-plugin docker-compose-plugin -y
```
Проверяем:
```
sudo docker run --rm hello-world
```
Опционально можно запускать без `sudo`,:
```
sudo usermod -aG docker $USER
# выйти из сессии и зайти снова
docker run --rm hello-world
```
Проверка версии:
```
docker version
docker info
```
## Базовый жизненный цикл контейнера
Скачать образ и запустить интерактивную оболочку, образ автоматически скачивается с Docker Registry:
```
docker run -it --rm debian:bookworm-slim bash
```
Установка образа:
Мы указали `bash` в команде, по этому зашли в интерактивный терминал:
В контейнере, так же как и в линуксе, можно управлять файлами, директориями, сервисами, но ограничено.
Выйти из bash - `exit`, контейнер остановится и `--rm` удалит его.
Коротко про флаги:
| Флаг | Смысл |
| -------------------------------------- | --------------------------------- |
| `-it` | интерактив + псевдо-TTY |
| `--rm` | удалить контейнер после остановки |
| `-d` | запустить контейнер в фон |
| `--name` | имя контейнера |
| `-p 8080:80` | проброс порта хост:контейнер |
| `-v /путь/на/хосте:/путь/в/контейнере` | том (данные на хосте) |
Запуск веб-сервера в фоне:
```
docker run -d --name demo-nginx -p 8080:80 nginx:alpine
```
Проверим работу на порту 8080:
Список контейнеров и логи:
`docker ps` - посмотреть список АКТИВНЫХ контейнеров
`docker ps -a` - посмотреть список ВСЕХ контейнеров
`docker logs demo-nginx` - вывод логов контейнера
`docker logs -f demo-nginx` - смотреть логи контейнера в реальном времени
Остановка и удаление:
`docker stop demo-nginx` - остановка контейнера
`docker rm demo-nginx` - удалить контейнер
## Вход в уже запущенный контейнер: `docker exec`
```
docker run -d --name my-nginx -p 8080:80 nginx:alpine
docker exec -it my-nginx sh
```
Внутри можно смотреть файлы конфигурации, например `cat /etc/nginx/nginx.conf`, ставить утилиты в отладочных образах и т.д.
Так же интересный факт: мы уже устанавливали образ `nginx:alpine`, по этому нам его повторно скачивать не нужно, он хранится в docker cache. Единственный минус - занимает место на диске.
## Копирование файлов **в** контейнер и **из** контейнера: `docker cp`
На хосте создадим простейший html файл:
`echo '
Обновим страницу в браузере - должен отобразиться новый html.
Скопировать **из** контейнера на хост:
```
docker cp my-nginx:/etc/nginx/nginx.conf /tmp/nginx.conf.from-container
```
Важно: `docker cp` удобен для отладки и разовых правок. Для поставки контента, файлов в образ используйте функцию `COPY` в `Dockerfile` или тома в Compose (смотреть дальше по гайду).
## Тома (volumes): данные переживают пересоздание контейнера
Именованный том:
```
docker volume create webdata
docker run -d --name nginx-vol -p 8081:80 -v webdata:/usr/share/nginx/html nginx:alpine
docker exec nginx-vol sh -c 'echo vol > /usr/share/nginx/html/index.html'
```
Про каждую команду:
`docker volume create webdata` - создаем именованный том `webdata` на нашем хосте.
`docker run -d --name nginx-vol -p 8081:80 -v webdata:/usr/share/nginx/html nginx:alpine` - создем контейнер `nginx-vol` на порту 8081 и монитируем ему том `webdata` в каталог контейнера `/usr/share/nginx/html`. В этом каталоге nginx подхватывает index.html файл.
`docker exec nginx-vol sh -c 'echo vol > /usr/share/nginx/html/index.html'` - выполняем команду ВНУТРИ контейнера `nginx-vol` через "sh -c" и передать ему команду `echo`.
Коротко, что мы сделали:
Мы создали том `webdata` - это физическое место на нашем хосте. Мы к контейнеру примонтировали этот том `-v webdata:/usr/share/nginx/html`. То есть у контейнера теперь есть доступ к этому физическому месту(тому). Затем мы создали файл `index.html` в каталоге `/usr/share/nginx/html`, это означает, что мы фактически создали файл у себя на хосте. Файлы в томах никуда не пропадают при удалении контейнеров.
Проверим на деле:
```
docker stop nginx-vol
docker rm nginx-vol
docker run -d --name nginx-vol -p 8081:80 -v webdata:/usr/share/nginx/html nginx:alpine
```
Новый контейнер взял файл с тома. Если мы зайдем в контейнер и удалим файл index.html, то он удалится и в томе(на хосте).
Bind-mount, привязка папки с хоста:
```
mkdir -p ~/docker-html && echo '
Теперь поменяем в нашей локальной папке файл и посмотрим что будет:
Всё работает.
## Сеть: как контейнеры видят друг друга
По умолчанию контейнеры в **bridge** могут общаться по IP. Удобнее создать **пользовательскую сеть** и обращаться по **имени контейнера**:
```
docker network create app-net
docker run -d --name backend --network app-net nginx:alpine
docker run -it --rm --network app-net alpine wget -qO- http://backend
```
Мы создали виртуальную докер сеть `app-net`, создали первый контейнер с nginx с сетью `app-net`, затем второй контейнер с такой же сетью. Со второго контейнера мы обратились к первому по его названию.
## Очистка среды
```
docker stop my-nginx nginx-vol nginx-bind backend
docker rm my-nginx nginx-vol nginx-bind backend
docker volume rm webdata
docker network rm app-net
```
## Что дальше
- Часть **3** — сборка своего образа, `Dockerfile`, многоэтапные сборки, `COPY` vs монтирование.
- Часть **4** — полный пример **Docker Compose** из репозитория `docker/examples/multi-service`.