mirror of
https://github.com/renovatebot/renovate.git
synced 2025-01-11 05:39:10 +00:00
730 lines
19 KiB
TypeScript
730 lines
19 KiB
TypeScript
import { Buffer } from 'node:buffer';
|
|
import type {
|
|
GetCommentsForPullRequestOutput,
|
|
ListRepositoriesOutput,
|
|
} from '@aws-sdk/client-codecommit';
|
|
import { PullRequestStatusEnum } from '@aws-sdk/client-codecommit';
|
|
import {
|
|
PLATFORM_BAD_CREDENTIALS,
|
|
REPOSITORY_EMPTY,
|
|
REPOSITORY_NOT_FOUND,
|
|
} from '../../../constants/error-messages';
|
|
import { logger } from '../../../logger';
|
|
import type { BranchStatus, PrState } from '../../../types';
|
|
import { coerceArray } from '../../../util/array';
|
|
import { parseJson } from '../../../util/common';
|
|
import * as git from '../../../util/git';
|
|
import { regEx } from '../../../util/regex';
|
|
import { sanitize } from '../../../util/sanitize';
|
|
import type {
|
|
BranchStatusConfig,
|
|
CreatePRConfig,
|
|
EnsureCommentConfig,
|
|
EnsureCommentRemovalConfig,
|
|
EnsureIssueConfig,
|
|
EnsureIssueResult,
|
|
FindPRConfig,
|
|
Issue,
|
|
MergePRConfig,
|
|
PlatformParams,
|
|
PlatformResult,
|
|
Pr,
|
|
RepoParams,
|
|
RepoResult,
|
|
UpdatePrConfig,
|
|
} from '../types';
|
|
import { getNewBranchName, repoFingerprint } from '../util';
|
|
import { smartTruncate } from '../utils/pr-body';
|
|
import * as client from './codecommit-client';
|
|
|
|
export interface CodeCommitPr extends Pr {
|
|
body: string;
|
|
destinationCommit: string;
|
|
sourceCommit: string;
|
|
}
|
|
|
|
interface Config {
|
|
repository?: string;
|
|
defaultBranch?: string;
|
|
region?: string;
|
|
prList?: CodeCommitPr[];
|
|
}
|
|
|
|
export const id = 'codecommit';
|
|
|
|
export const config: Config = {};
|
|
|
|
export async function initPlatform({
|
|
endpoint,
|
|
username,
|
|
password,
|
|
token: awsToken,
|
|
}: PlatformParams): Promise<PlatformResult> {
|
|
const accessKeyId = username;
|
|
const secretAccessKey = password;
|
|
let region: string | undefined;
|
|
|
|
if (accessKeyId) {
|
|
process.env.AWS_ACCESS_KEY_ID = accessKeyId;
|
|
}
|
|
if (secretAccessKey) {
|
|
process.env.AWS_SECRET_ACCESS_KEY = secretAccessKey;
|
|
}
|
|
if (awsToken) {
|
|
process.env.AWS_SESSION_TOKEN = awsToken;
|
|
}
|
|
|
|
if (endpoint) {
|
|
const regionReg = regEx(/.*codecommit\.(?<region>.+)\.amazonaws\.com/);
|
|
const codeCommitMatch = regionReg.exec(endpoint);
|
|
region = codeCommitMatch?.groups?.region;
|
|
if (region) {
|
|
process.env.AWS_REGION = region;
|
|
} else {
|
|
logger.warn("Can't parse region, make sure your endpoint is correct");
|
|
}
|
|
}
|
|
|
|
// If any of the below fails, it will throw an exception stopping the program.
|
|
client.buildCodeCommitClient();
|
|
// To check if we have permission to codecommit, throws exception if failed.
|
|
await client.listRepositories();
|
|
|
|
const platformConfig: PlatformResult = {
|
|
endpoint:
|
|
endpoint ??
|
|
`https://git-codecommit.${
|
|
process.env.AWS_REGION ?? 'us-east-1'
|
|
}.amazonaws.com/`,
|
|
};
|
|
return Promise.resolve(platformConfig);
|
|
}
|
|
|
|
export async function initRepo({
|
|
repository,
|
|
endpoint,
|
|
}: RepoParams): Promise<RepoResult> {
|
|
logger.debug(`initRepo("${repository}")`);
|
|
|
|
config.repository = repository;
|
|
|
|
let repo;
|
|
try {
|
|
repo = await client.getRepositoryInfo(repository);
|
|
} catch (err) {
|
|
logger.error({ err }, 'Could not find repository');
|
|
throw new Error(REPOSITORY_NOT_FOUND);
|
|
}
|
|
|
|
if (!repo?.repositoryMetadata) {
|
|
logger.error({ repository }, 'Could not find repository');
|
|
throw new Error(REPOSITORY_NOT_FOUND);
|
|
}
|
|
logger.debug({ repositoryDetails: repo }, 'Repository details');
|
|
const metadata = repo.repositoryMetadata;
|
|
|
|
const url = client.getCodeCommitUrl(metadata, repository);
|
|
try {
|
|
await git.initRepo({
|
|
url,
|
|
});
|
|
} catch (err) {
|
|
logger.debug({ err }, 'Failed to git init');
|
|
throw new Error(PLATFORM_BAD_CREDENTIALS);
|
|
}
|
|
|
|
if (!metadata.defaultBranch || !metadata.repositoryId) {
|
|
logger.debug('Repo is empty');
|
|
throw new Error(REPOSITORY_EMPTY);
|
|
}
|
|
|
|
const defaultBranch = metadata.defaultBranch;
|
|
config.defaultBranch = defaultBranch;
|
|
logger.debug(`${repository} default branch = ${defaultBranch}`);
|
|
|
|
return {
|
|
repoFingerprint: repoFingerprint(metadata.repositoryId, endpoint),
|
|
defaultBranch,
|
|
isFork: false,
|
|
};
|
|
}
|
|
|
|
export async function getPrList(): Promise<CodeCommitPr[]> {
|
|
logger.debug('getPrList()');
|
|
|
|
if (config.prList) {
|
|
return config.prList;
|
|
}
|
|
|
|
const listPrsResponse = await client.listPullRequests(config.repository!);
|
|
const fetchedPrs: CodeCommitPr[] = [];
|
|
|
|
if (listPrsResponse && !listPrsResponse.pullRequestIds) {
|
|
return fetchedPrs;
|
|
}
|
|
|
|
const prIds = coerceArray(listPrsResponse.pullRequestIds);
|
|
|
|
for (const prId of prIds) {
|
|
const prRes = await client.getPr(prId);
|
|
|
|
if (!prRes?.pullRequest) {
|
|
continue;
|
|
}
|
|
const prInfo = prRes.pullRequest;
|
|
const pr: CodeCommitPr = {
|
|
targetBranch: prInfo.pullRequestTargets![0].destinationReference!,
|
|
sourceBranch: prInfo.pullRequestTargets![0].sourceReference!,
|
|
destinationCommit: prInfo.pullRequestTargets![0].destinationCommit!,
|
|
sourceCommit: prInfo.pullRequestTargets![0].sourceCommit!,
|
|
state:
|
|
prInfo.pullRequestStatus === PullRequestStatusEnum.OPEN
|
|
? 'open'
|
|
: 'closed',
|
|
number: Number.parseInt(prId),
|
|
title: prInfo.title!,
|
|
body: prInfo.description!,
|
|
};
|
|
fetchedPrs.push(pr);
|
|
}
|
|
|
|
config.prList = fetchedPrs;
|
|
|
|
logger.debug(`Retrieved Pull Requests, count: ${fetchedPrs.length}`);
|
|
return fetchedPrs;
|
|
}
|
|
|
|
export async function findPr({
|
|
branchName,
|
|
prTitle,
|
|
state = 'all',
|
|
}: FindPRConfig): Promise<CodeCommitPr | null> {
|
|
let prsFiltered: CodeCommitPr[] = [];
|
|
try {
|
|
const prs = await getPrList();
|
|
const refsHeadBranchName = getNewBranchName(branchName);
|
|
prsFiltered = prs.filter(
|
|
(item) => item.sourceBranch === refsHeadBranchName,
|
|
);
|
|
|
|
if (prTitle) {
|
|
prsFiltered = prsFiltered.filter(
|
|
(item) => item.title.toUpperCase() === prTitle.toUpperCase(),
|
|
);
|
|
}
|
|
|
|
switch (state) {
|
|
case 'all':
|
|
break;
|
|
case '!open':
|
|
prsFiltered = prsFiltered.filter((item) => item.state !== 'open');
|
|
break;
|
|
default:
|
|
prsFiltered = prsFiltered.filter((item) => item.state === 'open');
|
|
break;
|
|
}
|
|
} catch (err) {
|
|
logger.error({ err }, 'findPr error');
|
|
}
|
|
if (prsFiltered.length === 0) {
|
|
return null;
|
|
}
|
|
return prsFiltered[0];
|
|
}
|
|
|
|
export async function getBranchPr(
|
|
branchName: string,
|
|
): Promise<CodeCommitPr | null> {
|
|
logger.debug(`getBranchPr(${branchName})`);
|
|
const existingPr = await findPr({
|
|
branchName,
|
|
state: 'open',
|
|
});
|
|
return existingPr ? getPr(existingPr.number) : null;
|
|
}
|
|
|
|
export async function getPr(
|
|
pullRequestId: number,
|
|
): Promise<CodeCommitPr | null> {
|
|
logger.debug(`getPr(${pullRequestId})`);
|
|
const prRes = await client.getPr(`${pullRequestId}`);
|
|
|
|
if (!prRes?.pullRequest) {
|
|
return null;
|
|
}
|
|
|
|
const prInfo = prRes.pullRequest;
|
|
let prState: PrState;
|
|
if (prInfo.pullRequestTargets![0].mergeMetadata?.isMerged) {
|
|
prState = 'merged';
|
|
} else {
|
|
prState =
|
|
prInfo.pullRequestStatus === PullRequestStatusEnum.OPEN
|
|
? 'open'
|
|
: 'closed';
|
|
}
|
|
|
|
return {
|
|
sourceBranch: prInfo.pullRequestTargets![0].sourceReference!,
|
|
sourceCommit: prInfo.pullRequestTargets![0].sourceCommit!,
|
|
state: prState,
|
|
number: pullRequestId,
|
|
title: prInfo.title!,
|
|
targetBranch: prInfo.pullRequestTargets![0].destinationReference!,
|
|
destinationCommit: prInfo.pullRequestTargets![0].destinationCommit!,
|
|
body: prInfo.description!,
|
|
};
|
|
}
|
|
|
|
export async function getRepos(): Promise<string[]> {
|
|
logger.debug('Autodiscovering AWS CodeCommit repositories');
|
|
|
|
let reposRes: ListRepositoriesOutput;
|
|
try {
|
|
reposRes = await client.listRepositories();
|
|
//todo do we need pagination? maximum number of repos is 1000 without pagination, also the same for free account
|
|
} catch (error) {
|
|
logger.error({ error }, 'Could not retrieve repositories');
|
|
return [];
|
|
}
|
|
|
|
const res: string[] = [];
|
|
|
|
const repoNames = coerceArray(reposRes?.repositories);
|
|
|
|
for (const repo of repoNames) {
|
|
if (repo.repositoryName) {
|
|
res.push(repo.repositoryName);
|
|
}
|
|
}
|
|
|
|
return res;
|
|
}
|
|
|
|
export function massageMarkdown(input: string): string {
|
|
// Remove any HTML we use
|
|
return input
|
|
.replace(
|
|
'you tick the rebase/retry checkbox',
|
|
'PR is renamed to start with "rebase!"',
|
|
)
|
|
.replace(
|
|
'checking the rebase/retry box above',
|
|
'renaming the PR to start with "rebase!"',
|
|
)
|
|
.replace(regEx(/<\/?summary>/g), '**')
|
|
.replace(regEx(/<\/?details>/g), '')
|
|
.replace(regEx(`\n---\n\n.*?<!-- rebase-check -->.*?\n`), '')
|
|
.replace(regEx(/\]\(\.\.\/pull\//g), '](../../pull-requests/')
|
|
.replace(
|
|
regEx(/(?<hiddenComment><!--renovate-(?:debug|config-hash):.*?-->)/g),
|
|
'[//]: # ($<hiddenComment>)',
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Unsed, no Dashboard
|
|
*/
|
|
export function maxBodyLength(): number {
|
|
return Infinity;
|
|
}
|
|
|
|
export async function getJsonFile(
|
|
fileName: string,
|
|
repoName?: string,
|
|
branchOrTag?: string,
|
|
): Promise<any> {
|
|
const raw = await getRawFile(fileName, repoName, branchOrTag);
|
|
return parseJson(raw, fileName);
|
|
}
|
|
|
|
export async function getRawFile(
|
|
fileName: string,
|
|
repoName?: string,
|
|
branchOrTag?: string,
|
|
): Promise<string | null> {
|
|
const fileRes = await client.getFile(
|
|
repoName ?? config.repository,
|
|
fileName,
|
|
branchOrTag,
|
|
);
|
|
if (!fileRes?.fileContent) {
|
|
return null;
|
|
}
|
|
const buf = Buffer.from(fileRes.fileContent);
|
|
return buf.toString();
|
|
}
|
|
|
|
const AMAZON_MAX_BODY_LENGTH = 10239;
|
|
|
|
export async function createPr({
|
|
sourceBranch,
|
|
targetBranch,
|
|
prTitle: title,
|
|
prBody: body,
|
|
}: CreatePRConfig): Promise<CodeCommitPr> {
|
|
const description = smartTruncate(sanitize(body), AMAZON_MAX_BODY_LENGTH);
|
|
|
|
const prCreateRes = await client.createPr(
|
|
title,
|
|
sanitize(description),
|
|
sourceBranch,
|
|
targetBranch,
|
|
config.repository,
|
|
);
|
|
|
|
if (
|
|
!prCreateRes.pullRequest?.title ||
|
|
!prCreateRes.pullRequest?.pullRequestId ||
|
|
!prCreateRes.pullRequest?.description ||
|
|
!prCreateRes.pullRequest?.pullRequestTargets?.length
|
|
) {
|
|
throw new Error('Could not create pr, missing PR info');
|
|
}
|
|
|
|
return {
|
|
number: Number.parseInt(prCreateRes.pullRequest.pullRequestId),
|
|
state: 'open',
|
|
title: prCreateRes.pullRequest.title,
|
|
sourceBranch,
|
|
targetBranch,
|
|
sourceCommit: prCreateRes.pullRequest.pullRequestTargets[0].sourceCommit!,
|
|
destinationCommit:
|
|
prCreateRes.pullRequest.pullRequestTargets[0].destinationCommit!,
|
|
sourceRepo: config.repository,
|
|
body: prCreateRes.pullRequest.description,
|
|
};
|
|
}
|
|
|
|
export async function updatePr({
|
|
number: prNo,
|
|
prTitle: title,
|
|
prBody: body,
|
|
state,
|
|
}: UpdatePrConfig): Promise<void> {
|
|
logger.debug(`updatePr(${prNo}, ${title}, body)`);
|
|
|
|
let cachedPr: CodeCommitPr | undefined = undefined;
|
|
const cachedPrs = config.prList ?? [];
|
|
for (const p of cachedPrs) {
|
|
if (p.number === prNo) {
|
|
cachedPr = p;
|
|
}
|
|
}
|
|
|
|
if (body && cachedPr?.body !== body) {
|
|
await client.updatePrDescription(
|
|
`${prNo}`,
|
|
smartTruncate(sanitize(body), AMAZON_MAX_BODY_LENGTH),
|
|
);
|
|
}
|
|
|
|
if (title && cachedPr?.title !== title) {
|
|
await client.updatePrTitle(`${prNo}`, title);
|
|
}
|
|
|
|
const prStatusInput =
|
|
state === 'closed'
|
|
? PullRequestStatusEnum.CLOSED
|
|
: PullRequestStatusEnum.OPEN;
|
|
if (cachedPr?.state !== prStatusInput) {
|
|
try {
|
|
await client.updatePrStatus(`${prNo}`, prStatusInput);
|
|
} catch {
|
|
// safety check
|
|
// do nothing, it's ok to fail sometimes when trying to update from open to open or from closed to closed.
|
|
}
|
|
}
|
|
}
|
|
|
|
// Auto-Merge not supported currently.
|
|
/* istanbul ignore next */
|
|
export async function mergePr({
|
|
branchName,
|
|
id: prNo,
|
|
strategy,
|
|
}: MergePRConfig): Promise<boolean> {
|
|
logger.debug(`mergePr(${prNo}, ${branchName!})`);
|
|
await client.getPr(`${prNo}`);
|
|
return Promise.resolve(false);
|
|
//
|
|
// // istanbul ignore if
|
|
// if (!prOut) {
|
|
// return false;
|
|
// }
|
|
// const pReq = prOut.pullRequest;
|
|
// const targets = pReq?.pullRequestTargets;
|
|
//
|
|
// // istanbul ignore if
|
|
// if (!targets) {
|
|
// return false;
|
|
// }
|
|
//
|
|
// if (strategy === 'rebase') {
|
|
// logger.warn('CodeCommit does not support a "rebase" strategy.');
|
|
// return false;
|
|
// }
|
|
//
|
|
// try {
|
|
// if (strategy === 'auto' || strategy === 'squash') {
|
|
// await client.squashMerge(
|
|
// targets[0].repositoryName!,
|
|
// targets[0].sourceReference!,
|
|
// targets[0].destinationReference!,
|
|
// pReq?.title
|
|
// );
|
|
// } else if (strategy === 'fast-forward') {
|
|
// await client.fastForwardMerge(
|
|
// targets[0].repositoryName!,
|
|
// targets[0].sourceReference!,
|
|
// targets[0].destinationReference!
|
|
// );
|
|
// } else {
|
|
// logger.debug(`unsupported strategy`);
|
|
// return false;
|
|
// }
|
|
// } catch (err) {
|
|
// logger.debug({ err }, `PR merge error`);
|
|
// logger.info({ pr: prNo }, 'PR automerge failed');
|
|
// return false;
|
|
// }
|
|
//
|
|
// logger.trace(`Updating PR ${prNo} to status ${PullRequestStatusEnum.CLOSED}`);
|
|
//
|
|
// try {
|
|
// const response = await client.updatePrStatus(
|
|
// `${prNo}`,
|
|
// PullRequestStatusEnum.CLOSED
|
|
// );
|
|
// const isClosed =
|
|
// response.pullRequest?.pullRequestStatus === PullRequestStatusEnum.CLOSED;
|
|
//
|
|
// if (!isClosed) {
|
|
// logger.warn(
|
|
// {
|
|
// pullRequestId: prNo,
|
|
// status: response.pullRequest?.pullRequestStatus,
|
|
// },
|
|
// `Expected PR to have status`
|
|
// );
|
|
// }
|
|
// return true;
|
|
// } catch (err) {
|
|
// logger.debug({ err }, 'Failed to set the PR as Closed.');
|
|
// return false;
|
|
// }
|
|
}
|
|
|
|
export async function addReviewers(
|
|
prNo: number,
|
|
reviewers: string[],
|
|
): Promise<void> {
|
|
const numberOfApprovers = reviewers.length;
|
|
const approvalRuleContents = `{"Version":"2018-11-08","Statements": [{"Type": "Approvers","NumberOfApprovalsNeeded":${numberOfApprovers},"ApprovalPoolMembers": ${JSON.stringify(
|
|
reviewers,
|
|
)}}]}`;
|
|
const res = await client.createPrApprovalRule(
|
|
`${prNo}`,
|
|
approvalRuleContents,
|
|
);
|
|
if (res) {
|
|
const approvalRule = res.approvalRule;
|
|
logger.debug({ approvalRule }, `Approval Rule Added to PR #${prNo}:`);
|
|
}
|
|
}
|
|
|
|
/* istanbul ignore next */
|
|
export function addAssignees(iid: number, assignees: string[]): Promise<void> {
|
|
// CodeCommit does not support adding reviewers
|
|
return Promise.resolve();
|
|
}
|
|
|
|
/* istanbul ignore next */
|
|
export function findIssue(title: string): Promise<Issue | null> {
|
|
// CodeCommit does not have issues
|
|
return Promise.resolve(null);
|
|
}
|
|
|
|
/* istanbul ignore next */
|
|
export function ensureIssue({
|
|
title,
|
|
}: EnsureIssueConfig): Promise<EnsureIssueResult | null> {
|
|
// CodeCommit does not have issues
|
|
return Promise.resolve(null);
|
|
}
|
|
|
|
/* istanbul ignore next */
|
|
export function getIssueList(): Promise<Issue[]> {
|
|
// CodeCommit does not have issues
|
|
return Promise.resolve([]);
|
|
}
|
|
|
|
/* istanbul ignore next */
|
|
export function ensureIssueClosing(title: string): Promise<void> {
|
|
// CodeCommit does not have issues
|
|
return Promise.resolve();
|
|
}
|
|
|
|
/* istanbul ignore next */
|
|
export function deleteLabel(prNumber: number, label: string): Promise<void> {
|
|
return Promise.resolve();
|
|
}
|
|
|
|
// Returns the combined status for a branch.
|
|
/* istanbul ignore next */
|
|
export function getBranchStatus(branchName: string): Promise<BranchStatus> {
|
|
logger.debug(`getBranchStatus(${branchName})`);
|
|
logger.debug(
|
|
'returning branch status yellow, because getBranchStatus isnt supported on aws yet',
|
|
);
|
|
return Promise.resolve('yellow');
|
|
}
|
|
|
|
/* istanbul ignore next */
|
|
export function getBranchStatusCheck(
|
|
branchName: string,
|
|
context: string,
|
|
): Promise<BranchStatus | null> {
|
|
logger.debug(`getBranchStatusCheck(${branchName}, context=${context})`);
|
|
logger.debug(
|
|
'returning null, because getBranchStatusCheck is not supported on aws yet',
|
|
);
|
|
return Promise.resolve(null);
|
|
}
|
|
|
|
/* istanbul ignore next */
|
|
export function setBranchStatus({
|
|
branchName,
|
|
context,
|
|
description,
|
|
state,
|
|
url: targetUrl,
|
|
}: BranchStatusConfig): Promise<void> {
|
|
return Promise.resolve();
|
|
}
|
|
|
|
export async function ensureComment({
|
|
number,
|
|
topic,
|
|
content,
|
|
}: EnsureCommentConfig): Promise<boolean> {
|
|
logger.debug(`ensureComment(${number}, ${topic!}, content)`);
|
|
const header = topic ? `### ${topic}\n\n` : '';
|
|
const body = `${header}${sanitize(content)}`;
|
|
let prCommentsResponse: GetCommentsForPullRequestOutput;
|
|
try {
|
|
prCommentsResponse = await client.getPrComments(`${number}`);
|
|
} catch (err) {
|
|
logger.debug({ err }, 'Unable to retrieve pr comments');
|
|
return false;
|
|
}
|
|
|
|
let commentId: string | undefined = undefined;
|
|
let commentNeedsUpdating = false;
|
|
|
|
if (!prCommentsResponse?.commentsForPullRequestData) {
|
|
return false;
|
|
}
|
|
|
|
for (const commentObj of prCommentsResponse.commentsForPullRequestData) {
|
|
if (!commentObj?.comments) {
|
|
continue;
|
|
}
|
|
const firstCommentContent = commentObj.comments[0].content;
|
|
if (
|
|
(topic && firstCommentContent?.startsWith(header)) === true ||
|
|
(!topic && firstCommentContent === body)
|
|
) {
|
|
commentId = commentObj.comments[0].commentId;
|
|
commentNeedsUpdating = firstCommentContent !== body;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (!commentId) {
|
|
const prs = await getPrList();
|
|
const thisPr = prs.filter((item) => item.number === number);
|
|
|
|
if (!thisPr[0].sourceCommit || !thisPr[0].destinationCommit) {
|
|
return false;
|
|
}
|
|
|
|
await client.createPrComment(
|
|
`${number}`,
|
|
config.repository,
|
|
body,
|
|
thisPr[0].destinationCommit,
|
|
thisPr[0].sourceCommit,
|
|
);
|
|
logger.info(
|
|
{ repository: config.repository, prNo: number, topic },
|
|
'Comment added',
|
|
);
|
|
} else if (commentNeedsUpdating && commentId) {
|
|
await client.updateComment(commentId, body);
|
|
|
|
logger.debug(
|
|
{ repository: config.repository, prNo: number, topic },
|
|
'Comment updated',
|
|
);
|
|
} else {
|
|
logger.debug(
|
|
{ repository: config.repository, prNo: number, topic },
|
|
'Comment is already update-to-date',
|
|
);
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
export async function ensureCommentRemoval(
|
|
removeConfig: EnsureCommentRemovalConfig,
|
|
): Promise<void> {
|
|
const { number: prNo } = removeConfig;
|
|
const key =
|
|
removeConfig.type === 'by-topic'
|
|
? removeConfig.topic
|
|
: removeConfig.content;
|
|
logger.debug(`Ensuring comment "${key}" in #${prNo} is removed`);
|
|
|
|
let prCommentsResponse: GetCommentsForPullRequestOutput;
|
|
try {
|
|
prCommentsResponse = await client.getPrComments(`${prNo}`);
|
|
} catch (err) {
|
|
logger.debug({ err }, 'Unable to retrieve pr comments');
|
|
return;
|
|
}
|
|
|
|
if (!prCommentsResponse?.commentsForPullRequestData) {
|
|
logger.debug('commentsForPullRequestData not found');
|
|
return;
|
|
}
|
|
|
|
let commentIdToRemove: string | undefined;
|
|
for (const commentObj of prCommentsResponse.commentsForPullRequestData) {
|
|
if (!commentObj?.comments) {
|
|
logger.debug(
|
|
'comments object not found under commentsForPullRequestData',
|
|
);
|
|
continue;
|
|
}
|
|
|
|
for (const comment of commentObj.comments) {
|
|
if (
|
|
(removeConfig.type === 'by-topic' &&
|
|
comment.content?.startsWith(`### ${removeConfig.topic}\n\n`)) ===
|
|
true ||
|
|
(removeConfig.type === 'by-content' &&
|
|
removeConfig.content === comment.content?.trim())
|
|
) {
|
|
commentIdToRemove = comment.commentId;
|
|
break;
|
|
}
|
|
}
|
|
if (commentIdToRemove) {
|
|
await client.deleteComment(commentIdToRemove);
|
|
logger.debug(`comment "${key}" in PR #${prNo} was removed`);
|
|
break;
|
|
}
|
|
}
|
|
}
|