mirror of
				https://github.com/NginxProxyManager/nginx-proxy-manager.git
				synced 2025-11-04 01:15:14 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			254 lines
		
	
	
		
			7.5 KiB
		
	
	
	
		
			Groovy
		
	
	
	
	
	
			
		
		
	
	
			254 lines
		
	
	
		
			7.5 KiB
		
	
	
	
		
			Groovy
		
	
	
	
	
	
pipeline {
 | 
						|
	agent {
 | 
						|
		label 'docker-multiarch'
 | 
						|
	}
 | 
						|
	options {
 | 
						|
		buildDiscarder(logRotator(numToKeepStr: '5'))
 | 
						|
		disableConcurrentBuilds()
 | 
						|
		ansiColor('xterm')
 | 
						|
	}
 | 
						|
	environment {
 | 
						|
		IMAGE                      = "nginx-proxy-manager"
 | 
						|
		BUILD_VERSION              = getVersion()
 | 
						|
		MAJOR_VERSION              = "2"
 | 
						|
		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}"
 | 
						|
	}
 | 
						|
	stages {
 | 
						|
		stage('Environment') {
 | 
						|
			parallel {
 | 
						|
				stage('Master') {
 | 
						|
					when {
 | 
						|
						branch 'master'
 | 
						|
					}
 | 
						|
					steps {
 | 
						|
						script {
 | 
						|
							env.BUILDX_PUSH_TAGS = "-t docker.io/jc21/${IMAGE}:${BUILD_VERSION} -t docker.io/jc21/${IMAGE}:${MAJOR_VERSION} -t docker.io/jc21/${IMAGE}:latest"
 | 
						|
						}
 | 
						|
					}
 | 
						|
				}
 | 
						|
				stage('Other') {
 | 
						|
					when {
 | 
						|
						not {
 | 
						|
							branch 'master'
 | 
						|
						}
 | 
						|
					}
 | 
						|
					steps {
 | 
						|
						script {
 | 
						|
							// Defaults to the Branch name, which is applies to all branches AND pr's
 | 
						|
							env.BUILDX_PUSH_TAGS = "-t docker.io/jc21/${IMAGE}:github-${BRANCH_LOWER}"
 | 
						|
						}
 | 
						|
					}
 | 
						|
				}
 | 
						|
				stage('Versions') {
 | 
						|
					steps {
 | 
						|
						sh 'cat frontend/package.json | jq --arg BUILD_VERSION "${BUILD_VERSION}" \'.version = $BUILD_VERSION\' | sponge frontend/package.json'
 | 
						|
						sh 'echo -e "\\E[1;36mFrontend Version is:\\E[1;33m $(cat frontend/package.json | jq -r .version)\\E[0m"'
 | 
						|
						sh 'cat backend/package.json | jq --arg BUILD_VERSION "${BUILD_VERSION}" \'.version = $BUILD_VERSION\' | sponge backend/package.json'
 | 
						|
						sh 'echo -e "\\E[1;36mBackend Version is:\\E[1;33m  $(cat backend/package.json | jq -r .version)\\E[0m"'
 | 
						|
						sh 'sed -i -E "s/(version-)[0-9]+\\.[0-9]+\\.[0-9]+(-green)/\\1${BUILD_VERSION}\\2/" README.md'
 | 
						|
					}
 | 
						|
				}
 | 
						|
			}
 | 
						|
		}
 | 
						|
		stage('Frontend') {
 | 
						|
			steps {
 | 
						|
				sh './scripts/frontend-build'
 | 
						|
			}
 | 
						|
		}
 | 
						|
		stage('Backend') {
 | 
						|
			steps {
 | 
						|
				echo 'Checking Syntax ...'
 | 
						|
				sh 'docker pull nginxproxymanager/nginx-full:certbot-node'
 | 
						|
				// See: https://github.com/yarnpkg/yarn/issues/3254
 | 
						|
				sh '''docker run --rm \\
 | 
						|
					-v "$(pwd)/backend:/app" \\
 | 
						|
					-v "$(pwd)/global:/app/global" \\
 | 
						|
					-w /app \\
 | 
						|
					nginxproxymanager/nginx-full:certbot-node \\
 | 
						|
					sh -c "yarn install && yarn eslint . && rm -rf node_modules"
 | 
						|
				'''
 | 
						|
 | 
						|
				echo 'Docker Build ...'
 | 
						|
				sh '''docker build --pull --no-cache --squash --compress \\
 | 
						|
					-t "${IMAGE}:ci-${BUILD_NUMBER}" \\
 | 
						|
					-f docker/Dockerfile \\
 | 
						|
					--build-arg TARGETPLATFORM=linux/amd64 \\
 | 
						|
					--build-arg BUILDPLATFORM=linux/amd64 \\
 | 
						|
					--build-arg BUILD_VERSION="${BUILD_VERSION}" \\
 | 
						|
					--build-arg BUILD_COMMIT="${BUILD_COMMIT}" \\
 | 
						|
					--build-arg BUILD_DATE="$(date '+%Y-%m-%d %T %Z')" \\
 | 
						|
					.
 | 
						|
				'''
 | 
						|
			}
 | 
						|
		}
 | 
						|
		stage('Integration Tests Sqlite') {
 | 
						|
			steps {
 | 
						|
				// Bring up a stack
 | 
						|
				sh 'docker-compose up -d fullstack-sqlite'
 | 
						|
				sh './scripts/wait-healthy $(docker-compose ps -q fullstack-sqlite) 120'
 | 
						|
 | 
						|
				// Run tests
 | 
						|
				sh 'rm -rf test/results'
 | 
						|
				sh 'docker-compose up cypress-sqlite'
 | 
						|
				// Get results
 | 
						|
				sh 'docker cp -L "$(docker-compose ps -q cypress-sqlite):/test/results" test/'
 | 
						|
			}
 | 
						|
			post {
 | 
						|
				always {
 | 
						|
					// Dumps to analyze later
 | 
						|
					sh 'mkdir -p debug'
 | 
						|
					sh 'docker-compose logs fullstack-sqlite | gzip > debug/docker_fullstack_sqlite.log.gz'
 | 
						|
					sh 'docker-compose logs db | gzip > debug/docker_db.log.gz'
 | 
						|
					// Cypress videos and screenshot artifacts
 | 
						|
					dir(path: 'test/results') {
 | 
						|
						archiveArtifacts allowEmptyArchive: true, artifacts: '**/*', excludes: '**/*.xml'
 | 
						|
					}
 | 
						|
					junit 'test/results/junit/*'
 | 
						|
				}
 | 
						|
			}
 | 
						|
		}
 | 
						|
		stage('Integration Tests Mysql') {
 | 
						|
			steps {
 | 
						|
				// Bring up a stack
 | 
						|
				sh 'docker-compose up -d fullstack-mysql'
 | 
						|
				sh './scripts/wait-healthy $(docker-compose ps -q fullstack-mysql) 120'
 | 
						|
 | 
						|
				// Run tests
 | 
						|
				sh 'rm -rf test/results'
 | 
						|
				sh 'docker-compose up cypress-mysql'
 | 
						|
				// Get results
 | 
						|
				sh 'docker cp -L "$(docker-compose ps -q cypress-mysql):/test/results" test/'
 | 
						|
			}
 | 
						|
			post {
 | 
						|
				always {
 | 
						|
					// Dumps to analyze later
 | 
						|
					sh 'mkdir -p debug'
 | 
						|
					sh 'docker-compose logs fullstack-mysql | gzip > debug/docker_fullstack_mysql.log.gz'
 | 
						|
					sh 'docker-compose logs db | gzip > debug/docker_db.log.gz'
 | 
						|
					// Cypress videos and screenshot artifacts
 | 
						|
					dir(path: 'test/results') {
 | 
						|
						archiveArtifacts allowEmptyArchive: true, artifacts: '**/*', excludes: '**/*.xml'
 | 
						|
					}
 | 
						|
					junit 'test/results/junit/*'
 | 
						|
				}
 | 
						|
			}
 | 
						|
		}
 | 
						|
		stage('Docs') {
 | 
						|
			when {
 | 
						|
				not {
 | 
						|
					equals expected: 'UNSTABLE', actual: currentBuild.result
 | 
						|
				}
 | 
						|
			}
 | 
						|
			steps {
 | 
						|
				dir(path: 'docs') {
 | 
						|
					sh 'yarn install'
 | 
						|
					sh 'yarn build'
 | 
						|
				}
 | 
						|
 | 
						|
				dir(path: 'docs/.vuepress/dist') {
 | 
						|
					sh 'tar -czf ../../docs.tgz *'
 | 
						|
				}
 | 
						|
 | 
						|
				archiveArtifacts(artifacts: 'docs/docs.tgz', allowEmptyArchive: false)
 | 
						|
			}
 | 
						|
		}
 | 
						|
		stage('MultiArch Build') {
 | 
						|
			when {
 | 
						|
				not {
 | 
						|
					equals expected: 'UNSTABLE', actual: currentBuild.result
 | 
						|
				}
 | 
						|
			}
 | 
						|
			steps {
 | 
						|
				withCredentials([usernamePassword(credentialsId: 'jc21-dockerhub', passwordVariable: 'dpass', usernameVariable: 'duser')]) {
 | 
						|
					// Docker Login
 | 
						|
					sh "docker login -u '${duser}' -p '${dpass}'"
 | 
						|
					// Buildx with push from cache
 | 
						|
					sh "./scripts/buildx --push ${BUILDX_PUSH_TAGS}"
 | 
						|
				}
 | 
						|
			}
 | 
						|
		}
 | 
						|
		stage('Docs Deploy') {
 | 
						|
			when {
 | 
						|
				allOf {
 | 
						|
					branch 'master'
 | 
						|
					not {
 | 
						|
						equals expected: 'UNSTABLE', actual: currentBuild.result
 | 
						|
					}
 | 
						|
				}
 | 
						|
			}
 | 
						|
			steps {
 | 
						|
				withCredentials([[$class: 'AmazonWebServicesCredentialsBinding', accessKeyVariable: 'AWS_ACCESS_KEY_ID', credentialsId: 'npm-s3-docs', secretKeyVariable: 'AWS_SECRET_ACCESS_KEY']]) {
 | 
						|
					sh """docker run --rm \\
 | 
						|
						--name \${COMPOSE_PROJECT_NAME}-docs-upload \\
 | 
						|
						-e S3_BUCKET=jc21-npm-site \\
 | 
						|
						-e AWS_ACCESS_KEY_ID=$AWS_ACCESS_KEY_ID \\
 | 
						|
						-e AWS_SECRET_ACCESS_KEY=$AWS_SECRET_ACCESS_KEY \\
 | 
						|
						-v \$(pwd):/app \\
 | 
						|
						-w /app \\
 | 
						|
						jc21/ci-tools \\
 | 
						|
						scripts/docs-upload /app/docs/.vuepress/dist/
 | 
						|
					"""
 | 
						|
 | 
						|
					sh """docker run --rm \\
 | 
						|
						--name \${COMPOSE_PROJECT_NAME}-docs-invalidate \\
 | 
						|
						-e AWS_ACCESS_KEY_ID=$AWS_ACCESS_KEY_ID \\
 | 
						|
						-e AWS_SECRET_ACCESS_KEY=$AWS_SECRET_ACCESS_KEY \\
 | 
						|
						jc21/ci-tools \\
 | 
						|
						aws cloudfront create-invalidation --distribution-id EN1G6DEWZUTDT --paths '/*'
 | 
						|
					"""
 | 
						|
				}
 | 
						|
			}
 | 
						|
		}
 | 
						|
		stage('PR Comment') {
 | 
						|
			when {
 | 
						|
				allOf {
 | 
						|
					changeRequest()
 | 
						|
					not {
 | 
						|
						equals expected: 'UNSTABLE', actual: currentBuild.result
 | 
						|
					}
 | 
						|
				}
 | 
						|
			}
 | 
						|
			steps {
 | 
						|
				script {
 | 
						|
					def comment = pullRequest.comment("This is an automated message from CI:\n\nDocker Image for build ${BUILD_NUMBER} is available on [DockerHub](https://cloud.docker.com/repository/docker/jc21/${IMAGE}) as `jc21/${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.")
 | 
						|
				}
 | 
						|
			}
 | 
						|
		}
 | 
						|
	}
 | 
						|
	post {
 | 
						|
		always {
 | 
						|
			sh 'docker-compose down --remove-orphans --volumes -t 30'
 | 
						|
			sh 'echo Reverting ownership'
 | 
						|
			sh 'docker run --rm -v $(pwd):/data jc21/ci-tools chown -R $(id -u):$(id -g) /data'
 | 
						|
		}
 | 
						|
		success {
 | 
						|
			juxtapose event: 'success'
 | 
						|
			sh 'figlet "SUCCESS"'
 | 
						|
		}
 | 
						|
		failure {
 | 
						|
			archiveArtifacts(artifacts: 'debug/**.*', allowEmptyArchive: true)
 | 
						|
			juxtapose event: 'failure'
 | 
						|
			sh 'figlet "FAILURE"'
 | 
						|
		}
 | 
						|
		unstable {
 | 
						|
			archiveArtifacts(artifacts: 'debug/**.*', allowEmptyArchive: true)
 | 
						|
			juxtapose event: 'unstable'
 | 
						|
			sh 'figlet "UNSTABLE"'
 | 
						|
		}
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
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()
 | 
						|
}
 |