ci-release-build.js 9.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289
  1. if (!process.env.CI) require('dotenv-safe').load();
  2. const assert = require('assert');
  3. const got = require('got');
  4. const { Octokit } = require('@octokit/rest');
  5. const BUILD_APPVEYOR_URL = 'https://ci.appveyor.com/api/builds';
  6. const CIRCLECI_PIPELINE_URL = 'https://circleci.com/api/v2/project/gh/electron/electron/pipeline';
  7. const CIRCLECI_WAIT_TIME = process.env.CIRCLECI_WAIT_TIME || 30000;
  8. const appVeyorJobs = {
  9. 'electron-x64': 'electron-x64-release',
  10. 'electron-ia32': 'electron-ia32-release',
  11. 'electron-woa': 'electron-woa-release'
  12. };
  13. const circleCIPublishWorkflows = [
  14. 'linux-publish',
  15. 'macos-publish'
  16. ];
  17. const circleCIPublishIndividualArches = {
  18. 'macos-publish': ['osx-x64', 'mas-x64', 'osx-arm64', 'mas-arm64'],
  19. 'linux-publish': ['arm', 'arm64', 'x64']
  20. };
  21. let jobRequestedCount = 0;
  22. async function makeRequest ({ auth, username, password, url, headers, body, method }) {
  23. const clonedHeaders = {
  24. ...(headers || {})
  25. };
  26. if (auth?.bearer) {
  27. clonedHeaders.Authorization = `Bearer ${auth.bearer}`;
  28. }
  29. const options = {
  30. headers: clonedHeaders,
  31. body,
  32. method
  33. };
  34. if (username || password) {
  35. options.username = username;
  36. options.password = password;
  37. }
  38. const response = await got(url, options);
  39. if (response.statusCode < 200 || response.statusCode >= 300) {
  40. console.error('Error: ', `(status ${response.statusCode})`, response.body);
  41. throw new Error(`Unexpected status code ${response.statusCode} from ${url}`);
  42. }
  43. return JSON.parse(response.body);
  44. }
  45. async function circleCIcall (targetBranch, workflowName, options) {
  46. console.log(`Triggering CircleCI to run build job: ${workflowName} on branch: ${targetBranch} with release flag.`);
  47. const buildRequest = {
  48. branch: targetBranch,
  49. parameters: {}
  50. };
  51. if (options.ghRelease) {
  52. buildRequest.parameters['upload-to-storage'] = '0';
  53. } else {
  54. buildRequest.parameters['upload-to-storage'] = '1';
  55. }
  56. buildRequest.parameters[`run-${workflowName}`] = true;
  57. if (options.arch) {
  58. const validArches = circleCIPublishIndividualArches[workflowName];
  59. assert(validArches.includes(options.arch), `Unknown CircleCI architecture "${options.arch}". Valid values are ${JSON.stringify(validArches)}`);
  60. buildRequest.parameters['macos-publish-arch-limit'] = options.arch;
  61. }
  62. jobRequestedCount++;
  63. // The logic below expects that the CircleCI workflows for releases each
  64. // contain only one job in order to maintain compatibility with sudowoodo.
  65. // If the workflows are changed in the CircleCI config.yml, this logic will
  66. // also need to be changed as well as possibly changing sudowoodo.
  67. try {
  68. const circleResponse = await circleCIRequest(CIRCLECI_PIPELINE_URL, 'POST', buildRequest);
  69. console.log(`CircleCI release build pipeline ${circleResponse.id} for ${workflowName} triggered.`);
  70. const workflowId = await getCircleCIWorkflowId(circleResponse.id);
  71. if (workflowId === -1) {
  72. return;
  73. }
  74. const workFlowUrl = `https://circleci.com/workflow-run/${workflowId}`;
  75. if (options.runningPublishWorkflows) {
  76. console.log(`CircleCI release workflow request for ${workflowName} successful. Check ${workFlowUrl} for status.`);
  77. } else {
  78. console.log(`CircleCI release build workflow running at https://circleci.com/workflow-run/${workflowId} for ${workflowName}.`);
  79. const jobNumber = await getCircleCIJobNumber(workflowId);
  80. if (jobNumber === -1) {
  81. return;
  82. }
  83. const jobUrl = `https://circleci.com/gh/electron/electron/${jobNumber}`;
  84. console.log(`CircleCI release build request for ${workflowName} successful. Check ${jobUrl} for status.`);
  85. }
  86. } catch (err) {
  87. console.log('Error calling CircleCI: ', err);
  88. }
  89. }
  90. async function getCircleCIWorkflowId (pipelineId) {
  91. const pipelineInfoUrl = `https://circleci.com/api/v2/pipeline/${pipelineId}`;
  92. let workflowId = 0;
  93. while (workflowId === 0) {
  94. const pipelineInfo = await circleCIRequest(pipelineInfoUrl, 'GET');
  95. switch (pipelineInfo.state) {
  96. case 'created': {
  97. const workflows = await circleCIRequest(`${pipelineInfoUrl}/workflow`, 'GET');
  98. // The logic below expects three workflow.items: publish, lint, & setup
  99. if (workflows.items.length === 3) {
  100. workflowId = workflows.items.find(item => item.name.includes('publish')).id;
  101. break;
  102. }
  103. console.log('Unexpected number of workflows, response was:', workflows);
  104. workflowId = -1;
  105. break;
  106. }
  107. case 'error': {
  108. console.log('Error retrieving workflows, response was:', pipelineInfo);
  109. workflowId = -1;
  110. break;
  111. }
  112. }
  113. await new Promise(resolve => setTimeout(resolve, CIRCLECI_WAIT_TIME));
  114. }
  115. return workflowId;
  116. }
  117. async function getCircleCIJobNumber (workflowId) {
  118. const jobInfoUrl = `https://circleci.com/api/v2/workflow/${workflowId}/job`;
  119. let jobNumber = 0;
  120. while (jobNumber === 0) {
  121. const jobInfo = await circleCIRequest(jobInfoUrl, 'GET');
  122. if (!jobInfo.items) {
  123. continue;
  124. }
  125. if (jobInfo.items.length !== 1) {
  126. console.log('Unexpected number of jobs, response was:', jobInfo);
  127. jobNumber = -1;
  128. break;
  129. }
  130. switch (jobInfo.items[0].status) {
  131. case 'not_running':
  132. case 'queued':
  133. case 'running': {
  134. if (jobInfo.items[0].job_number && !isNaN(jobInfo.items[0].job_number)) {
  135. jobNumber = jobInfo.items[0].job_number;
  136. }
  137. break;
  138. }
  139. case 'canceled':
  140. case 'error':
  141. case 'infrastructure_fail':
  142. case 'timedout':
  143. case 'not_run':
  144. case 'failed': {
  145. console.log(`Error job returned a status of ${jobInfo.items[0].status}, response was:`, jobInfo);
  146. jobNumber = -1;
  147. break;
  148. }
  149. }
  150. await new Promise(resolve => setTimeout(resolve, CIRCLECI_WAIT_TIME));
  151. }
  152. return jobNumber;
  153. }
  154. async function circleCIRequest (url, method, requestBody) {
  155. const requestOpts = {
  156. username: process.env.CIRCLE_TOKEN,
  157. password: '',
  158. method,
  159. url,
  160. headers: {
  161. 'Content-Type': 'application/json',
  162. Accept: 'application/json'
  163. }
  164. };
  165. if (requestBody) {
  166. requestOpts.body = JSON.stringify(requestBody);
  167. }
  168. return makeRequest(requestOpts, true).catch(err => {
  169. console.log('Error calling CircleCI:', err);
  170. });
  171. }
  172. function buildAppVeyor (targetBranch, options) {
  173. const validJobs = Object.keys(appVeyorJobs);
  174. if (options.job) {
  175. assert(validJobs.includes(options.job), `Unknown AppVeyor CI job name: ${options.job}. Valid values are: ${validJobs}.`);
  176. callAppVeyor(targetBranch, options.job, options);
  177. } else {
  178. validJobs.forEach((job) => callAppVeyor(targetBranch, job, options));
  179. }
  180. }
  181. async function callAppVeyor (targetBranch, job, options) {
  182. console.log(`Triggering AppVeyor to run build job: ${job} on branch: ${targetBranch} with release flag.`);
  183. const environmentVariables = {
  184. ELECTRON_RELEASE: 1,
  185. APPVEYOR_BUILD_WORKER_CLOUD: 'electronhq-16-core'
  186. };
  187. if (!options.ghRelease) {
  188. environmentVariables.UPLOAD_TO_STORAGE = 1;
  189. }
  190. const requestOpts = {
  191. url: BUILD_APPVEYOR_URL,
  192. auth: {
  193. bearer: process.env.APPVEYOR_CLOUD_TOKEN
  194. },
  195. headers: {
  196. 'Content-Type': 'application/json'
  197. },
  198. body: JSON.stringify({
  199. accountName: 'electron-bot',
  200. projectSlug: appVeyorJobs[job],
  201. branch: targetBranch,
  202. commitId: options.commit || undefined,
  203. environmentVariables
  204. }),
  205. method: 'POST'
  206. };
  207. jobRequestedCount++;
  208. try {
  209. const { version } = await makeRequest(requestOpts, true);
  210. const buildUrl = `https://ci.appveyor.com/project/electron-bot/${appVeyorJobs[job]}/build/${version}`;
  211. console.log(`AppVeyor release build request for ${job} successful. Check build status at ${buildUrl}`);
  212. } catch (err) {
  213. console.log('Could not call AppVeyor: ', err);
  214. }
  215. }
  216. function buildCircleCI (targetBranch, options) {
  217. if (options.job) {
  218. assert(circleCIPublishWorkflows.includes(options.job), `Unknown CircleCI workflow name: ${options.job}. Valid values are: ${circleCIPublishWorkflows}.`);
  219. circleCIcall(targetBranch, options.job, options);
  220. } else {
  221. assert(!options.arch, 'Cannot provide a single architecture while building all workflows, please specify a single workflow via --workflow');
  222. options.runningPublishWorkflows = true;
  223. circleCIPublishWorkflows.forEach((job) => circleCIcall(targetBranch, job, options));
  224. }
  225. }
  226. function runRelease (targetBranch, options) {
  227. if (options.ci) {
  228. switch (options.ci) {
  229. case 'CircleCI': {
  230. buildCircleCI(targetBranch, options);
  231. break;
  232. }
  233. case 'AppVeyor': {
  234. buildAppVeyor(targetBranch, options);
  235. break;
  236. }
  237. default: {
  238. console.log(`Error! Unknown CI: ${options.ci}.`);
  239. process.exit(1);
  240. }
  241. }
  242. } else {
  243. buildCircleCI(targetBranch, options);
  244. buildAppVeyor(targetBranch, options);
  245. }
  246. console.log(`${jobRequestedCount} jobs were requested.`);
  247. }
  248. module.exports = runRelease;
  249. if (require.main === module) {
  250. const args = require('minimist')(process.argv.slice(2), {
  251. boolean: ['ghRelease']
  252. });
  253. const targetBranch = args._[0];
  254. if (args._.length < 1) {
  255. console.log(`Trigger CI to build release builds of electron.
  256. Usage: ci-release-build.js [--job=CI_JOB_NAME] [--arch=INDIVIDUAL_ARCH] [--ci=CircleCI|AppVeyor]
  257. [--ghRelease] [--circleBuildNum=xxx] [--appveyorJobId=xxx] [--commit=sha] TARGET_BRANCH
  258. `);
  259. process.exit(0);
  260. }
  261. runRelease(targetBranch, args);
  262. }