Merge pull request #11153 from electron/update-ci-release
Move release build process to API calls
This commit is contained in:
commit
f469059e90
4 changed files with 206 additions and 53 deletions
|
@ -12,10 +12,6 @@ jobs:
|
||||||
- run:
|
- run:
|
||||||
name: Check for release
|
name: Check for release
|
||||||
command: |
|
command: |
|
||||||
MESSAGE="$(git log --format=%B -n 1 HEAD)"
|
|
||||||
case ${MESSAGE} in
|
|
||||||
Bump* ) echo 'export ELECTRON_RELEASE=1' >> $BASH_ENV
|
|
||||||
esac
|
|
||||||
if [ -n "${RUN_RELEASE_BUILD}" ]; then
|
if [ -n "${RUN_RELEASE_BUILD}" ]; then
|
||||||
echo 'release build triggered from api'
|
echo 'release build triggered from api'
|
||||||
echo 'export ELECTRON_RELEASE=1 TRIGGERED_BY_API=1' >> $BASH_ENV
|
echo 'export ELECTRON_RELEASE=1 TRIGGERED_BY_API=1' >> $BASH_ENV
|
||||||
|
@ -73,10 +69,6 @@ jobs:
|
||||||
- run:
|
- run:
|
||||||
name: Check for release
|
name: Check for release
|
||||||
command: |
|
command: |
|
||||||
MESSAGE="$(git log --format=%B -n 1 HEAD)"
|
|
||||||
case ${MESSAGE} in
|
|
||||||
Bump* ) echo 'export ELECTRON_RELEASE=1' >> $BASH_ENV
|
|
||||||
esac
|
|
||||||
if [ -n "${RUN_RELEASE_BUILD}" ]; then
|
if [ -n "${RUN_RELEASE_BUILD}" ]; then
|
||||||
echo 'release build triggered from api'
|
echo 'release build triggered from api'
|
||||||
echo 'export ELECTRON_RELEASE=1 TRIGGERED_BY_API=1' >> $BASH_ENV
|
echo 'export ELECTRON_RELEASE=1 TRIGGERED_BY_API=1' >> $BASH_ENV
|
||||||
|
@ -134,10 +126,6 @@ jobs:
|
||||||
- run:
|
- run:
|
||||||
name: Check for release
|
name: Check for release
|
||||||
command: |
|
command: |
|
||||||
MESSAGE="$(git log --format=%B -n 1 HEAD)"
|
|
||||||
case ${MESSAGE} in
|
|
||||||
Bump* ) echo 'export ELECTRON_RELEASE=1' >> $BASH_ENV
|
|
||||||
esac
|
|
||||||
if [ -n "${RUN_RELEASE_BUILD}" ]; then
|
if [ -n "${RUN_RELEASE_BUILD}" ]; then
|
||||||
echo 'release build triggered from api'
|
echo 'release build triggered from api'
|
||||||
echo 'export ELECTRON_RELEASE=1 TRIGGERED_BY_API=1' >> $BASH_ENV
|
echo 'export ELECTRON_RELEASE=1 TRIGGERED_BY_API=1' >> $BASH_ENV
|
||||||
|
@ -200,10 +188,6 @@ jobs:
|
||||||
- run:
|
- run:
|
||||||
name: Check for release
|
name: Check for release
|
||||||
command: |
|
command: |
|
||||||
MESSAGE="$(git log --format=%B -n 1 HEAD)"
|
|
||||||
case ${MESSAGE} in
|
|
||||||
Bump* ) echo 'export ELECTRON_RELEASE=1' >> $BASH_ENV
|
|
||||||
esac
|
|
||||||
if [ -n "${RUN_RELEASE_BUILD}" ]; then
|
if [ -n "${RUN_RELEASE_BUILD}" ]; then
|
||||||
echo 'release build triggered from api'
|
echo 'release build triggered from api'
|
||||||
echo 'export ELECTRON_RELEASE=1 TRIGGERED_BY_API=1' >> $BASH_ENV
|
echo 'export ELECTRON_RELEASE=1 TRIGGERED_BY_API=1' >> $BASH_ENV
|
||||||
|
|
|
@ -1,56 +1,210 @@
|
||||||
const args = require('minimist')(process.argv.slice(2))
|
|
||||||
const assert = require('assert')
|
const assert = require('assert')
|
||||||
const request = require('request')
|
const request = require('request')
|
||||||
|
const buildAppVeyorURL = 'https://windows-ci.electronjs.org/api/builds'
|
||||||
|
const jenkinsServer = 'https://mac-ci.electronjs.org'
|
||||||
|
|
||||||
const ciJobs = [
|
const circleCIJobs = [
|
||||||
'electron-linux-arm64',
|
'electron-linux-arm64',
|
||||||
'electron-linux-ia32',
|
'electron-linux-ia32',
|
||||||
'electron-linux-x64',
|
'electron-linux-x64',
|
||||||
'electron-linux-arm'
|
'electron-linux-arm'
|
||||||
]
|
]
|
||||||
|
|
||||||
const CIcall = (buildUrl, targetBranch, job) => {
|
const jenkinsJobs = [
|
||||||
console.log(`Triggering CircleCI to run build job: ${job} on branch: ${targetBranch} with release flag.`)
|
'electron-mas-x64-release',
|
||||||
|
'electron-osx-x64-release'
|
||||||
|
]
|
||||||
|
|
||||||
request({
|
async function makeRequest (requestOptions, parseResponse) {
|
||||||
|
return new Promise((resolve, reject) => {
|
||||||
|
request(requestOptions, (err, res, body) => {
|
||||||
|
if (!err && res.statusCode >= 200 && res.statusCode < 300) {
|
||||||
|
if (parseResponse) {
|
||||||
|
const build = JSON.parse(body)
|
||||||
|
resolve(build)
|
||||||
|
} else {
|
||||||
|
resolve(body)
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (parseResponse) {
|
||||||
|
console.log('Error: ', `(status ${res.statusCode})`, err || JSON.parse(res.body), requestOptions)
|
||||||
|
} else {
|
||||||
|
console.log('Error: ', `(status ${res.statusCode})`, err || res.body, requestOptions)
|
||||||
|
}
|
||||||
|
reject()
|
||||||
|
}
|
||||||
|
})
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
async function circleCIcall (buildUrl, targetBranch, job, ghRelease) {
|
||||||
|
assert(process.env.CIRCLE_TOKEN, 'CIRCLE_TOKEN not found in environment')
|
||||||
|
console.log(`Triggering CircleCI to run build job: ${job} on branch: ${targetBranch} with release flag.`)
|
||||||
|
let buildRequest = {
|
||||||
|
'build_parameters': {
|
||||||
|
'CIRCLE_JOB': job
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (ghRelease) {
|
||||||
|
buildRequest.build_parameters.ELECTRON_RELEASE = 1
|
||||||
|
} else {
|
||||||
|
buildRequest.build_parameters.RUN_RELEASE_BUILD = 'true'
|
||||||
|
}
|
||||||
|
|
||||||
|
let circleResponse = await makeRequest({
|
||||||
method: 'POST',
|
method: 'POST',
|
||||||
url: buildUrl,
|
url: buildUrl,
|
||||||
headers: {
|
headers: {
|
||||||
'Content-Type': 'application/json',
|
'Content-Type': 'application/json',
|
||||||
'Accept': 'application/json'
|
'Accept': 'application/json'
|
||||||
},
|
},
|
||||||
body: JSON.stringify({
|
body: JSON.stringify(buildRequest)
|
||||||
'build_parameters': {
|
}, true).catch(err => {
|
||||||
'RUN_RELEASE_BUILD': 'true',
|
console.log('Error calling CircleCI:', err)
|
||||||
'CIRCLE_JOB': job
|
|
||||||
}
|
|
||||||
})
|
|
||||||
}, (err, res, body) => {
|
|
||||||
if (!err && res.statusCode >= 200 && res.statusCode < 300) {
|
|
||||||
const build = JSON.parse(body)
|
|
||||||
console.log(`Check ${build.build_url} for status. (${job})`)
|
|
||||||
} else {
|
|
||||||
console.log('Error: ', `(status ${res.statusCode})`, err || JSON.parse(res.body), job)
|
|
||||||
}
|
|
||||||
})
|
})
|
||||||
|
console.log(`Check ${circleResponse.build_url} for status. (${job})`)
|
||||||
}
|
}
|
||||||
|
|
||||||
if (args._.length < 1) {
|
async function buildAppVeyor (targetBranch, ghRelease) {
|
||||||
console.log(`Trigger Circle CI to build release builds of electron.
|
console.log(`Triggering AppVeyor to run build on branch: ${targetBranch} with release flag.`)
|
||||||
Usage: ci-release-build.js [--job=CI_JOB_NAME] TARGET_BRANCH
|
assert(process.env.APPVEYOR_TOKEN, 'APPVEYOR_TOKEN not found in environment')
|
||||||
`)
|
let environmentVariables = {}
|
||||||
process.exit(0)
|
|
||||||
|
if (ghRelease) {
|
||||||
|
environmentVariables.ELECTRON_RELEASE = 1
|
||||||
|
} else {
|
||||||
|
environmentVariables.RUN_RELEASE_BUILD = 'true'
|
||||||
|
}
|
||||||
|
|
||||||
|
const requestOpts = {
|
||||||
|
url: buildAppVeyorURL,
|
||||||
|
auth: {
|
||||||
|
bearer: process.env.APPVEYOR_TOKEN
|
||||||
|
},
|
||||||
|
headers: {
|
||||||
|
'Content-Type': 'application/json'
|
||||||
|
},
|
||||||
|
body: JSON.stringify({
|
||||||
|
accountName: 'AppVeyor',
|
||||||
|
projectSlug: 'electron',
|
||||||
|
branch: targetBranch,
|
||||||
|
environmentVariables
|
||||||
|
}),
|
||||||
|
method: 'POST'
|
||||||
|
}
|
||||||
|
let appVeyorResponse = await makeRequest(requestOpts, true).catch(err => {
|
||||||
|
console.log('Error calling AppVeyor:', err)
|
||||||
|
})
|
||||||
|
const buildUrl = `https://windows-ci.electronjs.org/project/AppVeyor/electron/build/${appVeyorResponse.version}`
|
||||||
|
console.log(`AppVeyor release build request successful. Check build status at ${buildUrl}`)
|
||||||
}
|
}
|
||||||
|
|
||||||
assert(process.env.CIRCLE_TOKEN, 'CIRCLE_TOKEN not found in environment')
|
function buildCircleCI (targetBranch, ghRelease, job) {
|
||||||
|
const circleBuildUrl = `https://circleci.com/api/v1.1/project/github/electron/electron/tree/${targetBranch}?circle-token=${process.env.CIRCLE_TOKEN}`
|
||||||
const targetBranch = args._[0]
|
if (job) {
|
||||||
const job = args['job']
|
assert(circleCIJobs.includes(job), `Unknown CI job name: ${job}.`)
|
||||||
const circleBuildUrl = `https://circleci.com/api/v1.1/project/github/electron/electron/tree/${targetBranch}?circle-token=${process.env.CIRCLE_TOKEN}`
|
circleCIcall(circleBuildUrl, targetBranch, job, ghRelease)
|
||||||
|
} else {
|
||||||
if (job) {
|
circleCIJobs.forEach((job) => circleCIcall(circleBuildUrl, targetBranch, job, ghRelease))
|
||||||
assert(ciJobs.includes(job), `Unknown CI job name: ${job}.`)
|
}
|
||||||
CIcall(circleBuildUrl, targetBranch, job)
|
}
|
||||||
} else {
|
|
||||||
ciJobs.forEach((job) => CIcall(circleBuildUrl, targetBranch, job))
|
async function buildJenkins (targetBranch, ghRelease, job) {
|
||||||
|
assert(process.env.JENKINS_AUTH_TOKEN, 'JENKINS_AUTH_TOKEN not found in environment')
|
||||||
|
assert(process.env.JENKINS_BUILD_TOKEN, 'JENKINS_BUILD_TOKEN not found in environment')
|
||||||
|
let jenkinsCrumb = await getJenkinsCrumb()
|
||||||
|
|
||||||
|
if (job) {
|
||||||
|
assert(jenkinsJobs.includes(job), `Unknown CI job name: ${job}.`)
|
||||||
|
callJenkinsBuild(job, jenkinsCrumb, targetBranch, ghRelease)
|
||||||
|
} else {
|
||||||
|
jenkinsJobs.forEach((job) => {
|
||||||
|
callJenkinsBuild(job, jenkinsCrumb, targetBranch, ghRelease)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function callJenkins (path, requestParameters, requestHeaders) {
|
||||||
|
let requestOptions = {
|
||||||
|
url: `${jenkinsServer}/${path}`,
|
||||||
|
auth: {
|
||||||
|
user: 'build',
|
||||||
|
pass: process.env.JENKINS_AUTH_TOKEN
|
||||||
|
},
|
||||||
|
qs: requestParameters
|
||||||
|
}
|
||||||
|
if (requestHeaders) {
|
||||||
|
requestOptions.headers = requestHeaders
|
||||||
|
}
|
||||||
|
let jenkinsResponse = await makeRequest(requestOptions).catch(err => {
|
||||||
|
console.log(`Error calling Jenkins:`, err)
|
||||||
|
})
|
||||||
|
return jenkinsResponse
|
||||||
|
}
|
||||||
|
|
||||||
|
async function callJenkinsBuild (job, jenkinsCrumb, targetBranch, ghRelease) {
|
||||||
|
console.log(`Triggering Jenkins to run build job: ${job} on branch: ${targetBranch} with release flag.`)
|
||||||
|
let jenkinsParams = {
|
||||||
|
token: process.env.JENKINS_BUILD_TOKEN,
|
||||||
|
BRANCH: targetBranch
|
||||||
|
}
|
||||||
|
if (!ghRelease) {
|
||||||
|
jenkinsParams.RUN_RELEASE_BUILD = 1
|
||||||
|
}
|
||||||
|
await callJenkins(`job/${job}/buildWithParameters`, jenkinsParams, jenkinsCrumb)
|
||||||
|
.catch(err => {
|
||||||
|
console.log(`Error calling Jenkins build`, err)
|
||||||
|
})
|
||||||
|
let buildUrl = `${jenkinsServer}/job/${job}/lastBuild/`
|
||||||
|
console.log(`Jenkins build request successful. Check build status at ${buildUrl}.`)
|
||||||
|
}
|
||||||
|
|
||||||
|
async function getJenkinsCrumb () {
|
||||||
|
let crumbResponse = await callJenkins('crumbIssuer/api/xml', {
|
||||||
|
xpath: 'concat(//crumbRequestField,":",//crumb)'
|
||||||
|
}).catch(err => {
|
||||||
|
console.log(`Error getting jenkins crumb:`, err)
|
||||||
|
})
|
||||||
|
let crumbDetails = crumbResponse.split(':')
|
||||||
|
let crumbHeader = {}
|
||||||
|
crumbHeader[crumbDetails[0]] = crumbDetails[1]
|
||||||
|
return crumbHeader
|
||||||
|
}
|
||||||
|
|
||||||
|
function runRelease (targetBranch, options) {
|
||||||
|
if (options.ci) {
|
||||||
|
switch (options.ci) {
|
||||||
|
case 'CircleCI': {
|
||||||
|
buildCircleCI(targetBranch, options.ghRelease, options.job)
|
||||||
|
break
|
||||||
|
}
|
||||||
|
case 'AppVeyor': {
|
||||||
|
buildAppVeyor(targetBranch, options.ghRelease)
|
||||||
|
break
|
||||||
|
}
|
||||||
|
case 'Jenkins': {
|
||||||
|
buildJenkins(targetBranch, options.ghRelease, options.job)
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
buildCircleCI(targetBranch, options.ghRelease, options.job)
|
||||||
|
buildAppVeyor(targetBranch, options.ghRelease)
|
||||||
|
buildJenkins(targetBranch, options.ghRelease, options.job)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
module.exports = runRelease
|
||||||
|
|
||||||
|
if (require.main === module) {
|
||||||
|
const args = require('minimist')(process.argv.slice(2))
|
||||||
|
const targetBranch = args._[0]
|
||||||
|
if (args._.length < 1) {
|
||||||
|
console.log(`Trigger CI to build release builds of electron.
|
||||||
|
Usage: ci-release-build.js [--job=CI_JOB_NAME] [--ci=CircleCI|AppVeyor|Jenkins] [--ghRelease] TARGET_BRANCH
|
||||||
|
`)
|
||||||
|
process.exit(0)
|
||||||
|
}
|
||||||
|
runRelease(targetBranch, args)
|
||||||
}
|
}
|
||||||
|
|
|
@ -11,7 +11,8 @@ import stat
|
||||||
if sys.platform == "win32":
|
if sys.platform == "win32":
|
||||||
import _winreg
|
import _winreg
|
||||||
|
|
||||||
from lib.config import BASE_URL, PLATFORM, get_target_arch, get_zip_name
|
from lib.config import BASE_URL, PLATFORM, enable_verbose_mode, \
|
||||||
|
get_target_arch, get_zip_name
|
||||||
from lib.util import scoped_cwd, rm_rf, get_electron_version, make_zip, \
|
from lib.util import scoped_cwd, rm_rf, get_electron_version, make_zip, \
|
||||||
execute, electron_gyp
|
execute, electron_gyp
|
||||||
|
|
||||||
|
@ -79,6 +80,11 @@ TARGET_DIRECTORIES = {
|
||||||
|
|
||||||
|
|
||||||
def main():
|
def main():
|
||||||
|
args = parse_args()
|
||||||
|
|
||||||
|
if args.verbose:
|
||||||
|
enable_verbose_mode()
|
||||||
|
|
||||||
rm_rf(DIST_DIR)
|
rm_rf(DIST_DIR)
|
||||||
os.makedirs(DIST_DIR)
|
os.makedirs(DIST_DIR)
|
||||||
|
|
||||||
|
@ -92,8 +98,6 @@ def main():
|
||||||
copy_vcruntime_binaries()
|
copy_vcruntime_binaries()
|
||||||
copy_ucrt_binaries()
|
copy_ucrt_binaries()
|
||||||
|
|
||||||
args = parse_args()
|
|
||||||
|
|
||||||
if PLATFORM != 'win32' and not args.no_api_docs:
|
if PLATFORM != 'win32' and not args.no_api_docs:
|
||||||
create_api_json_schema()
|
create_api_json_schema()
|
||||||
create_typescript_definitions()
|
create_typescript_definitions()
|
||||||
|
@ -307,6 +311,9 @@ def parse_args():
|
||||||
parser.add_argument('--no_api_docs',
|
parser.add_argument('--no_api_docs',
|
||||||
action='store_true',
|
action='store_true',
|
||||||
help='Skip generating the Electron API Documentation!')
|
help='Skip generating the Electron API Documentation!')
|
||||||
|
parser.add_argument('-v', '--verbose',
|
||||||
|
action='store_true',
|
||||||
|
help='Prints the output of the subprocesses')
|
||||||
return parser.parse_args()
|
return parser.parse_args()
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -3,6 +3,7 @@
|
||||||
require('colors')
|
require('colors')
|
||||||
const args = require('minimist')(process.argv.slice(2))
|
const args = require('minimist')(process.argv.slice(2))
|
||||||
const assert = require('assert')
|
const assert = require('assert')
|
||||||
|
const ciReleaseBuild = require('./ci-release-build')
|
||||||
const { execSync } = require('child_process')
|
const { execSync } = require('child_process')
|
||||||
const fail = '\u2717'.red
|
const fail = '\u2717'.red
|
||||||
const { GitProcess, GitError } = require('dugite')
|
const { GitProcess, GitError } = require('dugite')
|
||||||
|
@ -158,6 +159,12 @@ async function pushRelease () {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async function runReleaseBuilds () {
|
||||||
|
await ciReleaseBuild('release', {
|
||||||
|
ghRelease: true
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
async function prepareRelease (isBeta, notesOnly) {
|
async function prepareRelease (isBeta, notesOnly) {
|
||||||
let currentBranch = await getCurrentBranch(gitDir)
|
let currentBranch = await getCurrentBranch(gitDir)
|
||||||
if (notesOnly) {
|
if (notesOnly) {
|
||||||
|
@ -167,6 +174,7 @@ async function prepareRelease (isBeta, notesOnly) {
|
||||||
await createReleaseBranch()
|
await createReleaseBranch()
|
||||||
await createRelease(currentBranch, isBeta)
|
await createRelease(currentBranch, isBeta)
|
||||||
await pushRelease()
|
await pushRelease()
|
||||||
|
await runReleaseBuilds()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue