if (!process.env.CI) require('dotenv-safe').load() const assert = require('assert') const request = require('request') const buildAppVeyorURL = 'https://ci.appveyor.com/api/builds' const circleCIPipelineURL = 'https://circleci.com/api/v2/project/gh/electron/electron/pipeline' const vstsURL = 'https://github.visualstudio.com/electron/_apis/build' const appVeyorJobs = { 'electron-x64': 'electron-x64-release', 'electron-ia32': 'electron-ia32-release', 'electron-woa': 'electron-woa-release' } const circleCIJobs = [ 'linux-arm-publish', 'linux-arm64-publish', 'linux-ia32-publish', 'linux-x64-publish', 'mas-publish', 'osx-publish' ] const vstsArmJobs = [ 'electron-arm-testing', 'electron-arm64-testing', 'electron-woa-testing' ] let jobRequestedCount = 0 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 { console.error('Error occurred while requesting:', requestOptions.url) if (parseResponse) { try { console.log('Error: ', `(status ${res.statusCode})`, err || JSON.parse(res.body)) } catch (err) { console.log('Error: ', `(status ${res.statusCode})`, res.body) } } else { console.log('Error: ', `(status ${res.statusCode})`, err || res.body) } reject(err) } }) }) } async function circleCIcall (targetBranch, job, options) { console.log(`Triggering CircleCI to run build job: ${job} on branch: ${targetBranch} with release flag.`) const buildRequest = { 'branch': targetBranch, 'parameters': { 'run-lint': false, 'run-build-linux': false, 'run-build-mac': false } } if (options.ghRelease) { buildRequest.parameters['upload-to-s3'] = '0' } else { buildRequest.parameters['upload-to-s3'] = '1' } buildRequest.parameters[`run-${job}`] = true jobRequestedCount++ // The logic below expects that the CircleCI workflows for releases each // contain only one job in order to maintain compatibility with sudowoodo. // If the workflows are changed in the CircleCI config.yml, this logic will // also need to be changed as well as possibly changing sudowoodo. try { const circleResponse = await circleCIRequest(circleCIPipelineURL, 'POST', buildRequest) console.log(`CircleCI release build pipeline ${circleResponse.id} for ${job} triggered.`) const pipelineInfoUrl = `https://circleci.com/api/v2/pipeline/${circleResponse.id}` const workflowId = await getCircleCIWorkflowId(circleResponse.id) if (workflowId === -1) { return } console.log(`CircleCI release build workflow running at https://circleci.com/workflow-run/${workflowId} for ${job}.`) const jobNumber = await getCircleCIJobNumber(workflowId) if (jobNumber === -1) { return } const jobUrl = `https://circleci.com/gh/electron/electron/${jobNumber}` console.log(`CircleCI release build request for ${job} successful. Check ${jobUrl} for status.`) } catch (err) { console.log('Error calling CircleCI: ', err) } } async function getCircleCIWorkflowId (pipelineId) { const pipelineInfoUrl = `https://circleci.com/api/v2/pipeline/${pipelineId}` for (let i = 0; i < 5; i++) { const pipelineInfo = await circleCIRequest(pipelineInfoUrl, 'GET') switch (pipelineInfo.state) { case 'created': { if (pipelineInfo.workflows.length === 1) { return pipelineInfo.workflows[0].id } console.log('Unxpected number of workflows, response was:', pipelineInfo) return -1 } case 'error': { console.log('Error retrieving workflows, response was:', pipelineInfo) return -1 } } await new Promise(resolve => setTimeout(resolve, 5000)) } return -1 } async function getCircleCIJobNumber (workflowId) { const jobInfoUrl = `https://circleci.com/api/v2/workflow/${workflowId}/jobs` for (let i = 0; i < 5; i++) { const jobInfo = await circleCIRequest(jobInfoUrl, 'GET') if (!jobInfo.items) { continue } if (jobInfo.items.length !== 1) { console.log('Unxpected number of jobs, response was:', jobInfo) return -1 } switch (jobInfo.items[0].status) { case 'not_running': case 'queued': case 'running': { if (jobInfo.items[0].job_number && !isNaN(jobInfo.items[0].job_number)) { return jobInfo.items[0].job_number } break } case 'error': { console.log('Error retrieving jobs, response was:', jobInfo) return -1 } } await new Promise(resolve => setTimeout(resolve, 5000)) } return -1 } async function circleCIRequest (url, method, requestBody) { return makeRequest({ auth: { username: process.env.CIRCLE_TOKEN, password: '' }, method, url, headers: { 'Content-Type': 'application/json', 'Accept': 'application/json' }, body: requestBody ? JSON.stringify(requestBody) : null }, true).catch(err => { console.log('Error calling CircleCI:', err) }) } function buildAppVeyor (targetBranch, options) { const validJobs = Object.keys(appVeyorJobs) if (options.job) { assert(validJobs.includes(options.job), `Unknown AppVeyor CI job name: ${options.job}. Valid values are: ${validJobs}.`) callAppVeyor(targetBranch, options.job, options) } else { validJobs.forEach((job) => callAppVeyor(targetBranch, job, options)) } } async function callAppVeyor (targetBranch, job, options) { console.log(`Triggering AppVeyor to run build job: ${job} on branch: ${targetBranch} with release flag.`) const environmentVariables = { ELECTRON_RELEASE: 1 } if (!options.ghRelease) { environmentVariables.UPLOAD_TO_S3 = 1 } const requestOpts = { url: buildAppVeyorURL, auth: { bearer: process.env.APPVEYOR_CLOUD_TOKEN }, headers: { 'Content-Type': 'application/json' }, body: JSON.stringify({ accountName: 'electron-bot', projectSlug: appVeyorJobs[job], branch: targetBranch, environmentVariables }), method: 'POST' } jobRequestedCount++ const appVeyorResponse = await makeRequest(requestOpts, true).catch(err => { console.log('Error calling AppVeyor:', err) }) const buildUrl = `https://ci.appveyor.com/project/electron-bot/${appVeyorJobs[job]}/build/${appVeyorResponse.version}` console.log(`AppVeyor release build request for ${job} successful. Check build status at ${buildUrl}`) } function buildCircleCI (targetBranch, options) { if (options.job) { assert(circleCIJobs.includes(options.job), `Unknown CircleCI job name: ${options.job}. Valid values are: ${circleCIJobs}.`) circleCIcall(targetBranch, options.job, options) } else { circleCIJobs.forEach((job) => circleCIcall(targetBranch, job, options)) } } async function buildVSTS (targetBranch, options) { if (options.armTest) { assert(vstsArmJobs.includes(options.job), `Unknown VSTS CI arm test job name: ${options.job}. Valid values are: ${vstsArmJobs}.`) } console.log(`Triggering VSTS to run build on branch: ${targetBranch} with release flag.`) const environmentVariables = { ELECTRON_RELEASE: 1 } if (options.armTest) { if (options.circleBuildNum) { environmentVariables.CIRCLE_BUILD_NUM = options.circleBuildNum } else if (options.appveyorJobId) { environmentVariables.APPVEYOR_JOB_ID = options.appveyorJobId } } else { if (!options.ghRelease) { environmentVariables.UPLOAD_TO_S3 = 1 } } const requestOpts = { url: `${vstsURL}/definitions?api-version=4.1`, auth: { user: '', password: process.env.VSTS_TOKEN }, headers: { 'Content-Type': 'application/json' } } const vstsResponse = await makeRequest(requestOpts, true).catch(err => { console.log('Error calling VSTS to get build definitions:', err) }) const buildsToRun = vstsResponse.value.filter(build => build.name === options.job) buildsToRun.forEach((build) => callVSTSBuild(build, targetBranch, environmentVariables)) } async function callVSTSBuild (build, targetBranch, environmentVariables) { const buildBody = { definition: build, sourceBranch: targetBranch, priority: 'high' } if (Object.keys(environmentVariables).length !== 0) { buildBody.parameters = JSON.stringify(environmentVariables) } const requestOpts = { url: `${vstsURL}/builds?api-version=4.1`, auth: { user: '', password: process.env.VSTS_TOKEN }, headers: { 'Content-Type': 'application/json' }, body: JSON.stringify(buildBody), method: 'POST' } jobRequestedCount++ const vstsResponse = await makeRequest(requestOpts, true).catch(err => { console.log(`Error calling VSTS for job ${build.name}`, err) }) console.log(`VSTS release build request for ${build.name} successful. Check ${vstsResponse._links.web.href} for status.`) } function runRelease (targetBranch, options) { if (options.ci) { switch (options.ci) { case 'CircleCI': { buildCircleCI(targetBranch, options) break } case 'AppVeyor': { buildAppVeyor(targetBranch, options) break } case 'VSTS': { buildVSTS(targetBranch, options) break } default: { console.log(`Error! Unknown CI: ${options.ci}.`) process.exit(1) } } } else { buildCircleCI(targetBranch, options) buildAppVeyor(targetBranch, options) } console.log(`${jobRequestedCount} jobs were requested.`) } module.exports = runRelease if (require.main === module) { const args = require('minimist')(process.argv.slice(2), { boolean: ['ghRelease', 'armTest'] }) 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|VSTS] [--ghRelease] [--armTest] [--circleBuildNum=xxx] [--appveyorJobId=xxx] TARGET_BRANCH `) process.exit(0) } runRelease(targetBranch, args) }