| //------------------------------------------------------------------------------------------------------- |
| // Copyright (C) Microsoft. All rights reserved. |
| // Licensed under the MIT license. See LICENSE.txt file in the project root for full license information. |
| //------------------------------------------------------------------------------------------------------- |
| |
| // Import the utility functionality. |
| import jobs.generation.Utilities |
| |
| // Grab the github project name passed in |
| def project = GithubProject |
| def branch = GithubBranchName |
| |
| def msbuildTypeMap = [ |
| 'debug':'chk', |
| 'test':'test', |
| 'release':'fre' |
| ] |
| |
| // convert `machine` parameter to OS component of PR task name |
| def machineTypeToOSTagMap = [ |
| 'Windows 7': 'Windows 7', // 'latest-or-auto' -> Windows Server 2008 R2 ~= Windows 7 |
| 'Windows_NT': 'Windows 8.1', // 'latest-or-auto' -> Windows Server 2012 R2 ~= Windows 8.1 aka Blue |
| 'windows.10.amd64.clientrs4.devex.open': 'Windows 10', // = Windows 10 RS4 with Dev 15.7 |
| 'Ubuntu16.04': 'Ubuntu', |
| 'OSX.1011.Amd64.Chakra.Open': 'OSX' |
| ] |
| |
| def defaultMachineTag = 'latest-or-auto' |
| |
| def legacyWindows7Machine = 'Windows 7' |
| def legacyWindows7MachineTag = defaultMachineTag |
| |
| def legacyWindows8Machine = 'Windows_NT' |
| def legacyWindows8MachineTag = defaultMachineTag |
| |
| def latestWindowsMachine = 'windows.10.amd64.clientrs4.devex.open' // Windows 10 RS4 with Dev 15.7 |
| def latestWindowsMachineTag = null // all information is included in the machine name above |
| |
| def dailyRegex = 'dailies' |
| |
| // --------------- |
| // HELPER CLOSURES |
| // --------------- |
| |
| def CreateBuildTask = { isPR, buildArch, buildType, machine, machineTag, configTag, buildExtra, testExtra, runCodeAnalysis, excludeConfigIf, nonDefaultTaskSetup -> |
| if (excludeConfigIf && excludeConfigIf(isPR, buildArch, buildType)) { |
| return // early exit: we don't want to create a job for this configuration |
| } |
| |
| def config = "${buildArch}_${buildType}" |
| config = (configTag == null) ? config : "${configTag}_${config}" |
| |
| // params: Project, BaseTaskName, IsPullRequest (appends '_prtest') |
| def jobName = Utilities.getFullJobName(project, config, isPR) |
| |
| def testableConfig = buildType in ['debug', 'test'] && buildArch != 'arm' |
| def analysisConfig = buildType in ['release'] && runCodeAnalysis |
| |
| def buildScript = "call .\\jenkins\\buildone.cmd ${buildArch} ${buildType} " |
| buildScript += buildExtra ?: '' |
| buildScript += analysisConfig ? ' "/p:runcodeanalysis=true"' : '' |
| def testScript = "call .\\jenkins\\testone.cmd ${buildArch} ${buildType} " |
| testScript += testExtra ?: '' |
| def analysisScript = '.\\Build\\scripts\\check_prefast_error.ps1 . CodeAnalysis.err' |
| |
| def newJob = job(jobName) { |
| // This opens the set of build steps that will be run. |
| // This looks strange, but it is actually a method call, with a |
| // closure as a param, since Groovy allows method calls without parens. |
| // (Compare with '.each' method used above.) |
| steps { |
| batchFile(buildScript) // run the parameter as if it were a batch file |
| if (testableConfig) { |
| batchFile(testScript) |
| } |
| if (analysisConfig) { |
| powerShell(analysisScript) |
| } |
| } |
| } |
| |
| def msbuildType = msbuildTypeMap.get(buildType) |
| def msbuildFlavor = "build_${buildArch}${msbuildType}" |
| def archivalString = "test/${msbuildFlavor}.*,test/logs/**" |
| archivalString += analysisConfig ? ',CodeAnalysis.err' : '' |
| Utilities.addArchival(newJob, archivalString, |
| '', // no exclusions from archival |
| false, // doNotFailIfNothingArchived=false ~= failIfNothingArchived |
| false) // archiveOnlyIfSuccessful=false ~= archiveAlways |
| |
| if (machineTag == null) { |
| // note: this is a different overload and not equivalent to calling setMachineAffinity(_,_,null) |
| Utilities.setMachineAffinity(newJob, machine) |
| } else { |
| Utilities.setMachineAffinity(newJob, machine, machineTag) |
| } |
| |
| Utilities.standardJobSetup(newJob, project, isPR, "*/${branch}") |
| |
| if (nonDefaultTaskSetup == null) { |
| if (isPR) { |
| def osTag = machineTypeToOSTagMap.get(machine) |
| Utilities.addGithubPRTriggerForBranch(newJob, branch, "${osTag} ${config}") |
| } else { |
| Utilities.addGithubPushTrigger(newJob) |
| } |
| } else { |
| // nonDefaultTaskSetup is e.g. DailyBuildTaskSetup (which sets up daily builds) |
| // These jobs will only be configured for the branch specified below, |
| // which is the name of the branch netci.groovy was processed for. |
| // See list of such branches at: |
| // https://github.com/dotnet/dotnet-ci/blob/master/jobs/data/repolist.txt |
| nonDefaultTaskSetup(newJob, isPR, config) |
| } |
| } |
| |
| def CreateBuildTasks = { machine, machineTag, configTag, buildExtra, testExtra, runCodeAnalysis, excludeConfigIf, nonDefaultTaskSetup -> |
| [true, false].each { isPR -> |
| ['x86', 'x64', 'arm'].each { buildArch -> |
| ['debug', 'test', 'release'].each { buildType -> |
| CreateBuildTask(isPR, buildArch, buildType, machine, machineTag, configTag, buildExtra, testExtra, runCodeAnalysis, excludeConfigIf, nonDefaultTaskSetup) |
| } |
| } |
| } |
| } |
| |
| def CreateXPlatBuildTask = { isPR, buildType, staticBuild, machine, platform, configTag, |
| xplatBranch, nonDefaultTaskSetup, customOption, testVariant, extraBuildParams -> |
| |
| def config = (platform == "osx" ? "osx_${buildType}" : "linux_${buildType}") |
| def numConcurrentCommand = (platform == "osx" ? "sysctl -n hw.logicalcpu" : "nproc") |
| |
| config = (configTag == null) ? config : "${configTag}_${config}" |
| config = staticBuild ? "static_${config}" : "shared_${config}" |
| config = customOption ? customOption.replaceAll(/[-]+/, "_") + "_" + config : config |
| |
| // params: Project, BaseTaskName, IsPullRequest (appends '_prtest') |
| def jobName = Utilities.getFullJobName(project, config, isPR) |
| |
| def infoScript = "bash jenkins/get_system_info.sh --${platform}" |
| def buildFlag = buildType == "release" ? "" : (buildType == "debug" ? "--debug" : "--test-build") |
| def staticFlag = staticBuild ? "--static" : "" |
| def swbCheckFlag = (platform == "linux" && buildType == "debug" && !staticBuild) ? "--wb-check" : ""; |
| def icuFlag = (platform == "osx" ? "--icu=/Users/DDITLABS/homebrew/opt/icu4c/include" : "") |
| def compilerPaths = (platform == "osx") ? "" : "--cxx=/usr/bin/clang++-3.9 --cc=/usr/bin/clang-3.9" |
| def buildScript = "bash ./build.sh ${staticFlag} -j=`${numConcurrentCommand}` ${buildFlag} " + |
| "${swbCheckFlag} ${compilerPaths} ${icuFlag} ${customOption} ${extraBuildParams}" |
| def icuLibFlag = (platform == "osx" ? "--iculib=/Users/DDITLABS/homebrew/opt/icu4c" : "") |
| def testScript = "bash test/runtests.sh ${icuLibFlag} \"${testVariant}\"" |
| |
| def newJob = job(jobName) { |
| steps { |
| shell(infoScript) |
| shell(buildScript) |
| shell(testScript) |
| } |
| } |
| |
| def archivalString = "out/build.log" |
| Utilities.addArchival(newJob, archivalString, |
| '', // no exclusions from archival |
| true, // doNotFailIfNothingArchived=false ~= failIfNothingArchived (true ~= doNotFail) |
| false) // archiveOnlyIfSuccessful=false ~= archiveAlways |
| |
| if (platform == "osx") { |
| Utilities.setMachineAffinity(newJob, machine) // OSX machine string contains all info already |
| } else { |
| Utilities.setMachineAffinity(newJob, machine, defaultMachineTag) |
| } |
| Utilities.standardJobSetup(newJob, project, isPR, "*/${branch}") |
| |
| if (nonDefaultTaskSetup == null) { |
| if (isPR) { |
| def osTag = machineTypeToOSTagMap.get(machine) |
| Utilities.addGithubPRTriggerForBranch(newJob, xplatBranch, "${osTag} ${config}") |
| } else { |
| Utilities.addGithubPushTrigger(newJob) |
| } |
| } else { |
| // nonDefaultTaskSetup is e.g. DailyBuildTaskSetup (which sets up daily builds) |
| // These jobs will only be configured for the branch specified below, |
| // which is the name of the branch netci.groovy was processed for. |
| // See list of such branches at: |
| // https://github.com/dotnet/dotnet-ci/blob/master/jobs/data/repolist.txt |
| nonDefaultTaskSetup(newJob, isPR, config) |
| } |
| } |
| |
| // Generic task to trigger clang-based cross-plat build tasks |
| def CreateXPlatBuildTasks = { machine, platform, configTag, xplatBranch, nonDefaultTaskSetup, extraBuildParams -> |
| [true, false].each { isPR -> |
| CreateXPlatBuildTask(isPR, "test", "", machine, platform, |
| configTag, xplatBranch, nonDefaultTaskSetup, "--no-jit", "--variants disable_jit", extraBuildParams) |
| |
| ['debug', 'test', 'release'].each { buildType -> |
| def staticBuildConfigs = [true, false] |
| if (platform == "osx") { |
| staticBuildConfigs = [true] |
| } |
| |
| staticBuildConfigs.each { staticBuild -> |
| CreateXPlatBuildTask(isPR, buildType, staticBuild, machine, platform, |
| configTag, xplatBranch, nonDefaultTaskSetup, "", "", extraBuildParams) |
| } |
| } |
| } |
| } |
| |
| def DailyBuildTaskSetup = { newJob, isPR, triggerName, groupRegex -> |
| // The addition of triggers makes the job non-default in GitHub. |
| if (isPR) { |
| def triggerRegex = "(${dailyRegex}|${groupRegex}|${triggerName})" |
| Utilities.addGithubPRTriggerForBranch(newJob, branch, |
| triggerName, // GitHub task name |
| "(?i).*test\\W+${triggerRegex}.*") |
| } else { |
| Utilities.addPeriodicTrigger(newJob, '@daily') |
| } |
| } |
| |
| def CreateStyleCheckTasks = { taskString, taskName, checkName -> |
| [true, false].each { isPR -> |
| def jobName = Utilities.getFullJobName(project, taskName, isPR) |
| |
| def newJob = job(jobName) { |
| steps { |
| shell(taskString) |
| } |
| } |
| |
| Utilities.standardJobSetup(newJob, project, isPR, "*/${branch}") |
| if (isPR) { |
| // Set PR trigger. |
| Utilities.addGithubPRTriggerForBranch(newJob, branch, checkName) |
| } else { |
| // Set a push trigger |
| Utilities.addGithubPushTrigger(newJob) |
| } |
| |
| Utilities.setMachineAffinity(newJob, 'Ubuntu16.04', defaultMachineTag) |
| } |
| } |
| |
| // ---------------- |
| // INNER LOOP TASKS |
| // ---------------- |
| |
| // The latest machine seems to have a configuration problem preventing us from building ARM. |
| // For now, build ARM on the LKG config, Legacy Windows 8.1 (Blue) config. |
| // TODO When the Windows 10 configuration is updated to fix ARM builds, unify this config split. |
| CreateBuildTasks(latestWindowsMachine, latestWindowsMachineTag, null, null, "-win10", true, |
| /* excludeConfigIf */ { isPR, buildArch, buildType -> (buildArch == 'arm') }, null) // configures everything except ARM |
| CreateBuildTasks(legacyWindows8Machine, legacyWindows8MachineTag, null, null, "-winBlue", true, |
| /* excludeConfigIf */ { isPR, buildArch, buildType -> (buildArch != 'arm') }, null) // configures ARM |
| |
| // Add some additional daily configs to trigger per-PR as a quality gate: |
| // x64_debug Slow Tests |
| CreateBuildTask(true, 'x64', 'debug', |
| latestWindowsMachine, latestWindowsMachineTag, 'ci_slow', null, '-win10 -includeSlow', false, null, null) |
| // x64_debug DisableJIT |
| CreateBuildTask(true, 'x64', 'debug', |
| latestWindowsMachine, latestWindowsMachineTag, 'ci_disablejit', '"/p:BuildJIT=false"', '-win10 -disablejit', false, null, null) |
| // x64_debug Lite |
| CreateBuildTask(true, 'x64', 'debug', |
| latestWindowsMachine, latestWindowsMachineTag, 'ci_lite', '"/p:BuildLite=true"', '-win10 -lite', false, null, null) |
| // x64_debug Legacy (Windows 7) |
| CreateBuildTask(true, 'x64', 'debug', |
| legacyWindows7Machine, legacyWindows7MachineTag, 'ci_legacy7', 'msbuild14', '-win7 -includeSlow', false, null, null) |
| // x64_debug Legacy (Windows 8.1 (Blue)) |
| CreateBuildTask(true, 'x64', 'debug', |
| legacyWindows8Machine, legacyWindows8MachineTag, 'ci_legacy8', 'msbuild14', '-winBlue -includeSlow', false, null, null) |
| |
| // ----------------- |
| // DAILY BUILD TASKS |
| // ----------------- |
| |
| if (!branch.endsWith('-ci')) { |
| // build and test on the legacy configuration (Windows 7 + VS 2015 (Dev14)) |
| CreateBuildTasks(legacyWindows7Machine, legacyWindows7MachineTag, 'daily_legacy7', 'msbuild14', '-win7 -includeSlow', false, |
| /* excludeConfigIf */ { isPR, buildArch, buildType -> (buildArch == 'arm') }, // excludes ARM |
| /* nonDefaultTaskSetup */ { newJob, isPR, config -> |
| DailyBuildTaskSetup(newJob, isPR, |
| "Windows 7 ${config}", |
| 'legacy7?\\s+tests)')}) |
| |
| // build and test on the legacy configuration (Windows 8.1 (Blue) + VS 2015 (Dev14)) |
| CreateBuildTasks(legacyWindows8Machine, legacyWindows8MachineTag, 'daily_legacy8', 'msbuild14', '-winBlue -includeSlow', false, |
| /* excludeConfigIf */ null, // ARM builds previously worked on this configuration, so don't exclude them unless we explicitly drop support |
| /* nonDefaultTaskSetup */ { newJob, isPR, config -> |
| DailyBuildTaskSetup(newJob, isPR, |
| "Windows 8 ${config}", |
| 'legacy8?\\s+tests')}) |
| |
| // build and test on the latest configuration (RS4 + VS 2017 Dev 15.7) with -includeSlow |
| // TODO When the Windows 10 configuration is updated to fix ARM builds, unify this config split. |
| CreateBuildTasks(latestWindowsMachine, latestWindowsMachineTag, 'daily_slow', null, '-win10 -includeSlow', false, |
| /* excludeConfigIf */ { isPR, buildArch, buildType -> (buildArch == 'arm') }, // configures everything except ARM |
| /* nonDefaultTaskSetup */ { newJob, isPR, config -> |
| DailyBuildTaskSetup(newJob, isPR, |
| "Windows ${config}", |
| 'slow\\s+tests')}) |
| CreateBuildTasks(legacyWindows8Machine, legacyWindows8MachineTag, 'daily_slow', null, '-winBlue -includeSlow', false, |
| /* excludeConfigIf */ { isPR, buildArch, buildType -> (buildArch != 'arm') }, // configures ARM |
| /* nonDefaultTaskSetup */ { newJob, isPR, config -> |
| DailyBuildTaskSetup(newJob, isPR, |
| "Windows ${config}", |
| 'slow\\s+tests')}) |
| |
| // build and test on the latest configuration (RS4 + VS 2017 Dev 15.7) with JIT disabled |
| // TODO When the Windows 10 configuration is updated to fix ARM builds, unify this config split. |
| CreateBuildTasks(latestWindowsMachine, latestWindowsMachineTag, 'daily_disablejit', '"/p:BuildJIT=false"', '-win10 -disablejit', true, |
| /* excludeConfigIf */ { isPR, buildArch, buildType -> (buildArch == 'arm') }, // configures everything except ARM |
| /* nonDefaultTaskSetup */ { newJob, isPR, config -> |
| DailyBuildTaskSetup(newJob, isPR, |
| "Windows ${config}", |
| '(disablejit|nojit)\\s+tests')}) |
| CreateBuildTasks(legacyWindows8Machine, legacyWindows8MachineTag, 'daily_disablejit', '"/p:BuildJIT=false"', '-winBlue -disablejit', true, |
| /* excludeConfigIf */ { isPR, buildArch, buildType -> (buildArch != 'arm') }, // configures ARM |
| /* nonDefaultTaskSetup */ { newJob, isPR, config -> |
| DailyBuildTaskSetup(newJob, isPR, |
| "Windows ${config}", |
| '(disablejit|nojit)\\s+tests')}) |
| |
| // build and test on the latest configuration (RS4 + VS 2017 Dev 15.7) with Lite build |
| // TODO When the Windows 10 configuration is updated to fix ARM builds, unify this config split. |
| CreateBuildTasks(latestWindowsMachine, latestWindowsMachineTag, 'daily_lite', '"/p:BuildLite=true"', '-win10 -lite', true, |
| /* excludeConfigIf */ { isPR, buildArch, buildType -> (buildArch == 'arm') }, // configures everything except ARM |
| /* nonDefaultTaskSetup */ { newJob, isPR, config -> |
| DailyBuildTaskSetup(newJob, isPR, |
| "Windows ${config}", |
| 'lite\\s+tests')}) |
| CreateBuildTasks(legacyWindows8Machine, legacyWindows8MachineTag, 'daily_lite', '"/p:BuildLite=true"', '-winBlue -lite', true, |
| /* excludeConfigIf */ { isPR, buildArch, buildType -> (buildArch != 'arm') }, // configures ARM |
| /* nonDefaultTaskSetup */ { newJob, isPR, config -> |
| DailyBuildTaskSetup(newJob, isPR, |
| "Windows ${config}", |
| 'lite\\s+tests')}) |
| } |
| |
| // ---------------- |
| // CODE STYLE TASKS |
| // ---------------- |
| |
| CreateStyleCheckTasks('./jenkins/check_copyright.sh', 'ubuntu_check_copyright', 'Copyright Check') |
| CreateStyleCheckTasks('./jenkins/check_eol.sh', 'ubuntu_check_eol', 'EOL Check') |
| CreateStyleCheckTasks('./jenkins/check_tabs.sh', 'ubuntu_check_tabs', 'Tab Check') |
| |
| // -------------- |
| // XPLAT BRANCHES |
| // -------------- |
| |
| // Explicitly enumerate xplat-incompatible branches, because we don't anticipate any future incompatible branches |
| def isXPlatCompatibleBranch = !(branch in ['release/1.1', 'release/1.1-ci', 'release/1.2', 'release/1.2-ci']) |
| |
| // Include these explicitly-named branches |
| def isXPlatDailyBranch = branch in ['master', 'linux', 'xplat'] |
| // Include some release/* branches (ignore branches ending in '-ci') |
| if (branch.startsWith('release') && !branch.endsWith('-ci')) { |
| // Allows all current and future release/* branches on which we should run daily builds of XPlat configs. |
| // RegEx matches branch names we should ignore (e.g. release/1.1, release/1.2, release/1.2-pre) |
| includeReleaseBranch = !(branch =~ /^release\/(1\.[12](\D.*)?)$/) |
| isXPlatDailyBranch |= includeReleaseBranch |
| } |
| |
| // ----------------- |
| // LINUX BUILD TASKS |
| // ----------------- |
| |
| if (isXPlatCompatibleBranch) { |
| def osString = 'Ubuntu16.04' |
| |
| // PR and CI checks |
| CreateXPlatBuildTasks(osString, "linux", "ubuntu", branch, null, "") |
| |
| // Create a PR/continuous task to check ubuntu/static/debug/no-icu |
| [true, false].each { isPR -> |
| CreateXPlatBuildTask(isPR, "debug", true, osString, "linux", |
| "ubuntu", branch, null, "--no-icu", "--not-tag exclude_noicu", "") |
| } |
| |
| // daily builds |
| if (isXPlatDailyBranch) { |
| CreateXPlatBuildTasks(osString, "linux", "daily_ubuntu", branch, |
| /* nonDefaultTaskSetup */ { newJob, isPR, config -> |
| DailyBuildTaskSetup(newJob, isPR, |
| "Ubuntu ${config}", |
| 'linux\\s+tests')}, |
| /* extraBuildParams */ "--extra-defines=PERFMAP_TRACE_ENABLED=1") |
| } |
| } |
| |
| // --------------- |
| // OSX BUILD TASKS |
| // --------------- |
| |
| if (isXPlatCompatibleBranch) { |
| def osString = 'OSX.1011.Amd64.Chakra.Open' |
| |
| // PR and CI checks |
| CreateXPlatBuildTasks(osString, "osx", "osx", branch, null, "") |
| |
| // daily builds |
| if (isXPlatDailyBranch) { |
| CreateXPlatBuildTasks(osString, "osx", "daily_osx", branch, |
| /* nonDefaultTaskSetup */ { newJob, isPR, config -> |
| DailyBuildTaskSetup(newJob, isPR, |
| "OSX ${config}", |
| 'linux\\s+tests')}, |
| /* extraBuildParams */ "") |
| } |
| } |
| |
| // ------------ |
| // HELP MESSAGE |
| // ------------ |
| |
| Utilities.createHelperJob(this, project, branch, |
| "Welcome to the ${project} Repository", // This is prepended to the help message |
| "For additional documentation on ChakraCore CI checks, please see:\n" + |
| "\n" + |
| "* https://github.com/Microsoft/ChakraCore/wiki/Jenkins-CI-Checks\n" + |
| "* https://github.com/Microsoft/ChakraCore/wiki/Jenkins-Build-Triggers\n" + |
| "* https://github.com/Microsoft/ChakraCore/wiki/Jenkins-Repro-Steps\n" + |
| "\n" + |
| "Have a nice day!") // This is appended to the help message. You might put known issues here. |