import * as core from '@actions/core' import * as github from '@actions/github' // TODO: Ensure this (and the Octokit client) works for non-github.com URLs, as well. // https://github.com/orgs|users//projects/ const urlParse = /^(?:https:\/\/)?github\.com\/(?orgs|users)\/(?[^/]+)\/projects\/(?\d+)/ interface ProjectNodeIDResponse { organization?: { projectV2: { id: string } } user?: { projectV2: { id: string } } } interface ProjectAddItemResponse { addProjectV2ItemById: { item: { id: string } } } export async function addToProject(): Promise { const projectUrl = core.getInput('project-url', {required: true}) const ghToken = core.getInput('github-token', {required: true}) const labeled = core .getInput('labeled') .split(',') .map(l => l.trim().toLowerCase()) .filter(l => l.length > 0) ?? [] const labelOperator = core.getInput('label-operator').trim().toLocaleLowerCase() const octokit = github.getOctokit(ghToken) const urlMatch = projectUrl.match(urlParse) const issue = github.context.payload.issue ?? github.context.payload.pull_request const issueLabels: string[] = (issue?.labels ?? []).map((l: {name: string}) => l.name.toLowerCase()) // Ensure the issue matches our `labeled` filter based on the label-operator. if (labelOperator === 'and') { if (!labeled.every(l => issueLabels.includes(l))) { core.info(`Skipping issue ${issue?.number} because it doesn't match all the labels: ${labeled.join(', ')}`) return } } else if (labelOperator === 'not') { if (labeled.length > 0 && issueLabels.some(l => labeled.includes(l))) { core.info(`Skipping issue ${issue?.number} because it contains one of the labels: ${labeled.join(', ')}`) return } } else { if (labeled.length > 0 && !issueLabels.some(l => labeled.includes(l))) { core.info(`Skipping issue ${issue?.number} because it does not have one of the labels: ${labeled.join(', ')}`) return } } core.debug(`Project URL: ${projectUrl}`) if (!urlMatch) { throw new Error( `Invalid project URL: ${projectUrl}. Project URL should match the format https://github.com///projects/` ) } const ownerName = urlMatch.groups?.ownerName const projectNumber = parseInt(urlMatch.groups?.projectNumber ?? '', 10) const ownerType = urlMatch.groups?.ownerType const ownerTypeQuery = mustGetOwnerTypeQuery(ownerType) core.debug(`Org name: ${ownerName}`) core.debug(`Project number: ${projectNumber}`) core.debug(`Owner type: ${ownerType}`) // First, use the GraphQL API to request the project's node ID. const idResp = await octokit.graphql( `query getProject($ownerName: String!, $projectNumber: Int!) { ${ownerTypeQuery}(login: $ownerName) { projectV2(number: $projectNumber) { id } } }`, { ownerName, projectNumber } ) const projectId = idResp[ownerTypeQuery]?.projectV2.id const contentId = issue?.node_id core.debug(`Project node ID: ${projectId}`) core.debug(`Content ID: ${contentId}`) // Next, use the GraphQL API to add the issue to the project. const addResp = await octokit.graphql( `mutation addIssueToProject($input: AddProjectV2ItemByIdInput!) { addProjectV2ItemById(input: $input) { item { id } } }`, { input: { projectId, contentId } } ) core.setOutput('itemId', addResp.addProjectV2ItemById.item.id) } export function mustGetOwnerTypeQuery(ownerType?: string): 'organization' | 'user' { const ownerTypeQuery = ownerType === 'orgs' ? 'organization' : ownerType === 'users' ? 'user' : null if (!ownerTypeQuery) { throw new Error(`Unsupported ownerType: ${ownerType}. Must be one of 'orgs' or 'users'`) } return ownerTypeQuery }