Skip to content
This repository has been archived by the owner on Nov 17, 2023. It is now read-only.

Refactor Jenkinsfiles #13344

Merged
merged 13 commits into from
Nov 21, 2018
26 changes: 26 additions & 0 deletions Jenkinsfile
Original file line number Diff line number Diff line change
Expand Up @@ -20,6 +20,32 @@
// Jenkins pipeline
// See documents at https://jenkins.io/doc/book/pipeline/jenkinsfile/


/***
* _____ _
* | __ \| |
* | |__) | | ___ __ _ ___ ___
* | ___/| |/ _ \/ _` / __|/ _ \
* | | | || | __/ (_| \__ \ __/
* |_|_| ||_|\___|\__,_|___/\___|
* / _` |/ _ \
* | (_| | (_) |_
* \__,_|\___/| |
* _ __ ___ | |_
* | '_ \ / _ \| __| _ _ __
* | | | | (_) | |_ | (_)/ _|
* |_|_|_|\___/_\__| __| |_| |_ _ _
* | '_ ` _ \ / _ \ / _` | | _| | | |
* | | | | | | (_) | (_| | | | | |_| |
* |_| |_| |_|\___/ \__,_|_|_| \__, |
* __/ |
* |___/
*
* This file is about to be deprecated! See https://github.com/apache/incubator-mxnet/pull/13344
* for more details
*/


// mxnet libraries
mx_lib = 'lib/libmxnet.so, lib/libmxnet.a, 3rdparty/dmlc-core/libdmlc.a, 3rdparty/tvm/nnvm/lib/libnnvm.a'

