每个人都知道软件测试的重要性和必要性-我相信许多人都坚持这样做。但出乎意料的是,我很难找到一个很好的例子来用GitLab和JUnit配置CI / CD。让我来填补这个空白。
首先,让我定义完整的上下文:
以下是我们的示例中常见的操作顺序:
让我们开始实践吧!
我们将从 .gitlab-ci.yaml
的代码开始,以下部分描述了应用程序的部署和运行测试。代码有点长,我在其中插入了详细的注释:
variables:
WERF_VERSION: "1.0 beta"
.base_deploy: &base_deploy
script:
- kubectl --context="${WERF_KUBE_CONTEXT}" get ns ${CI_ENVIRONMENT_SLUG} || kubectl create ns ${CI_ENVIRONMENT_SLUG}
- 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
--set "global.run_tests=${RUN_TESTS:-no}"
--set "global.env=${CI_ENVIRONMENT_SLUG}"
--set "global.ci_timeout=${CI_TIMEOUT:-900}"
--timeout ${CI_TIMEOUT:-900}
dependencies:
- Build
.test-base: &test-base
extends: .base_deploy
before_script:
- mkdir /mnt/tests/${CI_COMMIT_REF_SLUG} || true
- mkdir ./tests || true
- ln -s /mnt/tests/${CI_COMMIT_REF_SLUG} ./tests/${CI_COMMIT_REF_SLUG}
after_script:
- 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_KUBE_CONTEXT: 'admin@stage-cluster'
tags:
werf-runner
标签的runner - werf-runner
artifacts:
paths:
- ./tests/${CI_COMMIT_REF_SLUG}/*
expire_in: 7 day
reports:
junit: ./tests/${CI_COMMIT_REF_SLUG}/report.xml
stages:
- build
- tests
build:
stage: build
script:
- 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:
name: tests-${CI_COMMIT_REF_SLUG}
stage: tests
except:
- schedules
现在是时候创建一个YAML文件( tests-job.yaml
)了,这个文件用来做两件事
.helm / templates
请参阅以下说明:
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 ""
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:
apiVersion: v1
kind: PersistentVolumeClaim
metadata:
name: {{ .Chart.Name }}-pvc
spec:
accessModes:
- ReadWriteOnce
resources:
requests:
storage: 10Mi
storageClassName: {{ .Chart.Name }}-{{ .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}
),并在其中部署几个组件:
指令
的 Job 注意清单开头的初始 if
语句。 为了防止使用应用程序部署Helm图表的其他YAML文件,您必须插入以下相反条件:
Hey, I'm another YAML
{{- end }}
但是,如果一些测试需要 其他的基础设施 (如redis,rabbitmq, Mongo, PostgreSQL等等),那么你可以在相应的YAML文件中启动并部署这些组件到测试环境中。
当前,仅通过构建服务器(使用gitlab-runner)支持使用werf进行构建和部署。 但是,测试容器在主节点上运行。 在这种情况下,您必须在主节点上创建 /mnt/tests
文件夹并将其安装到运行器(runner), 例如 通过NFS 。 K8s文档中提供了 详细示例 。
我们将会得到以下结果
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
另一种可能性是直接在gitlab-runner上创建一个共享的NFS目录,然后将其安装到pod上。
您可能会问,如果可以轻松地在shell中运行测试脚本,那么创建Job的意义何在?答案很明显:
有些测试需要基础架构(例如MongoDB,RabbitMQ,PostgreSQL等)来检查功能。我的方法是一个统一的解决方案,可以轻松集成其他实例。另外,我们获得了 标准 的部署方法(即使用NFS和额外的目录)。
如果应用提前准备好的配置会是什么结果?
合并请求中将显示在其先前流水线中执行的测试的摘要:
单击错误获取更多信息:
*注意:细心的读者会注意到我们正在测试Node.js应用程序,但是屏幕截图上有一个.NET。不必感到惊讶:虽然我们在原始的应用程序中没有发现任何问题,但在另一个应用程序中暴露出了一些小问题。
如您所见,它是如此简单!
如果您已经有可以使用的Shell脚本并且不需要Kubernetes,则可以通过更简单的方法实现以上例子。 请查看 GitLab CI 文档 。CI文档中提供了Ruby,Go,Gradle,Maven和其他一些产品的示例。