2020-07-16 10:57:13 +02:00
|
|
|
import * as core from '@actions/core'
|
|
|
|
import {Inputs} from './create-pull-request'
|
|
|
|
import {Octokit, OctokitOptions} from './octokit-client'
|
|
|
|
|
|
|
|
const ERROR_PR_REVIEW_FROM_AUTHOR =
|
|
|
|
'Review cannot be requested from pull request author'
|
|
|
|
|
|
|
|
interface Repository {
|
|
|
|
owner: string
|
|
|
|
repo: string
|
|
|
|
}
|
|
|
|
|
2020-09-17 03:41:24 +02:00
|
|
|
interface Pull {
|
|
|
|
number: number
|
|
|
|
html_url: string
|
|
|
|
}
|
|
|
|
|
2020-07-16 10:57:13 +02:00
|
|
|
export class GitHubHelper {
|
|
|
|
private octokit: InstanceType<typeof Octokit>
|
|
|
|
|
|
|
|
constructor(token: string) {
|
|
|
|
const options: OctokitOptions = {}
|
|
|
|
if (token) {
|
|
|
|
options.auth = `${token}`
|
|
|
|
}
|
|
|
|
this.octokit = new Octokit(options)
|
|
|
|
}
|
|
|
|
|
|
|
|
private parseRepository(repository: string): Repository {
|
|
|
|
const [owner, repo] = repository.split('/')
|
|
|
|
return {
|
|
|
|
owner: owner,
|
|
|
|
repo: repo
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private async createOrUpdate(
|
|
|
|
inputs: Inputs,
|
|
|
|
baseRepository: string,
|
|
|
|
headBranch: string
|
2020-09-17 03:41:24 +02:00
|
|
|
): Promise<Pull> {
|
2020-07-16 10:57:13 +02:00
|
|
|
// Try to create the pull request
|
|
|
|
try {
|
|
|
|
const {data: pull} = await this.octokit.pulls.create({
|
|
|
|
...this.parseRepository(baseRepository),
|
|
|
|
title: inputs.title,
|
|
|
|
head: headBranch,
|
|
|
|
base: inputs.base,
|
|
|
|
body: inputs.body,
|
|
|
|
draft: inputs.draft
|
|
|
|
})
|
|
|
|
core.info(
|
|
|
|
`Created pull request #${pull.number} (${headBranch} => ${inputs.base})`
|
|
|
|
)
|
2020-09-17 03:41:24 +02:00
|
|
|
return {
|
|
|
|
number: pull.number,
|
|
|
|
html_url: pull.html_url
|
|
|
|
}
|
2020-07-16 10:57:13 +02:00
|
|
|
} catch (e) {
|
|
|
|
if (
|
2020-12-09 01:38:10 +01:00
|
|
|
e.message &&
|
|
|
|
e.message.includes(`A pull request already exists for ${headBranch}`)
|
2020-07-16 10:57:13 +02:00
|
|
|
) {
|
2020-12-09 01:38:10 +01:00
|
|
|
core.info(`A pull request already exists for ${headBranch}`)
|
|
|
|
} else {
|
2020-07-16 10:57:13 +02:00
|
|
|
throw e
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Update the pull request that exists for this branch and base
|
|
|
|
const {data: pulls} = await this.octokit.pulls.list({
|
|
|
|
...this.parseRepository(baseRepository),
|
|
|
|
state: 'open',
|
|
|
|
head: headBranch,
|
|
|
|
base: inputs.base
|
|
|
|
})
|
|
|
|
const {data: pull} = await this.octokit.pulls.update({
|
|
|
|
...this.parseRepository(baseRepository),
|
|
|
|
pull_number: pulls[0].number,
|
|
|
|
title: inputs.title,
|
|
|
|
body: inputs.body,
|
|
|
|
draft: inputs.draft
|
|
|
|
})
|
|
|
|
core.info(
|
|
|
|
`Updated pull request #${pull.number} (${headBranch} => ${inputs.base})`
|
|
|
|
)
|
2020-09-17 03:41:24 +02:00
|
|
|
return {
|
|
|
|
number: pull.number,
|
|
|
|
html_url: pull.html_url
|
|
|
|
}
|
2020-07-16 10:57:13 +02:00
|
|
|
}
|
|
|
|
|
2020-07-18 08:33:46 +02:00
|
|
|
async getRepositoryParent(headRepository: string): Promise<string> {
|
2020-07-16 10:57:13 +02:00
|
|
|
const {data: headRepo} = await this.octokit.repos.get({
|
|
|
|
...this.parseRepository(headRepository)
|
|
|
|
})
|
2020-07-18 08:33:46 +02:00
|
|
|
if (!headRepo.parent) {
|
2020-07-16 10:57:13 +02:00
|
|
|
throw new Error(
|
2020-07-18 08:33:46 +02:00
|
|
|
`Repository '${headRepository}' is not a fork. Unable to continue.`
|
2020-07-16 10:57:13 +02:00
|
|
|
)
|
|
|
|
}
|
2020-07-18 08:33:46 +02:00
|
|
|
return headRepo.parent.full_name
|
|
|
|
}
|
2020-07-16 10:57:13 +02:00
|
|
|
|
2020-07-18 08:33:46 +02:00
|
|
|
async createOrUpdatePullRequest(
|
|
|
|
inputs: Inputs,
|
|
|
|
baseRepository: string,
|
|
|
|
headRepository: string
|
|
|
|
): Promise<void> {
|
|
|
|
const [headOwner] = headRepository.split('/')
|
|
|
|
const headBranch = `${headOwner}:${inputs.branch}`
|
2020-07-16 10:57:13 +02:00
|
|
|
|
|
|
|
// Create or update the pull request
|
2020-09-17 03:41:24 +02:00
|
|
|
const pull = await this.createOrUpdate(inputs, baseRepository, headBranch)
|
2020-07-16 10:57:13 +02:00
|
|
|
|
2020-07-16 12:42:19 +02:00
|
|
|
// Set outputs
|
|
|
|
core.startGroup('Setting outputs')
|
2020-09-17 03:41:24 +02:00
|
|
|
core.setOutput('pull-request-number', pull.number)
|
|
|
|
core.setOutput('pull-request-url', pull.html_url)
|
|
|
|
// Deprecated
|
|
|
|
core.exportVariable('PULL_REQUEST_NUMBER', pull.number)
|
2020-07-16 12:42:19 +02:00
|
|
|
core.endGroup()
|
2020-07-16 10:57:13 +02:00
|
|
|
|
|
|
|
// Set milestone, labels and assignees
|
|
|
|
const updateIssueParams = {}
|
|
|
|
if (inputs.milestone) {
|
|
|
|
updateIssueParams['milestone'] = inputs.milestone
|
|
|
|
core.info(`Applying milestone '${inputs.milestone}'`)
|
|
|
|
}
|
|
|
|
if (inputs.labels.length > 0) {
|
|
|
|
updateIssueParams['labels'] = inputs.labels
|
|
|
|
core.info(`Applying labels '${inputs.labels}'`)
|
|
|
|
}
|
|
|
|
if (inputs.assignees.length > 0) {
|
|
|
|
updateIssueParams['assignees'] = inputs.assignees
|
|
|
|
core.info(`Applying assignees '${inputs.assignees}'`)
|
|
|
|
}
|
|
|
|
if (Object.keys(updateIssueParams).length > 0) {
|
|
|
|
await this.octokit.issues.update({
|
|
|
|
...this.parseRepository(baseRepository),
|
2020-09-17 03:41:24 +02:00
|
|
|
issue_number: pull.number,
|
2020-07-16 10:57:13 +02:00
|
|
|
...updateIssueParams
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
// Request reviewers and team reviewers
|
|
|
|
const requestReviewersParams = {}
|
|
|
|
if (inputs.reviewers.length > 0) {
|
|
|
|
requestReviewersParams['reviewers'] = inputs.reviewers
|
|
|
|
core.info(`Requesting reviewers '${inputs.reviewers}'`)
|
|
|
|
}
|
|
|
|
if (inputs.teamReviewers.length > 0) {
|
|
|
|
requestReviewersParams['team_reviewers'] = inputs.teamReviewers
|
|
|
|
core.info(`Requesting team reviewers '${inputs.teamReviewers}'`)
|
|
|
|
}
|
|
|
|
if (Object.keys(requestReviewersParams).length > 0) {
|
|
|
|
try {
|
|
|
|
await this.octokit.pulls.requestReviewers({
|
|
|
|
...this.parseRepository(baseRepository),
|
2020-09-17 03:41:24 +02:00
|
|
|
pull_number: pull.number,
|
2020-07-16 10:57:13 +02:00
|
|
|
...requestReviewersParams
|
|
|
|
})
|
|
|
|
} catch (e) {
|
|
|
|
if (e.message && e.message.includes(ERROR_PR_REVIEW_FROM_AUTHOR)) {
|
|
|
|
core.warning(ERROR_PR_REVIEW_FROM_AUTHOR)
|
|
|
|
} else {
|
|
|
|
throw e
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|