renovate/lib/modules/platform/azure/index.ts

1020 lines
27 KiB
TypeScript

import { setTimeout } from 'timers/promises';
import is from '@sindresorhus/is';
import type {
GitPullRequest,
GitPullRequestCommentThread,
GitStatus,
GitVersionDescriptor,
} from 'azure-devops-node-api/interfaces/GitInterfaces.js';
import {
GitPullRequestMergeStrategy,
GitStatusState,
PullRequestStatus,
} from 'azure-devops-node-api/interfaces/GitInterfaces.js';
import {
REPOSITORY_ARCHIVED,
REPOSITORY_EMPTY,
REPOSITORY_NOT_FOUND,
} from '../../../constants/error-messages';
import { logger } from '../../../logger';
import type { BranchStatus } from '../../../types';
import { ExternalHostError } from '../../../types/errors/external-host-error';
import { parseJson } from '../../../util/common';
import * as git from '../../../util/git';
import * as hostRules from '../../../util/host-rules';
import { regEx } from '../../../util/regex';
import { sanitize } from '../../../util/sanitize';
import { ensureTrailingSlash } from '../../../util/url';
import type {
BranchStatusConfig,
CreatePRConfig,
EnsureCommentConfig,
EnsureCommentRemovalConfig,
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 azureApi from './azure-got-wrapper';
import * as azureHelper from './azure-helper';
import type { AzurePr } from './types';
import { AzurePrVote } from './types';
import {
getBranchNameWithoutRefsheadsPrefix,
getGitStatusContextCombinedName,
getGitStatusContextFromCombinedName,
getRenovatePRFormat,
getRepoByName,
getStorageExtraCloneOpts,
mapMergeStrategy,
max4000Chars,
} from './util';
interface Config {
repoForceRebase: boolean;
mergeMethods: Record<string, GitPullRequestMergeStrategy>;
owner: string;
repoId: string;
project: string;
prList: AzurePr[];
fileList: null;
repository: string;
defaultBranch: string;
}
interface User {
id: string;
name: string;
isRequired: boolean;
}
let config: Config = {} as any;
const defaults: {
endpoint?: string;
hostType: string;
} = {
hostType: 'azure',
};
export const id = 'azure';
export function initPlatform({
endpoint,
token,
username,
password,
}: PlatformParams): Promise<PlatformResult> {
if (!endpoint) {
throw new Error('Init: You must configure an Azure DevOps endpoint');
}
if (!token && !(username && password)) {
throw new Error(
'Init: You must configure an Azure DevOps token, or a username and password',
);
}
// TODO: Add a connection check that endpoint/token combination are valid (#9593)
const res = {
endpoint: ensureTrailingSlash(endpoint),
};
defaults.endpoint = res.endpoint;
azureApi.setEndpoint(res.endpoint);
const platformConfig: PlatformResult = {
endpoint: defaults.endpoint,
};
return Promise.resolve(platformConfig);
}
export async function getRepos(): Promise<string[]> {
logger.debug('Autodiscovering Azure DevOps repositories');
const azureApiGit = await azureApi.gitApi();
const repos = await azureApiGit.getRepositories();
return repos
.filter((repo) => repo.isDisabled !== true)
.map((repo) => `${repo.project?.name}/${repo.name}`);
}
export async function getRawFile(
fileName: string,
repoName?: string,
branchOrTag?: string,
): Promise<string | null> {
try {
const azureApiGit = await azureApi.gitApi();
let repoId: string | undefined;
if (repoName) {
const repos = await azureApiGit.getRepositories();
const repo = getRepoByName(repoName, repos);
repoId = repo?.id;
} else {
repoId = config.repoId;
}
if (!repoId) {
logger.debug('No repoId so cannot getRawFile');
return null;
}
const versionDescriptor: GitVersionDescriptor = {
version: branchOrTag,
} satisfies GitVersionDescriptor;
const item = await azureApiGit.getItem(
repoId, // repositoryId
fileName, // path
undefined, // project
undefined, // scopePath
undefined, // recursionLevel
undefined, // includeContentMetadata
undefined, // latestProcessedChange
undefined, // download
branchOrTag ? versionDescriptor : undefined, // versionDescriptor
true, // includeContent
);
return item?.content ?? null;
} catch (err) /* istanbul ignore next */ {
if (
err.message?.includes('<title>Azure DevOps Services Unavailable</title>')
) {
logger.debug(
'Azure DevOps is currently unavailable when attempting to fetch file - throwing ExternalHostError',
);
throw new ExternalHostError(err, id);
}
if (err.code === 'ECONNRESET' || err.code === 'ETIMEDOUT') {
throw new ExternalHostError(err, id);
}
if (err.statusCode && err.statusCode >= 500 && err.statusCode < 600) {
throw new ExternalHostError(err, id);
}
throw err;
}
}
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 initRepo({
repository,
cloneSubmodules,
cloneSubmodulesFilter,
}: RepoParams): Promise<RepoResult> {
logger.debug(`initRepo("${repository}")`);
config = { repository } as Config;
const azureApiGit = await azureApi.gitApi();
const repos = await azureApiGit.getRepositories();
const repo = getRepoByName(repository, repos);
if (!repo) {
logger.error({ repos, repo }, 'Could not find repo in repo list');
throw new Error(REPOSITORY_NOT_FOUND);
}
logger.debug({ repositoryDetails: repo }, 'Repository details');
if (repo.isDisabled) {
logger.debug('Repository is disabled- throwing error to abort renovation');
throw new Error(REPOSITORY_ARCHIVED);
}
// istanbul ignore if
if (!repo.defaultBranch) {
logger.debug('Repo is empty');
throw new Error(REPOSITORY_EMPTY);
}
// TODO #22198
config.repoId = repo.id!;
config.project = repo.project!.name!;
config.owner = '?owner?';
logger.debug(`${repository} owner = ${config.owner}`);
const defaultBranch = repo.defaultBranch.replace('refs/heads/', '');
config.defaultBranch = defaultBranch;
logger.debug(`${repository} default branch = ${defaultBranch}`);
config.mergeMethods = {};
config.repoForceRebase = false;
const [projectName, repoName] = repository.split('/');
const opts = hostRules.find({
hostType: defaults.hostType,
url: defaults.endpoint,
});
// TODO: types (#22198)
const manualUrl = `${defaults.endpoint!}${encodeURIComponent(
projectName,
)}/_git/${encodeURIComponent(repoName)}`;
const url = repo.remoteUrl ?? manualUrl;
await git.initRepo({
...config,
url,
extraCloneOpts: getStorageExtraCloneOpts(opts),
cloneSubmodules,
cloneSubmodulesFilter,
});
const repoConfig: RepoResult = {
defaultBranch,
isFork: false,
repoFingerprint: repoFingerprint(repo.id!, defaults.endpoint),
};
return repoConfig;
}
export async function getPrList(): Promise<AzurePr[]> {
logger.debug('getPrList()');
if (!config.prList) {
const azureApiGit = await azureApi.gitApi();
let prs: GitPullRequest[] = [];
let fetchedPrs: GitPullRequest[];
let skip = 0;
do {
fetchedPrs = await azureApiGit.getPullRequests(
config.repoId,
{ status: 4 },
config.project,
0,
skip,
100,
);
prs = prs.concat(fetchedPrs);
skip += 100;
} while (fetchedPrs.length > 0);
config.prList = prs.map(getRenovatePRFormat);
logger.debug(`Retrieved Pull Requests count: ${config.prList.length}`);
}
return config.prList;
}
export async function getPr(pullRequestId: number): Promise<Pr | null> {
logger.debug(`getPr(${pullRequestId})`);
if (!pullRequestId) {
return null;
}
const azurePr = (await getPrList()).find(
(item) => item.number === pullRequestId,
);
if (!azurePr) {
return null;
}
const azureApiGit = await azureApi.gitApi();
const labels = await azureApiGit.getPullRequestLabels(
config.repoId,
pullRequestId,
);
azurePr.labels = labels
.filter((label) => label.active)
.map((label) => label.name)
.filter(is.string);
return azurePr;
}
export async function findPr({
branchName,
prTitle,
state = 'all',
targetBranch,
}: FindPRConfig): Promise<Pr | null> {
let prsFiltered: Pr[] = [];
try {
const prs = await getPrList();
prsFiltered = prs.filter(
(item) => item.sourceRefName === getNewBranchName(branchName),
);
if (prTitle) {
prsFiltered = prsFiltered.filter(
(item) => item.title.toUpperCase() === prTitle.toUpperCase(),
);
}
switch (state) {
case 'all':
// no more filter needed, we can go further...
break;
case '!open':
prsFiltered = prsFiltered.filter((item) => item.state !== 'open');
break;
default:
prsFiltered = prsFiltered.filter((item) => item.state === state);
break;
}
} catch (err) {
logger.error({ err }, 'findPr error');
}
if (prsFiltered.length === 0) {
return null;
}
if (targetBranch && prsFiltered.length > 1) {
const pr = prsFiltered.find((item) => item.targetBranch === targetBranch);
if (pr) {
return pr;
}
}
return prsFiltered[0];
}
export async function getBranchPr(
branchName: string,
targetBranch?: string,
): Promise<Pr | null> {
logger.debug(`getBranchPr(${branchName}, ${targetBranch})`);
const existingPr = await findPr({
branchName,
state: 'open',
targetBranch,
});
return existingPr ? getPr(existingPr.number) : null;
}
async function getStatusCheck(branchName: string): Promise<GitStatus[]> {
const azureApiGit = await azureApi.gitApi();
const branch = await azureApiGit.getBranch(
config.repoId,
// TODO: fix undefined (#22198)
getBranchNameWithoutRefsheadsPrefix(branchName)!,
);
// only grab the latest statuses, it will group any by context
return azureApiGit.getStatuses(
// TODO #22198
branch.commit!.commitId!,
config.repoId,
undefined,
undefined,
undefined,
true,
);
}
const azureToRenovateStatusMapping: Record<GitStatusState, BranchStatus> = {
[GitStatusState.Succeeded]: 'green',
[GitStatusState.NotApplicable]: 'green',
[GitStatusState.NotSet]: 'yellow',
[GitStatusState.Pending]: 'yellow',
[GitStatusState.PartiallySucceeded]: 'yellow',
[GitStatusState.Error]: 'red',
[GitStatusState.Failed]: 'red',
};
export async function getBranchStatusCheck(
branchName: string,
context: string,
): Promise<BranchStatus | null> {
const res = await getStatusCheck(branchName);
for (const check of res) {
if (getGitStatusContextCombinedName(check.context) === context) {
// TODO #22198
return azureToRenovateStatusMapping[check.state!] ?? 'yellow';
}
}
return null;
}
export async function getBranchStatus(
branchName: string,
internalChecksAsSuccess: boolean,
): Promise<BranchStatus> {
logger.debug(`getBranchStatus(${branchName})`);
const statuses = await getStatusCheck(branchName);
logger.debug({ branch: branchName, statuses }, 'branch status check result');
if (!statuses.length) {
logger.debug('empty branch status check result = returning "pending"');
return 'yellow';
}
const noOfFailures = statuses.filter(
(status) =>
status.state === GitStatusState.Error ||
status.state === GitStatusState.Failed,
).length;
if (noOfFailures) {
return 'red';
}
const noOfPending = statuses.filter(
(status) =>
status.state === GitStatusState.NotSet ||
status.state === GitStatusState.Pending,
).length;
if (noOfPending) {
return 'yellow';
}
if (
!internalChecksAsSuccess &&
statuses.every(
(status) =>
status.state === GitStatusState.Succeeded &&
status.context?.genre === 'renovate',
)
) {
logger.debug(
'Successful checks are all internal renovate/ checks, so returning "pending" branch status',
);
return 'yellow';
}
return 'green';
}
async function getMergeStrategy(
targetRefName: string,
): Promise<GitPullRequestMergeStrategy> {
return (
config.mergeMethods[targetRefName] ??
(config.mergeMethods[targetRefName] = await azureHelper.getMergeMethod(
config.repoId,
config.project,
targetRefName,
config.defaultBranch,
))
);
}
export async function createPr({
sourceBranch,
targetBranch,
prTitle: title,
prBody: body,
labels,
draftPR = false,
platformPrOptions,
}: CreatePRConfig): Promise<Pr> {
const sourceRefName = getNewBranchName(sourceBranch);
const targetRefName = getNewBranchName(targetBranch);
const description = max4000Chars(sanitize(body));
const azureApiGit = await azureApi.gitApi();
const workItemRefs = [
{
id: platformPrOptions?.azureWorkItemId?.toString(),
},
];
let pr: GitPullRequest = await azureApiGit.createPullRequest(
{
sourceRefName,
targetRefName,
title,
description,
workItemRefs,
isDraft: draftPR,
},
config.repoId,
);
if (platformPrOptions?.usePlatformAutomerge) {
const mergeStrategy =
platformPrOptions.automergeStrategy === 'auto'
? await getMergeStrategy(pr.targetRefName!)
: mapMergeStrategy(platformPrOptions.automergeStrategy);
pr = await azureApiGit.updatePullRequest(
{
autoCompleteSetBy: {
// TODO #22198
id: pr.createdBy!.id,
},
completionOptions: {
mergeStrategy,
deleteSourceBranch: true,
mergeCommitMessage: title,
},
},
config.repoId,
// TODO #22198
pr.pullRequestId!,
);
}
if (platformPrOptions?.autoApprove) {
await azureApiGit.createPullRequestReviewer(
{
reviewerUrl: pr.createdBy!.url,
vote: AzurePrVote.Approved,
isFlagged: false,
isRequired: false,
},
config.repoId,
// TODO #22198
pr.pullRequestId!,
pr.createdBy!.id!,
);
}
await Promise.all(
labels!.map((label) =>
azureApiGit.createPullRequestLabel(
{
name: label,
},
config.repoId,
// TODO #22198
pr.pullRequestId!,
),
),
);
const result = getRenovatePRFormat(pr);
if (config.prList) {
config.prList.push(result);
}
return result;
}
export async function updatePr({
number: prNo,
prTitle: title,
prBody: body,
state,
platformPrOptions,
targetBranch,
}: UpdatePrConfig): Promise<void> {
logger.debug(`updatePr(${prNo}, ${title}, body)`);
const azureApiGit = await azureApi.gitApi();
const objToUpdate: GitPullRequest = {
title,
};
if (targetBranch) {
objToUpdate.targetRefName = getNewBranchName(targetBranch);
}
if (body) {
objToUpdate.description = max4000Chars(sanitize(body));
}
if (state === 'open') {
await azureApiGit.updatePullRequest(
{
status: PullRequestStatus.Active,
},
config.repoId,
prNo,
);
} else if (state === 'closed') {
objToUpdate.status = PullRequestStatus.Abandoned;
}
if (platformPrOptions?.autoApprove) {
const pr = await azureApiGit.getPullRequestById(prNo, config.project);
await azureApiGit.createPullRequestReviewer(
{
reviewerUrl: pr.createdBy!.url,
vote: AzurePrVote.Approved,
isFlagged: false,
isRequired: false,
},
config.repoId,
// TODO #22198
pr.pullRequestId!,
pr.createdBy!.id!,
);
}
const updatedPr = await azureApiGit.updatePullRequest(
objToUpdate,
config.repoId,
prNo,
);
if (config.prList) {
const prToCache = getRenovatePRFormat(updatedPr);
// We need to update the cached entry for this PR
const existingIndex = config.prList.findIndex(
(item) => item.number === prNo,
);
// istanbul ignore if: should not happen
if (existingIndex === -1) {
logger.warn({ prNo }, 'PR not found in cache');
// Add to cache
config.prList.push(prToCache);
} else {
// overwrite existing PR in cache
config.prList[existingIndex] = prToCache;
}
}
}
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(massageMarkdown(content))}`;
const azureApiGit = await azureApi.gitApi();
const threads = await azureApiGit.getThreads(config.repoId, number);
let threadIdFound: number | undefined;
let commentIdFound: number | undefined;
let commentNeedsUpdating = false;
threads.forEach((thread) => {
const firstCommentContent = thread.comments?.[0].content;
if (
(topic && firstCommentContent?.startsWith(header)) === true ||
(!topic && firstCommentContent === body)
) {
threadIdFound = thread.id;
commentIdFound = thread.comments?.[0].id;
commentNeedsUpdating = firstCommentContent !== body;
}
});
if (!threadIdFound) {
await azureApiGit.createThread(
{
comments: [{ content: body, commentType: 1, parentCommentId: 0 }],
status: 1,
},
config.repoId,
number,
);
logger.info(
{ repository: config.repository, issueNo: number, topic },
'Comment added',
);
} else if (commentNeedsUpdating) {
await azureApiGit.updateComment(
{
content: body,
},
config.repoId,
number,
threadIdFound,
// TODO #22198
commentIdFound!,
);
logger.debug(
{ repository: config.repository, issueNo: number, topic },
'Comment updated',
);
} else {
logger.debug(
{ repository: config.repository, issueNo: number, topic },
'Comment is already update-to-date',
);
}
return true;
}
export async function ensureCommentRemoval(
removeConfig: EnsureCommentRemovalConfig,
): Promise<void> {
const { number: issueNo } = removeConfig;
const key =
removeConfig.type === 'by-topic'
? removeConfig.topic
: removeConfig.content;
logger.debug(`Ensuring comment "${key}" in #${issueNo} is removed`);
const azureApiGit = await azureApi.gitApi();
const threads = await azureApiGit.getThreads(config.repoId, issueNo);
let threadIdFound: number | null | undefined = null;
if (removeConfig.type === 'by-topic') {
const thread = threads.find(
(thread: GitPullRequestCommentThread): boolean =>
!!thread.comments?.[0].content?.startsWith(
`### ${removeConfig.topic}\n\n`,
),
);
threadIdFound = thread?.id;
} else {
const thread = threads.find(
(thread: GitPullRequestCommentThread): boolean =>
thread.comments?.[0].content?.trim() === removeConfig.content,
);
threadIdFound = thread?.id;
}
if (threadIdFound) {
await azureApiGit.updateThread(
{
status: 4, // close
},
config.repoId,
issueNo,
threadIdFound,
);
}
}
const renovateToAzureStatusMapping: Record<BranchStatus, GitStatusState> = {
['green']: GitStatusState.Succeeded,
['yellow']: GitStatusState.Pending,
['red']: GitStatusState.Failed,
};
export async function setBranchStatus({
branchName,
context,
description,
state,
url: targetUrl,
}: BranchStatusConfig): Promise<void> {
logger.debug(
`setBranchStatus(${branchName}, ${context}, ${description}, ${state}, ${targetUrl!})`,
);
const azureApiGit = await azureApi.gitApi();
const branch = await azureApiGit.getBranch(
config.repoId,
getBranchNameWithoutRefsheadsPrefix(branchName)!,
);
const statusToCreate: GitStatus = {
description,
context: getGitStatusContextFromCombinedName(context),
state: renovateToAzureStatusMapping[state],
targetUrl,
};
await azureApiGit.createCommitStatus(
statusToCreate,
// TODO #22198
branch.commit!.commitId!,
config.repoId,
);
logger.trace(`Created commit status of ${state} on branch ${branchName}`);
}
export async function mergePr({
branchName,
id: pullRequestId,
strategy,
}: MergePRConfig): Promise<boolean> {
logger.debug(`mergePr(${pullRequestId}, ${branchName!})`);
const azureApiGit = await azureApi.gitApi();
let pr = await azureApiGit.getPullRequestById(pullRequestId, config.project);
const mergeStrategy =
strategy === 'auto'
? await getMergeStrategy(pr.targetRefName!)
: mapMergeStrategy(strategy);
const objToUpdate: GitPullRequest = {
status: PullRequestStatus.Completed,
lastMergeSourceCommit: pr.lastMergeSourceCommit,
completionOptions: {
mergeStrategy,
deleteSourceBranch: true,
mergeCommitMessage: pr.title,
},
};
logger.trace(
`Updating PR ${pullRequestId} to status ${PullRequestStatus.Completed} (${
PullRequestStatus[PullRequestStatus.Completed]
}) with lastMergeSourceCommit ${
// TODO: types (#22198)
pr.lastMergeSourceCommit?.commitId
} using mergeStrategy ${mergeStrategy} (${
GitPullRequestMergeStrategy[mergeStrategy]
})`,
);
try {
const response = await azureApiGit.updatePullRequest(
objToUpdate,
config.repoId,
pullRequestId,
);
let retries = 0;
let isClosed = response.status === PullRequestStatus.Completed;
while (!isClosed && retries < 5) {
retries += 1;
const sleepMs = retries * 1000;
logger.trace(
{ pullRequestId, status: pr.status, retries },
`Updated PR to closed status but change has not taken effect yet. Retrying...`,
);
await setTimeout(sleepMs);
pr = await azureApiGit.getPullRequestById(pullRequestId, config.project);
isClosed = pr.status === PullRequestStatus.Completed;
}
if (!isClosed) {
logger.warn(
{ pullRequestId, status: pr.status },
`Expected PR to have status ${
PullRequestStatus[PullRequestStatus.Completed]
// TODO #22198
}. However, it is ${PullRequestStatus[pr.status!]}.`,
);
}
return true;
} catch (err) {
logger.debug({ err }, 'Failed to set the PR as completed.');
return false;
}
}
export function massageMarkdown(input: string): string {
// Remove any HTML we use
return smartTruncate(input, maxBodyLength())
.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(`\n---\n\n.*?<!-- rebase-check -->.*?\n`), '')
.replace(regEx(/<!--renovate-(?:debug|config-hash):.*?-->/g), '');
}
export function maxBodyLength(): number {
return 4000;
}
/* istanbul ignore next */
export function findIssue(): Promise<Issue | null> {
// TODO: Needs implementation (#9592)
logger.debug(`findIssue() is not implemented`);
return Promise.resolve(null);
}
/* istanbul ignore next */
export function ensureIssue(): Promise<EnsureIssueResult | null> {
// TODO: Needs implementation (#9592)
logger.debug(`ensureIssue() is not implemented`);
return Promise.resolve(null);
}
/* istanbul ignore next */
export function ensureIssueClosing(): Promise<void> {
return Promise.resolve();
}
/* istanbul ignore next */
export function getIssueList(): Promise<Issue[]> {
logger.debug(`getIssueList()`);
// TODO: Needs implementation (#9592)
return Promise.resolve([]);
}
async function getUserIds(users: string[]): Promise<User[]> {
const azureApiGit = await azureApi.gitApi();
const azureApiCore = await azureApi.coreApi();
const repos = await azureApiGit.getRepositories();
const repo = repos.filter((c) => c.id === config.repoId)[0];
const requiredReviewerPrefix = 'required:';
const validReviewers = new Set<string>();
// TODO #22198
const teams = await azureHelper.getAllProjectTeams(repo.project!.id!);
const members = await Promise.all(
teams.map(
async (t) =>
await azureApiCore.getTeamMembersWithExtendedProperties(
// TODO #22198
repo.project!.id!,
t.id!,
),
),
);
const ids: { id: string; name: string; isRequired: boolean }[] = [];
members.forEach((listMembers) => {
listMembers.forEach((m) => {
users.forEach((r) => {
let reviewer = r;
let isRequired = false;
if (reviewer.startsWith(requiredReviewerPrefix)) {
reviewer = reviewer.replace(requiredReviewerPrefix, '');
isRequired = true;
}
if (
reviewer.toLowerCase() === m.identity?.displayName?.toLowerCase() ||
reviewer.toLowerCase() === m.identity?.uniqueName?.toLowerCase()
) {
if (ids.filter((c) => c.id === m.identity?.id).length === 0) {
// TODO #22198
ids.push({
id: m.identity.id!,
name: reviewer,
isRequired,
});
validReviewers.add(reviewer);
}
}
});
});
});
teams.forEach((t) => {
users.forEach((r) => {
let reviewer = r;
let isRequired = false;
if (reviewer.startsWith(requiredReviewerPrefix)) {
reviewer = reviewer.replace(requiredReviewerPrefix, '');
isRequired = true;
}
if (reviewer.toLowerCase() === t.name?.toLowerCase()) {
if (ids.filter((c) => c.id === t.id).length === 0) {
// TODO #22198
ids.push({ id: t.id!, name: reviewer, isRequired });
validReviewers.add(reviewer);
}
}
});
});
for (const u of users) {
const reviewer = u.replace(requiredReviewerPrefix, '');
if (!validReviewers.has(reviewer)) {
logger.once.info(
`${reviewer} is neither an Azure DevOps Team nor a user associated with a Team`,
);
}
}
return ids;
}
/**
*
* @param {number} issueNo
* @param {string[]} assignees
*/
export async function addAssignees(
issueNo: number,
assignees: string[],
): Promise<void> {
logger.trace(`addAssignees(${issueNo}, [${assignees.join(', ')}])`);
const ids = await getUserIds(assignees);
await ensureComment({
number: issueNo,
topic: 'Add Assignees',
content: ids.map((a) => `@<${a.id}>`).join(', '),
});
}
/**
*
* @param {number} prNo
* @param {string[]} reviewers
*/
export async function addReviewers(
prNo: number,
reviewers: string[],
): Promise<void> {
logger.trace(`addReviewers(${prNo}, [${reviewers.join(', ')}])`);
const azureApiGit = await azureApi.gitApi();
const ids = await getUserIds(reviewers);
await Promise.all(
ids.map(async (obj) => {
await azureApiGit.createPullRequestReviewer(
{
isRequired: obj.isRequired,
},
config.repoId,
prNo,
obj.id,
);
logger.debug(`Reviewer added: ${obj.name}`);
}),
);
}
export async function deleteLabel(
prNumber: number,
label: string,
): Promise<void> {
logger.debug(`Deleting label ${label} from #${prNumber}`);
const azureApiGit = await azureApi.gitApi();
await azureApiGit.deletePullRequestLabels(config.repoId, prNumber, label);
}