ci-release-build.js 7.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236
  1. if (!process.env.CI) require('dotenv-safe').load()
  2. const assert = require('assert')
  3. const request = require('request')
  4. const buildAppVeyorURL = 'https://ci.appveyor.com/api/builds'
  5. const vstsURL = 'https://github.visualstudio.com/electron/_apis/build'
  6. const appVeyorJobs = {
  7. 'electron-x64': 'electron-x64-release',
  8. 'electron-ia32': 'electron-ia32-release'
  9. }
  10. const circleCIJobs = [
  11. 'linux-arm-publish',
  12. 'linux-arm64-publish',
  13. 'linux-ia32-publish',
  14. 'linux-x64-publish',
  15. 'mas-publish',
  16. 'osx-publish'
  17. ]
  18. const vstsArmJobs = [
  19. 'electron-arm-testing',
  20. 'electron-arm64-testing'
  21. ]
  22. async function makeRequest (requestOptions, parseResponse) {
  23. return new Promise((resolve, reject) => {
  24. request(requestOptions, (err, res, body) => {
  25. if (!err && res.statusCode >= 200 && res.statusCode < 300) {
  26. if (parseResponse) {
  27. const build = JSON.parse(body)
  28. resolve(build)
  29. } else {
  30. resolve(body)
  31. }
  32. } else {
  33. console.error('Error occurred while requesting:', requestOptions.url)
  34. if (parseResponse) {
  35. try {
  36. console.log('Error: ', `(status ${res.statusCode})`, err || JSON.parse(res.body), requestOptions)
  37. } catch (err) {
  38. console.log('Error: ', `(status ${res.statusCode})`, err || res.body, requestOptions)
  39. }
  40. } else {
  41. console.log('Error: ', `(status ${res.statusCode})`, err || res.body, requestOptions)
  42. }
  43. reject(err)
  44. }
  45. })
  46. })
  47. }
  48. async function circleCIcall (buildUrl, targetBranch, job, options) {
  49. console.log(`Triggering CircleCI to run build job: ${job} on branch: ${targetBranch} with release flag.`)
  50. const buildRequest = {
  51. 'build_parameters': {
  52. 'CIRCLE_JOB': job
  53. }
  54. }
  55. if (!options.ghRelease) {
  56. buildRequest.build_parameters.UPLOAD_TO_S3 = 1
  57. }
  58. const circleResponse = await makeRequest({
  59. method: 'POST',
  60. url: buildUrl,
  61. headers: {
  62. 'Content-Type': 'application/json',
  63. 'Accept': 'application/json'
  64. },
  65. body: JSON.stringify(buildRequest)
  66. }, true).catch(err => {
  67. console.log('Error calling CircleCI:', err)
  68. })
  69. console.log(`CircleCI release build request for ${job} successful. Check ${circleResponse.build_url} for status.`)
  70. }
  71. function buildAppVeyor (targetBranch, options) {
  72. const validJobs = Object.keys(appVeyorJobs)
  73. if (options.job) {
  74. assert(validJobs.includes(options.job), `Unknown AppVeyor CI job name: ${options.job}. Valid values are: ${validJobs}.`)
  75. callAppVeyor(targetBranch, options.job, options)
  76. } else {
  77. validJobs.forEach((job) => callAppVeyor(targetBranch, job, options))
  78. }
  79. }
  80. async function callAppVeyor (targetBranch, job, options) {
  81. console.log(`Triggering AppVeyor to run build job: ${job} on branch: ${targetBranch} with release flag.`)
  82. const environmentVariables = {
  83. ELECTRON_RELEASE: 1
  84. }
  85. if (!options.ghRelease) {
  86. environmentVariables.UPLOAD_TO_S3 = 1
  87. }
  88. const requestOpts = {
  89. url: buildAppVeyorURL,
  90. auth: {
  91. bearer: process.env.APPVEYOR_CLOUD_TOKEN
  92. },
  93. headers: {
  94. 'Content-Type': 'application/json'
  95. },
  96. body: JSON.stringify({
  97. accountName: 'electron-bot',
  98. projectSlug: appVeyorJobs[job],
  99. branch: targetBranch,
  100. environmentVariables
  101. }),
  102. method: 'POST'
  103. }
  104. const appVeyorResponse = await makeRequest(requestOpts, true).catch(err => {
  105. console.log('Error calling AppVeyor:', err)
  106. })
  107. const buildUrl = `https://ci.appveyor.com/project/electron-bot/${appVeyorJobs[job]}/build/${appVeyorResponse.version}`
  108. console.log(`AppVeyor release build request for ${job} successful. Check build status at ${buildUrl}`)
  109. }
  110. function buildCircleCI (targetBranch, options) {
  111. const circleBuildUrl = `https://circleci.com/api/v1.1/project/github/electron/electron/tree/${targetBranch}?circle-token=${process.env.CIRCLE_TOKEN}`
  112. if (options.job) {
  113. assert(circleCIJobs.includes(options.job), `Unknown CircleCI job name: ${options.job}. Valid values are: ${circleCIJobs}.`)
  114. circleCIcall(circleBuildUrl, targetBranch, options.job, options)
  115. } else {
  116. circleCIJobs.forEach((job) => circleCIcall(circleBuildUrl, targetBranch, job, options))
  117. }
  118. }
  119. async function buildVSTS (targetBranch, options) {
  120. if (options.armTest) {
  121. assert(vstsArmJobs.includes(options.job), `Unknown VSTS CI arm test job name: ${options.job}. Valid values are: ${vstsArmJobs}.`)
  122. }
  123. console.log(`Triggering VSTS to run build on branch: ${targetBranch} with release flag.`)
  124. const environmentVariables = {
  125. ELECTRON_RELEASE: 1
  126. }
  127. if (options.armTest) {
  128. environmentVariables.CIRCLE_BUILD_NUM = options.circleBuildNum
  129. } else {
  130. if (!options.ghRelease) {
  131. environmentVariables.UPLOAD_TO_S3 = 1
  132. }
  133. }
  134. const requestOpts = {
  135. url: `${vstsURL}/definitions?api-version=4.1`,
  136. auth: {
  137. user: '',
  138. password: process.env.VSTS_TOKEN
  139. },
  140. headers: {
  141. 'Content-Type': 'application/json'
  142. }
  143. }
  144. const vstsResponse = await makeRequest(requestOpts, true).catch(err => {
  145. console.log('Error calling VSTS to get build definitions:', err)
  146. })
  147. const buildsToRun = vstsResponse.value.filter(build => build.name === options.job)
  148. buildsToRun.forEach((build) => callVSTSBuild(build, targetBranch, environmentVariables))
  149. }
  150. async function callVSTSBuild (build, targetBranch, environmentVariables) {
  151. const buildBody = {
  152. definition: build,
  153. sourceBranch: targetBranch,
  154. priority: 'high'
  155. }
  156. if (Object.keys(environmentVariables).length !== 0) {
  157. buildBody.parameters = JSON.stringify(environmentVariables)
  158. }
  159. const requestOpts = {
  160. url: `${vstsURL}/builds?api-version=4.1`,
  161. auth: {
  162. user: '',
  163. password: process.env.VSTS_TOKEN
  164. },
  165. headers: {
  166. 'Content-Type': 'application/json'
  167. },
  168. body: JSON.stringify(buildBody),
  169. method: 'POST'
  170. }
  171. const vstsResponse = await makeRequest(requestOpts, true).catch(err => {
  172. console.log(`Error calling VSTS for job ${build.name}`, err)
  173. })
  174. console.log(`VSTS release build request for ${build.name} successful. Check ${vstsResponse._links.web.href} for status.`)
  175. }
  176. function runRelease (targetBranch, options) {
  177. if (options.ci) {
  178. switch (options.ci) {
  179. case 'CircleCI': {
  180. buildCircleCI(targetBranch, options)
  181. break
  182. }
  183. case 'AppVeyor': {
  184. buildAppVeyor(targetBranch, options)
  185. break
  186. }
  187. case 'VSTS': {
  188. buildVSTS(targetBranch, options)
  189. break
  190. }
  191. default: {
  192. console.log(`Error! Unknown CI: ${options.ci}.`)
  193. process.exit(1)
  194. }
  195. }
  196. } else {
  197. buildCircleCI(targetBranch, options)
  198. buildAppVeyor(targetBranch, options)
  199. }
  200. }
  201. module.exports = runRelease
  202. if (require.main === module) {
  203. const args = require('minimist')(process.argv.slice(2), {
  204. boolean: ['ghRelease', 'armTest']
  205. })
  206. const targetBranch = args._[0]
  207. if (args._.length < 1) {
  208. console.log(`Trigger CI to build release builds of electron.
  209. Usage: ci-release-build.js [--job=CI_JOB_NAME] [--ci=CircleCI|AppVeyor|VSTS]
  210. [--ghRelease] [--armTest] [--circleBuildNum=xxx] TARGET_BRANCH
  211. `)
  212. process.exit(0)
  213. }
  214. runRelease(targetBranch, args)
  215. }