====== Gitlab ======
Документация: https://docs.gitlab.com/ee/ci/\\
Get started with GitLab CI/CD: https://docs.gitlab.com/ee/ci/\\
Learn GitLab with tutorials: https://docs.gitlab.com/ee/tutorials/\\
GitLab with Git Essentials - Hands-On Lab: [[https://handbook.gitlab.com/handbook/customer-success/professional-services-engineering/education-services/gitbasicshandsonlab3/|Use GitLab To Merge Code]]\\
Удобен тем, что содержит всё необходимое, не нужно использовать кучу различных сервисов:
* Собственно, сам git-репозиторий
* wiki
* Issue board с привязкой к событиями внутри Гитлаба, этакая канбан-доска. Для каждого этапа можно создать список задач, а также метки для удобной фильтрации. Для каждого тикета можно назначить ответственного и этап.
* Container registry - хранилище образов контейнеров
* Package registry - хранилище ПО
* CI/CD ([[https://habr.com/ru/company/ruvds/blog/522334/|Разница между Jenkins и Gitlab CI]])
Ещё один комбайн типа Гитлаба - [[https://www.jetbrains.com/ru-ru/space/|Jetbrains Space]].
* Pipeline - описывается в ''.gitlab-ci.yml'', кладётся в корень репозитория.
* Job artifacts — результаты выполнения задач.
* Cache dependencies — кэш зависимостей можно сохранять для ускорения сборки.
* CI/CD variables — могут использоваться в ''.gitlab-ci.yml'', но лучше не задавать переменные в самом файле, а определять их в настройках CI/CD.
===== Workflow =====
^GitLab Component ^Function ^Also Known As... ^
|Project |The core building block where work is organized, managed, tracked and delivered to help the team to collaborate and plan work in the form of issues. |Repository |
|Group |A collection of projects and/or other groups. They are like folders. |Project |
|Issue |An issue is part of a project. It is the fundamental planning object where the team documents the use case in the description, discusses the approach, estimates the size/effort (issue weight), tracks actual time/effort, assigns work, and tracks progress. |Story, Narrative, Ticket |
|Epic |A collection of related issues across different groups and projects to help organize by theme |Initiatives, Themes |
|Merge Request |The linkage between the issue and the actual code changes. Captures the design, implementation details (code changes), discussions (code reviews), approvals, testing (CI Pipeline), and security scans. |Pull Request |
|Label |Used to tag and track work for a project or group and associate issues with different initiatives |Tag |
|Board |A visual listing of projects and issues useful for teams to manage their backlog of work, prioritize items, and move issues to the team or specific stage in the project. |Kanban |
|Milestone |A sprint or deliverable(s), helping you organize code, issues, and merge requests into a cohesive group |Release |
|Roadmap |A visual representation of the various epics for the group | |
**Поток**
{{:service:pasted:20240529-111219.png?800}}
- Issues - всё начинается с этого. Обсуждение и комментирование нововведения и его реализации. Проблема связана только с конкретным проектом, но если в группе несколько проектов, то можно видеть проблемы всех проектов на уровне группы.
- Merge Request - создаётся после создания проблемы. Мерж-реквесты позволяют визуализировать и совместно работать над предлагаемыми изменениями в исходном коде, которые существуют в виде коммитов в данной ветке Git. Иногда называется pull request.
- Когда мерж-реквест принят, его можно закоммитить, что запустит пайплайн.
- Выполнение пайплайна - сборка, тесты и деплой на тестовое окружение. Если пайплайн завершился неуспешно, можно прочесть логи для исправления проблемы.
- Review Apps - проверка приложения. Живой экземпляр новой версии приложения для ознакомления дизайнерам/менеджерам и т. п.
- Peer Review and Discussion - проверка и обсуждение с коллегами на предмет отсутствия каких-то конфликтов и т. п.
- Approve changes - одобрение изменений тем, у кого есть на это права.
- Merge; Issue Closed; CD Pipeline runs - после одобрения изменений проблема закрывается и приложение выкатывается в прод.
- Мониторинг - контроль приложения на предмет того, что изменения имеют желаемый эффект. В Гитлабе, если что, изменения можно откатить обратно.
**Code Review Workflow**
{{:service:pasted:20240529-115042.png}}
**Дополнительные инструменты**
* Snippets - хранение небольших кусков кода или текста и общий доступ к ним.
* Wiki - встроена в каждый проект Гитлаба. Используется, если не хочется хранить документацию к проекту в самом репозитории, но она должна быть близко. Её можно править как через веб-интерфейс, так и локально через git. Wiki является отдельным репозиторием.
* Web IDE - удобный встроенный редактор кода, можно работать прямо в браузере. [[https://about.gitlab.com/blog/2022/12/15/get-ready-for-new-gitlab-web-ide/|Обзор новых функций]].
===== Реестры =====
* [[https://docs.gitlab.com/ee/user/packages/container_registry/index.html|Container registry]] - хранение образов контейнеров
* [[https://docs.gitlab.com/ee/user/packages/package_registry/index.html|Package registry]] - хранение пакетов разных пакетных менеджеров и зависимостей.
Дополнительно: [[https://docs.gitlab.com/ee/user/packages/terraform_module_registry/index.html|Terraform Module Registry]], [[https://docs.gitlab.com/ee/user/packages/dependency_proxy/index.html|Dependency Proxy]] (local proxy for frequently-used upstream images and packages. The Dependency Proxy caches both the manifest and blobs for a given image, so when you request it again, Docker Hub does not have to be contacted.)
===== Релизы =====
Это снапшот проекта для конечных пользователей, включая инсталляционные пакеты и примечания к релизу. Релизы могут создаваться в любой ветке. Создание релиза также создаёт [[https://git-scm.com/book/en/v2/Git-Basics-Tagging|git-тэг]] для пометки точки релиза в исходном коде.
Релиз включает:
* Снапшот исходного кода репозитория.
* [[https://docs.gitlab.com/ee/user/packages/generic_packages/index.html|Пакеты (generic packages)]], созданные из артефактов.
* Метаданные, ассоциированные с релиз-версией кода.
* Примечания к релизу.
Когда релиз создаётся, то
* Гитлаб автоматически архивирует исходный код и ассоциирует его с релизом.
* Гитлаб автоматически создаёт json-файл ([[https://docs.gitlab.com/ee/user/project/releases/release_evidence.html|release evidence]]), где перечислено всё, что входит в релиз, что нужно для сравнения релизов и их аудита. Можно использовать API для генерации release evidence, их может быть несколько.
После релиза можно
* Добавить примечания к релизу.
* Добавить сообщение к гит-тэгу, ассоциированному с релизом.
* [[https://docs.gitlab.com/ee/user/project/releases/#associate-milestones-with-a-release|Ассоциировать вехи]] (milestones) с релизом.
* [[https://docs.gitlab.com/ee/user/project/releases/release_fields.html#release-assets|Добавить прочие ресурсы (assets)]] типа инструкций по запуску и пакетов.
[[https://docs.gitlab.com/ee/user/project/releases/#create-a-release|Creating a release]]\\
[[https://docs.gitlab.com/ee/user/project/releases/release_cicd_examples.html|Release CI/CD examples]]\\
[[https://docs.gitlab.com/ee/user/project/releases/release_cli.html|Release CLI tool]]
В Гитлабе есть несколько функций для удобства развертывания и теста релизов.
* [[https://docs.gitlab.com/ee/ci/review_apps/|Review apps]] - предпросмотр изменений, сделанных в ветке, с помощью развёртывания динамического окружения для мердж-реквеста. [[https://www.youtube.com/watch?v=HblSiPFamDI|Youtube]]
* [[https://docs.gitlab.com/ee/operations/feature_flags.html|Feature flags]] - переключатели, позволяющие включить и выключать функции для тех или иных групп пользователей.
* [[https://docs.gitlab.com/ee/user/project/pages/index.html|Gitlab Pages]] - встроенный хостинг статических веб-страниц прямо из репозитория. Таким образом можно хостить портфолио, документацию, презентации и т. п. [[https://www.youtube.com/watch?v=TWqh9MtT4Bg|Youtube]]
===== Pipeline =====
Пайплайн - это файл ''.gitlab-ci.yml'' в корне git-репозитория. Состоит из последовательных этапов (stages), в каждом этапе есть задачи (jobs). По умолчанию, если не заданы условия, задачи выполняются параллельно. Если какая-то задача выполнилась с ошибкой, то весь пайплайн будет провален.
Вид простейшего пайплайна:
variables:
ART_TOKEN: "37tui2v3pd8238PGd83g2"
stages:
- build
- release
- notify
build_a:
stage: build
build_b:
stage: build
# https://docs.gitlab.com/ee/ci/yaml/index.html#only--except
only: # срабатывать только, если:
- master # ветка master
- tags # может быть инициировано тэгом
changes:
- backend/* # при изменении файлов в каталоге backend
- frontend/* # при изменении файлов в каталоге frontend
except: # не срабатывать, если:
refs: # refs: - значение по умолчанию, можно не писать.
- schedules # этап вызван по расписанию
- triggers # или по триггеру (через вызов API)
variables:
- $CI_COMMIT_MESSAGE =~ /skip tests/ # или сообщение коммита содержит текст
release_a:
stage: release
needs: # зависимость задач друг от друга, в веб-интерфейсе GitLab есть их визуальное представление
- build_a
release_b:
stage: release
slack_notify_build_b:
stage: notify
only:
changes:
- backend/*
- frontend/*
script:
- |
curl -X POST -H 'Content-type: application/json' \
--data "{\"text\":\":building_construction: *$CI_PROJECT_NAME* ($CI_COMMIT_REF_NAME) backend - <$CI_API_V4_URL/projects/$CI_PROJECT_ID/jobs/artifacts/$CI_COMMIT_REF_NAME/download?job=build-backend-code-job&private_token=$ART_TOKEN|:package:>\"}" \
https://hooks.slack.com/services/XXXXXXXXX/XXXXXXXXXX/w8ydugdiug2p938
needs:
- build_b
Релиз - это скомпилированный и упакованный исходный код + файл json, содержащий информацию о выпуске. Дополнительно можно добавить описание и прочее, например, ссылку на объект в хранилище артефактов. Можно использовать встроенное хранилище (Package Registry), а можно другое, например, Nexus.
Использование спецсимволов в строке скрипта: https://docs.gitlab.com/ee/ci/yaml/script.html (как экранировать символы, брать строку в кавычки и т. д.)\\
Переменные: https://docs.gitlab.com/ee/ci/variables/
Средство для проверки синтаксиса внутри Гитлаба - CI Lint.
Токен можно сгенерировать как для пользователя в целом (Edit profile -> Access tokens), так и для отдельного репозитория (Settings -> Access tokens).
==== Heredoc и перенос строк ====
''- |-'' сохраняет переносы строк. Удобно для heredoc, if/else и т. п.\\
''- > '' превращает переносы строк в пробелы. Удобно для простого переноса длинной строки.
release:
image: node:12-stretch-slim
stage: release
before_script:
- apt-get update && apt-get install -y curl git jq
script:
- |-
PAYLOAD=$(cat << JSON
{
"branch": "master",
"commit_message": "some commit message",
"actions": [
{
"action": "create",
"file_path": "foo/bar",
"content": "some content"
}
]
}
JSON
)
- >
curl -X POST https://requestbin.io/1f84by61
--header 'Content-Type: application/json; charset=utf-8'
--data-binary "$PAYLOAD"
when: manual
only:
- /^release-.*$/
https://forum.gitlab.com/t/is-it-possible-to-use-a-here-document-from-within-a-multiline-command-in-gitlab-ci-yml/39329/2
===== Задачи (jobs) =====
Могут являться частью этапа (stage). В рамках одного этапа задачи выполняются одновременно. Этап не начинается, пока не закончится предыдущий. Зависимость задач от выполнения этапов:
stages:
- test
- build
- push
- deploy
lint-test:
stage: test
before_script:
- echo "prepare lint test"
script:
- echo "lint test"
after_script:
- echo "cleaing up lint test data"
unit-test:
stage: test
before_script:
- echo "prepare unit test"
script:
- echo "unit test"
after_script:
- echo "cleaing up unit test data"
build-windows:
stage: build
script:
# задача закончится ошибкой, и задача push-windows и deploy будут пропущены
- wrong-echo "build windows image"
- echo "tag windows image"
build-linux:
stage: build
script:
- echo "build linux image"
- echo "tag linux image"
push-windows:
stage: push
needs:
- build-windows
script:
- echo "log in to repository"
- echo "push windows image"
push-linux:
stage: push
needs:
- build-linux
script:
- echo "log in to repository"
- echo "push linux image"
deploy:
stage: deploy
script:
- echo "deploy"
{{:service:pasted:20220819-062835.png}}
Можно вызвать скрипт, но сначала его нужно сделать исполняемым:
script:
- chmod +x ./scriptfile.sh
- ./scriptfile.sh
Если скрипт что-то создал в каталоге на раннере, то можно это не удалять, т. к. пайплайн запускается каждый раз по новой и не использует старого окружения.
==== only/except ====
Определяют, при каких условиях задача выполняется.
# выполнять только в ветке main
job:
only:
- main
==== Workflow rules ====
''workflow'' определяет глобальные настройки пайплайна. Положим, чтобы не добавлять в каждую задачу ''only:'', можно задать это глобально:
workflow:
rules:
# если ветка не main и вызывается не с помощью merge/pull request, то
- if: $CI_COMMIT_BRANCH != "main" && $CI_PIPELINE_SOURCE != "merge_request_event"
when: never # никогда не выполнять
- when: always # в остальных случаях - выполнять
''$CI_COMMIT_BRANCH'' и ''$CI_PIPELINE_SOURCE'' - [[https://docs.gitlab.com/ee/ci/variables/predefined_variables.html|встроенные переменные]].
==== Переменные ====
Помимо встроенных переменных, есть произвольные.\\
Настройка: Settings (внутри проекта) -> CI/CD -> Variables.
Тип переменной может быть file. Например, можно занести туда конфигурационный файл для какого-либо сервиса. При вызове переменной в ней содержится полный путь к этому файлу, например,
# так выводится путь
echo "$CONF_FILE"
/builds/user/projectname.tmp/CONF_FILE
# а так - содержимое
cat $CONF_FILE
Можно задать переменные и внутри пайплайна
variables:
image_repo: docker.io/id/app
image_tag: v2.0
Если блок ''variables:'' находится внутри задачи, то они действуют только внутри этой задачи. Чтобы переменные действовали во всех задачах пайплайна, блок ''variables:'' нужно перенести на верхний уровень (вровень с самими задачами).
==== Использование в Dockerfile образов из локального реестра образов ====
В пайплайне нужно добавить ''%%--build-arg CI_REGISTRY_IMAGE=${CI_REGISTRY_IMAGE}%%'' в команду сборки.
build:
stage: build
image: docker:20.10.12-dind-rootless
before_script:
- until docker info; do sleep 1; done
- docker login -u $CI_REGISTRY_USER -p $CI_REGISTRY_PASSWORD $CI_REGISTRY
script:
- cd backend
- >
docker build
--build-arg VERSION=$VERSION
--build-arg CI_REGISTRY_IMAGE=${CI_REGISTRY_IMAGE}
--tag $CI_REGISTRY_IMAGE/backend:$CI_COMMIT_SHA
.
- docker push $CI_REGISTRY_IMAGE/backend:$CI_COMMIT_SHA
Dockerfile:
ARG CI_REGISTRY_IMAGE=${CI_REGISTRY_IMAGE}
FROM ${CI_REGISTRY_IMAGE}/node:16.18-alpine AS builder
==== Кэш ====
Хранение файлов, которые могут быть использованы в соседней задаче на раннере повторно вместо того, чтобы каждый раз качать их из интернета или с сервера, например, npm/ruby/python/go-пакеты.
Артефакты помещаются на сервер, кэш хранится на раннере. Если раннеров много, нужно настраивать распределённый кэш (distributed caching), хранящийся в S3. Хоть это всё равно удалённый сервер, это более эффективно, чем качать из интернета, и качается один zip-файл вместо множества мелких.
Настраивается соответствующим разделом в задаче. Если не задать ключ (имя кэша), то он будет называться default. Все задачи, где совпадает ключ, будут использовать один и тот же кэш.
run_unit_tests:
...
cache:
key: "npmdeps_$CI_COMMIT_REF_NAME"
paths:
- app/node_modules
policy: pull-push
"Pull-push" - значение по умолчанию, его можно не указывать, т. е., кэш используется и потом обновляется при необходимости. Если есть какая-то параллельно выполняющаяся задача, использующая тот же кэш, то политику нужно отрегулировать, чтобы обе задачи не пытались писать одни и те же файлы в кэш. В одной из задач ставится ''policy: pull'', чтобы она только использовала кэш без его обновления. Иногда, в больших пайплайнах, бывает выделенная задача для создания кэша, там тогда нужно ставить политику в ''push''.
Команду ''npm install'' всё равно нужно оставить, чтобы отсутствие кэша или какие-то проблемы с ним не влияли на общую работу пайплайна.
Если кэш создаётся с помощью Docker executor, то нужно настраивать volume на раннере, чтобы кэш не уничтожился вместе с контейнером. Путь в ''cache_dir'' и в ''volumes'' должен быть одним и тем же.
[[runners]]
...
executor = "docker"
cache_dir = "/cache"
...
[runners.docker]
volumes = ["/cache"]
Разница между первым и вторым запуском пайплайна. Первый раз кэш формировался, второй - использовался.\\
{{:service:pasted:20230530-081539.png}}
Почистить кэш можно кнопкой "Clear runner caches" в CI/CD -> Pipelines. "Почистить" в этом случае значит просто не использовать кэш в следующих запусках пайплайна, физически кэш не удаляется. [[https://docs.gitlab.com/ee/ci/caching/#where-the-caches-are-stored|Где хранится кэш]]?
[[https://docs.gitlab.com/ee/ci/caching/|Справка по кэшу]]
==== Шаблоны задач ====
Идут в комплекте с Гитлабом и могут быть включены в пайплайн в любом проекте, например, SAST (тест кода на безопасность).
Шаблоны: https://gitlab.com/gitlab-org/gitlab/-/tree/master/lib/gitlab/ci/templates
Включить SAST:
sast:
stage: test
tags:
- remote
- docker
include:
template: Jobs/SAST.gitlab-ci.yml
Можно включить до 100 шаблонов. Местоположение раздела ''include:'' в пайплайне неважно.
Если в проекте ещё нет пайплайна, то в разделе CI/CD -> Pipelines можно выбрать шаблон для быстрого его создания.
===== Deploy =====
- На сервере, куда будет ставиться приложение, сгенерить ключи SSH: ''%%ssh-keygen -t ed25519 -C "dev-server"%%''
- Засунуть закрытый ключ в переменную File на Гитлабе (в конце должна быть пустая строка!)
- Открытый ключ добавить на dev-server в ''~/.ssh/authorized_keys'' пользователя, под которым будет деплой.
- Добавить сертификат git.ca.crt в доверенные, так же как на раннере: ''nano /etc/ssl/certs/git.ca.crt # вставить содержимое сертификата с сервера''
- В пайплайне примерно следующее
deploy_to_dev:
stage: deploy
tags:
- remote
- shell
before_script:
- chmod 400 $SSH_PRIVATE_KEY
script:
- ssh -o StrictHostKeyChecking=no -i $SSH_PRIVATE_KEY ssu@$DEV_SERVER_HOST "
docker login -u $CI_REGISTRY_USER -p $CI_REGISTRY_PASSWORD $CI_REGISTRY &&
docker run -d -p 3000:3000 $IMAGE_NAME:$IMAGE_TAG
"
Вышеприведённый вариант с Докером неудобен, т. к. контейнер не заменяется, и при повторном запуске пайплайна будет ошибка, т. к. порт занят. Эту проблему решает docker-compose.
deploy_to_dev:
stage: deploy
tags:
- remote
- shell
before_script:
- chmod 400 $SSH_PRIVATE_KEY
- scp -o StrictHostKeyChecking=no -i $SSH_PRIVATE_KEY ./docker-compose.yaml ssu@$DEV_SERVER_HOST:~
script:
- ssh -o StrictHostKeyChecking=no -i $SSH_PRIVATE_KEY ssu@$DEV_SERVER_HOST "
export IMAGE_NAME=$IMAGE_NAME &&
export IMAGE_TAG=$IMAGE_TAG &&
docker login -u $CI_REGISTRY_USER -p $CI_REGISTRY_PASSWORD $CI_REGISTRY &&
docker-compose down &&
docker-compose up -d
"
environment:
name: development
url: $DEV_ENDPOINT
Так как сервер, где запускается контейнер, ничего не знает об окружении, надо пробрасывать значения переменных через ''export'' и копировать ''docker-compose.yaml'' по SSH. CСоответственно, в docker-compose.yaml можно сослаться на переменные
version: "3.3"
services:
app:
image: $IMAGE_NAME:$IMAGE_TAG
ports:
- 3000:3000
SSH keys, документация: https://docs.gitlab.com/ee/ci/ssh_keys/
==== Автоверсии ====
Канон - major.minor.patch. Здесь в примере npm и версия находится в файле app/version.json, откуда она будет браться:
{
"name": "bootcamp-node-project",
"version": "1.0",
..
},
Недостающая часть добавляется из ID пайплайна. ''jd'' надо поставить на воркере, если его там нет. Полученный файл передаётся через артефакт.
build_image:
...
before_script:
- export PACKAGE_JSON_VERSION=$(cat app/package.json | jq -r .version)
- export VERSION=$PACKAGE_JSON_VERSION.$CI_PIPELINE_IID
- echo $VERSION > version-file.txt
artifacts:
paths:
- version-file.txt
Артефакты автоматически доступны в последующих этапах, но не между задачами внутри одного этапа. Чтобы сделать артефакт доступным для соседней задачи, нужно настроить dependencies/needs.\\
Needs ждёт выполнения задач, указанных там, а dependencies берёт артефакты из указанных задач. Если указаны и needs, и dependencies одновременно, то needs должен содержать задачи, перечисленные в dependencies, иначе работать не будет. В то же время, если указать только needs, артефакты будут скачаны из задач, перечисленных там, поэтому нет необходимости указывать dependencies, если они совпадают с needs.
push_image:
...
needs:
- build_image
before_script:
- export VERSION=$(cat version-file.txt)
- docker login -u $CI_REGISTRY_USER -p $CI_REGISTRY_PASSWORD $CI_REGISTRY
script:
- docker push $IMAGE_NAME:$VERSION
Чтобы заблокировать скачивание артефактов, например, на этапе последующего тестирования, нужно указать пустой массив:
test_dev:
stage: deploy
dependencies: []
script:
- echo "testing dev"
В специфическом случае с npm можно провернуть вариант с .env вместо выгрузки артефактов. Это файл "ключ-значение". В блоке артефактов указывается reports -> dotenv.
build_image:
...
before_script:
- export PACKAGE_JSON_VERSION=$(cat app/package.json | jq -r .version)
- export VERSION=$PACKAGE_JSON_VERSION.$CI_PIPELINE_IID
- echo "VERSION=$VERSION" > build.env
- echo "DEVELOPER=Vasya" >> build.env
artifacts:
reports:
dotenv: build.env
После этого последующие соседние задачи (c соответствующими dependencies/needs) будут уже знать про эти переменные, ничего указывать не надо.
==== Развёртывание на несколько окружений ====
^dev -> ^staging -> ^prod ^
|Functional tests\\ Integration tests\\ SAST tests |Performance tests\\ DAST tests | |
Для примера развёртывание будет на одну и ту же машину с использованием разных портов. Для начала нужно задать переменную для порта в докер-композе, чтобы управлять ей из пайплайна.
ports:
- ${APP_PORT}:3000
Задать переменную в пайплайне в задаче deploy_to_dev: ''export APP_PORT=3000'', а также ''export COMPOSE_PROJECT_NAME=dev'', чтобы задать имя проекта, иначе ''docker-compose down'' в другом деплое будет останавливать один и тот же контейнер (файл Докер-композ используется ведь тот же самый, и имя по умолчанию генерируется одинаковое).
Ну а затем копируется задача deploy_to_dev в deploy_to_staging и там меняется всё, что нужно. Создаются этапы deploy_dev и deploy_staging, меняются/дописываются переменные. Перед deploy_to_staging должна стоять задача каких-нибудь functional tests, чтобы, если тесты не прошли, развётрывание на тестовое окружение уже не шло. Чтобы не плодить этапов, эти тесты могут входить в этап deploy_dev с ''needs: - deploy_to_dev''.
Чтобы не копировать практически одинаковые задачи деплоя много раз, нужно сделать //шаблон задачи.//
Положим, взяли задачу deploy_to_dev, поменяли название (точка спереди - чтобы эта "задача" не выполнялась), удалён ''stage:'' и все параметры, которые меняются от одного окружения к другому ($SSH_PRIVATE_KEY), указываются в блоке ''variables:'' как задаваемые извне.
.deploy:
tags:
- remote
- shell
variables:
SSH_PRIVATE_KEY: ""
SERVER_HOST: ""
DEPLOY_ENV: ""
APP_PORT: ""
ENDPOINT: ""
before_script:
- chmod 400 $SSH_PRIVATE_KEY
- export VERSION=$(cat version-file.txt)
- scp -o StrictHostKeyChecking=no -i $SSH_PRIVATE_KEY ./docker-compose.yaml ubuntu@$SERVER_HOST:~
script:
- ssh -o StrictHostKeyChecking=no -i $SSH_PRIVATE_KEY ubuntu@$SERVER_HOST "
export IMAGE_NAME=$IMAGE_NAME &&
export IMAGE_TAG=$VERSION &&
export APP_PORT=$APP_PORT &&
export COMPOSE_PROJECT_NAME=$DEPLOY_ENV &&
docker login -u $CI_REGISTRY_USER -p $CI_REGISTRY_PASSWORD $CI_REGISTRY &&
docker-compose down &&
docker-compose up -d
"
environment:
name: $DEPLOY_ENV
url: $ENDPOINT
А потом можно ссылаться на этот шаблон, задавая недостающие параметры и значения переменных. Имена переменных лучше делать разными, например, так работать не будет: ''SSH_PRIVATE_KEY: $SSH_PRIVATE_KEY'', если ''$SSH_PRIVATE_KEY'' имеет тип File. В этом случае при присвоении вместо пути к файлу в переменную будет пихаться его содержимое.
deploy_to_dev:
extends: .deploy
stage: deploy_dev
variables:
SSH_KEY: $DEV_SSH_PRIVATE_KEY
SERVER_HOST: $DEV_SERVER_HOST
DEPLOY_ENV: development
APP_PORT: 3000
ENDPOINT: $DEV_ENDPOINT
deploy_to_staging:
extends: .deploy
stage: deploy_staging
variables:
SSH_KEY: $STAGING_SSH_PRIVATE_KEY
SERVER_HOST: $STAGING_SERVER_HOST
DEPLOY_ENV: staging
APP_PORT: 4000
ENDPOINT: $STAGING_ENDPOINT
===== Install =====
:!: По состоянию на 2023 год репозитории для установки закрыты для России, качать и ставить надо пакетом.\\
https://packages.gitlab.com/app/gitlab/gitlab-ce/search?q=amd64&dist=jammy (для Ubuntu 22.04)
wget --content-disposition https://packages.gitlab.com/gitlab/gitlab-ce/packages/ubuntu/jammy/gitlab-ce_16.0.5-ce.0_amd64.deb/download.deb
dpkg -i gitlab-ce_16.0.5-ce.0_amd64.deb
DockerHub: https://hub.docker.com/u/gitlab\\
Install on Docker: https://docs.gitlab.com/ee/install/docker.html\\
Private CI/CD using Docker: https://oramind.com/private-cicd-using-gitlab-docker/
https://packages.gitlab.com/gitlab/gitlab-ce\\
https://docs.gitlab.com/ee/update/package/index.html#upgrade-using-a-manually-downloaded-package
==== Runner ====
Скачать: https://gitlab.com/gitlab-org/gitlab-runner/-/releases\\
Установить:
curl -LJO "https://gitlab-runner-downloads.s3.amazonaws.com/latest/deb/gitlab-runner_amd64.deb"
dpkg -i gitlab-runner_amd64.deb
Регистрация: ''gitlab-runner register''
# Регистрировать можно несколько раз, если нужно несколько видов раннера на одной и той же машине.
# URL и token берутся из Settings -> CI/CD -> Runners
URL='http://git.example.com/'
REG_TOKEN='GD1358941-xY4DE8k7Mr3dffgLLun'
gitlab-runner register \
--url $URL \
--registration-token $REG_TOKEN \
--executor shell \
--tag-list "shell" \
--description t-docker2
gitlab-runner register \
--url $URL \
--registration-token $REG_TOKEN \
--executor docker \
--tag-list "docker" \
--docker-image "alpine:3.18" \
--description t-docker2
# Перезапустить сервис, если раннер не готов сразу
systemctl restart gitlab-runner
# Для докера нужно добавить пользователя gitlab-runner в группу docker
sudo usermod -aG docker gitlab-runner
reboot
:!: Если задачи застревают (stack), надо поставить галку Run untagged jobs в Admin -> Settings -> CI/CD -> Runners -> Свойства раннера. Но лучше прописывать тэги раннера в задачах пайплайна.
https://docs.gitlab.com/runner/install/linux-manually.html\\
https://docs.gitlab.com/runner/register/
==== Container registry ====
* Package registry - для пакетов поддерживаемых менеджеров пакетов (Maven, npm, NuGet,PyPI, Rube gems, etc)
* Container registry - для хранения контейнеров
* Terraform modules (Infrastructure registry) - пакеты IaC
Здесь:
- Gitlab был установлен из пакета (Omnibus GitLab installations)
- Тот же домен, порт 5050
- Сертификат самоподписанный
CRT_DIR=/etc/gitlab/ssl
CN=git.example.com
PORT=5050
mkdir -p -m 700 $CRT_DIR
cd $CRT_DIR
# Сгенерить сертификаты
openssl genrsa -out $CN.ca.key 2048
openssl req -new -x509 -days 36500 -key $CN.ca.key -subj "/C=CN/ST=GD/L=SZ/O=$CN/CN=$CN Root CA" -out $CN.ca.crt
openssl req -newkey rsa:2048 -nodes -keyout $CN.key -subj "/C=CN/ST=GD/L=SZ/O=$CN/CN=$CN" -out $CN.csr
openssl x509 -req -extfile <(printf "subjectAltName=DNS:$CN") -days 36500 -in $CN.csr -CA $CN.ca.crt -CAkey $CN.ca.key -CAcreateserial -out $CN.crt
chmod 600 $CRT_DIR/$CN.*
###################################
# В конфиге /etc/gitlab/gitlab.rb
###################################
# Включить SSL для самого Гитлаба
sed -i "/[^_]external_url /c external_url 'https://$CN'
/nginx\['ssl_certificate'\] /c nginx['ssl_certificate'] = \"/etc/gitlab/ssl/$CN.crt\"
/nginx\['ssl_certificate_key'\] /c nginx['ssl_certificate_key'] = \"/etc/gitlab/ssl/$CN.key\"
" /etc/gitlab/gitlab.rb
# Включить реестр контейнеров
sed -i "/registry_external_url /c registry_external_url 'https://$CN:$PORT'
/registry_nginx\['enable'\] /c registry_nginx['enable'] = true
/registry_nginx\['listen_port'\] /c registry_nginx['listen_port'] = $PORT
" /etc/gitlab/gitlab.rb
# Перечитать настройки
sudo gitlab-ctl reconfigure
https://docs.gitlab.com/ee/administration/packages/container_registry.html#configure-container-registry-under-an-existing-gitlab-domain
На раннере:
# Добавить сертификат git.ca.crt в доверенные
nano /etc/ssl/certs/git.ca.crt # вставить содержимое сертификата с сервера
update-ca-certificates
# После этого можно регистрировать раннер
Если не добавить в доверенные на уровне самой системы, надо будет париться с регистрацией раннера на сервере и с докером:
### Let Docker accept your self-signed certificate
# Per default, Docker will not accept your self-signed certificate. You need to create a folder with your CA in order to make Docker aware that
# your certificate is valid. For that reason, you create a folder of your trusted Docker registry and copy your CA into the folder.
# If you will not copy the CA in the folder. You will receive the following error:
# Registry fails with x509 certificate signed by unknown authority
sudo mkdir -p /etc/docker/certs.d/git.example.com:5050
sudo cp /etc/gitlab/ssl/git.example.com.ca.crt /etc/docker/certs.d/git.example.com:5050/
### Let GitLab runners accept your self-signed certificate
# The same error will occur when you want to register your GitLab runner:
# Post “https://git.example.com/api/v4/runners”: x509: certificate signed by unknown authority
# Please use the following command in order to register your GitLab runner successfully:
sudo gitlab-runner register --tls-ca-file="/etc/gitlab/ssl/git.example.com.ca.crt"
Справка: ''/help/administration/packages/container_registry.md''\\
[[https://gitlab.com/gitlab-org/gitlab-runner/-/issues/28841#note_1004765355|x509: certificate relies on legacy Common Name field, use SANs instead]]\\
[[https://ubuntu.com/server/docs/security-trust-store|Installing a root CA certificate in the trust store]]\\
[[https://beye.blog/gitlab-server-with-a-self-signed-certificate-and-embedded-docker-registry/|GitLab server with a self-signed certificate and embedded docker registry]]
===== Подход к работе с микросервисами =====
Если микросервисов в приложении больше чем один, возникает 2 подхода к хранению кода
- Один репозиторий (monorepo) - для каждого микросервиса используется свой каталог. Минусы - опасность написания связанного кода между микросервисами, большой объём кода, сложность управления пайплайнами, одним коммитом можно поломать всё сразу. Подходит для маленьких проектов и одной команды.
- Несколько (polyrepo) - каждый микросервис находится в своём репозитории. В Гитлабе связанные проекты можно объединить в группу.
==== Сеть ====
Чтобы контейнеры могли взаимодействовать, их нужно поместить в одну сеть. Для этого в докер-композ-файле указывается определённая сеть, которая должна уже присутствовать (параметр external)
version: "3.3"
services:
app:
image: ${DC_IMAGE_NAME}:${DC_IMAGE_TAG}
ports:
- ${DC_APP_PORT}:${DC_APP_PORT}
networks:
- micro_service
networks:
micro_service:
external:
name: micro_service
:!: Параметр ''name:'' в опциях сети предписывает не прибавлять к её названию имя проекта, т. е. будет именно ''micro_service'', а не ''project_micro_service''.
А в пайплайне в шаблоне задачи деплоя прописать создание этой сети и указание, чтобы задача не обламывалась, если сеть уже есть:
docker network create micro_service || true &&
docker-compose down &&
docker-compose up -d
++++ Полный листинг .gitlab-ci.yml (monorepo) |
workflow:
rules:
- if: $CI_COMMIT_BRANCH != "main" && $CI_PIPELINE_SOURCE != "merge_request_event"
when: never
- when: always
variables:
DEPLOYMENT_SERVER_HOST: "server"
APP_ENDPOINT: http://server:3000
stages:
- build
- deploy
.build:
stage: build
tags:
- remote
- shell
variables:
MICRO_SERVICE: ""
SERVICE_VERSION: ""
before_script:
- cd $MICRO_SERVICE
- export IMAGE_NAME=$CI_REGISTRY_IMAGE/microservice/$MICRO_SERVICE
- export IMAGE_TAG=$SERVICE_VERSION
script:
- docker login -u $CI_REGISTRY_USER -p $CI_REGISTRY_PASSWORD $CI_REGISTRY
- docker build -t $IMAGE_NAME:$IMAGE_TAG .
- docker push $IMAGE_NAME:$IMAGE_TAG
build_frontend:
extends: .build
variables:
MICRO_SERVICE: frontend
SERVICE_VERSION: "1.3"
only:
changes:
- "frontend/**/*"
build_products:
extends: .build
variables:
MICRO_SERVICE: products
SERVICE_VERSION: "1.8"
only:
changes:
- "products/**/*"
build_shopping_cart:
extends: .build
variables:
MICRO_SERVICE: shopping-cart
SERVICE_VERSION: "2.1"
only:
changes:
- "shopping-cart/**/*"
.deploy:
stage: deploy
tags:
- remote
- shell
variables:
MICRO_SERVICE: ""
SERVICE_VERSION: ""
APP_PORT: ""
before_script:
- chmod 400 $SSH_PRIVATE_KEY
- export IMAGE_NAME=$CI_REGISTRY_IMAGE/microservice/$MICRO_SERVICE
- export IMAGE_TAG=$SERVICE_VERSION
script:
- scp -o StrictHostKeyChecking=no -i $SSH_PRIVATE_KEY ./docker-compose.yaml ubuntu@$DEPLOYMENT_SERVER_HOST:~
- ssh -o StrictHostKeyChecking=no -i $SSH_PRIVATE_KEY ubuntu@$DEPLOYMENT_SERVER_HOST "
docker login -u $CI_REGISTRY_USER -p $CI_REGISTRY_PASSWORD $CI_REGISTRY &&
export COMPOSE_PROJECT_NAME=$MICRO_SERVICE &&
export DC_IMAGE_NAME=$IMAGE_NAME &&
export DC_IMAGE_TAG=$IMAGE_TAG &&
export DC_APP_PORT=$APP_PORT &&
docker network create micro_service || true &&
docker-compose down &&
docker-compose up -d"
environment:
name: development
url: $APP_ENDPOINT
deploy_frontend:
extends: .deploy
variables:
MICRO_SERVICE: frontend
SERVICE_VERSION: "1.3"
APP_PORT: 3000
only:
changes:
- "frontend/**/*"
deploy_products:
extends: .deploy
variables:
MICRO_SERVICE: products
SERVICE_VERSION: "1.8"
APP_PORT: 3001
only:
changes:
- "products/**/*"
deploy_shopping_cart:
extends: .deploy
variables:
MICRO_SERVICE: shopping-cart
SERVICE_VERSION: "2.1"
APP_PORT: 3002
only:
changes:
- "shopping-cart/**/*"
++++
==== Polyrepo ====
Подготовка:
- Создаётся **группа,** куда добавляются все нужные проекты. Удобство группы в том, что можно задавать переменные, раннеры и т. п. для всей группы, а не для каждого проекта.
- Раннер для группы регистрируется отдельно в настройках группы. Потом нужно в группе настроить переменную для закрытого ключа SSH сервера деплоя.
Дальше в каждую репу копируются docker-compose.yml и .gitlab-ci.yml и редактируются соответственно. Шаблоны задач можно переделать в реальные задачи, т. к. сервис в репе только один, отредактировав их и убрав задачи, вызывающие эти шаблоны, удалить из них блоки переменных, ожидаемых извне. Убрать детекты срабатывания в подкаталогах и заходы в подкаталоги из задач.
++++ Полный листинг .gitlab-ci.yml (polyrepo) сервиса frontend |
workflow:
rules:
- if: $CI_COMMIT_BRANCH != "main" && $CI_PIPELINE_SOURCE != "merge_request_event"
when: never
- when: always
variables:
DEPLOYMENT_SERVER_HOST: "server"
APP_ENDPOINT: http://server:3000
MICRO_SERVICE: frontend
SERVICE_VERSION: "1.3"
APP_PORT: 3000
stages:
- build
- deploy
build:
stage: build
tags:
- group
- shell
before_script:
- export IMAGE_NAME=$CI_REGISTRY_IMAGE/microservice/$MICRO_SERVICE
- export IMAGE_TAG=$SERVICE_VERSION
script:
- docker login -u $CI_REGISTRY_USER -p $CI_REGISTRY_PASSWORD $CI_REGISTRY
- docker build -t $IMAGE_NAME:$IMAGE_TAG .
- docker push $IMAGE_NAME:$IMAGE_TAG
deploy:
stage: deploy
tags:
- group
- shell
before_script:
- chmod 400 $SSH_PRIVATE_KEY
- export IMAGE_NAME=$CI_REGISTRY_IMAGE/microservice/$MICRO_SERVICE
- export IMAGE_TAG=$SERVICE_VERSION
script:
- scp -o StrictHostKeyChecking=no -i $SSH_PRIVATE_KEY ./docker-compose.yaml ubuntu@$DEPLOYMENT_SERVER_HOST:~
- ssh -o StrictHostKeyChecking=no -i $SSH_PRIVATE_KEY ubuntu@$DEPLOYMENT_SERVER_HOST "
docker login -u $CI_REGISTRY_USER -p $CI_REGISTRY_PASSWORD $CI_REGISTRY &&
export COMPOSE_PROJECT_NAME=$MICRO_SERVICE &&
export DC_IMAGE_NAME=$IMAGE_NAME &&
export DC_IMAGE_TAG=$IMAGE_TAG &&
export DC_APP_PORT=$APP_PORT &&
docker network create micro_service || true &&
docker-compose down &&
docker-compose up -d"
environment:
name: development
url: $APP_ENDPOINT
++++
Проблема множества репозиториев в том, что одна и та же конфигурация повторяется из раза в раз, а если работают разные команды разрабов, то эти конфигурации начинают разниться в подходах, появляются разные тесты и т. п. и управлять этим барахлом становится тяжело. Для решения этой проблемы есть шаблоны задач, но не в том же пайплайне, а вынесенные в отдельный проект, на которые можно ссылаться как на шаблон SAST, к примеру.
==== Шаблоны задач ====
Надо вынести повторяющиеся задачи в отдельные .yml-файлы, например
# Можно также вписать блок с переменными, которые нужно передать извне для большей наглядности, но это необязательно
variables:
MICRO_SERVICE: ""
SERVICE_VERSION: ""
build:
stage: build
before_script:
- export IMAGE_NAME=$CI_REGISTRY_IMAGE/microservice/$MICRO_SERVICE
- export IMAGE_TAG=$SERVICE_VERSION
script:
- docker login -u $CI_REGISTRY_USER -p $CI_REGISTRY_PASSWORD $CI_REGISTRY
- docker build -t $IMAGE_NAME:$IMAGE_TAG .
- docker push $IMAGE_NAME:$IMAGE_TAG
В файле шаблона всё должно быть параметризовано по-максимуму. В основном пайплайне нужно сослаться на файлы шаблона, а в самой задае добавить только те разделы, которых нет в шаблоне.\\
Местоположение раздела ''include:'' в пайплайне не имеет значения.\\
:!: Если нужно что-то добавить в раздел, который уже есть в шаблоне (например, before_script), то придётся писать его полностью, т. к. при наличии раздела в основном пайплайне раздел шаблона полностью заменяется.
include:
- local: '.build-template.yml'
- local: '.deploy-template.yml'
...
build:
tags:
- group
- shell
before_script:
- echo "Добавленные действия, а дальше то, что уже есть в шаблоне, иначе не заработает"
- export IMAGE_NAME=$CI_REGISTRY_IMAGE/microservice/$MICRO_SERVICE
- export IMAGE_TAG=$SERVICE_VERSION
Файлы шаблонов можно вынести в отдельный репозиторий, например, ci-templates. В этом случае ссылаться на эти шаблоны нужно так:
include:
- project: mymicroservice-cicd/ci-templates
ref: main
file:
- build.yml
- deploy.yml
''ref: main'' указывает на ветку.\\
Т. к. репозиторий в той же группе, указывается ''группа/репозиторий''. Если бы он был вне группы, надо было бы указывать ''имя пользователя/репозиторий''.
===== Mail =====
If you would rather send application email via an SMTP server instead of via Sendmail or Postfix, add the following configuration information to ''/etc/gitlab/gitlab.rb'' and run ''gitlab-ctl reconfigure''.\\
https://docs.gitlab.com/omnibus/settings/smtp.html
gitlab_rails['smtp_enable'] = true
gitlab_rails['smtp_address'] = "mail.example.com"
gitlab_rails['smtp_port'] = 25
gitlab_rails['smtp_domain'] = "example.com"
gitlab_rails['smtp_enable_starttls_auto'] = true
gitlab_rails['smtp_openssl_verify_mode'] = 'none'
### Email Settings
gitlab_rails['gitlab_email_from'] = 'git@example.com'
gitlab_rails['gitlab_email_display_name'] = 'Git'
Проверка отправки писем из консоли
gitlab-rails console -e production
Notify.test_email('user@example.com', 'Hello World', 'This is a test message').deliver_now
===== Решение проблем =====
==== 502 - Whoops, GitLab is taking too much time to respond is normal during GitLab startup and goes away after couple minutes ====
После перезапуска возникает ошибка 502. Нужно подождать, т. к. Gitlab запускается небыстро.
См. статус: ''gitlab-ctl''
==== OpenSSL::SSL::SSLError (hostname does not match the server certificate) ====
В конфигурацию /etc/gitlab/gitlab.rb добавить
gitlab_rails['smtp_openssl_verify_mode'] = 'none'
# Затем
gitlab-ctl reconfigure
gitlab-rake cache:clear RAILS_ENV=production
https://gitlab.com/gitlab-org/gitlab-foss/-/issues/446
==== Got permission denied while trying to connect to the Docker daemon socket at unix:///var/run/docker.sock ====
Возникает при сборке образа на раннере (shell executor). Решение:
usermod -aG docker gitlab-runner
service docker restart
https://gitlab.com/gitlab-org/gitlab-runner/-/issues/3492
==== CI_REGISTRY error during connect status 255: Permission denied, please try again ====
Раннер не может подключиться к реестру контейнеров:
$ echo "$CI_REGISTRY_PASSWORD" |docker login -u $CI_REGISTRY_USER --password-stdin $CI_REGISTRY
error during connect: Post "http://docker.example.com/v1.24/auth": command [ssh -l cicd -- 10.1.0.138 docker system dial-stdio] has exited with exit status 255,
please make sure the URL is valid, and Docker 18.09 or later is installed on the remote host: stderr=Permission denied, please try again.
Permission denied, please try again.
cicd@10.1.0.138: Permission denied (publickey,password).
Решение: не раннере удалить всё в каталоге ''/home/gitlab-runner''.
===== Безопасность =====
Shifting Security Left - GitLab DevSecOps Overview: https://www.youtube.com/watch?v=XnYstHObqlA\\
Гитлаб предлагает сканеры безопасности:
* [[https://docs.gitlab.com/ee/user/application_security/sast/|Static Application Security Testing]] (SAST) - обнаружение проблем в самом исходном коде, например, пользовательский ввод позволяет выполнить инъекцию команды. [[https://www.youtube.com/watch?v=8sOjvlkl8QY|Демо]]
* [[https://docs.gitlab.com/ee/user/application_security/secret_detection/|Secret Detection]] - обнаружение вбитых паролей в исходном коде. [[https://www.youtube.com/watch?v=W2tjcQreDwQ|Демо]]
* [[https://docs.gitlab.com/ee/user/application_security/dast/|Dynamic Application Security Testing]] (DAST) - сканирование на уязвимости через вызов приложения или API. [[https://www.youtube.com/watch?v=ywme200L7LU|Демо]]
* [[https://docs.gitlab.com/ee/user/application_security/iac_scanning/|Infrastructure-as-Code (IaC) Scanning]] - сканирование IaC-файлов (Ansible, Terraform) на уязвимости.
* [[https://docs.gitlab.com/ee/user/application_security/dependency_scanning/|Dependency Scanning]] - отчёт по уязвимостям зависимостей проекта по их версиям. [[https://www.youtube.com/watch?v=39RvTMLDszc|Демо]]
* [[https://docs.gitlab.com/ee/user/application_security/container_scanning/|Container Scanning]] - сканирование докер-образов на уязвимости. [[https://www.youtube.com/watch?v=C0jn2eN5MAs|Демо]]
* [[https://docs.gitlab.com/ee/user/application_security/api_fuzzing/|Fuzz Testing]] - шлёт на вход функций всякую ерунду на предмет неожиданных глюков. [[https://www.youtube.com/watch?v=oUHsfvLGhDk|Демо]]
Отчёты / управление:
* [[https://docs.gitlab.com/ee/user/application_security/security_dashboard/|Security Reports]] - сводный отчёт по уязвимостям. [[https://www.youtube.com/watch?v=QHQHN4luNpc|Демо]]
* [[https://docs.gitlab.com/ee/user/application_security/vulnerability_report/|Vulnerability Management]] [[https://www.youtube.com/watch?v=8SJHz6BCgXM|Демо]]
* [[https://docs.gitlab.com/ee/user/application_security/policies/|Policies]] [[https://www.youtube.com/watch?v=IZbCIKXz-wM|Демо]]
===== Прочее =====
==== Тестовый проект Gitlab from zero to hero ====
https://gitlab.com/nanuchi/mynodeapp-cicd-project
Monorepo: https://gitlab.com/nanuchi/mymicroservice-cicd
==== Example CI/CD Pipelines ====
[[https://gitlab.com/gitlab-examples/ruby-autodeploy|Ruby Auto Deploy]]\\
[[https://gitlab.com/gitlab-examples/multi-project-pipelines/simple-maven-app|Simple Maven App]]\\
[[https://www.youtube.com/watch?v=0Tc0YYBxqi4&ab_channel=GitLab|AutoDevOps]]\\
[[https://gitlab.com/gitlab-examples|More Project Examples]]\\
===== Заметки о реальных проектах =====
==== Сервис личного кабинета в Docker Swarm ====
# SSH / Context
ssh -o StrictHostKeyChecking=no -i $SSH_PRIVATE_KEY $DEPLOY_USER@$DEPLOY_HOST
docker context create remote --docker "host=ssh://cicd@host"
docker context use remote
# Вход в репозиторий
docker login -u mail@example.com -p P@ssw0rd git.example.com:5050
# Пулл для локального тестирования
docker pull git.example.com:5050/personal-account/backend-laravel/lk/backend_app:1.0
docker pull git.example.com:5050/personal-account/backend-laravel/lk/backend_nginx:1.0
# Сети надо создавать overlay и swarm, иначе не будет работать внутренний DNS
docker network create --driver=overlay --scope=swarm lk_rabbitmq
docker network create --driver=overlay --scope=swarm lk_backend
# Запуск
docker stack deploy -c ./docker-compose.yml lk
#docker stack rm lk
docker exec $(docker ps -qf name=lk_app) mkdir -p /var/www/html/storage/framework/sessions
mkdir -p /var/www/html/storage/framework/sessions
chown -R 82:82 /var/www/html/storage
version: "3.9"
services:
app: &app
image: git.example.com:5050/personal-account/backend-laravel/lk/backend_app:1.0
healthcheck:
test: ["CMD", "netstat", "-an", "|fgrep", ":9000"]
volumes:
- /docker/lk/www/storage:/var/www/html/storage
networks:
- lk_backend
- lk_rabbitmq
nginx:
image: git.example.com:5050/personal-account/backend-laravel/lk/backend_nginx:1.0
environment:
NGINX_ROOT: /var/www/html/public
NGINX_FASTCGI_PASS: app
healthcheck:
test: ["CMD", "curl", "-f", "http://localhost:80"]
ports:
- 8080:80
networks:
- lk_backend
db:
image: mysql:8.0
environment:
MYSQL_DATABASE: personal-area
MYSQL_ROOT_PASSWORD: password
MYSQL_PASSWORD: password
MYSQL_USER: personal-area-user
SERVICE_TAGS: dev
SERVICE_NAME: mysql
healthcheck:
test: ["CMD", "mysqladmin", "ping", "-h", "localhost", "-uroot", "-p$$MYSQL_ROOT_PASSWORD"]
volumes:
- /docker/lk/db:/docker-entrypoint-initdb.d
networks:
- lk_backend
rabbitmq:
image: rabbitmq:management-alpine
hostname: rabbitmq
environment:
RABBITMQ_DEFAULT_USER: admin
RABBITMQ_DEFAULT_PASS: admin
healthcheck:
test: ["CMD", "rabbitmq-diagnostics", "-q", "ping"]
ports:
- 5672:5672
- 15672:15672
volumes:
- /docker/lk/rabbitmq:/var/lib/rabbitmq
networks:
- lk_rabbitmq
queue-lk-job-broadcast: &queue
<<: *app
command: php artisan queue:work --queue=lk.job.broadcast
healthcheck:
disable: true
queue-lk-job-catalog_order:
<<: *queue
command: php artisan queue:work --queue=lk.job.catalog_order
queue-lk-job-notification:
<<: *queue
command: php artisan queue:work --queue=lk.job.notification
queue-lk-job-schedule:
<<: *queue
command: php artisan queue:work --queue=lk.job.schedule
# И ещё куча таких же, слушающих очереди
websockets:
<<: *queue
command: php artisan websockets:serve
networks:
lk_backend:
external: true
name: lk_backend
lk_rabbitmq:
external: true
name: lk_rabbitmq