mirror of
				https://github.com/NginxProxyManager/nginx-proxy-manager.git
				synced 2025-10-30 23:33:34 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			280 lines
		
	
	
		
			7.9 KiB
		
	
	
	
		
			Groovy
		
	
	
	
	
	
			
		
		
	
	
			280 lines
		
	
	
		
			7.9 KiB
		
	
	
	
		
			Groovy
		
	
	
	
	
	
| 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()
 | |
| }
 | |
| 
 | |
| pipeline {
 | |
| 	agent {
 | |
| 		label 'docker-multiarch'
 | |
| 	}
 | |
| 	options {
 | |
| 		buildDiscarder(logRotator(numToKeepStr: '5'))
 | |
| 		disableConcurrentBuilds()
 | |
| 		ansiColor('xterm')
 | |
| 	}
 | |
| 	environment {
 | |
| 		DOCKER_ORG                 = 'jc21'
 | |
| 		IMAGE                      = 'nginx-proxy-manager'
 | |
| 		BUILD_VERSION              = getVersion()
 | |
| 		BUILD_COMMIT               = getCommit()
 | |
| 		MAJOR_VERSION              = '3'
 | |
| 		BRANCH_LOWER               = "${BRANCH_NAME.toLowerCase().replaceAll('/', '-')}"
 | |
| 		COMPOSE_PROJECT_NAME       = "npm_${BRANCH_LOWER}_${BUILD_NUMBER}"
 | |
| 		COMPOSE_FILE               = 'docker/docker-compose.ci.yml'
 | |
| 		COMPOSE_INTERACTIVE_NO_CLI = 1
 | |
| 		BUILDX_NAME                = "${COMPOSE_PROJECT_NAME}"
 | |
| 		DOCS_BUCKET                = 'jc21-npm-site-next' // TODO: change to prod when official
 | |
| 		DOCS_CDN                   = 'E2Z0128EHS0Q23'     // TODO: same
 | |
| 	}
 | |
| 	stages {
 | |
| 		stage('Environment') {
 | |
| 			parallel {
 | |
| 				stage('Master') {
 | |
| 					when {
 | |
| 						branch 'master'
 | |
| 					}
 | |
| 					steps {
 | |
| 						script {
 | |
| 							buildxPushTags = "-t docker.io/${DOCKER_ORG}/${IMAGE}:${BUILD_VERSION} -t docker.io/${DOCKER_ORG}/${IMAGE}:${MAJOR_VERSION} -t docker.io/${DOCKER_ORG}/${IMAGE}:latest"
 | |
| 							echo 'Building on Master is disabled!'
 | |
| 							sh 'exit 1'
 | |
| 						}
 | |
| 					}
 | |
| 				}
 | |
| 				stage('Other') {
 | |
| 					when {
 | |
| 						not {
 | |
| 							branch 'master'
 | |
| 						}
 | |
| 					}
 | |
| 					steps {
 | |
| 						script {
 | |
| 							// Defaults to the Branch name, which is applies to all branches AND pr's
 | |
| 							// buildxPushTags = "-t docker.io/jc21/${IMAGE}:github-${BRANCH_LOWER}"
 | |
| 							buildxPushTags = "-t docker.io/${DOCKER_ORG}/${IMAGE}:v3"
 | |
| 						}
 | |
| 					}
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 		stage('Frontend') {
 | |
| 			steps {
 | |
| 				script {
 | |
| 					def shStatusCode = sh(label: 'build-frontend', returnStatus: true, script: '''
 | |
| 						set -e
 | |
| 						./scripts/ci/build-frontend | sed -r "s/\\x1B\\[([0-9]{1,2}(;[0-9]{1,2})?)?[m|K]//g" > ${WORKSPACE}/tmp-sh-build 2>&1
 | |
| 					''')
 | |
| 					shOutput = readFile "${env.WORKSPACE}/tmp-sh-build"
 | |
| 					if (shStatusCode != 0) {
 | |
| 						error "${shOutput}"
 | |
| 					}
 | |
| 				}
 | |
| 			}
 | |
| 			post {
 | |
| 				always {
 | |
| 					sh 'rm -f ${WORKSPACE}/tmp-sh-build'
 | |
| 					// junit 'frontend/eslint.xml'
 | |
| 					// junit 'frontend/junit.xml'
 | |
| 				}
 | |
| 				failure {
 | |
| 					npmGithubPrComment("CI Error:\n\n```\n${shOutput}\n```", true)
 | |
| 				}
 | |
| 				success {
 | |
| 					script {
 | |
| 						shOutput = ""
 | |
| 					}
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 		stage('Backend') {
 | |
| 			steps {
 | |
| 				withCredentials([string(credentialsId: 'npm-sentry-dsn', variable: 'SENTRY_DSN')]) {
 | |
| 					withCredentials([usernamePassword(credentialsId: 'oss-index-token', passwordVariable: 'NANCY_TOKEN', usernameVariable: 'NANCY_USER')]) {
 | |
| 						script {
 | |
| 							def shStatusCode = sh(label: 'test-backend', returnStatus: true, script: '''
 | |
| 								set -e
 | |
| 								./scripts/ci/test-backend | sed -r "s/\\x1B\\[([0-9]{1,2}(;[0-9]{1,2})?)?[m|K]//g" > ${WORKSPACE}/tmp-sh-build 2>&1
 | |
| 							''')
 | |
| 							shOutput = readFile "${env.WORKSPACE}/tmp-sh-build"
 | |
| 							if (shStatusCode != 0) {
 | |
| 								error "${shOutput}"
 | |
| 							}
 | |
| 						}
 | |
| 					}
 | |
| 					// Build all the golang binaries
 | |
| 					script {
 | |
| 						def shStatusCode = sh(label: 'build-backend', returnStatus: true, script: '''
 | |
| 							set -e
 | |
| 							./scripts/ci/build-backend > ${WORKSPACE}/tmp-sh-build 2>&1
 | |
| 						''')
 | |
| 						shOutput = readFile "${env.WORKSPACE}/tmp-sh-build"
 | |
| 						if (shStatusCode != 0) {
 | |
| 							error "${shOutput}"
 | |
| 						}
 | |
| 					}
 | |
| 					// Build the docker image used for testing below
 | |
| 					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}" \\
 | |
| 						.
 | |
| 					'''
 | |
| 				}
 | |
| 			}
 | |
| 			post {
 | |
| 				always {
 | |
| 					sh 'rm -f ${WORKSPACE}/tmp-sh-build'
 | |
| 				}
 | |
| 				failure {
 | |
| 					npmGithubPrComment("CI Error:\n\n```\n${shOutput}\n```", true)
 | |
| 				}
 | |
| 				success {
 | |
| 					archiveArtifacts allowEmptyArchive: false, artifacts: 'bin/*'
 | |
| 					script {
 | |
| 						shOutput = ""
 | |
| 					}
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 		stage('Test') {
 | |
| 			when {
 | |
| 				not {
 | |
| 					equals expected: 'UNSTABLE', actual: currentBuild.result
 | |
| 				}
 | |
| 			}
 | |
| 			steps {
 | |
| 				// Docker image check
 | |
| 				/*
 | |
| 				sh '''docker run --rm \
 | |
| 					-v /var/run/docker.sock:/var/run/docker.sock \
 | |
| 					-v "$(pwd)/docker:/app" \
 | |
| 					-e CI=true \
 | |
| 					wagoodman/dive:latest --ci-config /app/.dive-ci \
 | |
| 					"${IMAGE}:${BRANCH_LOWER}-ci-${BUILD_NUMBER}"
 | |
| 				'''
 | |
| 				*/
 | |
| 				sh './scripts/ci/fulltest-cypress'
 | |
| 			}
 | |
| 			post {
 | |
| 				always {
 | |
| 					// Dumps to analyze later
 | |
| 					sh 'mkdir -p debug'
 | |
| 					sh 'docker logs $(docker-compose ps -q fullstack) > debug/docker_fullstack.log 2>&1'
 | |
| 					sh 'docker logs $(docker-compose ps -q stepca) > debug/docker_stepca.log 2>&1'
 | |
| 					sh 'docker logs $(docker-compose ps -q pdns) > debug/docker_pdns.log 2>&1'
 | |
| 					sh 'docker logs $(docker-compose ps -q pdns-db) > debug/docker_pdns-db.log 2>&1'
 | |
| 					sh 'docker logs $(docker-compose ps -q dnsrouter) > debug/docker_dnsrouter.log 2>&1'
 | |
| 					junit 'test/results/junit/*'
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 		stage('Docs') {
 | |
| 			when {
 | |
| 				not {
 | |
| 					equals expected: 'UNSTABLE', actual: currentBuild.result
 | |
| 				}
 | |
| 			}
 | |
| 			steps {
 | |
| 				dir(path: 'docs') {
 | |
| 					sh 'yarn install'
 | |
| 					sh 'yarn build'
 | |
| 				}
 | |
| 
 | |
| 				// API Docs:
 | |
| 				sh 'docker-compose exec -T fullstack curl -s --output /temp-docs/api-schema.json "http://fullstack:81/api/schema"'
 | |
| 				sh 'mkdir -p "docs/.vuepress/dist/api"'
 | |
| 				sh 'mv docs/api-schema.json docs/.vuepress/dist/api/'
 | |
| 
 | |
| 				dir(path: 'docs/.vuepress/dist') {
 | |
| 					sh 'tar -czf ../../docs.tgz *'
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 		stage('MultiArch Build') {
 | |
| 			when {
 | |
| 				not {
 | |
| 					equals expected: 'UNSTABLE', actual: currentBuild.result
 | |
| 				}
 | |
| 			}
 | |
| 			steps {
 | |
| 				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}"'
 | |
| 						sh "./scripts/buildx --push ${buildxPushTags}"
 | |
| 						// sh './scripts/buildx -o type=local,dest=docker-build'
 | |
| 					}
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 		stage('Docs Deploy') {
 | |
| 			when {
 | |
| 				allOf {
 | |
| 					branch 'v3' // TODO: change to master when ready
 | |
| 					not {
 | |
| 						equals expected: 'UNSTABLE', actual: currentBuild.result
 | |
| 					}
 | |
| 				}
 | |
| 			}
 | |
| 			steps {
 | |
| 				npmDocsRelease("$DOCS_BUCKET", "$DOCS_CDN")
 | |
| 			}
 | |
| 		}
 | |
| 		stage('PR Comment') {
 | |
| 			when {
 | |
| 				allOf {
 | |
| 					changeRequest()
 | |
| 					not {
 | |
| 						equals expected: 'UNSTABLE', actual: currentBuild.result
 | |
| 					}
 | |
| 				}
 | |
| 			}
 | |
| 			steps {
 | |
| 				script {
 | |
| 					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)
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 	post {
 | |
| 		always {
 | |
| 			sh 'docker-compose down --rmi all --remove-orphans --volumes -t 30 || true'
 | |
| 			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'
 | |
| 		}
 | |
| 		success {
 | |
| 			juxtapose event: 'success'
 | |
| 			sh 'figlet "SUCCESS"'
 | |
| 		}
 | |
| 		failure {
 | |
| 			dir(path: 'test') {
 | |
| 				archiveArtifacts allowEmptyArchive: true, artifacts: 'results/**/*', excludes: '**/*.xml'
 | |
| 			}
 | |
| 			archiveArtifacts(artifacts: 'debug/**.*', allowEmptyArchive: true)
 | |
| 			juxtapose event: 'failure'
 | |
| 			sh 'figlet "FAILURE"'
 | |
| 		}
 | |
| 		unstable {
 | |
| 			dir(path: 'test') {
 | |
| 				archiveArtifacts allowEmptyArchive: true, artifacts: 'results/**/*', excludes: '**/*.xml'
 | |
| 			}
 | |
| 			archiveArtifacts(artifacts: 'debug/**.*', allowEmptyArchive: true)
 | |
| 			juxtapose event: 'unstable'
 | |
| 			sh 'figlet "UNSTABLE"'
 | |
| 		}
 | |
| 	}
 | |
| }
 |