electron/script/release/ci-release-build.js

366 lines
12 KiB
JavaScript
Raw Normal View History

2020-03-20 20:28:31 +00:00
if (!process.env.CI) require('dotenv-safe').load();
2018-07-11 18:02:03 +00:00
2020-03-20 20:28:31 +00:00
const assert = require('assert');
const got = require('got');
2020-03-20 20:28:31 +00:00
const BUILD_APPVEYOR_URL = 'https://ci.appveyor.com/api/builds';
const CIRCLECI_PIPELINE_URL = 'https://circleci.com/api/v2/project/gh/electron/electron/pipeline';
const VSTS_URL = 'https://github.visualstudio.com/electron/_apis/build';
const DEVOPS_URL = 'https://dev.azure.com/electron-ci/electron/_apis/build';
2020-03-20 20:28:31 +00:00
const CIRCLECI_WAIT_TIME = process.env.CIRCLECI_WAIT_TIME || 30000;
const appVeyorJobs = {
'electron-x64': 'electron-x64-release',
'electron-ia32': 'electron-ia32-release',
'electron-woa': 'electron-woa-release'
2020-03-20 20:28:31 +00:00
};
const circleCIPublishWorkflows = [
'linux-publish',
'macos-publish'
];
const circleCIPublishIndividualArches = {
'macos-publish': ['osx-x64', 'mas-x64', 'osx-arm64', 'mas-arm64'],
chore: bump chromium to 102.0.4999.0 (main) (#33731) * chore: bump chromium in DEPS to 102.0.4999.0 * 3576640: Set OOM handler during V8 initialization https://chromium-review.googlesource.com/c/chromium/src/+/3576640 * 3574964: Remove deprecated base::Value usage in print_settings_conversion code. https://chromium-review.googlesource.com/c/chromium/src/+/3574964 * 3570062: Replicate Active state to render process for all RenderViews. https://chromium-review.googlesource.com/c/chromium/src/+/3570062 * chore: fixup patch indices * 3380402: Remove legacy SwiftShader https://chromium-review.googlesource.com/c/chromium/src/+/3380402 * 3570254: [Local Fonts] Rename permission name from FONT_ACCESS to LOCAL_FONTS. https://chromium-review.googlesource.com/c/chromium/src/+/3570254 * 3572172: Rename or remove several parameters involved in creation of MimeHandler streams https://chromium-review.googlesource.com/c/chromium/src/+/3572172 * fix: add missing base/bits include * chore: fix lint * chore: remove ia32 Linux support * chore: patch out swift-format cipd dep on macOS * build: apply patch better * build: reset all caches * build: update zip manifests to remove swiftshared libraries Refs: https://chromium-review.googlesource.com/c/chromium/src/+/3380402 * Revert "build: update zip manifests to remove swiftshared libraries" This reverts commit 6aeec01ef1a79425a7b7d8c1cfb131a26b91c494. * Revert "3380402: Remove legacy SwiftShader" This reverts commit 4c7eebbbf2d0a459cc192959e17ae20f970c2da2. * build: remove unused swiftshader egl libraries Co-authored-by: electron-roller[bot] <84116207+electron-roller[bot]@users.noreply.github.com> Co-authored-by: Shelley Vohr <shelley.vohr@gmail.com> Co-authored-by: Samuel Attard <sattard@salesforce.com>
2022-04-22 22:36:22 +00:00
'linux-publish': ['arm', 'arm64', 'x64']
};
const vstsArmJobs = [
'electron-arm-testing',
'electron-osx-arm64-testing',
'electron-mas-arm64-testing',
'electron-arm64-testing',
'electron-woa-testing'
2020-03-20 20:28:31 +00:00
];
2020-03-20 20:28:31 +00:00
let jobRequestedCount = 0;
async function makeRequest ({ auth, url, headers, body, method }) {
const clonedHeaders = {
...(headers || {})
};
if (auth && auth.bearer) {
clonedHeaders.Authorization = `Bearer ${auth.bearer}`;
}
const response = await got(url, {
headers: clonedHeaders,
body,
method,
auth: auth && (auth.username || auth.password) ? `${auth.username}:${auth.password}` : undefined
2020-03-20 20:28:31 +00:00
});
if (response.statusCode < 200 || response.statusCode >= 300) {
console.error('Error: ', `(status ${response.statusCode})`, response.body);
throw new Error(`Unexpected status code ${response.statusCode} from ${url}`);
}
return JSON.parse(response.body);
2017-11-16 21:04:50 +00:00
}
async function circleCIcall (targetBranch, workflowName, options) {
console.log(`Triggering CircleCI to run build job: ${workflowName} on branch: ${targetBranch} with release flag.`);
const buildRequest = {
branch: targetBranch,
parameters: {}
2020-03-20 20:28:31 +00:00
};
if (options.ghRelease) {
buildRequest.parameters['upload-to-storage'] = '0';
} else {
buildRequest.parameters['upload-to-storage'] = '1';
}
buildRequest.parameters[`run-${workflowName}`] = true;
if (options.arch) {
const validArches = circleCIPublishIndividualArches[workflowName];
assert(validArches.includes(options.arch), `Unknown CircleCI architecture "${options.arch}". Valid values are ${JSON.stringify(validArches)}`);
buildRequest.parameters['macos-publish-arch-limit'] = options.arch;
}
2020-03-20 20:28:31 +00:00
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 {
2020-03-20 20:28:31 +00:00
const circleResponse = await circleCIRequest(CIRCLECI_PIPELINE_URL, 'POST', buildRequest);
console.log(`CircleCI release build pipeline ${circleResponse.id} for ${workflowName} triggered.`);
2020-03-20 20:28:31 +00:00
const workflowId = await getCircleCIWorkflowId(circleResponse.id);
if (workflowId === -1) {
2020-03-20 20:28:31 +00:00
return;
}
2020-03-20 20:28:31 +00:00
const workFlowUrl = `https://circleci.com/workflow-run/${workflowId}`;
if (options.runningPublishWorkflows) {
console.log(`CircleCI release workflow request for ${workflowName} successful. Check ${workFlowUrl} for status.`);
} else {
console.log(`CircleCI release build workflow running at https://circleci.com/workflow-run/${workflowId} for ${workflowName}.`);
2020-03-20 20:28:31 +00:00
const jobNumber = await getCircleCIJobNumber(workflowId);
if (jobNumber === -1) {
2020-03-20 20:28:31 +00:00
return;
}
2020-03-20 20:28:31 +00:00
const jobUrl = `https://circleci.com/gh/electron/electron/${jobNumber}`;
console.log(`CircleCI release build request for ${workflowName} successful. Check ${jobUrl} for status.`);
}
} catch (err) {
2020-03-20 20:28:31 +00:00
console.log('Error calling CircleCI: ', err);
}
}
async function getCircleCIWorkflowId (pipelineId) {
2020-03-20 20:28:31 +00:00
const pipelineInfoUrl = `https://circleci.com/api/v2/pipeline/${pipelineId}`;
let workflowId = 0;
while (workflowId === 0) {
2020-03-20 20:28:31 +00:00
const pipelineInfo = await circleCIRequest(pipelineInfoUrl, 'GET');
switch (pipelineInfo.state) {
case 'created': {
2020-03-20 20:28:31 +00:00
const workflows = await circleCIRequest(`${pipelineInfoUrl}/workflow`, 'GET');
// The logic below expects three workflow.items: publish, lint, & setup
if (workflows.items.length === 3) {
workflowId = workflows.items.find(item => item.name.includes('publish')).id;
2020-03-20 20:28:31 +00:00
break;
}
console.log('Unexpected number of workflows, response was:', workflows);
2020-03-20 20:28:31 +00:00
workflowId = -1;
break;
}
case 'error': {
2020-03-20 20:28:31 +00:00
console.log('Error retrieving workflows, response was:', pipelineInfo);
workflowId = -1;
break;
}
}
2020-03-20 20:28:31 +00:00
await new Promise(resolve => setTimeout(resolve, CIRCLECI_WAIT_TIME));
}
2020-03-20 20:28:31 +00:00
return workflowId;
}
async function getCircleCIJobNumber (workflowId) {
2020-03-20 20:28:31 +00:00
const jobInfoUrl = `https://circleci.com/api/v2/workflow/${workflowId}/job`;
let jobNumber = 0;
while (jobNumber === 0) {
2020-03-20 20:28:31 +00:00
const jobInfo = await circleCIRequest(jobInfoUrl, 'GET');
if (!jobInfo.items) {
2020-03-20 20:28:31 +00:00
continue;
}
if (jobInfo.items.length !== 1) {
console.log('Unexpected number of jobs, response was:', jobInfo);
2020-03-20 20:28:31 +00:00
jobNumber = -1;
break;
}
switch (jobInfo.items[0].status) {
case 'not_running':
case 'queued':
case 'running': {
if (jobInfo.items[0].job_number && !isNaN(jobInfo.items[0].job_number)) {
2020-03-20 20:28:31 +00:00
jobNumber = jobInfo.items[0].job_number;
}
2020-03-20 20:28:31 +00:00
break;
}
case 'canceled':
case 'error':
case 'infrastructure_fail':
case 'timedout':
case 'not_run':
case 'failed': {
2020-03-20 20:28:31 +00:00
console.log(`Error job returned a status of ${jobInfo.items[0].status}, response was:`, jobInfo);
jobNumber = -1;
break;
}
}
2020-03-20 20:28:31 +00:00
await new Promise(resolve => setTimeout(resolve, CIRCLECI_WAIT_TIME));
}
2020-03-20 20:28:31 +00:00
return jobNumber;
}
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
2017-11-16 21:04:50 +00:00
}, true).catch(err => {
2020-03-20 20:28:31 +00:00
console.log('Error calling CircleCI:', err);
});
2017-11-16 21:04:50 +00:00
}
function buildAppVeyor (targetBranch, options) {
2020-03-20 20:28:31 +00:00
const validJobs = Object.keys(appVeyorJobs);
if (options.job) {
2020-03-20 20:28:31 +00:00
assert(validJobs.includes(options.job), `Unknown AppVeyor CI job name: ${options.job}. Valid values are: ${validJobs}.`);
callAppVeyor(targetBranch, options.job, options);
} else {
2020-03-20 20:28:31 +00:00
validJobs.forEach((job) => callAppVeyor(targetBranch, job, options));
}
}
async function callAppVeyor (targetBranch, job, options) {
2020-03-20 20:28:31 +00:00
console.log(`Triggering AppVeyor to run build job: ${job} on branch: ${targetBranch} with release flag.`);
const environmentVariables = {
ELECTRON_RELEASE: 1,
APPVEYOR_BUILD_WORKER_CLOUD: 'libcc-20'
2020-03-20 20:28:31 +00:00
};
2017-11-16 21:04:50 +00:00
if (!options.ghRelease) {
environmentVariables.UPLOAD_TO_STORAGE = 1;
}
2017-11-16 21:04:50 +00:00
const requestOpts = {
url: BUILD_APPVEYOR_URL,
2017-11-16 21:04:50 +00:00
auth: {
bearer: process.env.APPVEYOR_CLOUD_TOKEN
2017-11-16 21:04:50 +00:00
},
headers: {
'Content-Type': 'application/json'
},
body: JSON.stringify({
accountName: 'electron-bot',
projectSlug: appVeyorJobs[job],
2017-11-16 21:04:50 +00:00
branch: targetBranch,
commitId: options.commit || undefined,
2017-11-16 21:04:50 +00:00
environmentVariables
}),
method: 'POST'
2020-03-20 20:28:31 +00:00
};
jobRequestedCount++;
try {
const { version } = await makeRequest(requestOpts, true);
const buildUrl = `https://ci.appveyor.com/project/electron-bot/${appVeyorJobs[job]}/build/${version}`;
console.log(`AppVeyor release build request for ${job} successful. Check build status at ${buildUrl}`);
} catch (err) {
console.log('Could not call AppVeyor: ', err);
}
2017-11-16 21:04:50 +00:00
}
function buildCircleCI (targetBranch, options) {
if (options.job) {
assert(circleCIPublishWorkflows.includes(options.job), `Unknown CircleCI workflow name: ${options.job}. Valid values are: ${circleCIPublishWorkflows}.`);
2020-03-20 20:28:31 +00:00
circleCIcall(targetBranch, options.job, options);
2017-11-16 21:04:50 +00:00
} else {
assert(!options.arch, 'Cannot provide a single architecture while building all workflows, please specify a single workflow via --workflow');
2020-03-20 20:28:31 +00:00
options.runningPublishWorkflows = true;
circleCIPublishWorkflows.forEach((job) => circleCIcall(targetBranch, job, options));
2017-11-16 21:04:50 +00:00
}
}
async function buildVSTS (targetBranch, options) {
assert(options.armTest, `${options.ci} only works with the --armTest option.`);
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}.`);
const environmentVariables = {};
if (options.circleBuildNum) {
environmentVariables.CIRCLE_BUILD_NUM = options.circleBuildNum;
} else if (options.appveyorJobId) {
environmentVariables.APPVEYOR_JOB_ID = options.appveyorJobId;
}
let vstsURL = VSTS_URL;
let vstsToken = process.env.VSTS_TOKEN;
assert(vstsToken, `${options.ci} requires the $VSTS_TOKEN environment variable to be provided`);
if (options.ci === 'DevOps') {
vstsURL = DEVOPS_URL;
vstsToken = process.env.DEVOPS_TOKEN;
}
const requestOpts = {
url: `${vstsURL}/definitions?api-version=4.1`,
auth: {
user: '',
password: vstsToken
},
headers: {
'Content-Type': 'application/json'
}
2020-03-20 20:28:31 +00:00
};
jobRequestedCount++;
try {
const vstsResponse = await makeRequest(requestOpts, true);
const buildToRun = vstsResponse.value.find(build => build.name === options.job);
callVSTSBuild(buildToRun, targetBranch, environmentVariables, vstsURL, vstsToken);
} catch (err) {
console.log('Problem calling VSTS to get build definitions: ', err);
}
}
async function callVSTSBuild (build, targetBranch, environmentVariables, vstsURL, vstsToken) {
const buildBody = {
definition: build,
sourceBranch: targetBranch,
priority: 'high'
2020-03-20 20:28:31 +00:00
};
if (Object.keys(environmentVariables).length !== 0) {
2020-03-20 20:28:31 +00:00
buildBody.parameters = JSON.stringify(environmentVariables);
}
const requestOpts = {
url: `${vstsURL}/builds?api-version=4.1`,
auth: {
user: '',
password: vstsToken
},
headers: {
'Content-Type': 'application/json'
},
body: JSON.stringify(buildBody),
method: 'POST'
2020-03-20 20:28:31 +00:00
};
try {
const { _links } = await makeRequest(requestOpts, true);
console.log(`VSTS release build request for ${build.name} successful. Check ${_links.web.href} for status.`);
} catch (err) {
console.log(`Could not call VSTS for job ${build.name}: `, err);
}
}
function runRelease (targetBranch, options) {
2017-11-16 21:04:50 +00:00
if (options.ci) {
switch (options.ci) {
case 'CircleCI': {
2020-03-20 20:28:31 +00:00
buildCircleCI(targetBranch, options);
break;
2017-11-16 21:04:50 +00:00
}
case 'AppVeyor': {
2020-03-20 20:28:31 +00:00
buildAppVeyor(targetBranch, options);
break;
2017-11-16 21:04:50 +00:00
}
case 'DevOps':
case 'VSTS': {
2020-03-20 20:28:31 +00:00
buildVSTS(targetBranch, options);
break;
}
default: {
2020-03-20 20:28:31 +00:00
console.log(`Error! Unknown CI: ${options.ci}.`);
process.exit(1);
}
2017-11-16 21:04:50 +00:00
}
} else {
2020-03-20 20:28:31 +00:00
buildCircleCI(targetBranch, options);
buildAppVeyor(targetBranch, options);
2017-11-16 21:04:50 +00:00
}
2020-03-20 20:28:31 +00:00
console.log(`${jobRequestedCount} jobs were requested.`);
2017-11-16 21:04:50 +00:00
}
2020-03-20 20:28:31 +00:00
module.exports = runRelease;
2017-11-16 21:04:50 +00:00
if (require.main === module) {
const args = require('minimist')(process.argv.slice(2), {
boolean: ['ghRelease', 'armTest']
2020-03-20 20:28:31 +00:00
});
const targetBranch = args._[0];
2017-11-16 21:04:50 +00:00
if (args._.length < 1) {
console.log(`Trigger CI to build release builds of electron.
Usage: ci-release-build.js [--job=CI_JOB_NAME] [--arch=INDIVIDUAL_ARCH] [--ci=CircleCI|AppVeyor|VSTS|DevOps]
[--ghRelease] [--armTest] [--circleBuildNum=xxx] [--appveyorJobId=xxx] [--commit=sha] TARGET_BRANCH
2020-03-20 20:28:31 +00:00
`);
process.exit(0);
2017-11-16 21:04:50 +00:00
}
2020-03-20 20:28:31 +00:00
runRelease(targetBranch, args);
}