305 lines
		
	
	
	
		
			9.6 KiB
			
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			305 lines
		
	
	
	
		
			9.6 KiB
			
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
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 jobInfoUrl = `https://circleci.com/api/v2/workflow/${workflowId}/jobs`
 | 
						|
    const jobInfo = await circleCIRequest(jobInfoUrl, 'GET')
 | 
						|
    if (!jobInfo.items || jobInfo.items.length !== 1) {
 | 
						|
      console.log('Error retrieving job for workflow, response was:', jobInfo)
 | 
						|
      return
 | 
						|
    }
 | 
						|
    const jobUrl = `https://circleci.com/gh/electron/electron/${jobInfo.items[0].job_number}`
 | 
						|
    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 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)
 | 
						|
    buildVSTS(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)
 | 
						|
}
 |