แแแแกแแ แแแฃแฎแแแแแแ, แ แแ แงแแแแแ แแจแแแแแแ แแ แแชแแก, แ แแ แแฅแแแแ แแ แแแ แแแฃแแ แฃแแ แฃแแแแแงแแคแแก แขแแกแขแแ แแแ แแแแจแแแแแแแแแแ แแ แแฃแชแแแแแแแแ, แแ แแแแ แ แแแแก แแแแแแแก แแแขแแแแขแฃแ แแ แแแแ แฎแแแก แแแแแแแแแแแจแ, แฐแแแ แแก แฃแแแ แแแแแ แกแแแ แชแแจแ แแ แแ แกแแแแแแ แแ แช แแ แแ แ แแชแแแขแ แแกแแแ แแแแฃแแแ แฃแแ แแ แแแฃแฅแขแแแแก แแแแแแแแชแแแก แแแกแแงแแแแแแแ. แแก แแแจแ, แ แแแแ แช (แฉแแแแ แกแแงแแแ แแแ) GitLab แแ JUnit. แจแแแแแกแแ แแก แฎแแ แแแแ!
แจแแกแแแแแ
แแแ แแแ แ แแแจแ, แแแแ แแแแแซแแ แแแแชแแ แแแแขแแฅแกแขแ:
- แแแแแแแแ แงแแแแ แฉแแแแ แแแแแแแชแแ แแฃแจแแแแก Kubernetes-แแ, แแแแแแฎแแแแแ แขแแกแขแแแแก แแแจแแแแแก แจแแกแแแแแแก แแแคแ แแกแขแ แฃแฅแขแฃแ แแแ.
- แจแแแ แแแแกแ แแ แแแแแแแแแแกแแแแก แฉแแแ แแแงแแแแแ
แแแ แค (แแแคแ แแกแขแ แฃแฅแขแฃแ แฃแแ แแแแแแแแแขแแแแก แแแแแกแแแ แแกแแ, แแก แแกแแแ แแแขแแแแขแฃแ แแ แแแจแแแแก, แ แแ Helm แแ แแก แฉแแ แแฃแแ). - แแ แแ แจแแแฃแแแแแ แขแแกแขแแแแก แคแแฅแขแแแ แแแ แจแแฅแแแแก แแแขแแแแแก: แฉแแแแก แจแแแแฎแแแแแจแ, แแแแแแขแ แแแแแ แฌแแ แก แขแแกแขแแแก แแ แฉแแแ แแฎแแแแ แฃแแ แฃแแแแแแงแแคแ แแแ แแแจแแแแแก (แแ แจแแกแแแแแแกแ แแแแแ แแจแแก แแ แกแแแแแแก แจแแ แฌแงแแแก แแแแฎแแแแแจแ).
แ แแแแ แ แแฅแแแแ แแแฅแแแแแแแแแก แแแแแแ แแแแแแแแแแ แแแ?
- แแแแแแแชแแแก แแแแแ - แแ แแขแแแแก แแฆแฌแแ แแก แแแแแแขแแแแแ.
- แแแแแแแแกแแ แแแแแแแชแแ Kubernetes แแแแกแขแแ แแก แชแแแแแฃแ แกแแฎแแแแ แกแแแ แชแแจแ แแ แแแแฌแงแแ แขแแกแขแแ แแแ.
- แแ แขแแคแแฅแขแแแแก แซแแแแ แแ JUnit แแแแแ แแจแแแแก แแแแแแแ GitLab-แแ.
- แแแ แ แจแแฅแแแแแ แกแแฎแแแแแแก แกแแแ แชแแก แฌแแจแแ.
แแฎแแ - แแแแฎแแ แชแแแแแแแแแ!
แ แแแฃแแแ แแแ
GitLab CI
แแแแแฌแงแแ แคแ แแแแแแขแแ .gitlab-ci.yaml
, แ แแแแแแช แแฆแฌแแ แก แแแแแแแชแแแก แแแงแแแแแแก แแ แขแแกแขแแแแก แแแจแแแแแก. แฉแแแแแแแแแแ แกแแแแแแ แแแชแฃแแแแแแ แแฆแแแฉแแแ, แแแแขแแ แแแ แกแแคแฃแซแแแแแแแ แแแแแแขแ แแแแแแขแแ แแแแ:
variables:
# ะพะฑััะฒะปัะตะผ ะฒะตััะธั werf, ะบะพัะพััั ัะพะฑะธัะฐะตะผัั ะธัะฟะพะปัะทะพะฒะฐัั
WERF_VERSION: "1.0 beta"
.base_deploy: &base_deploy
script:
# ัะพะทะดะฐะตะผ namespace ะฒ K8s, ะตัะปะธ ะตะณะพ ะฝะตั
- kubectl --context="${WERF_KUBE_CONTEXT}" get ns ${CI_ENVIRONMENT_SLUG} || kubectl create ns ${CI_ENVIRONMENT_SLUG}
# ะทะฐะณััะถะฐะตะผ werf ะธ ะดะตะฟะปะพะธะผ โ ะฟะพะดัะพะฑะฝะตะต ะพะฑ ััะพะผ ัะผ. ะฒ ะดะพะบัะผะตะฝัะฐัะธะธ
# (https://werf.io/how_to/gitlab_ci_cd_integration.html#deploy-stage)
- type multiwerf && source <(multiwerf use ${WERF_VERSION})
- werf version
- type werf && source <(werf ci-env gitlab --tagging-strategy tag-or-branch --verbose)
- werf deploy --stages-storage :local
--namespace ${CI_ENVIRONMENT_SLUG}
--set "global.commit_ref_slug=${CI_COMMIT_REF_SLUG:-''}"
# ะฟะตัะตะดะฐะตะผ ะฟะตัะตะผะตะฝะฝัั `run_tests`
# ะพะฝะฐ ะฑัะดะตั ะธัะฟะพะปัะทะพะฒะฐัััั ะฒ ัะตะฝะดะตัะต Helm-ัะตะปะธะทะฐ
--set "global.run_tests=${RUN_TESTS:-no}"
--set "global.env=${CI_ENVIRONMENT_SLUG}"
# ะธะทะผะตะฝัะตะผ timeout (ะฑัะฒะฐัั ะดะพะปะณะธะต ัะตััั) ะธ ะฟะตัะตะดะฐะตะผ ะตะณะพ ะฒ ัะตะปะธะท
--set "global.ci_timeout=${CI_TIMEOUT:-900}"
--timeout ${CI_TIMEOUT:-900}
dependencies:
- Build
.test-base: &test-base
extends: .base_deploy
before_script:
# ัะพะทะดะฐะตะผ ะดะธัะตะบัะพัะธั ะดะปั ะฑัะดััะตะณะพ ะพััะตัะฐ, ะธัั
ะพะดั ะธะท $CI_COMMIT_REF_SLUG
- mkdir /mnt/tests/${CI_COMMIT_REF_SLUG} || true
# ะฒัะฝัะถะดะตะฝะฝัะน ะบะพัััะปั, ั.ะบ. GitLab ั
ะพัะตั ะฟะพะปััะธัั ะฐััะตัะฐะบัั ะฒ ัะฒะพะตะผ build-dirโะต
- mkdir ./tests || true
- ln -s /mnt/tests/${CI_COMMIT_REF_SLUG} ./tests/${CI_COMMIT_REF_SLUG}
after_script:
# ะฟะพัะปะต ะพะบะพะฝัะฐะฝะธั ัะตััะพะฒ ัะดะฐะปัะตะผ ัะตะปะธะท ะฒะผะตััะต ั Jobโะพะผ
# (ะธ, ะฒะพะทะผะพะถะฝะพ, ะตะณะพ ะธะฝััะฐััััะบัััะพะน)
- type multiwerf && source <(multiwerf use ${WERF_VERSION})
- werf version
- type werf && source <(werf ci-env gitlab --tagging-strategy tag-or-branch --verbose)
- werf dismiss --namespace ${CI_ENVIRONMENT_SLUG} --with-namespace
# ะผั ัะฐะทัะตัะฐะตะผ ะฟะฐะดะตะฝะธั, ะฝะพ ะฒั ะผะพะถะตัะต ัะดะตะปะฐัั ะธะฝะฐัะต
allow_failure: true
variables:
RUN_TESTS: 'yes'
# ะทะฐะดะฐะตะผ ะบะพะฝัะตะบัั ะฒ werf
# (https://werf.io/how_to/gitlab_ci_cd_integration.html#infrastructure)
WERF_KUBE_CONTEXT: 'admin@stage-cluster'
tags:
# ะธัะฟะพะปัะทัะตะผ ัะฐะฝะฝะตั ั ัะตะณะพะผ `werf-runner`
- werf-runner
artifacts:
# ััะตะฑัะตััั ัะพะฑัะฐัั ะฐััะตัะฐะบั ะดะปั ัะพะณะพ, ััะพะฑั ะตะณะพ ะผะพะถะฝะพ ะฑัะปะพ ัะฒะธะดะตัั
# ะฒ ะฟะฐะนะฟะปะฐะนะฝะต ะธ ัะบะฐัะฐัั โ ะฝะฐะฟัะธะผะตั, ะดะปั ะฑะพะปะตะต ะฒะดัะผัะธะฒะพะณะพ ะธะทััะตะฝะธั
paths:
- ./tests/${CI_COMMIT_REF_SLUG}/*
# ะฐััะตัะฐะบัั ััะฐััะต ะฝะตะดะตะปะธ ะฑัะดัั ัะดะฐะปะตะฝั
expire_in: 7 day
# ะฒะฐะถะฝะพ: ััะธ ัััะพะบะธ ะพัะฒะตัะฐัั ะทะฐ ะฟะฐััะธะฝะณ ะพััะตัะฐ GitLabโะพะผ
reports:
junit: ./tests/${CI_COMMIT_REF_SLUG}/report.xml
# ะดะปั ัะฟัะพัะตะฝะธั ะทะดะตัั ะฟะพะบะฐะทะฐะฝั ะฒัะตะณะพ ะดะฒะต ััะฐะดะธะธ
# ะฒ ัะตะฐะปัะฝะพััะธ ะถะต ั ะฒะฐั ะธั
ะฑัะดะตั ะฑะพะปััะต โ ะบะฐะบ ะผะธะฝะธะผัะผ ะธะท-ะทะฐ ะดะตะฟะปะพั
stages:
- build
- tests
build:
stage: build
script:
# ัะฑะพัะบะฐ โ ัะฝะพะฒะฐ ะฟะพ ะดะพะบัะผะตะฝัะฐัะธะธ ะฟะพ werf
# (https://werf.io/how_to/gitlab_ci_cd_integration.html#build-stage)
- type multiwerf && source <(multiwerf use ${WERF_VERSION})
- werf version
- type werf && source <(werf ci-env gitlab --tagging-strategy tag-or-branch --verbose)
- werf build-and-publish --stages-storage :local
tags:
- werf-runner
except:
- schedules
run tests:
<<: *test-base
environment:
# "ัะฐะผะฐ ัะพะปั" ะธะผะตะฝะพะฒะฐะฝะธั namespaceโะฐ
# (https://docs.gitlab.com/ce/ci/variables/predefined_variables.html)
name: tests-${CI_COMMIT_REF_SLUG}
stage: tests
except:
- schedules
แแฃแแแ แแแขแแแ
แแฎแแ แแแ แแฅแขแแ แแแจแ .helm/templates
แแแแแ แจแแแฅแแแแ YAML แกแแแฃแจแแแกแแแ แแ แแแ - tests-job.yaml
โ แขแแกแขแแแแกแ แแ แแฃแแแ แแแขแแก แกแแญแแ แ แ แแกแฃแ แกแแแแก แแแกแแจแแแแแ. แแฎแแแแ แแแแแแ แขแแแแแ แฉแแแแแแแแแแแก แจแแแแแ:
{{- if eq .Values.global.run_tests "yes" }}
---
apiVersion: v1
kind: ConfigMap
metadata:
name: tests-script
data:
tests.sh: |
echo "======================"
echo "${APP_NAME} TESTS"
echo "======================"
cd /app
npm run test:ci
cp report.xml /app/test_results/${CI_COMMIT_REF_SLUG}/
echo ""
echo ""
echo ""
chown -R 999:999 /app/test_results/${CI_COMMIT_REF_SLUG}
---
apiVersion: batch/v1
kind: Job
metadata:
name: {{ .Chart.Name }}-test
annotations:
"helm.sh/hook": post-install,post-upgrade
"helm.sh/hook-weight": "2"
"werf/watch-logs": "true"
spec:
activeDeadlineSeconds: {{ .Values.global.ci_timeout }}
backoffLimit: 1
template:
metadata:
name: {{ .Chart.Name }}-test
spec:
containers:
- name: test
command: ['bash', '-c', '/app/tests.sh']
{{ tuple "application" . | include "werf_container_image" | indent 8 }}
env:
- name: env
value: {{ .Values.global.env }}
- name: CI_COMMIT_REF_SLUG
value: {{ .Values.global.commit_ref_slug }}
- name: APP_NAME
value: {{ .Chart.Name }}
{{ tuple "application" . | include "werf_container_env" | indent 8 }}
volumeMounts:
- mountPath: /app/test_results/
name: data
- mountPath: /app/tests.sh
name: tests-script
subPath: tests.sh
tolerations:
- key: dedicated
operator: Exists
- key: node-role.kubernetes.io/master
operator: Exists
restartPolicy: OnFailure
volumes:
- name: data
persistentVolumeClaim:
claimName: {{ .Chart.Name }}-pvc
- name: tests-script
configMap:
name: tests-script
---
apiVersion: v1
kind: PersistentVolumeClaim
metadata:
name: {{ .Chart.Name }}-pvc
spec:
accessModes:
- ReadWriteOnce
resources:
requests:
storage: 10Mi
storageClassName: {{ .Chart.Name }}-{{ .Values.global.commit_ref_slug }}
volumeName: {{ .Values.global.commit_ref_slug }}
---
apiVersion: v1
kind: PersistentVolume
metadata:
name: {{ .Values.global.commit_ref_slug }}
spec:
accessModes:
- ReadWriteOnce
capacity:
storage: 10Mi
local:
path: /mnt/tests/
nodeAffinity:
required:
nodeSelectorTerms:
- matchExpressions:
- key: kubernetes.io/hostname
operator: In
values:
- kube-master
persistentVolumeReclaimPolicy: Delete
storageClassName: {{ .Chart.Name }}-{{ .Values.global.commit_ref_slug }}
{{- end }}
แ แแแแ แ แ แแกแฃแ แกแแแ แแฆแฌแแ แแแแ แแ แแแแคแแแฃแ แแชแแแจแ? แแแแแแแแแแกแแก แฉแแแ แแฅแแแแ แฃแแแแแแฃแ แกแแฎแแแแ แกแแแ แชแแก แแ แแแฅแขแแกแแแแก (แแก แแแแแแแแฃแแแ .gitlab-ci.yaml
- tests-${CI_COMMIT_REF_SLUG}
) แแ แแแแคแแ แแแแแ:
- ConfigMap แขแแกแขแแก แกแแ แแแขแแ;
- แกแแแฃแจแแ แแแแแก แแฆแฌแแ แแแแแแ แแ แแแแแแแแฃแแ แแแ แแฅแขแแแแ
command
, แ แแแแแแช แฃแแ แแแแ แแขแแ แแแก แขแแกแขแแแก; - PV แแ PVC, แ แแแแแแช แกแแจแฃแแแแแแก แแแซแแแแ แจแแแแแฎแแ แขแแกแขแแก แแแแแชแแแแแ.
แงแฃแ แแแฆแแแ แแแแฅแชแแแ แจแแกแแแแแ แแแ แแแแก if
แแแแแคแแกแขแแก แแแกแแฌแงแแกแจแ - แจแแกแแแแแแกแแ, Helm แกแฅแแแแก แกแฎแแ YAML แคแแแแแแ แแแแแแแชแแแกแแแ แแ แแแ แฃแแแ แแงแแก แจแแคแฃแแฃแแ แฃแแฃ แแแแแแแ แแกแ, แ แแ แแกแแแ แแ แแแแแแแแแแ แขแแกแขแแ แแแแก แแ แแก. แแแฃ:
{{- if ne .Values.global.run_tests "yes" }}
---
ั ะดััะณะพะน ัะผะปะธะบ
{{- end }}
แแฃแแชแ แแฃ แขแแกแขแแแ แแแแแฎแแแก แแแ แแแแฃแ แแแคแ แแกแขแ แฃแฅแขแฃแ แแก (แแแแแแแแแ, Redis, RabbitMQ, Mongo, PostgreSQL...) - แแแแ YAML แจแแแซแแแแ แแงแแก แแ แแ แแก แแแแแ แแแ. แแแแแแแแกแแ แแกแแแ แกแแขแแกแขแ แแแ แแแแจแแช... แแแแ แแแฃแแแ แแ แแกแแแ, แ แแแแ แช แแฅแแแแแแแก แจแแกแแคแแ แแกแแ, แ แ แแฅแแ แฃแแแ.
แกแแแแแแ แจแแฎแแแ
แแแแขแแ แ แแ แจแแแ แแแ แแ แแแแแแแแแ werf แกแแแฃแจแแแแแแก แแแแแงแแแแแแ แแแแแแแ แแฎแแแแ build แกแแ แแแ แแ (gitlab-runner-แแ) แแ แขแแกแขแแแแก แแฅแแแ pod แแแจแแแแฃแแแ แแแกแขแแ แแ, แแฅแแแ แฃแแแ แจแแฅแแแแ แแแ แแฅแขแแ แแ /mnt/tests
แแกแขแแขแแ แแ แแแแชแ แแแ แแแแแแก, แแแแแแแแแ, NFS-แแก แกแแจแฃแแแแแแ. แแแขแแแฃแ แ แแแแแแแแ แแแแแแ แขแแแแแแ แจแแแแซแแแแ แแฎแแแแ แแฅ
แจแแแแแ แแฅแแแแ:
user@kube-master:~$ cat /etc/exports | grep tests
/mnt/tests IP_gitlab-builder/32(rw,nohide,insecure,no_subtree_check,sync,all_squash,anonuid=999,anongid=998)
user@gitlab-runner:~$ cat /etc/fstab | grep tests
IP_kube-master:/mnt/tests /mnt/tests nfs4 _netdev,auto 0 0
แแ แแแแ แแ แซแแแแแก NFS แแแแแแ แแแแก แแแแแแแแแก แแแ แแแแแ gitlab-runner-แแ, แจแแแแแ แแ แแแก แแแแแแจแ แแแแแแขแแแแแแก.
แจแแแแจแแแ
แจแแแซแแแแ แแแแแฎแแ, แ แแขแแ แแ แแฃแแแแ แงแแแแแคแแ แก แกแแแฃแจแแแก แจแแฅแแแแ, แแฃ แแฅแแแ แจแแแแซแแแแ แฃแแ แแแแ แแฌแแ แแแแ แกแแ แแแขแ แขแแกแขแแแแ แแแ แแแแแ shell runner-แแ? แแแกแฃแฎแ แกแแแแแแ แขแ แแแแแแฃแ แแ...
แแแแแแ แแ แขแแกแขแ แกแแญแแ แแแแก แฌแแแแแแก แแแคแ แแกแขแ แฃแฅแขแฃแ แแแ (MongoDB, RabbitMQ, PostgreSQL แแ แ.แจ.), แ แแแ แแแแแแกแขแฃแ แแก, แ แแ แแกแแแ แกแฌแแ แแ แแฃแจแแแแแ. แฉแแแ แขแแกแขแแ แแแแก แแฅแแแแ แฃแแแคแแชแแ แแแฃแแก - แแ แแแแแแแแ แแแแแแ แฎแแแแ แแกแแแ แแแแแขแแแแแ แแ แแแฃแแแแแก แฉแแ แแแ. แแแแก แแแ แแ แแแฆแแแ แกแขแแแแแ แขแ แแแแแแแแแแก แแแแแแแ (แแฃแแแแช NFS-แแก แแแแแงแแแแแแ, แแแ แแฅแขแแ แแแแแก แแแแแขแแแแแ แแแแขแแแ).
แจแแแแแ
แ แแก แแแแแแแฎแแแ แแแแแแแแแฃแแ แแแแคแแแฃแ แแชแแแก แแแแแงแแแแแแกแแก?
แจแแ แฌแงแแแก แแแแฎแแแแ แแฉแแแแแแก แจแแแแฏแแแแแแ แกแขแแขแแกแขแแแแก แแแก แฃแแฎแแแก แแแแกแแแแแจแ แฉแแขแแ แแแฃแแ แขแแกแขแแแแกแแแแก:
แแแแแแฃแแ แจแแชแแแแ แจแแแซแแแแ แแแแฌแแแแฃแแแ แแฅ แแแขแแแแแแกแแแแก:
NB: แงแฃแ แแแฆแแแแแแ แแแแแฎแแแแ แจแแแแฉแแแแก, แ แแ แแแแแฌแแแแ NodeJS แแแแแแแชแแแก, แฎแแแ แกแแ แแแจแแขแแแจแ - .NET... แแฃ แแแแแแแแ แแแแแ: แฃแแ แแแแ, แกแขแแขแแแก แแแแแแแแแแกแแก แแแ แแแแ แแแแแแแชแแแก แขแแกแขแแ แแแแจแ แจแแชแแแแแแ แแ แแฆแแแฉแแแ, แแแแ แแ แแกแแแ แกแฎแแแจแ แแฆแแแแฉแแแแก.
แแแกแแแแ
แ แแแแ แช แฎแแแแแ, แแ แแคแแ แแ แ แแฃแแ!
แแ แแแชแแแจแ, แแฃ แแฅแแแ แฃแแแ แแแฅแแ แญแฃแ แแแก แแแแแฅแขแแ แ แแ แแก แแฃแจแแแแก, แแแแ แแ แแ แแญแแ แแแแแ Kubernetes, แแแกแแ แขแแกแขแแ แแแแก แแแแแแ แแแ แแแแแ แฃแคแ แ แแแ แขแแแ แแแแชแแแ แแฅแแแแ, แแแแ แ แแฅ แแฆแฌแแ แแแแ. แฒแ แจแ
PS
แแกแแแ แฌแแแแแแฎแแ แฉแแแแก แแแแแแ:
- ยซ
แกแแฃแแแแแกแ CI/CD แแ แแฅแขแแแ Kubernetes-แแแ แแ GitLab-แแแ (แแแแแฎแแแแ แแ แแแแแ แแแแแ แแจแ) "; - ยซ
แ แฉแแแแแ GitLab CI-แจแ แแแ แแแแฃแแ แกแแแฃแจแแ แแแแแแแแแก แจแแกแแฅแแแแแแ "; - ยซ
GitLab CI แฌแแ แแแแแแจแ แฃแฌแงแแแขแ แแแขแแแ แแชแแแกแ แแ แแแฌแแแแแแกแแแแก ".
แฌแงแแ แ: www.habr.com