bd0f276d30
Signed-off-by: Guillaume Coguiec <guillaume@logical.work>
251 lines
8.0 KiB
CUE
251 lines
8.0 KiB
CUE
// STARTING POINT: https://docs.dagger.io/1012/ci
|
|
// + ../../../.circleci/config.yml
|
|
package main
|
|
|
|
import (
|
|
"alpha.dagger.io/dagger"
|
|
"alpha.dagger.io/docker"
|
|
"alpha.dagger.io/os"
|
|
)
|
|
|
|
app_src: dagger.#Artifact
|
|
prod_dockerfile: dagger.#Input & {string}
|
|
docker_host: dagger.#Input & {string}
|
|
dockerhub_username: dagger.#Input & {string}
|
|
dockerhub_password: dagger.#Input & {dagger.#Secret}
|
|
// ⚠️ Keep this in sync with ../docker/Dockerfile.production
|
|
runtime_image_ref: dagger.#Input & {string | *"thechangelog/runtime:2021-05-29T10.17.12Z"}
|
|
prod_image_ref: dagger.#Input & {string | *"thechangelog/changelog.com:dagger"}
|
|
build_version: dagger.#Input & {string}
|
|
git_branch: dagger.#Input & {string | *""}
|
|
git_sha: dagger.#Input & {string}
|
|
git_author: dagger.#Input & {string}
|
|
app_version: dagger.#Input & {string}
|
|
build_url: dagger.#Input & {string}
|
|
// ⚠️ Keep this in sync with manifests/changelog/db.yml
|
|
test_db_image_ref: dagger.#Input & {string | *"circleci/postgres:12.6"}
|
|
test_db_container_name: "changelog_test_postgres"
|
|
|
|
// STORY #######################################################################
|
|
//
|
|
// 1. Migrate from CircleCI to GitHub Actions
|
|
// - extract existing build pipeline into Dagger
|
|
// - run the pipeline locally
|
|
// - run the pipeline in GitHub Actions
|
|
//
|
|
// 2. Pipeline is 2x quicker (110s vs 228s)
|
|
// - optimistic branching, as pipelines were originally intended
|
|
// - use our own hardware - Linode g6-dedicated-8
|
|
// - predictable runs, no queuing
|
|
// - caching is buildkit layers
|
|
//
|
|
// 3. Open Telemetry integration out-of-the-box
|
|
// - visualize all steps in Jaeger UI
|
|
|
|
// PIPELINE OVERVIEW ###########################################################
|
|
//
|
|
// app
|
|
// |
|
|
// v
|
|
// test_db_start deps ----------------------------------------\
|
|
// | | | |
|
|
// | v v v
|
|
// | deps_compile_test deps_compile_prod assets_compile
|
|
// | | | | |
|
|
// | | | \-----|
|
|
// | v v |
|
|
// \--------------> test image_prod_cache assets_digest
|
|
// | | |
|
|
// test_db_stop <---| v |
|
|
// | image_prod <----------/
|
|
// | |
|
|
// | v
|
|
// \--------------------> image_prod_tag
|
|
//
|
|
// ========================== BEFORE | AFTER | CHANGE ===================================
|
|
// Test, build & push 370s | 10s | 37.00x | https://app.circleci.com/pipelines/github/thechangelog/changelog.com/520/workflows/fbb7c701-d25a-42c1-b42c-db514cd770b4
|
|
// + app compile 220s | 100s | 2.20x | https://app.circleci.com/pipelines/github/thechangelog/changelog.com/582/workflows/65500f3d-eccc-49da-9ab0-69846bc812a7
|
|
// + deps compile 480s | 110s | 4.36x | https://app.circleci.com/pipelines/github/thechangelog/changelog.com/532/workflows/94f5a339-52a1-45ba-b39b-1bbb69ed6488
|
|
//
|
|
// Uncached ???s | 245s | ?.??x |
|
|
//
|
|
// #############################################################################
|
|
|
|
test_db_start: docker.#Command & {
|
|
host: docker_host
|
|
env: {
|
|
CONTAINER_NAME: test_db_container_name
|
|
CONTAINER_IMAGE: test_db_image_ref
|
|
}
|
|
command: #"""
|
|
docker container inspect $CONTAINER_NAME \
|
|
--format 'Container "{{.Name}}" is "{{.State.Status}}"' \
|
|
|| docker container run \
|
|
--detach --rm --name $CONTAINER_NAME \
|
|
--publish 127.0.0.1:5432:5432 \
|
|
--env POSTGRES_USER=postgres \
|
|
--env POSTGRES_DB=changelog_test \
|
|
--env POSTGRES_PASSWORD=postgres \
|
|
$CONTAINER_IMAGE
|
|
|
|
docker container inspect $CONTAINER_NAME \
|
|
--format 'Container "{{.Name}}" is "{{.State.Status}}"'
|
|
"""#
|
|
}
|
|
|
|
app_image: docker.#Pull & {
|
|
from: runtime_image_ref
|
|
}
|
|
|
|
// Put app_src in the correct path, /app
|
|
app: os.#Container & {
|
|
image: app_image
|
|
copy: "/app": from: app_src
|
|
}
|
|
|
|
// https://github.com/moby/buildkit/blob/master/frontend/dockerfile/docs/syntax.md#run---mounttypecache
|
|
deps_mount: "--mount=type=cache,id=deps,target=/app/deps/,sharing=shared"
|
|
|
|
build_test_mount: "--mount=type=cache,id=build_test,target=/app/_build/test,sharing=shared"
|
|
build_prod_mount: "--mount=type=cache,id=build_prod,target=/app/_build/prod,sharing=shared"
|
|
|
|
node_modules_mount: "--mount=type=cache,id=assets_node_modules,target=/app/assets/node_modules,sharing=shared"
|
|
|
|
deps: docker.#Build & {
|
|
source: app
|
|
dockerfile: """
|
|
FROM \(runtime_image_ref)
|
|
COPY /app/ /app/
|
|
WORKDIR /app
|
|
RUN \(deps_mount) mix deps.get
|
|
"""
|
|
}
|
|
|
|
assets_compile: docker.#Build & {
|
|
source: deps
|
|
dockerfile: """
|
|
FROM \(runtime_image_ref)
|
|
COPY /app/ /app/
|
|
WORKDIR /app/assets
|
|
RUN \(deps_mount) \(node_modules_mount) yarn install --frozen-lockfile && yarn run compile
|
|
"""
|
|
}
|
|
|
|
#deps_compile: docker.#Build & {
|
|
source: deps
|
|
dockerfile: """
|
|
FROM \(runtime_image_ref)
|
|
ARG MIX_ENV
|
|
ENV MIX_ENV=$MIX_ENV
|
|
COPY /app/ /app/
|
|
WORKDIR /app
|
|
RUN \(deps_mount) \(build_test_mount) \(build_prod_mount) mix do deps.compile, compile
|
|
"""
|
|
}
|
|
|
|
deps_compile_test: #deps_compile & {
|
|
args: MIX_ENV: "test"
|
|
}
|
|
|
|
test: docker.#Build & {
|
|
source: deps_compile_test
|
|
dockerfile: """
|
|
FROM \(runtime_image_ref)
|
|
ENV MIX_ENV=test
|
|
COPY /app/ /app/
|
|
WORKDIR /app
|
|
RUN \(deps_mount) \(build_test_mount) mix test
|
|
"""
|
|
}
|
|
|
|
test_db_stop: docker.#Command & {
|
|
host: docker_host
|
|
env: {
|
|
DEP: test.dockerfile
|
|
CONTAINER_NAME: test_db_container_name
|
|
}
|
|
command: #"""
|
|
docker container rm --force $CONTAINER_NAME
|
|
"""#
|
|
}
|
|
|
|
deps_compile_prod: #deps_compile & {
|
|
args: MIX_ENV: "prod"
|
|
}
|
|
|
|
assets_digest: docker.#Build & {
|
|
source: assets_compile
|
|
args: ONLY_RUN_AFTER_DEPS_COMPILE_PROD_OK: deps_compile_prod.args.MIX_ENV
|
|
dockerfile: """
|
|
FROM \(runtime_image_ref)
|
|
COPY /app/ /app/
|
|
ENV MIX_ENV=prod
|
|
WORKDIR /app/
|
|
RUN \(deps_mount) \(build_prod_mount) \(node_modules_mount) mix phx.digest
|
|
"""
|
|
}
|
|
|
|
image_prod_cache: docker.#Build & {
|
|
source: deps_compile_prod
|
|
dockerfile: """
|
|
FROM \(runtime_image_ref)
|
|
COPY /app/ /app/
|
|
WORKDIR /app
|
|
RUN --mount=type=cache,id=deps,target=/mnt/app/deps,sharing=shared cp -Rp /mnt/app/deps/* /app/deps/
|
|
RUN --mount=type=cache,id=build_prod,target=/mnt/app/_build/prod,sharing=shared cp -Rp /mnt/app/_build/prod/* /app/_build/prod/
|
|
"""
|
|
}
|
|
|
|
image_prod: docker.#Command & {
|
|
host: docker_host
|
|
copy: {
|
|
"/tmp/app": from: os.#Dir & {
|
|
from: image_prod_cache
|
|
path: "/app"
|
|
}
|
|
|
|
"/tmp/app/priv/static": from: os.#Dir & {
|
|
from: assets_digest
|
|
path: "/app/priv/static"
|
|
}
|
|
}
|
|
env: {
|
|
GIT_AUTHOR: git_author
|
|
GIT_SHA: git_sha
|
|
APP_VERSION: app_version
|
|
BUILD_VERSION: build_version
|
|
BUILD_URL: build_url
|
|
PROD_IMAGE_REF: prod_image_ref
|
|
}
|
|
files: "/tmp/Dockerfile": prod_dockerfile
|
|
secret: "/run/secrets/dockerhub_password": dockerhub_password
|
|
command: #"""
|
|
cd /tmp
|
|
docker build \
|
|
--build-arg APP_FROM_PATH=/app \
|
|
--build-arg GIT_AUTHOR="$GIT_AUTHOR" \
|
|
--build-arg GIT_SHA="$GIT_SHA" \
|
|
--build-arg APP_VERSION="$APP_VERSION" \
|
|
--build-arg BUILD_URL="$BUILD_URL" \
|
|
--tag "$PROD_IMAGE_REF" .
|
|
"""#
|
|
}
|
|
|
|
if git_branch == "master" {
|
|
image_prod_tag: docker.#Command & {
|
|
host: docker_host
|
|
env: {
|
|
DOCKERHUB_USERNAME: dockerhub_username
|
|
PROD_IMAGE_REF: image_prod.env.PROD_IMAGE_REF
|
|
ONLY_RUN_AFTER_TEST_OK: test.dockerfile
|
|
}
|
|
secret: "/run/secrets/dockerhub_password": dockerhub_password
|
|
command: #"""
|
|
docker login --username "$DOCKERHUB_USERNAME" --password "$(cat /run/secrets/dockerhub_password)"
|
|
docker push "$PROD_IMAGE_REF" | tee docker.push.log
|
|
echo "$PROD_IMAGE_REF" > image.ref
|
|
awk '/digest/ { print $3 }' < docker.push.log > image.digest
|
|
"""#
|
|
}
|
|
}
|