Merge pull request #6729 from corda/INFRA-683-os-47-linear-builds-parallel-stages

INFRA-683: Switching linear regression build on two agents
This commit is contained in:
Waldemar Żurowski 2020-09-18 11:14:47 +02:00 committed by GitHub
commit da73cbf2a5
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23

View File

@ -1,29 +1,21 @@
#!groovy #!groovy
/** /**
* Jenkins pipeline to build Corda OS release branches and tags * Jenkins pipeline to build Corda OS release branches and tags.
* PLEASE NOTE: we DO want to run a build for each commit!!!
*/ */
/**
* Kill already started job.
* Assume new commit takes precendence and results from previous
* unfinished builds are not required.
* This feature doesn't play well with disableConcurrentBuilds() option
*/
@Library('corda-shared-build-pipeline-steps')
import static com.r3.build.BuildControl.killAllExistingBuildsForJob
killAllExistingBuildsForJob(env.JOB_NAME, env.BUILD_NUMBER.toInteger())
/** /**
* Sense environment * Sense environment
*/ */
boolean isReleaseTag = (env.TAG_NAME =~ /^release-.*(?<!_JDK11)$/) boolean isReleaseTag = (env.TAG_NAME =~ /^release-.*(?<!_JDK11)$/)
boolean isInternalRelease = (env.TAG_NAME =~ /^internal-release-.*$/) boolean isInternalRelease = (env.TAG_NAME =~ /^internal-release-.*$/)
boolean isReleaseCandidate = (env.TAG_NAME =~ /^(release-.*(RC|HC).*(?<!_JDK11))$/)
/* /*
** calculate the stage for NexusIQ evaluation ** calculate the stage for NexusIQ evaluation
** * build for snapshots ** * build for snapshots
** * stage-release: for release candidates and for health checks ** * stage-release: for release candidates and for health checks
** * operate: for final release ** * release: for GA release
*/ */
def nexusDefaultIqStage = "build" def nexusDefaultIqStage = "build"
if (isReleaseTag) { if (isReleaseTag) {
@ -47,40 +39,64 @@ def nexusIqStageChoices = [nexusDefaultIqStage].plus(
'operate' 'operate'
].minus([nexusDefaultIqStage])) ].minus([nexusDefaultIqStage]))
/**
* Common Gradle arguments for all Gradle executions
*/
String COMMON_GRADLE_PARAMS = [
'--no-daemon',
'--stacktrace',
'--info',
'-Pcompilation.warningsAsErrors=false',
'-Ptests.failFast=true',
].join(' ')
pipeline { pipeline {
agent { label 'k8s' } agent { label 'standard' }
/*
* List options in alphabetical order
*/
options { options {
timestamps()
disableConcurrentBuilds()
timeout(time: 3, unit: 'HOURS')
buildDiscarder(logRotator(daysToKeepStr: '14', artifactDaysToKeepStr: '14')) buildDiscarder(logRotator(daysToKeepStr: '14', artifactDaysToKeepStr: '14'))
parallelsAlwaysFailFast()
timeout(time: 6, unit: 'HOURS')
timestamps()
} }
parameters { parameters {
choice choices: nexusIqStageChoices, description: 'NexusIQ stage for code evaluation', name: 'nexusIqStage' choice choices: nexusIqStageChoices, description: 'NexusIQ stage for code evaluation', name: 'nexusIqStage'
} }
/*
* List environment variables in alphabetical order
*/
environment { environment {
DOCKER_TAG_TO_USE = "${env.GIT_COMMIT.subSequence(0, 8)}" ARTIFACTORY_BUILD_NAME = "Corda :: Publish :: Publish Release to Artifactory :: ${env.BRANCH_NAME}"
DOCKER_URL = "https://index.docker.io/v1/"
EXECUTOR_NUMBER = "${env.EXECUTOR_NUMBER}"
BUILD_ID = "${env.BUILD_ID}-${env.JOB_NAME}"
ARTIFACTORY_CREDENTIALS = credentials('artifactory-credentials') ARTIFACTORY_CREDENTIALS = credentials('artifactory-credentials')
ARTIFACTORY_BUILD_NAME = "Corda / Publish / Publish Release to Artifactory".replaceAll("/", "::")
CORDA_ARTIFACTORY_USERNAME = "${env.ARTIFACTORY_CREDENTIALS_USR}"
CORDA_ARTIFACTORY_PASSWORD = "${env.ARTIFACTORY_CREDENTIALS_PSW}" CORDA_ARTIFACTORY_PASSWORD = "${env.ARTIFACTORY_CREDENTIALS_PSW}"
CORDA_ARTIFACTORY_USERNAME = "${env.ARTIFACTORY_CREDENTIALS_USR}"
DOCKER_URL = "https://index.docker.io/v1/"
} }
stages { stages {
/* stage('Compile') {
* Temporarily disable Sonatype checks for regression builds steps {
*/ sh script: [
stage('Sonatype Check') { './gradlew',
when { COMMON_GRADLE_PARAMS,
expression { isReleaseTag } 'clean',
} 'jar'
].join(' ')
}
}
stage('Stash') {
steps {
stash name: 'compiled', useDefaultExcludes: false
}
}
stage('Sonatype Check') {
steps { steps {
sh "./gradlew --no-daemon clean jar"
script { script {
sh "./gradlew --no-daemon properties | grep -E '^(version|group):' >version-properties" sh "./gradlew --no-daemon properties | grep -E '^(version|group):' >version-properties"
/* every build related to Corda X.Y (GA, RC, HC, patch or snapshot) uses the same NexusIQ application */ /* every build related to Corda X.Y (GA, RC, HC, patch or snapshot) uses the same NexusIQ application */
@ -98,82 +114,130 @@ pipeline {
} }
} }
stage('Deploy Nodes') { stage('All Tests') {
steps {
sh "./gradlew --no-daemon jar deployNodes"
}
}
stage('Generate Build Image') {
steps {
withCredentials([string(credentialsId: 'container_reg_passwd', variable: 'DOCKER_PUSH_PWD')]) {
sh "./gradlew " +
"-Dkubenetize=true " +
"-Ddocker.push.password=\"\${DOCKER_PUSH_PWD}\" " +
"-Ddocker.work.dir=\"/tmp/\${EXECUTOR_NUMBER}\" " +
"-Ddocker.container.env.parameter.CORDA_ARTIFACTORY_USERNAME=\"\${ARTIFACTORY_CREDENTIALS_USR}\" " +
"-Ddocker.container.env.parameter.CORDA_ARTIFACTORY_PASSWORD=\"\${ARTIFACTORY_CREDENTIALS_PSW}\" " +
"-Ddocker.build.tag=\"\${DOCKER_TAG_TO_USE}\"" +
" clean preAllocateForAllParallelUnitTest preAllocateForAllParallelIntegrationTest " +
" preAllocateForAllParallelSlowIntegrationTest preAllocateForAllParallelSmokeTest " +
" pushBuildImage --stacktrace"
}
sh "kubectl auth can-i get pods"
}
}
stage('Testing phase') {
parallel { parallel {
stage('Another agent') {
agent {
label 'standard'
}
options {
skipDefaultCheckout true
}
post {
always {
archiveArtifacts artifacts: '**/*.log', fingerprint: false
junit testResults: '**/build/test-results/**/*.xml', keepLongStdio: true
/*
* Copy all JUnit results files into a single top level directory.
* This is necessary to stop the allure plugin from hitting out
* of memory errors due to being passed many directories with
* long paths.
*
* File names are pre-pended with a prefix when
* copied to avoid collisions between files where the same test
* classes have run on multiple agents.
*/
fileOperations([fileCopyOperation(
includes: '**/build/test-results/**/*.xml',
targetLocation: 'allure-input',
flattenFiles: true,
renameFiles: true,
sourceCaptureExpression: '.*/([^/]+)$',
targetNameExpression: 'other-agent-$1')])
stash name: 'allure-input', includes: 'allure-input/**', useDefaultExcludes: false
}
cleanup {
deleteDir() /* clean up our workspace */
}
}
stages {
stage('Unstash') {
steps {
unstash 'compiled'
}
}
stage('Recompile') {
steps {
sh script: [
'./gradlew',
COMMON_GRADLE_PARAMS,
'jar'
].join(' ')
}
}
stage('Unit Test') { stage('Unit Test') {
steps { steps {
sh "./gradlew " + sh script: [
"-DbuildId=\"\${BUILD_ID}\" " + './gradlew',
"-Dkubenetize=true " + COMMON_GRADLE_PARAMS,
"-Ddocker.run.tag=\"\${DOCKER_TAG_TO_USE}\" " + 'test'
"-Dartifactory.username=\"\${ARTIFACTORY_CREDENTIALS_USR}\" " + ].join(' ')
"-Dartifactory.password=\"\${ARTIFACTORY_CREDENTIALS_PSW}\" " +
"-Dgit.branch=\"\${GIT_BRANCH}\" " +
"-Dgit.target.branch=\"\${GIT_BRANCH}\" " +
" allParallelUnitTest --stacktrace"
}
}
stage('Integration Test') {
steps {
sh "./gradlew " +
"-DbuildId=\"\${BUILD_ID}\" " +
"-Dkubenetize=true " +
"-Ddocker.run.tag=\"\${DOCKER_TAG_TO_USE}\" " +
"-Dartifactory.username=\"\${ARTIFACTORY_CREDENTIALS_USR}\" " +
"-Dartifactory.password=\"\${ARTIFACTORY_CREDENTIALS_PSW}\" " +
"-Dgit.branch=\"\${GIT_BRANCH}\" " +
"-Dgit.target.branch=\"\${GIT_BRANCH}\" " +
" allParallelIntegrationTest --stacktrace"
} }
} }
stage('Smoke Test') { stage('Smoke Test') {
steps { steps {
sh "./gradlew " + sh script: [
"-DbuildId=\"\${BUILD_ID}\" " + './gradlew',
"-Dkubenetize=true " + COMMON_GRADLE_PARAMS,
"-Ddocker.run.tag=\"\${DOCKER_TAG_TO_USE}\" " + 'smokeTest'
"-Dartifactory.username=\"\${ARTIFACTORY_CREDENTIALS_USR}\" " + ].join(' ')
"-Dartifactory.password=\"\${ARTIFACTORY_CREDENTIALS_PSW}\" " +
"-Dgit.branch=\"\${GIT_BRANCH}\" " +
"-Dgit.target.branch=\"\${GIT_BRANCH}\" " +
" allParallelSmokeTest --stacktrace"
} }
} }
stage('Slow Integration Test') { stage('Slow Integration Test') {
steps { steps {
sh "./gradlew " + sh script: [
"-DbuildId=\"\${BUILD_ID}\" " + './gradlew',
"-Dkubenetize=true " + COMMON_GRADLE_PARAMS,
"-Ddocker.run.tag=\"\${DOCKER_TAG_TO_USE}\" " + 'slowIntegrationTest'
"-Dartifactory.username=\"\${ARTIFACTORY_CREDENTIALS_USR}\" " + ].join(' ')
"-Dartifactory.password=\"\${ARTIFACTORY_CREDENTIALS_PSW}\" " + }
"-Dgit.branch=\"\${GIT_BRANCH}\" " + }
"-Dgit.target.branch=\"\${GIT_BRANCH}\" " + }
" allParallelSlowIntegrationTest --stacktrace" }
stage('Same agent') {
post {
always {
archiveArtifacts artifacts: '**/*.log', fingerprint: false
junit testResults: '**/build/test-results/**/*.xml', keepLongStdio: true
/*
* Copy all JUnit results files into a single top level directory.
* This is necessary to stop the allure plugin from hitting out
* of memory errors due to being passed many directories with
* long paths.
*
* File names are pre-pended with a prefix when
* copied to avoid collisions between files where the same test
* classes have run on multiple agents.
*/
fileOperations([fileCopyOperation(
includes: '**/build/test-results/**/*.xml',
targetLocation: 'allure-input',
flattenFiles: true,
renameFiles: true,
sourceCaptureExpression: '.*/([^/]+)$',
targetNameExpression: 'same-agent-$1')])
}
}
stages {
stage('Integration Test') {
steps {
sh script: [
'./gradlew',
COMMON_GRADLE_PARAMS,
'integrationTest'
].join(' ')
}
}
stage('Deploy Node') {
steps {
sh script: [
'./gradlew',
COMMON_GRADLE_PARAMS,
'deployNode'
].join(' ')
}
}
} }
} }
} }
@ -211,14 +275,19 @@ pipeline {
stage('Publish Release to Docker Hub') { stage('Publish Release to Docker Hub') {
when { when {
expression { !isInternalRelease && isReleaseTag } expression { isReleaseTag && !isInternalRelease && !isReleaseCandidate}
} }
steps { steps {
withCredentials([ withCredentials([
usernamePassword(credentialsId: 'corda-publisher-docker-hub-credentials', usernamePassword(credentialsId: 'corda-publisher-docker-hub-credentials',
usernameVariable: 'DOCKER_USERNAME', usernameVariable: 'DOCKER_USERNAME',
passwordVariable: 'DOCKER_PASSWORD')]) { passwordVariable: 'DOCKER_PASSWORD')
sh "./gradlew pushOfficialImages" ]) {
sh script: [
'./gradlew',
COMMON_GRADLE_PARAMS,
'pushOfficialImages'
].join(' ')
} }
} }
} }
@ -226,28 +295,9 @@ pipeline {
post { post {
always { always {
archiveArtifacts artifacts: '**/pod-logs/**/*.log', fingerprint: false
junit testResults: '**/build/test-results-xml/**/*.xml', keepLongStdio: true
script { script {
try { try {
/* unstash 'allure-input'
* Copy all JUnit results files into a single top level directory.
* This is necessary to stop the allure plugin from hitting out
* of memory errors due to being passed many directories with
* long paths.
*
* File names are pre-pended with the pod number when
* copied to avoid collisions between files where the same test
* classes have run on multiple pods.
*/
fileOperations([fileCopyOperation(
includes: '**/test-results-xml/**/test-runs/test-reports/**',
targetLocation: 'allure-input',
flattenFiles: true,
renameFiles: true,
sourceCaptureExpression: '.*test-results-xml/.*-([\\d]+)/.*/([^/]+)$',
targetNameExpression: '$1-$2')])
allure includeProperties: false, allure includeProperties: false,
jdk: '', jdk: '',
results: [[path: '**/allure-input']] results: [[path: '**/allure-input']]
@ -267,7 +317,7 @@ pipeline {
// Comparing the dates of the previous and current builds achieves this, // Comparing the dates of the previous and current builds achieves this,
// i.e. we will only send an email for the first build on a given day. // i.e. we will only send an email for the first build on a given day.
def prevBuildDate = new Date( def prevBuildDate = new Date(
currentBuild?.previousBuild.timeInMillis ?: 0).clearTime() currentBuild.previousBuild?.timeInMillis ?: 0).clearTime()
def currentBuildDate = new Date( def currentBuildDate = new Date(
currentBuild.timeInMillis).clearTime() currentBuild.timeInMillis).clearTime()
@ -278,6 +328,8 @@ pipeline {
statusSymbol = '\u2705' statusSymbol = '\u2705'
break; break;
case 'UNSTABLE': case 'UNSTABLE':
statusSymbol = '\u26A0'
break;
case 'FAILURE': case 'FAILURE':
statusSymbol = '\u274c' statusSymbol = '\u274c'
break; break;