ci-release-build.js 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371
  1. if (!process.env.CI) require('dotenv-safe').load();
  2. const assert = require('assert');
  3. const request = require('request');
  4. const BUILD_APPVEYOR_URL = 'https://ci.appveyor.com/api/builds';
  5. const CIRCLECI_PIPELINE_URL = 'https://circleci.com/api/v2/project/gh/electron/electron/pipeline';
  6. const VSTS_URL = 'https://github.visualstudio.com/electron/_apis/build';
  7. const DEVOPS_URL = 'https://dev.azure.com/electron-ci/electron/_apis/build';
  8. const CIRCLECI_WAIT_TIME = process.env.CIRCLECI_WAIT_TIME || 30000;
  9. const appVeyorJobs = {
  10. 'electron-x64': 'electron-x64-release',
  11. 'electron-ia32': 'electron-ia32-release',
  12. 'electron-woa': 'electron-woa-release'
  13. };
  14. const circleCIPublishWorkflows = [
  15. 'linux-publish',
  16. 'macos-publish'
  17. ];
  18. const circleCIJobs = circleCIPublishWorkflows.concat([
  19. 'linux-arm-publish',
  20. 'linux-arm64-publish',
  21. 'linux-ia32-publish',
  22. 'linux-x64-publish',
  23. 'mas-publish',
  24. 'mas-publish-arm64',
  25. 'osx-publish',
  26. 'osx-publish-arm64'
  27. ]);
  28. const vstsArmJobs = [
  29. 'electron-arm-testing',
  30. 'electron-osx-arm64-testing',
  31. 'electron-mas-arm64-testing',
  32. 'electron-arm64-testing',
  33. 'electron-woa-testing'
  34. ];
  35. let jobRequestedCount = 0;
  36. async function makeRequest (requestOptions, parseResponse) {
  37. return new Promise((resolve, reject) => {
  38. request(requestOptions, (err, res, body) => {
  39. if (!err && res.statusCode >= 200 && res.statusCode < 300) {
  40. if (parseResponse) {
  41. const build = JSON.parse(body);
  42. resolve(build);
  43. } else {
  44. resolve(body);
  45. }
  46. } else {
  47. console.error('Error occurred while requesting:', requestOptions.url);
  48. if (parseResponse) {
  49. try {
  50. console.log('Error: ', `(status ${res.statusCode})`, err || JSON.parse(res.body));
  51. } catch (err) {
  52. console.log('Error: ', `(status ${res.statusCode})`, res.body);
  53. }
  54. } else {
  55. console.log('Error: ', `(status ${res.statusCode})`, err || res.body);
  56. }
  57. reject(err);
  58. }
  59. });
  60. });
  61. }
  62. async function circleCIcall (targetBranch, job, options) {
  63. console.log(`Triggering CircleCI to run build job: ${job} on branch: ${targetBranch} with release flag.`);
  64. const buildRequest = {
  65. branch: targetBranch,
  66. parameters: {}
  67. };
  68. if (options.ghRelease) {
  69. buildRequest.parameters['upload-to-s3'] = '0';
  70. } else {
  71. buildRequest.parameters['upload-to-s3'] = '1';
  72. }
  73. buildRequest.parameters[`run-${job}`] = true;
  74. jobRequestedCount++;
  75. // The logic below expects that the CircleCI workflows for releases each
  76. // contain only one job in order to maintain compatibility with sudowoodo.
  77. // If the workflows are changed in the CircleCI config.yml, this logic will
  78. // also need to be changed as well as possibly changing sudowoodo.
  79. try {
  80. const circleResponse = await circleCIRequest(CIRCLECI_PIPELINE_URL, 'POST', buildRequest);
  81. console.log(`CircleCI release build pipeline ${circleResponse.id} for ${job} triggered.`);
  82. const workflowId = await getCircleCIWorkflowId(circleResponse.id);
  83. if (workflowId === -1) {
  84. return;
  85. }
  86. const workFlowUrl = `https://circleci.com/workflow-run/${workflowId}`;
  87. if (options.runningPublishWorkflows) {
  88. console.log(`CircleCI release workflow request for ${job} successful. Check ${workFlowUrl} for status.`);
  89. } else {
  90. console.log(`CircleCI release build workflow running at https://circleci.com/workflow-run/${workflowId} for ${job}.`);
  91. const jobNumber = await getCircleCIJobNumber(workflowId);
  92. if (jobNumber === -1) {
  93. return;
  94. }
  95. const jobUrl = `https://circleci.com/gh/electron/electron/${jobNumber}`;
  96. console.log(`CircleCI release build request for ${job} successful. Check ${jobUrl} for status.`);
  97. }
  98. } catch (err) {
  99. console.log('Error calling CircleCI: ', err);
  100. }
  101. }
  102. async function getCircleCIWorkflowId (pipelineId) {
  103. const pipelineInfoUrl = `https://circleci.com/api/v2/pipeline/${pipelineId}`;
  104. let workflowId = 0;
  105. while (workflowId === 0) {
  106. const pipelineInfo = await circleCIRequest(pipelineInfoUrl, 'GET');
  107. switch (pipelineInfo.state) {
  108. case 'created': {
  109. const workflows = await circleCIRequest(`${pipelineInfoUrl}/workflow`, 'GET');
  110. // The logic below expects two workflow.items: publish [0] & setup [1]
  111. if (workflows.items.length === 2) {
  112. workflowId = workflows.items.find(item => item.name.includes('publish')).id;
  113. break;
  114. }
  115. console.log('Unxpected number of workflows, response was:', pipelineInfo);
  116. workflowId = -1;
  117. break;
  118. }
  119. case 'error': {
  120. console.log('Error retrieving workflows, response was:', pipelineInfo);
  121. workflowId = -1;
  122. break;
  123. }
  124. }
  125. await new Promise(resolve => setTimeout(resolve, CIRCLECI_WAIT_TIME));
  126. }
  127. return workflowId;
  128. }
  129. async function getCircleCIJobNumber (workflowId) {
  130. const jobInfoUrl = `https://circleci.com/api/v2/workflow/${workflowId}/job`;
  131. let jobNumber = 0;
  132. while (jobNumber === 0) {
  133. const jobInfo = await circleCIRequest(jobInfoUrl, 'GET');
  134. if (!jobInfo.items) {
  135. continue;
  136. }
  137. if (jobInfo.items.length !== 1) {
  138. console.log('Unxpected number of jobs, response was:', jobInfo);
  139. jobNumber = -1;
  140. break;
  141. }
  142. switch (jobInfo.items[0].status) {
  143. case 'not_running':
  144. case 'queued':
  145. case 'running': {
  146. if (jobInfo.items[0].job_number && !isNaN(jobInfo.items[0].job_number)) {
  147. jobNumber = jobInfo.items[0].job_number;
  148. }
  149. break;
  150. }
  151. case 'canceled':
  152. case 'error':
  153. case 'infrastructure_fail':
  154. case 'timedout':
  155. case 'not_run':
  156. case 'failed': {
  157. console.log(`Error job returned a status of ${jobInfo.items[0].status}, response was:`, jobInfo);
  158. jobNumber = -1;
  159. break;
  160. }
  161. }
  162. await new Promise(resolve => setTimeout(resolve, CIRCLECI_WAIT_TIME));
  163. }
  164. return jobNumber;
  165. }
  166. async function circleCIRequest (url, method, requestBody) {
  167. return makeRequest({
  168. auth: {
  169. username: process.env.CIRCLE_TOKEN,
  170. password: ''
  171. },
  172. method,
  173. url,
  174. headers: {
  175. 'Content-Type': 'application/json',
  176. Accept: 'application/json'
  177. },
  178. body: requestBody ? JSON.stringify(requestBody) : null
  179. }, true).catch(err => {
  180. console.log('Error calling CircleCI:', err);
  181. });
  182. }
  183. function buildAppVeyor (targetBranch, options) {
  184. const validJobs = Object.keys(appVeyorJobs);
  185. if (options.job) {
  186. assert(validJobs.includes(options.job), `Unknown AppVeyor CI job name: ${options.job}. Valid values are: ${validJobs}.`);
  187. callAppVeyor(targetBranch, options.job, options);
  188. } else {
  189. validJobs.forEach((job) => callAppVeyor(targetBranch, job, options));
  190. }
  191. }
  192. async function callAppVeyor (targetBranch, job, options) {
  193. console.log(`Triggering AppVeyor to run build job: ${job} on branch: ${targetBranch} with release flag.`);
  194. const environmentVariables = {
  195. ELECTRON_RELEASE: 1,
  196. APPVEYOR_BUILD_WORKER_CLOUD: 'libcc-20'
  197. };
  198. if (!options.ghRelease) {
  199. environmentVariables.UPLOAD_TO_S3 = 1;
  200. }
  201. const requestOpts = {
  202. url: BUILD_APPVEYOR_URL,
  203. auth: {
  204. bearer: process.env.APPVEYOR_CLOUD_TOKEN
  205. },
  206. headers: {
  207. 'Content-Type': 'application/json'
  208. },
  209. body: JSON.stringify({
  210. accountName: 'electron-bot',
  211. projectSlug: appVeyorJobs[job],
  212. branch: targetBranch,
  213. commitId: options.commit || undefined,
  214. environmentVariables
  215. }),
  216. method: 'POST'
  217. };
  218. jobRequestedCount++;
  219. try {
  220. const { version } = await makeRequest(requestOpts, true);
  221. const buildUrl = `https://ci.appveyor.com/project/electron-bot/${appVeyorJobs[job]}/build/${version}`;
  222. console.log(`AppVeyor release build request for ${job} successful. Check build status at ${buildUrl}`);
  223. } catch (err) {
  224. console.log('Could not call AppVeyor: ', err);
  225. }
  226. }
  227. function buildCircleCI (targetBranch, options) {
  228. if (options.job) {
  229. assert(circleCIJobs.includes(options.job), `Unknown CircleCI job name: ${options.job}. Valid values are: ${circleCIJobs}.`);
  230. circleCIcall(targetBranch, options.job, options);
  231. } else {
  232. options.runningPublishWorkflows = true;
  233. circleCIPublishWorkflows.forEach((job) => circleCIcall(targetBranch, job, options));
  234. }
  235. }
  236. async function buildVSTS (targetBranch, options) {
  237. assert(options.armTest, `${options.ci} only works with the --armTest option.`);
  238. assert(vstsArmJobs.includes(options.job), `Unknown VSTS CI arm test job name: ${options.job}. Valid values are: ${vstsArmJobs}.`);
  239. console.log(`Triggering VSTS to run build on branch: ${targetBranch}.`);
  240. const environmentVariables = {};
  241. if (options.circleBuildNum) {
  242. environmentVariables.CIRCLE_BUILD_NUM = options.circleBuildNum;
  243. } else if (options.appveyorJobId) {
  244. environmentVariables.APPVEYOR_JOB_ID = options.appveyorJobId;
  245. }
  246. let vstsURL = VSTS_URL;
  247. let vstsToken = process.env.VSTS_TOKEN;
  248. assert(vstsToken, `${options.ci} requires the $VSTS_TOKEN environment variable to be provided`);
  249. if (options.ci === 'DevOps') {
  250. vstsURL = DEVOPS_URL;
  251. vstsToken = process.env.DEVOPS_TOKEN;
  252. }
  253. const requestOpts = {
  254. url: `${vstsURL}/definitions?api-version=4.1`,
  255. auth: {
  256. user: '',
  257. password: vstsToken
  258. },
  259. headers: {
  260. 'Content-Type': 'application/json'
  261. }
  262. };
  263. jobRequestedCount++;
  264. try {
  265. const vstsResponse = await makeRequest(requestOpts, true);
  266. const buildToRun = vstsResponse.value.find(build => build.name === options.job);
  267. callVSTSBuild(buildToRun, targetBranch, environmentVariables, vstsURL, vstsToken);
  268. } catch (err) {
  269. console.log('Problem calling VSTS to get build definitions: ', err);
  270. }
  271. }
  272. async function callVSTSBuild (build, targetBranch, environmentVariables, vstsURL, vstsToken) {
  273. const buildBody = {
  274. definition: build,
  275. sourceBranch: targetBranch,
  276. priority: 'high'
  277. };
  278. if (Object.keys(environmentVariables).length !== 0) {
  279. buildBody.parameters = JSON.stringify(environmentVariables);
  280. }
  281. const requestOpts = {
  282. url: `${vstsURL}/builds?api-version=4.1`,
  283. auth: {
  284. user: '',
  285. password: vstsToken
  286. },
  287. headers: {
  288. 'Content-Type': 'application/json'
  289. },
  290. body: JSON.stringify(buildBody),
  291. method: 'POST'
  292. };
  293. try {
  294. const { _links } = await makeRequest(requestOpts, true);
  295. console.log(`VSTS release build request for ${build.name} successful. Check ${_links.web.href} for status.`);
  296. } catch (err) {
  297. console.log(`Could not call VSTS for job ${build.name}: `, err);
  298. }
  299. }
  300. function runRelease (targetBranch, options) {
  301. if (options.ci) {
  302. switch (options.ci) {
  303. case 'CircleCI': {
  304. buildCircleCI(targetBranch, options);
  305. break;
  306. }
  307. case 'AppVeyor': {
  308. buildAppVeyor(targetBranch, options);
  309. break;
  310. }
  311. case 'DevOps':
  312. case 'VSTS': {
  313. buildVSTS(targetBranch, options);
  314. break;
  315. }
  316. default: {
  317. console.log(`Error! Unknown CI: ${options.ci}.`);
  318. process.exit(1);
  319. }
  320. }
  321. } else {
  322. buildCircleCI(targetBranch, options);
  323. buildAppVeyor(targetBranch, options);
  324. }
  325. console.log(`${jobRequestedCount} jobs were requested.`);
  326. }
  327. module.exports = runRelease;
  328. if (require.main === module) {
  329. const args = require('minimist')(process.argv.slice(2), {
  330. boolean: ['ghRelease', 'armTest']
  331. });
  332. const targetBranch = args._[0];
  333. if (args._.length < 1) {
  334. console.log(`Trigger CI to build release builds of electron.
  335. Usage: ci-release-build.js [--job=CI_JOB_NAME] [--ci=CircleCI|AppVeyor|VSTS|DevOps]
  336. [--ghRelease] [--armTest] [--circleBuildNum=xxx] [--appveyorJobId=xxx] [--commit=sha] TARGET_BRANCH
  337. `);
  338. process.exit(0);
  339. }
  340. runRelease(targetBranch, args);
  341. }