2023-03-08 06:12:57 +00:00
import groovy.transform.Field
@Field
def shOutput = ""
def buildxPushTags = ""
def getVersion() {
ver = sh(script: 'cat .version', returnStdout: true)
return ver.trim()
}
def getCommit() {
ver = sh(script: 'git log -n 1 --format=%h', returnStdout: true)
return ver.trim()
}
2018-05-03 05:43:04 +00:00
pipeline {
2020-02-19 04:55:06 +00:00
agent {
label 'docker-multiarch'
}
options {
buildDiscarder(logRotator(numToKeepStr: '5'))
disableConcurrentBuilds()
2020-05-25 04:53:35 +00:00
ansiColor('xterm')
2020-02-19 04:55:06 +00:00
}
environment {
2022-05-11 22:47:31 +00:00
DOCKER_ORG = 'jc21'
IMAGE = 'nginx-proxy-manager'
2020-02-19 04:55:06 +00:00
BUILD_VERSION = getVersion()
2022-05-11 22:47:31 +00:00
BUILD_COMMIT = getCommit()
MAJOR_VERSION = '3'
2020-02-19 06:11:52 +00:00
BRANCH_LOWER = "${BRANCH_NAME.toLowerCase().replaceAll('/', '-')}"
COMPOSE_PROJECT_NAME = "npm_${BRANCH_LOWER}_${BUILD_NUMBER}"
2020-02-19 04:55:06 +00:00
COMPOSE_FILE = 'docker/docker-compose.ci.yml'
COMPOSE_INTERACTIVE_NO_CLI = 1
BUILDX_NAME = "${COMPOSE_PROJECT_NAME}"
2022-05-11 22:47:31 +00:00
DOCS_BUCKET = 'jc21-npm-site-next' // TODO: change to prod when official
DOCS_CDN = 'E2Z0128EHS0Q23' // TODO: same
2020-02-19 04:55:06 +00:00
}
stages {
stage('Environment') {
parallel {
stage('Master') {
when {
branch 'master'
}
steps {
script {
2023-03-08 06:12:57 +00:00
buildxPushTags = "-t docker.io/${DOCKER_ORG}/${IMAGE}:${BUILD_VERSION} -t docker.io/${DOCKER_ORG}/${IMAGE}:${MAJOR_VERSION} -t docker.io/${DOCKER_ORG}/${IMAGE}:latest"
2022-05-11 22:47:31 +00:00
echo 'Building on Master is disabled!'
sh 'exit 1'
2020-02-19 04:55:06 +00:00
}
}
}
2020-02-19 05:28:27 +00:00
stage('Other') {
when {
not {
branch 'master'
}
}
steps {
script {
// Defaults to the Branch name, which is applies to all branches AND pr's
2023-03-08 06:12:57 +00:00
// buildxPushTags = "-t docker.io/jc21/${IMAGE}:github-${BRANCH_LOWER}"
buildxPushTags = "-t docker.io/${DOCKER_ORG}/${IMAGE}:v3"
2020-02-19 05:28:27 +00:00
}
}
}
2020-02-19 04:55:06 +00:00
}
}
stage('Frontend') {
steps {
2023-04-11 11:36:13 +00:00
sh './scripts/ci/build-frontend'
2020-02-19 04:55:06 +00:00
}
}
2022-05-11 22:47:31 +00:00
stage('Backend') {
2020-02-19 04:55:06 +00:00
steps {
2022-05-11 22:47:31 +00:00
withCredentials([string(credentialsId: 'npm-sentry-dsn', variable: 'SENTRY_DSN')]) {
withCredentials([usernamePassword(credentialsId: 'oss-index-token', passwordVariable: 'NANCY_TOKEN', usernameVariable: 'NANCY_USER')]) {
2023-05-31 11:44:06 +00:00
sh './scripts/ci/test-backend'
2023-03-08 06:12:57 +00:00
}
// Build all the golang binaries
2023-05-31 11:44:06 +00:00
sh './scripts/ci/build-backend'
2023-03-08 06:12:57 +00:00
// Build the docker image used for testing below
2022-05-11 22:47:31 +00:00
sh '''docker build --pull --no-cache \\
-t "${IMAGE}:${BRANCH_LOWER}-ci-${BUILD_NUMBER}" \\
-f docker/Dockerfile \\
--build-arg BUILD_COMMIT="${BUILD_COMMIT}" \\
--build-arg BUILD_DATE="$(date '+%Y-%m-%d %T %Z')" \\
--build-arg BUILD_VERSION="${BUILD_VERSION}" \\
.
'''
}
2020-02-19 04:55:06 +00:00
}
post {
2022-05-11 22:47:31 +00:00
success {
archiveArtifacts allowEmptyArchive: false, artifacts: 'bin/*'
2023-03-08 06:15:20 +00:00
script {
shOutput = ""
}
2020-08-05 22:58:20 +00:00
}
}
}
2023-05-30 03:26:46 +00:00
stage('Test Sqlite') {
environment {
COMPOSE_PROJECT_NAME = "npm_${BRANCH_LOWER}_${BUILD_NUMBER}_sqlite"
COMPOSE_FILE = 'docker/docker-compose.ci.yml'
}
when {
not {
equals expected: 'UNSTABLE', actual: currentBuild.result
}
}
steps {
sh 'rm -rf ./test/results/junit/*'
sh './scripts/ci/fulltest-cypress'
2023-05-31 12:10:57 +00:00
// Adding this here as the schema needs to come from a running stack, but this will be used by docs later
sh 'docker-compose exec -T fullstack curl -s --output /temp-docs/api-schema.json "http://fullstack:81/api/schema"'
2023-05-30 03:26:46 +00:00
}
post {
always {
// Dumps to analyze later
sh 'mkdir -p debug/sqlite'
sh 'docker logs $(docker-compose ps --all -q fullstack) > debug/sqlite/docker_fullstack.log 2>&1'
sh 'docker logs $(docker-compose ps --all -q stepca) > debug/sqlite/docker_stepca.log 2>&1'
sh 'docker logs $(docker-compose ps --all -q pdns) > debug/sqlite/docker_pdns.log 2>&1'
sh 'docker logs $(docker-compose ps --all -q pdns-db) > debug/sqlite/docker_pdns-db.log 2>&1'
sh 'docker logs $(docker-compose ps --all -q dnsrouter) > debug/sqlite/docker_dnsrouter.log 2>&1'
junit 'test/results/junit/*'
2023-05-30 03:58:36 +00:00
sh 'docker-compose down --remove-orphans --volumes -t 30 || true'
2023-05-30 03:26:46 +00:00
}
}
}
stage('Test Mysql') {
environment {
COMPOSE_PROJECT_NAME = "npm_${BRANCH_LOWER}_${BUILD_NUMBER}_mysql"
2023-05-30 03:41:28 +00:00
COMPOSE_FILE = 'docker/docker-compose.ci.yml:docker/docker-compose.ci.mysql.yml'
2023-05-30 03:26:46 +00:00
}
when {
not {
equals expected: 'UNSTABLE', actual: currentBuild.result
}
}
steps {
sh 'rm -rf ./test/results/junit/*'
sh './scripts/ci/fulltest-cypress'
}
post {
always {
// Dumps to analyze later
sh 'mkdir -p debug/mysql'
sh 'docker logs $(docker-compose ps --all -q fullstack) > debug/mysql/docker_fullstack.log 2>&1'
sh 'docker logs $(docker-compose ps --all -q stepca) > debug/mysql/docker_stepca.log 2>&1'
sh 'docker logs $(docker-compose ps --all -q pdns) > debug/mysql/docker_pdns.log 2>&1'
sh 'docker logs $(docker-compose ps --all -q pdns-db) > debug/mysql/docker_pdns-db.log 2>&1'
sh 'docker logs $(docker-compose ps --all -q dnsrouter) > debug/mysql/docker_dnsrouter.log 2>&1'
junit 'test/results/junit/*'
2023-05-30 03:58:36 +00:00
sh 'docker-compose down --remove-orphans --volumes -t 30 || true'
2023-05-30 03:26:46 +00:00
}
}
}
stage('Test Postgres') {
environment {
COMPOSE_PROJECT_NAME = "npm_${BRANCH_LOWER}_${BUILD_NUMBER}_postgres"
2023-05-30 03:41:28 +00:00
COMPOSE_FILE = 'docker/docker-compose.ci.yml:docker/docker-compose.ci.postgres.yml'
2023-05-30 03:26:46 +00:00
}
2022-05-11 22:47:31 +00:00
when {
not {
equals expected: 'UNSTABLE', actual: currentBuild.result
}
}
2020-08-05 22:58:20 +00:00
steps {
2023-05-30 03:26:46 +00:00
sh 'rm -rf ./test/results/junit/*'
2022-05-11 22:47:31 +00:00
sh './scripts/ci/fulltest-cypress'
2020-08-05 22:58:20 +00:00
}
post {
always {
// Dumps to analyze later
2023-05-30 03:26:46 +00:00
sh 'mkdir -p debug/postgres'
sh 'docker logs $(docker-compose ps --all -q fullstack) > debug/postgres/docker_fullstack.log 2>&1'
sh 'docker logs $(docker-compose ps --all -q stepca) > debug/postgres/docker_stepca.log 2>&1'
sh 'docker logs $(docker-compose ps --all -q pdns) > debug/postgres/docker_pdns.log 2>&1'
sh 'docker logs $(docker-compose ps --all -q pdns-db) > debug/postgres/docker_pdns-db.log 2>&1'
sh 'docker logs $(docker-compose ps --all -q dnsrouter) > debug/postgres/docker_dnsrouter.log 2>&1'
2020-03-06 03:11:04 +00:00
junit 'test/results/junit/*'
2023-05-30 03:58:36 +00:00
sh 'docker-compose down --remove-orphans --volumes -t 30 || true'
2020-02-19 04:55:06 +00:00
}
}
}
2020-03-11 06:54:10 +00:00
stage('Docs') {
when {
not {
equals expected: 'UNSTABLE', actual: currentBuild.result
}
}
steps {
2020-05-25 04:53:35 +00:00
dir(path: 'docs') {
2021-03-10 23:29:08 +00:00
sh 'yarn install'
sh 'yarn build'
2020-05-25 04:53:35 +00:00
}
2020-03-11 06:54:10 +00:00
2022-05-11 22:47:31 +00:00
// API Docs:
sh 'mkdir -p "docs/.vuepress/dist/api"'
sh 'mv docs/api-schema.json docs/.vuepress/dist/api/'
2020-05-25 04:53:35 +00:00
dir(path: 'docs/.vuepress/dist') {
sh 'tar -czf ../../docs.tgz *'
2020-03-11 06:54:10 +00:00
}
}
}
2020-02-19 04:55:06 +00:00
stage('MultiArch Build') {
when {
not {
equals expected: 'UNSTABLE', actual: currentBuild.result
}
}
steps {
2022-05-11 22:47:31 +00:00
withCredentials([string(credentialsId: 'npm-sentry-dsn', variable: 'SENTRY_DSN')]) {
withCredentials([usernamePassword(credentialsId: 'jc21-dockerhub', passwordVariable: 'dpass', usernameVariable: 'duser')]) {
sh 'docker login -u "${duser}" -p "${dpass}"'
2023-03-08 06:12:57 +00:00
sh "./scripts/buildx --push ${buildxPushTags}"
2022-05-11 22:47:31 +00:00
// sh './scripts/buildx -o type=local,dest=docker-build'
}
2020-02-19 04:55:06 +00:00
}
}
}
2020-03-11 06:54:10 +00:00
stage('Docs Deploy') {
when {
allOf {
2023-03-08 06:12:57 +00:00
branch 'v3' // TODO: change to master when ready
2020-03-11 06:54:10 +00:00
not {
equals expected: 'UNSTABLE', actual: currentBuild.result
}
}
}
steps {
2023-03-08 06:12:57 +00:00
npmDocsRelease("$DOCS_BUCKET", "$DOCS_CDN")
2020-03-11 06:54:10 +00:00
}
}
2020-02-19 04:55:06 +00:00
stage('PR Comment') {
when {
allOf {
changeRequest()
not {
equals expected: 'UNSTABLE', actual: currentBuild.result
}
}
}
steps {
2020-05-25 04:53:35 +00:00
script {
2023-03-08 06:12:57 +00:00
npmGithubPrComment("Docker Image for build ${BUILD_NUMBER} is available on [DockerHub](https://cloud.docker.com/repository/docker/${DOCKER_ORG}/${IMAGE}) as `${DOCKER_ORG}/${IMAGE}:github-${BRANCH_LOWER}`\n\n**Note:** ensure you backup your NPM instance before testing this PR image! Especially if this PR contains database changes.", true)
2020-02-19 04:55:06 +00:00
}
}
}
}
post {
always {
2023-05-30 03:58:36 +00:00
sh 'docker-compose down --rmi all --remove-orphans --volumes -t 30 || true'
2022-05-11 22:47:31 +00:00
sh './scripts/ci/build-cleanup'
echo 'Reverting ownership'
sh 'docker run --rm -v $(pwd):/data jc21/gotools:latest chown -R "$(id -u):$(id -g)" /data'
2020-02-19 04:55:06 +00:00
}
success {
juxtapose event: 'success'
sh 'figlet "SUCCESS"'
}
failure {
2022-05-11 22:47:31 +00:00
dir(path: 'test') {
archiveArtifacts allowEmptyArchive: true, artifacts: 'results/**/*', excludes: '**/*.xml'
}
2023-05-31 11:34:05 +00:00
archiveArtifacts(artifacts: 'debug/*', allowEmptyArchive: true)
2020-02-19 04:55:06 +00:00
juxtapose event: 'failure'
sh 'figlet "FAILURE"'
}
unstable {
2022-05-11 22:47:31 +00:00
dir(path: 'test') {
archiveArtifacts allowEmptyArchive: true, artifacts: 'results/**/*', excludes: '**/*.xml'
}
2023-05-31 11:34:05 +00:00
archiveArtifacts(artifacts: 'debug/*', allowEmptyArchive: true)
2020-02-19 04:55:06 +00:00
juxtapose event: 'unstable'
sh 'figlet "UNSTABLE"'
}
}
}