2019-02-16 04:16:49 +08:00
|
|
|
#!/usr/bin/env groovy
|
|
|
|
|
2019-03-13 23:54:18 +08:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2019 - present Instructure, Inc.
|
|
|
|
*
|
|
|
|
* This file is part of Canvas.
|
|
|
|
*
|
|
|
|
* Canvas is free software: you can redistribute it and/or modify it under
|
|
|
|
* the terms of the GNU Affero General Public License as published by the Free
|
|
|
|
* Software Foundation, version 3 of the License.
|
|
|
|
*
|
|
|
|
* Canvas is distributed in the hope that it will be useful, but WITHOUT ANY
|
|
|
|
* WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR
|
|
|
|
* A PARTICULAR PURPOSE. See the GNU Affero General Public License for more
|
|
|
|
* details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Affero General Public License along
|
|
|
|
* with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
2019-08-15 02:43:09 +08:00
|
|
|
def build_parameters = [
|
|
|
|
string(name: 'GERRIT_REFSPEC', value: "${env.GERRIT_REFSPEC}"),
|
|
|
|
string(name: 'GERRIT_EVENT_TYPE', value: "${env.GERRIT_EVENT_TYPE}"),
|
2020-01-18 04:26:48 +08:00
|
|
|
string(name: 'GERRIT_PROJECT', value: "${env.GERRIT_PROJECT}"),
|
2019-08-15 02:43:09 +08:00
|
|
|
string(name: 'GERRIT_BRANCH', value: "${env.GERRIT_BRANCH}"),
|
|
|
|
string(name: 'GERRIT_CHANGE_NUMBER', value: "${env.GERRIT_CHANGE_NUMBER}"),
|
|
|
|
string(name: 'GERRIT_PATCHSET_NUMBER', value: "${env.GERRIT_PATCHSET_NUMBER}"),
|
|
|
|
string(name: 'GERRIT_EVENT_ACCOUNT_NAME', value: "${env.GERRIT_EVENT_ACCOUNT_NAME}"),
|
2019-11-16 02:12:18 +08:00
|
|
|
string(name: 'GERRIT_EVENT_ACCOUNT_EMAIL', value: "${env.GERRIT_EVENT_ACCOUNT_EMAIL}"),
|
|
|
|
string(name: 'GERRIT_CHANGE_COMMIT_MESSAGE', value: "${env.GERRIT_CHANGE_COMMIT_MESSAGE}"),
|
|
|
|
string(name: 'GERRIT_HOST', value: "${env.GERRIT_HOST}"),
|
2019-12-20 02:55:08 +08:00
|
|
|
string(name: 'GERGICH_PUBLISH', value: "${env.GERGICH_PUBLISH}"),
|
|
|
|
string(name: 'MASTER_BOUNCER_RUN', value: "${env.MASTER_BOUNCER_RUN}")
|
2019-08-15 02:43:09 +08:00
|
|
|
]
|
|
|
|
|
2019-11-16 02:12:18 +08:00
|
|
|
def getImageTagVersion() {
|
|
|
|
def flags = load 'build/new-jenkins/groovy/commit-flags.groovy'
|
|
|
|
return flags.getImageTagVersion()
|
|
|
|
}
|
|
|
|
|
2020-01-03 00:39:27 +08:00
|
|
|
def runBuildImageMaybe(saveSuccess, block) {
|
2019-11-16 02:12:18 +08:00
|
|
|
def flags = load 'build/new-jenkins/groovy/commit-flags.groovy'
|
|
|
|
if (flags.hasFlag('skip-docker-build')) {
|
2020-01-03 00:39:27 +08:00
|
|
|
echo "Skip building image requested"
|
|
|
|
} else {
|
2019-11-23 04:57:11 +08:00
|
|
|
def skips = load 'build/new-jenkins/groovy/successes.groovy'
|
2020-01-03 00:39:27 +08:00
|
|
|
skips.skipIfPreviouslySuccessful("build-and-push-image", saveSuccess, block)
|
2019-11-16 02:12:18 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-11-23 04:57:11 +08:00
|
|
|
def skipIfPreviouslySuccessful(name, block) {
|
2020-01-03 00:39:27 +08:00
|
|
|
def skips = load 'build/new-jenkins/groovy/successes.groovy'
|
|
|
|
skips.skipIfPreviouslySuccessful(name, true, block)
|
2019-11-23 04:57:11 +08:00
|
|
|
}
|
|
|
|
|
2019-02-16 04:16:49 +08:00
|
|
|
pipeline {
|
2019-09-07 02:03:00 +08:00
|
|
|
agent { label 'canvas-docker' }
|
2019-02-16 04:16:49 +08:00
|
|
|
|
2019-02-21 01:06:06 +08:00
|
|
|
options {
|
|
|
|
ansiColor('xterm')
|
|
|
|
}
|
2019-02-20 23:41:00 +08:00
|
|
|
|
2019-02-21 01:06:06 +08:00
|
|
|
environment {
|
2019-08-22 21:21:41 +08:00
|
|
|
// include selenium while smoke is running locally
|
|
|
|
COMPOSE_FILE = 'docker-compose.new-jenkins.yml:docker-compose.new-jenkins-selenium.yml'
|
2019-03-12 09:55:34 +08:00
|
|
|
GERRIT_PORT = '29418'
|
2019-02-21 01:22:05 +08:00
|
|
|
GERRIT_URL = "$GERRIT_HOST:$GERRIT_PORT"
|
2019-08-10 06:30:33 +08:00
|
|
|
|
2019-11-16 02:12:18 +08:00
|
|
|
NAME = getImageTagVersion()
|
2019-03-15 01:44:25 +08:00
|
|
|
PATCHSET_TAG = "$DOCKER_REGISTRY_FQDN/jenkins/canvas-lms:$NAME"
|
|
|
|
MERGE_TAG = "$DOCKER_REGISTRY_FQDN/jenkins/canvas-lms:$GERRIT_BRANCH"
|
2019-09-27 05:00:38 +08:00
|
|
|
CACHE_TAG = "canvas-lms:previous-image"
|
2019-10-16 05:12:45 +08:00
|
|
|
POSTGRES_CACHE_TAG = "canvas-lms:previous-postgres-image"
|
|
|
|
CASSANDRA_CACHE_TAG = "canvas-lms:previous-cassandra-image"
|
|
|
|
DYNAMODB_CACHE_TAG = "canvas-lms:previous-dynamodb-image"
|
2019-02-21 01:06:06 +08:00
|
|
|
}
|
2019-02-16 04:16:49 +08:00
|
|
|
|
2019-02-21 01:06:06 +08:00
|
|
|
stages {
|
2019-07-29 23:55:07 +08:00
|
|
|
stage('Print Env Variables') {
|
2019-02-26 00:14:35 +08:00
|
|
|
steps {
|
2019-03-12 09:55:34 +08:00
|
|
|
timeout(time: 20, unit: 'SECONDS') {
|
2020-01-30 05:43:54 +08:00
|
|
|
sh 'build/new-jenkins/print-env-excluding-secrets.sh'
|
2019-03-12 09:55:34 +08:00
|
|
|
}
|
2019-02-26 00:14:35 +08:00
|
|
|
}
|
|
|
|
}
|
2019-02-21 01:22:05 +08:00
|
|
|
|
2019-12-10 02:37:05 +08:00
|
|
|
stage ('Pre-Cleanup') {
|
|
|
|
steps {
|
|
|
|
timeout(time: 2) {
|
|
|
|
sh 'build/new-jenkins/docker-cleanup.sh'
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-03-12 09:55:34 +08:00
|
|
|
stage('Plugins and Config Files') {
|
|
|
|
steps {
|
|
|
|
timeout(time: 3) {
|
|
|
|
script {
|
2019-12-06 06:28:52 +08:00
|
|
|
if (env.CANVAS_LMS_REFSPEC) {
|
|
|
|
// the plugin builds require the canvas lms refspec to be different. so only
|
|
|
|
// set this refspec if the main build is requesting it to be set.
|
|
|
|
// NOTE: this is only being set in main-from-plugin build. so main-canvas wont run this.
|
|
|
|
build_parameters += string(name: 'CANVAS_LMS_REFSPEC', value: "${env.CANVAS_LMS_REFSPEC}")
|
|
|
|
}
|
2019-11-16 02:12:18 +08:00
|
|
|
def credentials = load 'build/new-jenkins/groovy/credentials.groovy'
|
|
|
|
credentials.fetchFromGerrit('gerrit_builder', '.', '', 'canvas-lms/config')
|
2019-08-22 03:07:46 +08:00
|
|
|
gems = readFile('gerrit_builder/canvas-lms/config/plugins_list').split()
|
|
|
|
println "Plugin list: ${gems}"
|
|
|
|
/* fetch plugins */
|
2019-10-22 03:55:38 +08:00
|
|
|
gems.each { gem ->
|
|
|
|
if (env.GERRIT_PROJECT == gem) {
|
|
|
|
/* this is the commit we're testing */
|
2019-11-16 02:12:18 +08:00
|
|
|
credentials.fetchFromGerrit(gem, 'gems/plugins', null, null, env.GERRIT_REFSPEC)
|
2019-10-22 03:55:38 +08:00
|
|
|
} else {
|
2019-11-16 02:12:18 +08:00
|
|
|
credentials.fetchFromGerrit(gem, 'gems/plugins')
|
2019-10-22 03:55:38 +08:00
|
|
|
}
|
|
|
|
}
|
2019-12-19 04:27:32 +08:00
|
|
|
credentials.fetchFromGerrit('gergich_user_config', '.')
|
2019-11-16 02:12:18 +08:00
|
|
|
credentials.fetchFromGerrit('qti_migration_tool', 'vendor', 'QTIMigrationTool')
|
2019-08-22 03:07:46 +08:00
|
|
|
sh '''
|
2019-12-07 06:32:11 +08:00
|
|
|
mv -v gerrit_builder/canvas-lms/config/* config/
|
|
|
|
mv -v config/knapsack_rspec_report.json ./
|
|
|
|
rm -v config/cache_store.yml
|
2019-08-22 03:07:46 +08:00
|
|
|
rmdir -p gerrit_builder/canvas-lms/config
|
2019-12-07 06:32:11 +08:00
|
|
|
cp -v docker-compose/config/selenium.yml config/
|
|
|
|
cp -vR docker-compose/config/new-jenkins config/new-jenkins
|
|
|
|
cp -v config/delayed_jobs.yml.example config/delayed_jobs.yml
|
|
|
|
cp -v config/domain.yml.example config/domain.yml
|
|
|
|
cp -v config/external_migration.yml.example config/external_migration.yml
|
|
|
|
cp -v config/outgoing_mail.yml.example config/outgoing_mail.yml
|
2019-12-19 04:27:32 +08:00
|
|
|
cp -v ./gergich_user_config/gergich_user_config.yml ./gems/dr_diff/config/gergich_user_config.yml
|
2019-08-22 03:07:46 +08:00
|
|
|
'''
|
2019-02-21 01:22:05 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-03-05 05:09:25 +08:00
|
|
|
stage('Rebase') {
|
2020-01-21 01:18:19 +08:00
|
|
|
when { expression { env.GERRIT_EVENT_TYPE == 'patchset-created' && env.GERRIT_PROJECT == 'canvas-lms' } }
|
2019-03-05 05:09:25 +08:00
|
|
|
steps {
|
2019-03-20 23:16:16 +08:00
|
|
|
timeout(time: 2) {
|
2019-09-04 03:29:25 +08:00
|
|
|
script {
|
2019-11-16 02:12:18 +08:00
|
|
|
def credentials = load 'build/new-jenkins/groovy/credentials.groovy'
|
|
|
|
credentials.withGerritCredentials({ ->
|
2019-09-04 03:29:25 +08:00
|
|
|
sh '''
|
|
|
|
GIT_SSH_COMMAND='ssh -i \"$SSH_KEY_PATH\" -l \"$SSH_USERNAME\"' \
|
|
|
|
git fetch origin $GERRIT_BRANCH
|
|
|
|
|
2019-09-27 06:33:42 +08:00
|
|
|
git config user.name "$GERRIT_EVENT_ACCOUNT_NAME"
|
|
|
|
git config user.email "$GERRIT_EVENT_ACCOUNT_EMAIL"
|
2019-09-04 03:29:25 +08:00
|
|
|
|
|
|
|
# this helps current build issues where cleanup is needed before proceeding.
|
|
|
|
# however the later git rebase --abort should be enough once this has
|
|
|
|
# been on jenkins for long enough to hit all nodes, maybe a couple days?
|
|
|
|
if [ -d .git/rebase-merge ]; then
|
|
|
|
echo "A previous build's rebase failed and the build exited without cleaning up. Aborting the previous rebase now..."
|
|
|
|
git rebase --abort
|
2019-09-12 05:38:18 +08:00
|
|
|
git checkout $GERRIT_REFSPEC
|
2019-09-04 03:29:25 +08:00
|
|
|
fi
|
|
|
|
|
|
|
|
# store exit_status inline to ensures the script doesn't exit here on failures
|
|
|
|
git rebase --preserve-merges origin/$GERRIT_BRANCH; exit_status=$?
|
|
|
|
if [ $exit_status != 0 ]; then
|
|
|
|
echo "Warning: Rebase couldn't resolve changes automatically, please resolve these conflicts locally."
|
|
|
|
git rebase --abort
|
|
|
|
exit $exit_status
|
|
|
|
fi
|
|
|
|
'''
|
|
|
|
})
|
|
|
|
}
|
2019-03-20 23:16:16 +08:00
|
|
|
}
|
2019-03-05 05:09:25 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-02-21 01:22:05 +08:00
|
|
|
stage('Build Image') {
|
2019-02-21 01:06:06 +08:00
|
|
|
steps {
|
2019-11-16 02:12:18 +08:00
|
|
|
runBuildImageMaybe(false) {
|
2019-10-31 23:22:09 +08:00
|
|
|
timeout(time: 36) { /* this timeout is `2 * average build time` which currently: 18m * 2 = 36m */
|
2020-01-03 00:39:27 +08:00
|
|
|
script {
|
|
|
|
def flags = load 'build/new-jenkins/groovy/commit-flags.groovy'
|
|
|
|
if (!flags.hasFlag('skip-cache')) {
|
|
|
|
dockerCacheLoad(image: "$CACHE_TAG")
|
|
|
|
}
|
|
|
|
sh 'docker build -t $PATCHSET_TAG .'
|
|
|
|
sh 'docker tag $PATCHSET_TAG $CACHE_TAG'
|
|
|
|
}
|
2019-10-31 23:22:09 +08:00
|
|
|
}
|
2019-02-16 04:16:49 +08:00
|
|
|
}
|
2019-02-21 01:06:06 +08:00
|
|
|
}
|
|
|
|
}
|
2019-02-16 04:16:49 +08:00
|
|
|
|
2019-07-29 23:55:07 +08:00
|
|
|
stage('Publish Patchset Image') {
|
2019-03-20 23:16:16 +08:00
|
|
|
steps {
|
2019-11-16 02:12:18 +08:00
|
|
|
runBuildImageMaybe(true) {
|
2019-10-31 23:22:09 +08:00
|
|
|
timeout(time: 5) {
|
|
|
|
// always push the patchset tag otherwise when a later
|
|
|
|
// patchset is merged this patchset tag is overwritten
|
|
|
|
sh 'docker push $PATCHSET_TAG'
|
|
|
|
}
|
2019-07-29 23:55:07 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
stage('Parallel Run Tests') {
|
|
|
|
parallel {
|
2019-11-06 02:02:37 +08:00
|
|
|
stage('Linters') {
|
2020-01-21 01:18:19 +08:00
|
|
|
when { expression { env.GERRIT_EVENT_TYPE != 'change-merged' && env.GERRIT_PROJECT == 'canvas-lms' } }
|
2019-11-06 02:02:37 +08:00
|
|
|
steps {
|
|
|
|
skipIfPreviouslySuccessful("linters") {
|
|
|
|
build(
|
|
|
|
job: 'test-suites/linters',
|
|
|
|
parameters: build_parameters
|
|
|
|
)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-11-08 01:09:34 +08:00
|
|
|
stage('Vendored Gems') {
|
|
|
|
steps {
|
2019-10-31 23:22:09 +08:00
|
|
|
skipIfPreviouslySuccessful("vendored-gems") {
|
|
|
|
build(
|
|
|
|
job: 'test-suites/vendored-gems',
|
2019-11-19 02:32:31 +08:00
|
|
|
parameters: build_parameters
|
2019-10-31 23:22:09 +08:00
|
|
|
)
|
|
|
|
}
|
2019-11-08 01:09:34 +08:00
|
|
|
}
|
|
|
|
}
|
2019-11-24 04:34:22 +08:00
|
|
|
|
|
|
|
stage('JS') {
|
|
|
|
steps {
|
|
|
|
skipIfPreviouslySuccessful("js") {
|
|
|
|
build(
|
|
|
|
job: 'test-suites/JS',
|
|
|
|
parameters: build_parameters
|
|
|
|
)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2019-11-27 00:36:22 +08:00
|
|
|
|
|
|
|
stage('Contract Tests') {
|
|
|
|
steps {
|
|
|
|
skipIfPreviouslySuccessful("contract-tests") {
|
|
|
|
build(
|
|
|
|
job: 'test-suites/contract-tests',
|
|
|
|
parameters: build_parameters
|
|
|
|
)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2019-12-06 06:28:52 +08:00
|
|
|
|
2019-12-05 03:57:36 +08:00
|
|
|
stage('Selenium Chrome') {
|
|
|
|
steps {
|
|
|
|
skipIfPreviouslySuccessful("selenium-chrome") {
|
|
|
|
build(
|
|
|
|
job: 'test-suites/selenium-chrome',
|
|
|
|
parameters: build_parameters
|
|
|
|
)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-11-16 01:49:57 +08:00
|
|
|
stage('Rspec') {
|
|
|
|
steps {
|
|
|
|
skipIfPreviouslySuccessful("rspec") {
|
|
|
|
build(
|
|
|
|
job: 'test-suites/rspec',
|
|
|
|
parameters: build_parameters
|
|
|
|
)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-11-25 14:06:34 +08:00
|
|
|
stage ('Flakey Spec Catcher') {
|
2020-02-14 23:20:23 +08:00
|
|
|
when { expression { env.GERRIT_EVENT_TYPE != 'change-merged' } }
|
|
|
|
steps {
|
|
|
|
skipIfPreviouslySuccessful("flakey_spec_catcher") {
|
2019-11-25 14:06:34 +08:00
|
|
|
build(
|
|
|
|
job: 'test-suites/flakey-spec-catcher',
|
2020-02-28 00:55:06 +08:00
|
|
|
parameters: build_parameters,
|
|
|
|
propagate: false,
|
|
|
|
wait: false
|
2019-11-25 14:06:34 +08:00
|
|
|
)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-12-06 06:28:52 +08:00
|
|
|
// keep this around in case there is changes to the subbuilds that need to happen
|
|
|
|
// and you have no other way to test it except by running a test build.
|
|
|
|
// stage('Test Subbuild') {
|
|
|
|
// steps {
|
|
|
|
// skipIfPreviouslySuccessful("test-subbuild") {
|
|
|
|
// build(
|
|
|
|
// job: 'test-suites/test-subbuild',
|
|
|
|
// parameters: build_parameters
|
|
|
|
// )
|
|
|
|
// }
|
|
|
|
// }
|
|
|
|
// }
|
2019-10-18 04:04:26 +08:00
|
|
|
/*
|
|
|
|
* Don't run these on all patch sets until we have them ready to report results.
|
|
|
|
* Uncomment stage to run when developing.
|
|
|
|
*
|
|
|
|
*
|
|
|
|
* stage('Xbrowser') {
|
|
|
|
* steps {
|
2019-10-31 23:22:09 +08:00
|
|
|
* skipIfPreviouslySuccessful("xbrowser") {
|
|
|
|
* // propagate set to false until we can get tests passing
|
|
|
|
* build(
|
|
|
|
* job: 'test-suites/xbrowser',
|
|
|
|
* propagate: false,
|
|
|
|
* parameters: build_parameters
|
|
|
|
* )
|
|
|
|
* }
|
2019-10-18 04:04:26 +08:00
|
|
|
* }
|
2019-10-31 23:22:09 +08:00
|
|
|
* }
|
2019-10-18 04:04:26 +08:00
|
|
|
*/
|
2019-03-20 23:16:16 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-07-29 23:55:07 +08:00
|
|
|
stage('Publish Merged Image') {
|
2019-11-20 02:24:38 +08:00
|
|
|
when { expression { env.GERRIT_EVENT_TYPE == 'change-merged' } }
|
2019-02-21 01:06:06 +08:00
|
|
|
steps {
|
2019-11-20 02:24:38 +08:00
|
|
|
timeout(time: 10) {
|
2019-03-15 01:44:25 +08:00
|
|
|
script {
|
2020-03-03 02:56:47 +08:00
|
|
|
// Retriggers won't have an image to tag/push, pull that image if doesn't exist
|
|
|
|
// If image is not found it will return NULL
|
|
|
|
if (!sh (script: 'docker images -q $PATCHSET_TAG')) {
|
|
|
|
sh 'docker pull $PATCHSET_TAG'
|
|
|
|
}
|
2019-11-20 02:24:38 +08:00
|
|
|
sh '''
|
|
|
|
docker tag $PATCHSET_TAG $MERGE_TAG
|
|
|
|
docker push $MERGE_TAG
|
|
|
|
'''
|
|
|
|
dockerCacheStore(image: "$CACHE_TAG")
|
2019-03-15 01:44:25 +08:00
|
|
|
}
|
2019-02-16 04:16:49 +08:00
|
|
|
}
|
2019-02-21 01:06:06 +08:00
|
|
|
}
|
2019-02-16 04:16:49 +08:00
|
|
|
}
|
2019-02-21 01:06:06 +08:00
|
|
|
}
|
2019-03-12 09:55:34 +08:00
|
|
|
|
|
|
|
post {
|
2019-11-21 02:53:17 +08:00
|
|
|
failure {
|
|
|
|
script {
|
|
|
|
if ( env.GERRIT_EVENT_TYPE == 'change-merged' ) {
|
2020-01-03 00:39:27 +08:00
|
|
|
slackSend(
|
|
|
|
channel: '#canvas_builds',
|
|
|
|
color: 'danger',
|
|
|
|
message: "${env.JOB_NAME} failed on merge (<${env.BUILD_URL}|${env.BUILD_NUMBER}>)"
|
|
|
|
)
|
2019-11-21 02:53:17 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2019-12-18 01:39:44 +08:00
|
|
|
success {
|
|
|
|
script {
|
|
|
|
def successes = load 'build/new-jenkins/groovy/successes.groovy'
|
|
|
|
successes.markBuildAsSuccessful()
|
|
|
|
}
|
|
|
|
}
|
2019-03-20 23:16:16 +08:00
|
|
|
cleanup {
|
2019-12-18 01:39:44 +08:00
|
|
|
sh 'build/new-jenkins/docker-cleanup.sh --allow-failure'
|
2019-03-20 23:16:16 +08:00
|
|
|
}
|
2019-03-12 09:55:34 +08:00
|
|
|
}
|
2019-02-16 04:16:49 +08:00
|
|
|
}
|