Expand Down
105 changes: 95 additions & 10 deletions ci/Jenkinsfile_utils.groovy
Original file line number Diff line number Diff line change
Expand Up @@ -65,8 +65,10 @@ def init_git_win() {
// pack libraries for later use
def pack_lib(name, libs, include_gcov_data = false) {
sh """
set +e
echo "Packing ${libs} into ${name}"
echo ${libs} | sed -e 's/,/ /g' | xargs md5sum
return 0
"""
stash includes: libs, name: name

Expand All @@ -82,32 +84,43 @@ def unpack_and_init(name, libs, include_gcov_data = false) {
init_git()
unstash name
sh """
set +e
echo "Unpacked ${libs} from ${name}"
echo ${libs} | sed -e 's/,/ /g' | xargs md5sum
return 0
"""
if (include_gcov_data) {
// Restore GCNO files that are required for GCOV to operate during runtime
unstash "${name}_gcov_data"
}
}

def get_jenkins_master_url() {
return env.BUILD_URL.split('/')[2].split(':')[0]
}

def get_git_commit_hash() {
lastCommitMessage = sh (script: "git log -1 --pretty=%B", returnStdout: true)
lastCommitMessage = lastCommitMessage.trim()
if (lastCommitMessage.startsWith("Merge commit '") && lastCommitMessage.endsWith("' into HEAD")) {
// Merge commit applied by Jenkins, skip that commit
git_commit_hash = sh (script: "git rev-parse @~", returnStdout: true)
} else {
git_commit_hash = sh (script: "git rev-parse @", returnStdout: true)
}
return git_commit_hash
}

def publish_test_coverage() {
// CodeCovs auto detection has trouble with our CIs PR validation due the merging strategy
lastCommitMessage = sh (script: "git log -1 --pretty=%B", returnStdout: true)
lastCommitMessage = lastCommitMessage.trim()
if (lastCommitMessage.startsWith("Merge commit '") && lastCommitMessage.endsWith("' into HEAD")) {
// Merge commit applied by Jenkins, skip that commit
GIT_COMMIT_HASH = sh (script: "git rev-parse @~", returnStdout: true)
} else {
GIT_COMMIT_HASH = sh (script: "git rev-parse @", returnStdout: true)
}
git_commit_hash = get_git_commit_hash()

if (env.CHANGE_ID) {
// PR execution
codecovArgs = "-B ${env.CHANGE_TARGET} -C ${GIT_COMMIT_HASH} -P ${env.CHANGE_ID}"
codecovArgs = "-B ${env.CHANGE_TARGET} -C ${git_commit_hash} -P ${env.CHANGE_ID}"
} else {
// Branch execution
codecovArgs = "-B ${env.BRANCH_NAME} -C ${GIT_COMMIT_HASH}"
codecovArgs = "-B ${env.BRANCH_NAME} -C ${git_commit_hash}"
}

// To make sure we never fail because test coverage reporting is not available
Expand Down Expand Up @@ -144,9 +157,78 @@ def docker_run(platform, function_name, use_nvidia, shared_mem = '500m') {
sh command
}

// Allow publishing to GitHub with a custom context (the status shown under a PR)
// Credit to https://plugins.jenkins.io/github
def get_repo_url() {
checkout scm
sh "git config --get remote.origin.url > .git/remote-url"
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Why not .execute() writing a file doesn't seem the best way.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

There is no way to directly access shell output in Jenkins https://stackoverflow.com/questions/36507410/is-it-possible-to-capture-the-stdout-from-the-sh-dsl-command-in-the-pipeline

I could've used other methods, but that would then have triggered the security mechanisms of Jenkins due to the sandboxing mechanism. Execute (although I don't know exactly which function you're referring to) sounds like it would probably trigger that restriction.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Ah, they've recently added a redirect parameter. I've replaced it at #13355. Thanks for the heads up!

return readFile(".git/remote-url").trim()
}

def update_github_commit_status(state, message) {
node(NODE_LINUX_CPU) {
// NOTE: https://issues.jenkins-ci.org/browse/JENKINS-39482
//The GitHubCommitStatusSetter requires that the Git Server is defined under
//*Manage Jenkins > Configure System > GitHub > GitHub Servers*.
//Otherwise the GitHubCommitStatusSetter is not able to resolve the repository name
//properly and you would see an empty list of repos:
//[Set GitHub commit status (universal)] PENDING on repos [] (sha:xxxxxxx) with context:test/mycontext
//See https://cwiki.apache.org/confluence/display/MXNET/Troubleshooting#Troubleshooting-GitHubcommit/PRstatusdoesnotgetpublished
repoUrl = get_repo_url()
commitSha = get_git_commit_hash()
context = get_github_context()

step([
$class: 'GitHubCommitStatusSetter',
reposSource: [$class: "ManuallyEnteredRepositorySource", url: repoUrl],
contextSource: [$class: "ManuallyEnteredCommitContextSource", context: context],
commitShaSource: [$class: "ManuallyEnteredShaSource", sha: commitSha],
statusBackrefSource: [$class: "ManuallyEnteredBackrefSource", backref: "${env.RUN_DISPLAY_URL}"],
errorHandlers: [[$class: 'ShallowAnyErrorHandler']],
statusResultSource: [
$class: 'ConditionalStatusResultSource',
results: [[$class: "AnyBuildResult", message: message, state: state]]
]
])
}
}

def get_github_context() {
// Since we use multi-branch pipelines, Jenkins appends the branch name to the job name
if (JOB_NAME.contains('/')) {
short_job_name = JOB_NAME.split('/')[0]
} else {
short_job_name = JOB_NAME
}

return "ci/jenkins/${short_job_name}"
}

def parallel_stage(stage_name, steps) {
// Allow to pass an array of steps that will be executed in parallel in a stage
new_map = [:]

for (def step in steps) {
new_map = new_map << step
}

stage(stage_name) {
parallel new_map
}
}

def assign_node_labels(args) {
// This function allows to assign instance labels to the generalized placeholders.
// This serves two purposes:
// 1. Allow generalized placeholders (e.g. NODE_WINDOWS_CPU) in the job definition
// in order to abstract away the underlying node label. This allows to schedule a job
// onto a different node for testing or security reasons. This could be, for example,
// when you want to test a new set of slaves on separate labels or when a job should
// only be run on restricted slaves
// 2. Restrict the allowed job types within a Jenkinsfile. For example, a UNIX-CPU-only
// Jenkinsfile should not allowed access to Windows or GPU instances. This prevents
// users from just copy&pasting something into an existing Jenkinsfile without
// knowing about the limitations.
NODE_LINUX_CPU = args.linux_cpu
NODE_LINUX_GPU = args.linux_gpu
NODE_LINUX_GPU_P3 = args.linux_gpu_p3
Expand All @@ -164,15 +246,18 @@ def main_wrapper(args) {
// assign any caught errors here
err = null
try {
update_github_commit_status('PENDING', 'Job has been enqueued')
args['core_logic']()

// set build status to success at the end
currentBuild.result = "SUCCESS"
update_github_commit_status('SUCCESS', 'Job succeeded')
} catch (caughtError) {
node(NODE_LINUX_CPU) {
sh "echo caught ${caughtError}"
err = caughtError
currentBuild.result = "FAILURE"
update_github_commit_status('FAILURE', 'Job failed')
}
} finally {
node(NODE_LINUX_CPU) {
Expand Down
Loading