renovate 43.76.0 → 43.76.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -398,7 +398,15 @@ async function updatePRAndAddReviewers(prNo, reviewers) {
398
398
  if (err.statusCode === 404) throw new Error(REPOSITORY_NOT_FOUND);
399
399
  else if (err.statusCode === 409) if (isInvalidReviewersResponse(err)) {
400
400
  const invalidReviewers = getInvalidReviewers(err);
401
- await updatePRAndAddReviewers(prNo, reviewers.filter((name) => !invalidReviewers.includes(name)));
401
+ const filteredReviewers = reviewers.filter((name) => !invalidReviewers.includes(name));
402
+ if (filteredReviewers.length < reviewers.length) await updatePRAndAddReviewers(prNo, filteredReviewers);
403
+ else {
404
+ logger.warn({
405
+ invalidReviewers,
406
+ reviewers
407
+ }, "Could not filter invalid reviewers from list, aborting to prevent infinite recursion");
408
+ throw err;
409
+ }
402
410
  } else {
403
411
  logger.debug("409 response to adding reviewers - has repository changed?");
404
412
  throw new Error(REPOSITORY_CHANGED);
@@ -1 +1 @@
1
- {"version":3,"file":"index.js","names":["hostRules.find","utils.getRepoGitUrl","git.initRepo","utils.prInfo","git.getBranchCommit","git.branchExists","utils.isInvalidReviewersResponse","utils.getInvalidReviewers"],"sources":["../../../../lib/modules/platform/bitbucket-server/index.ts"],"sourcesContent":["import { isNonEmptyStringAndNotWhitespace } from '@sindresorhus/is';\nimport ignore from 'ignore';\nimport semver from 'semver';\nimport { setTimeout } from 'timers/promises';\nimport type { PartialDeep } from 'type-fest';\nimport { GlobalConfig } from '../../../config/global.ts';\nimport {\n REPOSITORY_CHANGED,\n REPOSITORY_EMPTY,\n REPOSITORY_NOT_FOUND,\n} from '../../../constants/error-messages.ts';\nimport { logger } from '../../../logger/index.ts';\nimport type { BranchStatus } from '../../../types/index.ts';\nimport type { FileData } from '../../../types/platform/bitbucket-server/index.ts';\nimport { parseJson } from '../../../util/common.ts';\nimport { getEnv } from '../../../util/env.ts';\nimport * as git from '../../../util/git/index.ts';\nimport { deleteBranch } from '../../../util/git/index.ts';\nimport * as hostRules from '../../../util/host-rules.ts';\nimport {\n BitbucketServerHttp,\n type BitbucketServerHttpOptions,\n setBaseUrl,\n} from '../../../util/http/bitbucket-server.ts';\nimport { memCacheProvider } from '../../../util/http/cache/memory-http-cache-provider.ts';\nimport type { HttpOptions, HttpResponse } from '../../../util/http/types.ts';\nimport { newlineRegex, regEx } from '../../../util/regex.ts';\nimport { sampleSize } from '../../../util/sample.ts';\nimport { sanitize } from '../../../util/sanitize.ts';\nimport { isEmailAdress } from '../../../util/schema-utils/index.ts';\nimport { ensureTrailingSlash, getQueryString } from '../../../util/url.ts';\nimport type {\n BranchStatusConfig,\n CreatePRConfig,\n EnsureCommentConfig,\n EnsureCommentRemovalConfig,\n EnsureIssueConfig,\n EnsureIssueResult,\n FileOwnerRule,\n FindPRConfig,\n Issue,\n MergePRConfig,\n PlatformParams,\n PlatformResult,\n Pr,\n RepoParams,\n RepoResult,\n UpdatePrConfig,\n} from '../types.ts';\nimport { getNewBranchName, repoFingerprint } from '../util.ts';\nimport { smartTruncate } from '../utils/pr-body.ts';\nimport { BbsPrCache } from './pr-cache.ts';\nimport type {\n Comment,\n PullRequestActivity,\n PullRequestCommentActivity,\n PullRequestMerge,\n} from './schema.ts';\nimport { ReviewerGroups, User, Users } from './schema.ts';\nimport type {\n BbsConfig,\n BbsPr,\n BbsRestBranch,\n BbsRestPr,\n BbsRestRepo,\n BbsRestUserRef,\n} from './types.ts';\nimport * as utils from './utils.ts';\nimport {\n getExtraCloneOpts,\n parseModifier,\n splitEscapedSpaces,\n} from './utils.ts';\n\n/*\n * Version: 5.3 (EOL Date: 15 Aug 2019)\n * See following docs for api information:\n * https://docs.atlassian.com/bitbucket-server/rest/5.3.0/bitbucket-rest.html\n * https://docs.atlassian.com/bitbucket-server/rest/5.3.0/bitbucket-build-rest.html\n *\n * See following page for uptodate supported versions\n * https://confluence.atlassian.com/support/atlassian-support-end-of-life-policy-201851003.html#AtlassianSupportEndofLifePolicy-BitbucketServer\n */\n\nexport const id = 'bitbucket-server';\n\nlet config: BbsConfig = {} as any;\n\nconst bitbucketServerHttp = new BitbucketServerHttp();\n\nconst defaults: {\n endpoint?: string;\n hostType: string;\n version: string;\n} = {\n hostType: 'bitbucket-server',\n version: '0.0.0',\n};\n\n/* v8 ignore next */\nfunction updatePrVersion(pr: number, version: number): number {\n const res = Math.max(config.prVersions.get(pr) ?? 0, version);\n config.prVersions.set(pr, res);\n return res;\n}\n\nexport async function initPlatform({\n endpoint,\n token,\n username,\n password,\n gitAuthor,\n}: PlatformParams): Promise<PlatformResult> {\n if (!endpoint) {\n throw new Error('Init: You must configure a Bitbucket Server endpoint');\n }\n if (!(username && password) && !token) {\n throw new Error(\n 'Init: You must either configure a Bitbucket Server username/password or a HTTP access token',\n );\n } else if (password && token) {\n throw new Error(\n 'Init: You must configure either a Bitbucket Server password or a HTTP access token, not both',\n );\n }\n // TODO: Add a connection check that endpoint/username/password combination are valid (#9595)\n defaults.endpoint = ensureTrailingSlash(endpoint);\n setBaseUrl(defaults.endpoint);\n const platformConfig: PlatformResult = {\n endpoint: defaults.endpoint,\n };\n try {\n const env = getEnv();\n let bitbucketServerVersion = env.RENOVATE_X_PLATFORM_VERSION;\n const { body, headers } = await bitbucketServerHttp.getJsonUnchecked<{\n version: string;\n }>(`./rest/api/1.0/application-properties`, { ...(token && { token }) });\n\n bitbucketServerVersion ??= body.version;\n if (isNonEmptyStringAndNotWhitespace(headers['x-ausername']) && !username) {\n logger.debug(\n { 'x-ausername': headers['x-ausername'] },\n 'Platform: No username configured using headers[\"x-ausername\"]',\n );\n config.username = headers['x-ausername'];\n }\n logger.debug('Bitbucket Server version is: ' + bitbucketServerVersion);\n\n // v8 ignore else -- TODO: add test #40625\n if (semver.valid(bitbucketServerVersion)) {\n defaults.version = bitbucketServerVersion;\n }\n } catch (err) {\n logger.debug(\n { err },\n 'Error authenticating with Bitbucket. Check that your token includes \"api\" permissions',\n );\n }\n\n if (!gitAuthor && username) {\n logger.debug(`Attempting to confirm gitAuthor from username`);\n const options: HttpOptions = {\n memCache: false,\n };\n\n if (token) {\n options.token = token;\n } else {\n options.username = username;\n options.password = password;\n }\n\n try {\n const { displayName, emailAddress } = (\n await bitbucketServerHttp.getJson(\n `./rest/api/1.0/users/${username}`,\n options,\n User,\n )\n ).body;\n\n if (!emailAddress?.length) {\n throw new Error(`No email address configured for username ${username}`);\n }\n\n platformConfig.gitAuthor = `${displayName} <${emailAddress}>`;\n\n logger.debug(`Detected gitAuthor: ${platformConfig.gitAuthor}`);\n } catch (err) {\n logger.debug(\n { err },\n 'Failed to get user info, fallback gitAuthor will be used',\n );\n }\n }\n\n return platformConfig;\n}\n\n// Get all repositories that the user has access to\nexport async function getRepos(): Promise<string[]> {\n logger.debug('Autodiscovering Bitbucket Server repositories');\n try {\n const repos = (\n await bitbucketServerHttp.getJsonUnchecked<BbsRestRepo[]>(\n `./rest/api/1.0/repos?permission=REPO_WRITE&state=AVAILABLE`,\n { paginate: true },\n )\n ).body;\n\n const result = repos.map((repo) => `${repo.project.key}/${repo.slug}`);\n logger.debug({ result }, 'result of getRepos()');\n return result;\n } catch (err) /* v8 ignore next */ {\n logger.error({ err }, `bitbucket getRepos error`);\n throw err;\n }\n}\n\nexport async function getRawFile(\n fileName: string,\n repoName?: string,\n branchOrTag?: string,\n): Promise<string | null> {\n const repo = repoName ?? config.repository;\n const [project, slug] = repo.split('/');\n const fileUrl =\n `./rest/api/1.0/projects/${project}/repos/${slug}/browse/${fileName}?limit=20000` +\n (branchOrTag ? '&at=' + branchOrTag : '');\n const res = await bitbucketServerHttp.getJsonUnchecked<FileData>(fileUrl);\n const { isLastPage, lines, size } = res.body;\n if (isLastPage) {\n return lines.map(({ text }) => text).join('\\n');\n }\n logger.warn({ size }, 'The file is too big');\n throw new Error(`The file is too big (${size}B)`);\n}\n\nexport async function getJsonFile(\n fileName: string,\n repoName?: string,\n branchOrTag?: string,\n): Promise<any> {\n // TODO #22198\n const raw = await getRawFile(fileName, repoName, branchOrTag);\n return parseJson(raw, fileName);\n}\n\n// Initialize Bitbucket Server by getting base branch\nexport async function initRepo({\n repository,\n cloneSubmodules,\n cloneSubmodulesFilter,\n gitUrl,\n}: RepoParams): Promise<RepoResult> {\n logger.debug(`initRepo(\"${JSON.stringify({ repository }, null, 2)}\")`);\n const opts = hostRules.find({\n hostType: defaults.hostType,\n url: defaults.endpoint,\n });\n\n const [projectKey, repositorySlug] = repository.split('/');\n\n config = {\n projectKey,\n repositorySlug,\n repository,\n prVersions: new Map<number, number>(),\n username: opts.username,\n ignorePrAuthor: GlobalConfig.get('ignorePrAuthor', false),\n } as any;\n\n try {\n const info = (\n await bitbucketServerHttp.getJsonUnchecked<BbsRestRepo>(\n `./rest/api/1.0/projects/${config.projectKey}/repos/${config.repositorySlug}`,\n )\n ).body;\n config.owner = info.project.key;\n logger.debug(`${repository} owner = ${config.owner}`);\n const branchRes = await bitbucketServerHttp.getJsonUnchecked<BbsRestBranch>(\n `./rest/api/1.0/projects/${config.projectKey}/repos/${config.repositorySlug}/branches/default`,\n );\n\n // 204 means empty, 404 means repo not found or missing default branch. repo must exist here.\n if ([204, 404].includes(branchRes.statusCode)) {\n throw new Error(REPOSITORY_EMPTY);\n }\n\n const url = utils.getRepoGitUrl(\n config.repositorySlug,\n // TODO #22198\n defaults.endpoint!,\n gitUrl,\n info,\n opts,\n );\n\n await git.initRepo({\n ...config,\n url,\n extraCloneOpts: getExtraCloneOpts(opts),\n cloneSubmodules,\n cloneSubmodulesFilter,\n fullClone: semver.lte(defaults.version, '8.0.0'),\n });\n\n config.mergeMethod = 'merge';\n const repoConfig: RepoResult = {\n defaultBranch: branchRes.body.displayId,\n isFork: !!info.origin,\n repoFingerprint: repoFingerprint(info.id, defaults.endpoint),\n };\n\n return repoConfig;\n } catch (err) /* v8 ignore next */ {\n if (err.statusCode === 404) {\n throw new Error(REPOSITORY_NOT_FOUND);\n }\n if (err.message === REPOSITORY_EMPTY) {\n throw err;\n }\n\n logger.debug({ err }, 'Unknown Bitbucket initRepo error');\n throw err;\n }\n}\n\nexport async function getBranchForceRebase(\n _branchName: string,\n): Promise<boolean> {\n // https://docs.atlassian.com/bitbucket-server/rest/7.0.1/bitbucket-rest.html#idp342\n const res = await bitbucketServerHttp.getJsonUnchecked<{\n mergeConfig: { defaultStrategy: { id: string } };\n }>(\n `./rest/api/1.0/projects/${config.projectKey}/repos/${config.repositorySlug}/settings/pull-requests`,\n );\n\n // If the default merge strategy contains `ff-only` the PR can only be merged\n // if it is up to date with the base branch.\n // The current options for id are:\n // no-ff, ff, ff-only, rebase-no-ff, rebase-ff-only, squash, squash-ff-only\n return Boolean(\n res.body?.mergeConfig?.defaultStrategy?.id.includes('ff-only'),\n );\n}\n\n// Gets details for a PR\nexport async function getPr(\n prNo: number,\n refreshCache?: boolean,\n): Promise<BbsPr | null> {\n logger.debug(`getPr(${prNo})`);\n if (!prNo) {\n return null;\n }\n\n // Disables memCache (which is enabled by default) to be replaced by\n // memCacheProvider.\n const opts: HttpOptions = { memCache: false };\n // TODO: should refresh the cache rather than just ignore it\n if (!refreshCache) {\n opts.cacheProvider = memCacheProvider;\n }\n\n const res = await bitbucketServerHttp.getJsonUnchecked<BbsRestPr>(\n `./rest/api/1.0/projects/${config.projectKey}/repos/${config.repositorySlug}/pull-requests/${prNo}`,\n opts,\n );\n\n const pr: BbsPr = {\n ...utils.prInfo(res.body),\n reviewers: res.body.reviewers.map((r) => r.user.name),\n };\n // TODO #22198\n pr.version = updatePrVersion(pr.number, pr.version!);\n\n return pr;\n}\n\n// TODO: coverage (#40625)\n/* v8 ignore next */\nfunction matchesState(state: string, desiredState: string): boolean {\n if (desiredState === 'all') {\n return true;\n }\n if (desiredState.startsWith('!')) {\n return state !== desiredState.substring(1);\n }\n return state === desiredState;\n}\n\n// TODO: coverage (#40625)\n/* v8 ignore next */\nfunction isRelevantPr(\n branchName: string,\n prTitle: string | null | undefined,\n state: string,\n) {\n return (p: Pr): boolean =>\n p.sourceBranch === branchName &&\n (!prTitle || p.title.toUpperCase() === prTitle.toUpperCase()) &&\n matchesState(p.state, state);\n}\n\n// TODO: coverage (#40625)\nexport async function getPrList(): Promise<Pr[]> {\n logger.debug(`getPrList()`);\n return await BbsPrCache.getPrs(\n bitbucketServerHttp,\n config.projectKey,\n config.repositorySlug,\n config.ignorePrAuthor,\n config.username,\n );\n}\n\n// TODO: coverage (#40625)\n/* v8 ignore next */\nexport async function findPr({\n branchName,\n prTitle,\n state = 'all',\n includeOtherAuthors,\n}: FindPRConfig): Promise<Pr | null> {\n logger.debug(`findPr(${branchName}, \"${prTitle!}\", \"${state}\")`);\n\n if (includeOtherAuthors) {\n // PR might have been created by anyone, so don't use the cached Renovate PR list\n const searchParams: Record<string, string> = {\n state: 'OPEN',\n };\n searchParams.direction = 'outgoing';\n searchParams.at = `refs/heads/${branchName}`;\n\n const query = getQueryString(searchParams);\n const prs = (\n await bitbucketServerHttp.getJsonUnchecked<BbsRestPr[]>(\n `./rest/api/1.0/projects/${config.projectKey}/repos/${config.repositorySlug}/pull-requests?${query}`,\n {\n paginate: true,\n limit: 1, // only fetch the latest pr\n },\n )\n ).body;\n\n if (!prs.length) {\n logger.debug(`No PR found for branch ${branchName}`);\n return null;\n }\n\n return utils.prInfo(prs[0]);\n }\n\n const prList = await getPrList();\n const pr = prList.find(isRelevantPr(branchName, prTitle, state));\n if (pr) {\n logger.debug(`Found PR #${pr.number}`);\n } else {\n logger.debug(`Renovate did not find a PR for branch #${branchName}`);\n }\n return pr ?? null;\n}\n\n// Returns the Pull Request for a branch. Null if not exists.\nexport async function getBranchPr(branchName: string): Promise<BbsPr | null> {\n logger.debug(`getBranchPr(${branchName})`);\n const existingPr = await findPr({\n branchName,\n state: 'open',\n });\n return existingPr ? getPr(existingPr.number) : null;\n}\n\n/* v8 ignore next */\nexport async function refreshPr(number: number): Promise<void> {\n // wait for pr change propagation\n await setTimeout(1000);\n // refresh cache\n await getPr(number, true);\n}\n\nasync function getStatus(\n branchName: string,\n memCache = true,\n): Promise<utils.BitbucketCommitStatus> {\n const branchCommit = git.getBranchCommit(branchName);\n\n /* v8 ignore next: temporary code */\n const opts: HttpOptions = memCache\n ? { cacheProvider: memCacheProvider }\n : { memCache: false };\n\n return (\n await bitbucketServerHttp.getJsonUnchecked<utils.BitbucketCommitStatus>(\n // TODO: types (#22198)\n `./rest/build-status/1.0/commits/stats/${branchCommit!}`,\n opts,\n )\n ).body;\n}\n\n// Returns the combined status for a branch.\n// umbrella for status checks\n// https://docs.atlassian.com/bitbucket-server/rest/6.0.0/bitbucket-build-rest.html#idp2\nexport async function getBranchStatus(\n branchName: string,\n): Promise<BranchStatus> {\n logger.debug(`getBranchStatus(${branchName})`);\n\n if (!git.branchExists(branchName)) {\n logger.debug('Branch does not exist - cannot fetch status');\n throw new Error(REPOSITORY_CHANGED);\n }\n\n try {\n const commitStatus = await getStatus(branchName);\n\n logger.debug({ commitStatus }, 'branch status check result');\n\n if (commitStatus.failed > 0) {\n return 'red';\n }\n if (commitStatus.inProgress > 0) {\n return 'yellow';\n }\n return commitStatus.successful > 0 ? 'green' : 'yellow';\n } catch (err) {\n logger.warn({ err }, `Failed to get branch status`);\n return 'red';\n }\n}\n\nasync function getStatusCheck(\n branchName: string,\n memCache = true,\n): Promise<utils.BitbucketStatus[]> {\n const branchCommit = git.getBranchCommit(branchName);\n\n const opts: BitbucketServerHttpOptions = { paginate: true };\n /* v8 ignore next: temporary code */\n if (memCache) {\n opts.cacheProvider = memCacheProvider;\n } else {\n opts.memCache = false;\n }\n\n return (\n await bitbucketServerHttp.getJsonUnchecked<utils.BitbucketStatus[]>(\n `./rest/build-status/1.0/commits/${branchCommit!}`,\n opts,\n )\n ).body;\n}\n\n// https://docs.atlassian.com/bitbucket-server/rest/6.0.0/bitbucket-build-rest.html#idp2\nexport async function getBranchStatusCheck(\n branchName: string,\n context: string,\n): Promise<BranchStatus | null> {\n logger.debug(`getBranchStatusCheck(${branchName}, context=${context})`);\n\n try {\n const states = await getStatusCheck(branchName);\n\n for (const state of states) {\n if (state.key === context) {\n switch (state.state) {\n case 'SUCCESSFUL':\n return 'green';\n case 'INPROGRESS':\n return 'yellow';\n case 'FAILED':\n default:\n return 'red';\n }\n }\n }\n } catch (err) {\n logger.warn({ err }, `Failed to check branch status`);\n }\n return null;\n}\n\nexport async function setBranchStatus({\n branchName,\n context,\n description,\n state,\n url: targetUrl,\n}: BranchStatusConfig): Promise<void> {\n logger.debug(`setBranchStatus(${branchName})`);\n\n const existingStatus = await getBranchStatusCheck(branchName, context);\n if (existingStatus === state) {\n return;\n }\n logger.debug({ branch: branchName, context, state }, 'Setting branch status');\n\n const branchCommit = git.getBranchCommit(branchName);\n\n try {\n const body: any = {\n key: context,\n description,\n url: targetUrl ?? 'https://renovatebot.com',\n };\n\n switch (state) {\n case 'green':\n body.state = 'SUCCESSFUL';\n break;\n case 'yellow':\n body.state = 'INPROGRESS';\n break;\n case 'red':\n default:\n body.state = 'FAILED';\n break;\n }\n\n await bitbucketServerHttp.postJson(\n // TODO: types (#22198)\n `./rest/build-status/1.0/commits/${branchCommit!}`,\n { body },\n );\n\n // update status cache\n await getStatus(branchName, false);\n await getStatusCheck(branchName, false);\n } catch (err) {\n logger.warn({ err }, `Failed to set branch status`);\n }\n}\n\n// Issue\n\n/* v8 ignore next */\nexport function findIssue(title: string): Promise<Issue | null> {\n logger.debug(`findIssue(${title})`);\n // This is used by Renovate when creating its own issues,\n // e.g. for deprecated package warnings,\n // config error notifications, or \"dependencyDashboard\"\n //\n // Bitbucket Server does not have issues\n return Promise.resolve(null);\n}\n\n/* v8 ignore next */\nexport function ensureIssue({\n title,\n}: EnsureIssueConfig): Promise<EnsureIssueResult | null> {\n logger.warn({ title }, 'Cannot ensure issue');\n // This is used by Renovate when creating its own issues,\n // e.g. for deprecated package warnings,\n // config error notifications, or \"dependencyDashboard\"\n //\n // Bitbucket Server does not have issues\n return Promise.resolve(null);\n}\n\n/* v8 ignore next */\nexport function getIssueList(): Promise<Issue[]> {\n logger.debug(`getIssueList()`);\n // This is used by Renovate when creating its own issues,\n // e.g. for deprecated package warnings,\n // config error notifications, or \"dependencyDashboard\"\n //\n // Bitbucket Server does not have issues\n return Promise.resolve([]);\n}\n\n/* v8 ignore next */\nexport function ensureIssueClosing(title: string): Promise<void> {\n logger.debug(`ensureIssueClosing(${title})`);\n // This is used by Renovate when creating its own issues,\n // e.g. for deprecated package warnings,\n // config error notifications, or \"dependencyDashboard\"\n //\n // Bitbucket Server does not have issues\n return Promise.resolve();\n}\n\nexport function addAssignees(iid: number, assignees: string[]): Promise<void> {\n logger.debug(`addAssignees(${iid}, [${assignees.join(', ')}])`);\n // This is used by Renovate when creating its own issues,\n // e.g. for deprecated package warnings,\n // config error notifications, or \"dependencyDashboard\"\n //\n // Bitbucket Server does not have issues\n return Promise.resolve();\n}\n\nexport async function addReviewers(\n prNo: number,\n reviewers: string[],\n): Promise<void> {\n logger.debug(`Adding reviewers '${reviewers.join(', ')}' to #${prNo}`);\n\n const reviewerNames = new Set<string>();\n\n for (const entry of reviewers) {\n // If entry is an email address, resolve username\n if (isEmailAdress(entry)) {\n const names = await getUsernamesByEmail(entry);\n for (const name of names) {\n reviewerNames.add(name);\n }\n } else {\n reviewerNames.add(entry);\n }\n }\n\n await retry(updatePRAndAddReviewers, [prNo, Array.from(reviewerNames)], 3, [\n REPOSITORY_CHANGED,\n ]);\n}\n\n/**\n * Resolves Bitbucket users by email address,\n * restricted to users who have REPO_READ permission on the target repository.\n *\n * @param emailAddress - A string that could be the user's email address.\n * @returns List of usernames for active, matched users.\n */\nexport async function getUsernamesByEmail(\n emailAddress: string,\n): Promise<string[]> {\n try {\n const filterUrl =\n `./rest/api/1.0/users?filter=${emailAddress}` +\n `&permission.1=REPO_READ` +\n `&permission.1.projectKey=${config.projectKey}` +\n `&permission.1.repositorySlug=${config.repositorySlug}`;\n\n const users = await bitbucketServerHttp.getJson(\n filterUrl,\n { paginate: true, limit: 100 },\n Users,\n );\n\n if (users.body.length) {\n return users.body\n .filter((u) => u.active && u.emailAddress === emailAddress)\n .map((u) => u.name);\n }\n } catch (err) {\n logger.warn(\n { err, emailAddress },\n `Failed to resolve email address to username`,\n );\n throw err;\n }\n logger.debug({ userinfo: emailAddress }, 'No users found for email-address');\n return [];\n}\n\nasync function updatePRAndAddReviewers(\n prNo: number,\n reviewers: string[],\n): Promise<void> {\n try {\n const pr = await getPr(prNo);\n if (!pr) {\n throw new Error(REPOSITORY_NOT_FOUND);\n }\n\n // TODO: can `reviewers` be undefined? (#40625)\n const reviewersSet = new Set([...pr.reviewers!, ...reviewers]);\n\n await bitbucketServerHttp.putJson(\n `./rest/api/1.0/projects/${config.projectKey}/repos/${config.repositorySlug}/pull-requests/${prNo}`,\n {\n body: {\n title: pr.title,\n version: pr.version,\n reviewers: Array.from(reviewersSet).map((name) => ({\n user: { name },\n })),\n },\n },\n );\n await getPr(prNo, true);\n } catch (err) {\n logger.warn({ err, reviewers, prNo }, `Failed to add reviewers`);\n if (err.statusCode === 404) {\n throw new Error(REPOSITORY_NOT_FOUND);\n } else if (err.statusCode === 409) {\n if (utils.isInvalidReviewersResponse(err)) {\n // Retry again with invalid reviewers being removed\n const invalidReviewers = utils.getInvalidReviewers(err);\n const filteredReviewers = reviewers.filter(\n (name) => !invalidReviewers.includes(name),\n );\n await updatePRAndAddReviewers(prNo, filteredReviewers);\n } else {\n logger.debug(\n '409 response to adding reviewers - has repository changed?',\n );\n throw new Error(REPOSITORY_CHANGED);\n }\n } else {\n throw err;\n }\n }\n}\n\nasync function retry<T extends (...arg0: any[]) => Promise<any>>(\n fn: T,\n args: Parameters<T>,\n maxTries: number,\n retryErrorMessages: string[],\n): Promise<Awaited<ReturnType<T>>> {\n const maxAttempts = Math.max(maxTries, 1);\n let lastError: Error | undefined;\n for (let attempt = 0; attempt < maxAttempts; attempt++) {\n try {\n return await fn(...args);\n } catch (e) {\n lastError = e;\n if (\n retryErrorMessages.length !== 0 &&\n !retryErrorMessages.includes(e.message)\n ) {\n logger.debug(`Error not marked for retry`);\n throw e;\n }\n }\n }\n\n logger.debug(`All ${maxAttempts} retry attempts exhausted`);\n throw lastError!;\n}\n\nexport function deleteLabel(issueNo: number, label: string): Promise<void> {\n logger.debug(`deleteLabel(${issueNo}, ${label})`);\n // Only used for the \"request Renovate to rebase a PR using a label\" feature\n //\n // Bitbucket Server does not have issues\n return Promise.resolve();\n}\n\nasync function getComments(prNo: number): Promise<Comment[]> {\n // GET /rest/api/1.0/projects/{projectKey}/repos/{repositorySlug}/pull-requests/{pullRequestId}/activities\n const activities = (\n await bitbucketServerHttp.getJsonUnchecked<PullRequestActivity[]>(\n `./rest/api/1.0/projects/${config.projectKey}/repos/${config.repositorySlug}/pull-requests/${prNo}/activities`,\n { paginate: true },\n )\n ).body;\n\n const comments = activities\n .filter(\n (a): a is PullRequestCommentActivity =>\n a.action === 'COMMENTED' && 'comment' in a && 'commentAction' in a,\n )\n .filter((a) => a.commentAction === 'ADDED')\n .map((a) => a.comment);\n\n logger.debug(`Found ${comments.length} comments`);\n\n return comments;\n}\n\nasync function addComment(prNo: number, text: string): Promise<void> {\n // POST /rest/api/1.0/projects/{projectKey}/repos/{repositorySlug}/pull-requests/{pullRequestId}/comments\n await bitbucketServerHttp.postJson(\n `./rest/api/1.0/projects/${config.projectKey}/repos/${config.repositorySlug}/pull-requests/${prNo}/comments`,\n {\n body: { text },\n },\n );\n}\n\nasync function getCommentVersion(\n prNo: number,\n commentId: number,\n): Promise<number> {\n // GET /rest/api/1.0/projects/{projectKey}/repos/{repositorySlug}/pull-requests/{pullRequestId}/comments/{commentId}\n const { version } = (\n await bitbucketServerHttp.getJsonUnchecked<{ version: number }>(\n `./rest/api/1.0/projects/${config.projectKey}/repos/${config.repositorySlug}/pull-requests/${prNo}/comments/${commentId}`,\n )\n ).body;\n\n return version;\n}\n\nasync function editComment(\n prNo: number,\n commentId: number,\n text: string,\n): Promise<void> {\n const version = await getCommentVersion(prNo, commentId);\n\n // PUT /rest/api/1.0/projects/{projectKey}/repos/{repositorySlug}/pull-requests/{pullRequestId}/comments/{commentId}\n await bitbucketServerHttp.putJson(\n `./rest/api/1.0/projects/${config.projectKey}/repos/${config.repositorySlug}/pull-requests/${prNo}/comments/${commentId}`,\n {\n body: { text, version },\n },\n );\n}\n\nasync function deleteComment(prNo: number, commentId: number): Promise<void> {\n const version = await getCommentVersion(prNo, commentId);\n\n // DELETE /rest/api/1.0/projects/{projectKey}/repos/{repositorySlug}/pull-requests/{pullRequestId}/comments/{commentId}\n await bitbucketServerHttp.deleteJson(\n `./rest/api/1.0/projects/${config.projectKey}/repos/${config.repositorySlug}/pull-requests/${prNo}/comments/${commentId}?version=${version}`,\n );\n}\n\nexport async function ensureComment({\n number,\n topic,\n content,\n}: EnsureCommentConfig): Promise<boolean> {\n const sanitizedContent = sanitize(content);\n try {\n const comments = await getComments(number);\n let body: string;\n let commentId: number | undefined;\n let commentNeedsUpdating: boolean | undefined;\n if (topic) {\n logger.debug(`Ensuring comment \"${topic}\" in #${number}`);\n body = `### ${topic}\\n\\n${sanitizedContent}`;\n comments.forEach((comment) => {\n if (comment.text.startsWith(`### ${topic}\\n\\n`)) {\n commentId = comment.id;\n commentNeedsUpdating = comment.text !== body;\n }\n });\n } else {\n logger.debug(`Ensuring content-only comment in #${number}`);\n body = `${sanitizedContent}`;\n comments.forEach((comment) => {\n if (comment.text === body) {\n commentId = comment.id;\n commentNeedsUpdating = false;\n }\n });\n }\n if (!commentId) {\n await addComment(number, body);\n logger.info(\n { repository: config.repository, prNo: number, topic },\n 'Comment added',\n );\n } else if (commentNeedsUpdating) {\n await editComment(number, commentId, body);\n logger.debug(\n { repository: config.repository, prNo: number },\n 'Comment updated',\n );\n } else {\n logger.debug('Comment is already up-to-date');\n }\n return true;\n } catch (err) /* v8 ignore next */ {\n logger.warn({ err }, 'Error ensuring comment');\n return false;\n }\n}\n\nexport async function ensureCommentRemoval(\n deleteConfig: EnsureCommentRemovalConfig,\n): Promise<void> {\n try {\n const { number: prNo } = deleteConfig;\n const key =\n deleteConfig.type === 'by-topic'\n ? deleteConfig.topic\n : deleteConfig.content;\n logger.debug(`Ensuring comment \"${key}\" in #${prNo} is removed`);\n const comments = await getComments(prNo);\n\n let commentId: number | null | undefined = null;\n // v8 ignore else -- TODO: add test #40625\n if (deleteConfig.type === 'by-topic') {\n const byTopic = (comment: Comment): boolean =>\n comment.text.startsWith(`### ${deleteConfig.topic}\\n\\n`);\n commentId = comments.find(byTopic)?.id;\n } else if (deleteConfig.type === 'by-content') {\n const byContent = (comment: Comment): boolean =>\n comment.text.trim() === deleteConfig.content;\n commentId = comments.find(byContent)?.id;\n }\n\n if (commentId) {\n await deleteComment(prNo, commentId);\n }\n } catch (err) /* v8 ignore next */ {\n logger.warn({ err }, 'Error ensuring comment removal');\n }\n}\n\n// Pull Request\n\nconst escapeHash = (input: string): string =>\n input?.replace(regEx(/#/g), '%23');\n\nexport async function createPr({\n sourceBranch,\n targetBranch,\n prTitle: title,\n prBody: rawDescription,\n platformPrOptions,\n}: CreatePRConfig): Promise<Pr> {\n const description = sanitize(rawDescription);\n logger.debug(`createPr(${sourceBranch}, title=${title})`);\n const base = targetBranch;\n let reviewers: BbsRestUserRef[] = [];\n\n if (platformPrOptions?.bbUseDefaultReviewers) {\n logger.debug(`fetching default reviewers`);\n const { id } = (\n await bitbucketServerHttp.getJsonUnchecked<{ id: number }>(\n `./rest/api/1.0/projects/${config.projectKey}/repos/${config.repositorySlug}`,\n )\n ).body;\n\n const defReviewers = (\n await bitbucketServerHttp.getJsonUnchecked<{ name: string }[]>(\n `./rest/default-reviewers/1.0/projects/${config.projectKey}/repos/${\n config.repositorySlug\n }/reviewers?sourceRefId=refs/heads/${escapeHash(\n sourceBranch,\n )}&targetRefId=refs/heads/${base}&sourceRepoId=${id}&targetRepoId=${id}`,\n )\n ).body;\n\n reviewers = defReviewers.map((u) => ({\n user: { name: u.name },\n }));\n }\n\n const body: PartialDeep<BbsRestPr> = {\n title,\n description,\n fromRef: {\n id: `refs/heads/${sourceBranch}`,\n },\n toRef: {\n id: `refs/heads/${base}`,\n },\n reviewers,\n };\n let prInfoRes: HttpResponse<BbsRestPr>;\n try {\n prInfoRes = await bitbucketServerHttp.postJson<BbsRestPr>(\n `./rest/api/1.0/projects/${config.projectKey}/repos/${config.repositorySlug}/pull-requests`,\n { body },\n );\n } catch (err) /* v8 ignore next */ {\n if (\n err.body?.errors?.[0]?.exceptionName ===\n 'com.atlassian.bitbucket.pull.EmptyPullRequestException'\n ) {\n logger.debug(\n 'Empty pull request - deleting branch so it can be recreated next run',\n );\n await deleteBranch(sourceBranch);\n throw new Error(REPOSITORY_CHANGED);\n }\n throw err;\n }\n\n const pr: BbsPr = {\n ...utils.prInfo(prInfoRes.body),\n };\n\n // TODO #22198\n updatePrVersion(pr.number, pr.version!);\n await BbsPrCache.setPr(\n bitbucketServerHttp,\n config.projectKey,\n config.repositorySlug,\n config.ignorePrAuthor,\n config.username,\n pr,\n );\n\n if (platformPrOptions?.usePlatformAutomerge) {\n await tryPrAutomerge(pr.number, pr.version!);\n }\n\n return pr;\n}\n\nexport async function updatePr({\n number: prNo,\n prTitle: title,\n prBody: rawDescription,\n state,\n bitbucketInvalidReviewers,\n targetBranch,\n}: UpdatePrConfig & {\n bitbucketInvalidReviewers?: string[] | undefined;\n}): Promise<void> {\n const description = sanitize(rawDescription);\n logger.debug(`updatePr(${prNo}, title=${title})`);\n\n try {\n const pr = await getPr(prNo);\n if (!pr) {\n throw Object.assign(new Error(REPOSITORY_NOT_FOUND), { statusCode: 404 });\n }\n\n const body: any = {\n title,\n description,\n version: pr.version,\n reviewers: pr.reviewers\n ?.filter((name: string) => !bitbucketInvalidReviewers?.includes(name))\n .map((name: string) => ({ user: { name } })),\n };\n if (targetBranch) {\n body.toRef = {\n id: getNewBranchName(targetBranch),\n };\n }\n\n const { body: updatedPr } = await bitbucketServerHttp.putJson<\n BbsRestPr & {\n version: number;\n }\n >(\n `./rest/api/1.0/projects/${config.projectKey}/repos/${config.repositorySlug}/pull-requests/${prNo}`,\n { body },\n );\n\n updatePrVersion(prNo, updatedPr.version);\n\n const currentState = updatedPr.state;\n // TODO #22198\n const newState = {\n ['open']: 'OPEN',\n ['closed']: 'DECLINED',\n }[state!];\n\n let finalState: 'open' | 'closed' =\n currentState === 'OPEN' ? 'open' : 'closed';\n\n if (\n newState &&\n ['OPEN', 'DECLINED'].includes(currentState) &&\n currentState !== newState\n ) {\n const command = state === 'open' ? 'reopen' : 'decline';\n const { body: updatedStatePr } = await bitbucketServerHttp.postJson<{\n version: number;\n }>(\n `./rest/api/1.0/projects/${config.projectKey}/repos/${config.repositorySlug}/pull-requests/${pr.number}/${command}?version=${updatedPr.version}`,\n );\n\n finalState = state!;\n\n updatePrVersion(pr.number, updatedStatePr.version);\n }\n\n const bbsPr = utils.prInfo(updatedPr);\n await BbsPrCache.setPr(\n bitbucketServerHttp,\n config.projectKey,\n config.repositorySlug,\n config.ignorePrAuthor,\n config.username,\n { ...bbsPr, state: finalState },\n );\n } catch (err) {\n logger.debug({ err, prNo }, `Failed to update PR`);\n if (err.statusCode === 404) {\n throw new Error(REPOSITORY_NOT_FOUND);\n } else if (err.statusCode === 409) {\n if (utils.isInvalidReviewersResponse(err) && !bitbucketInvalidReviewers) {\n // Retry again with invalid reviewers being removed\n const invalidReviewers = utils.getInvalidReviewers(err);\n await updatePr({\n number: prNo,\n prTitle: title,\n prBody: rawDescription,\n state,\n bitbucketInvalidReviewers: invalidReviewers,\n });\n } else {\n throw new Error(REPOSITORY_CHANGED);\n }\n } else {\n throw err;\n }\n }\n}\n\n// https://docs.atlassian.com/bitbucket-server/rest/6.0.0/bitbucket-rest.html#idp261\nexport async function mergePr({\n branchName,\n id: prNo,\n}: MergePRConfig): Promise<boolean> {\n logger.debug(`mergePr(${prNo}, ${branchName!})`);\n // Used for \"automerge\" feature\n try {\n const pr = await getPr(prNo);\n if (!pr) {\n throw Object.assign(new Error(REPOSITORY_NOT_FOUND), { statusCode: 404 });\n }\n const { body } = await bitbucketServerHttp.postJson<{ version: number }>(\n // TODO: types (#22198)\n `./rest/api/1.0/projects/${config.projectKey}/repos/${\n config.repositorySlug\n }/pull-requests/${prNo}/merge?version=${pr.version!}`,\n );\n updatePrVersion(prNo, body.version);\n } catch (err) {\n if (err.statusCode === 404) {\n throw new Error(REPOSITORY_NOT_FOUND);\n } else if (err.statusCode === 409) {\n logger.warn({ err }, `Failed to merge PR`);\n return false;\n } else {\n logger.warn({ err }, `Failed to merge PR`);\n return false;\n }\n }\n\n logger.debug(`PR merged, PrNo:${prNo}`);\n return true;\n}\n\n/**\n * Enables Bitbucket Server-native automerge for the given PR.\n * https://confluence.atlassian.com/bitbucketserver094/merge-a-pull-request-1489802114.html#Mergeapullrequest-Auto-mergeapullrequest\n */\nasync function tryPrAutomerge(\n prNumber: number,\n prVersion: number,\n): Promise<void> {\n logger.debug(`automergePr(${prNumber})`);\n\n if (semver.lt(defaults.version, '8.15.0')) {\n logger.debug(\n { prNumber },\n 'Bitbucket Server-native automerge: not supported on this version of Bitbucket. Use 8.15.0 or newer.',\n );\n return;\n }\n\n try {\n const body: PullRequestMerge = { autoMerge: true };\n await bitbucketServerHttp.postJson(\n `./rest/api/1.0/projects/${config.projectKey}/repos/${config.repositorySlug}/pull-requests/${prNumber}/merge?version=${prVersion}`,\n { body },\n );\n\n // enabling auto-merge doesn't increase PR version, so we omit updating the cache\n logger.debug({ prNumber }, 'Bitbucket Server-native automerge: success');\n } catch (err) {\n logger.warn({ err, prNumber }, 'Bitbucket Server-native automerge: fail');\n }\n}\n\nexport async function expandGroupMembers(\n reviewers: string[],\n): Promise<string[]> {\n logger.debug(`expandGroupMembers(${reviewers.join(', ')})`);\n const expandedUsers: string[] = [];\n const reviewerGroupPrefix = '@reviewer-group/';\n\n for (const reviewer of reviewers) {\n const [baseEntry, modifier] = reviewer.split(':');\n\n if (baseEntry.startsWith(reviewerGroupPrefix)) {\n const groupName = baseEntry.replace(reviewerGroupPrefix, '');\n const groupUsers = await getUsersFromReviewerGroup(groupName);\n if (!groupUsers.length) {\n continue;\n }\n\n if (modifier) {\n const randomCount = parseModifier(modifier);\n if (randomCount) {\n expandedUsers.push(...sampleSize(groupUsers, randomCount));\n continue;\n }\n }\n\n expandedUsers.push(...groupUsers);\n } else {\n expandedUsers.push(baseEntry); // Add the user entry\n }\n }\n\n return [...new Set(expandedUsers)];\n}\n\nexport function extractRulesFromCodeOwnersLines(\n cleanedLines: string[],\n): FileOwnerRule[] {\n const results: FileOwnerRule[] = [];\n\n const reversedLines = cleanedLines\n .filter((line) => line.trim() !== '' && !line.trim().startsWith('#'))\n .reverse();\n\n for (const line of reversedLines) {\n const [pattern, ...entries] = splitEscapedSpaces(line);\n const matcher = ignore().add(pattern);\n results.push({\n pattern,\n usernames: [...new Set(entries)],\n score: pattern.length,\n match: (path: string) => matcher.ignores(path),\n });\n }\n\n return results;\n}\n\n// Gets active users by name, from a reviewer group\n// Returns an empty array if the group is not found or has no active users\n// As there is no direct API to get group by name, we get all reviewer groups per repo and filter them\n// This is not efficient, but it is the only way to get users from a group by name\n// Supports both repository-scoped and project-scoped groups following the BitBucket server logic described here:\n// https://confluence.atlassian.com/bitbucketserver/code-owners-1296171116.html#Codeowners-Whatifaprojectandrepositorycontainareviewergroupwiththesamename?\nasync function getUsersFromReviewerGroup(groupName: string): Promise<string[]> {\n const allGroups = [];\n\n try {\n const reviewerGroups = await bitbucketServerHttp.getJson(\n `./rest/api/1.0/projects/${config.projectKey}/repos/${config.repositorySlug}/settings/reviewer-groups`,\n { paginate: true },\n ReviewerGroups,\n );\n\n allGroups.push(...reviewerGroups.body);\n } catch (err) {\n logger.debug({ err, groupName }, 'Failed to get reviewer groups for repo');\n return [];\n }\n\n // First, try to find a repo-scoped group with this name\n const repoGroup = allGroups.find(\n (group) => group.name === groupName && group.scope?.type === 'REPOSITORY',\n );\n\n if (repoGroup) {\n return repoGroup.users\n .filter((user) => user.active)\n .map((user) => user.name);\n }\n\n // If no repo-level group, fall back to project-level group\n const projectGroup = allGroups.find(\n (group) => group.name === groupName && group.scope?.type === 'PROJECT',\n );\n\n if (projectGroup) {\n return projectGroup.users\n .filter((user) => user.active)\n .map((user) => user.name);\n }\n\n // Group not found at either level\n logger.warn(\n { groupName },\n 'Reviewer group not found at repo or project level',\n );\n return [];\n}\n\nexport function massageMarkdown(input: string): string {\n logger.debug(`massageMarkdown(${input.split(newlineRegex)[0]})`);\n // Remove any HTML we use\n return smartTruncate(input, maxBodyLength())\n .replace(\n 'you tick the rebase/retry checkbox',\n 'PR is renamed to start with \"rebase!\"',\n )\n .replace(\n 'checking the rebase/retry box above',\n 'renaming the PR to start with \"rebase!\"',\n )\n .replace(regEx(/<\\/?summary>/g), '**')\n .replace(regEx(/<\\/?details>/g), '')\n .replace(regEx(`\\n---\\n\\n.*?<!-- rebase-check -->.*?(\\n|$)`), '')\n .replace(regEx(/<!--.*?-->/gs), '')\n .replace(\n regEx(\n /(!\\[.+?\\]\\(https:\\/\\/developer\\.mend\\.io\\/api\\/mc\\/badges\\/.+?\\))/g,\n ),\n '$1{height=20}',\n );\n}\n\nexport function maxBodyLength(): number {\n return 30000;\n}\n"],"mappings":"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"}
1
+ {"version":3,"file":"index.js","names":["hostRules.find","utils.getRepoGitUrl","git.initRepo","utils.prInfo","git.getBranchCommit","git.branchExists","utils.isInvalidReviewersResponse","utils.getInvalidReviewers"],"sources":["../../../../lib/modules/platform/bitbucket-server/index.ts"],"sourcesContent":["import { isNonEmptyStringAndNotWhitespace } from '@sindresorhus/is';\nimport ignore from 'ignore';\nimport semver from 'semver';\nimport { setTimeout } from 'timers/promises';\nimport type { PartialDeep } from 'type-fest';\nimport { GlobalConfig } from '../../../config/global.ts';\nimport {\n REPOSITORY_CHANGED,\n REPOSITORY_EMPTY,\n REPOSITORY_NOT_FOUND,\n} from '../../../constants/error-messages.ts';\nimport { logger } from '../../../logger/index.ts';\nimport type { BranchStatus } from '../../../types/index.ts';\nimport type { FileData } from '../../../types/platform/bitbucket-server/index.ts';\nimport { parseJson } from '../../../util/common.ts';\nimport { getEnv } from '../../../util/env.ts';\nimport * as git from '../../../util/git/index.ts';\nimport { deleteBranch } from '../../../util/git/index.ts';\nimport * as hostRules from '../../../util/host-rules.ts';\nimport {\n BitbucketServerHttp,\n type BitbucketServerHttpOptions,\n setBaseUrl,\n} from '../../../util/http/bitbucket-server.ts';\nimport { memCacheProvider } from '../../../util/http/cache/memory-http-cache-provider.ts';\nimport type { HttpOptions, HttpResponse } from '../../../util/http/types.ts';\nimport { newlineRegex, regEx } from '../../../util/regex.ts';\nimport { sampleSize } from '../../../util/sample.ts';\nimport { sanitize } from '../../../util/sanitize.ts';\nimport { isEmailAdress } from '../../../util/schema-utils/index.ts';\nimport { ensureTrailingSlash, getQueryString } from '../../../util/url.ts';\nimport type {\n BranchStatusConfig,\n CreatePRConfig,\n EnsureCommentConfig,\n EnsureCommentRemovalConfig,\n EnsureIssueConfig,\n EnsureIssueResult,\n FileOwnerRule,\n FindPRConfig,\n Issue,\n MergePRConfig,\n PlatformParams,\n PlatformResult,\n Pr,\n RepoParams,\n RepoResult,\n UpdatePrConfig,\n} from '../types.ts';\nimport { getNewBranchName, repoFingerprint } from '../util.ts';\nimport { smartTruncate } from '../utils/pr-body.ts';\nimport { BbsPrCache } from './pr-cache.ts';\nimport type {\n Comment,\n PullRequestActivity,\n PullRequestCommentActivity,\n PullRequestMerge,\n} from './schema.ts';\nimport { ReviewerGroups, User, Users } from './schema.ts';\nimport type {\n BbsConfig,\n BbsPr,\n BbsRestBranch,\n BbsRestPr,\n BbsRestRepo,\n BbsRestUserRef,\n} from './types.ts';\nimport * as utils from './utils.ts';\nimport {\n getExtraCloneOpts,\n parseModifier,\n splitEscapedSpaces,\n} from './utils.ts';\n\n/*\n * Version: 5.3 (EOL Date: 15 Aug 2019)\n * See following docs for api information:\n * https://docs.atlassian.com/bitbucket-server/rest/5.3.0/bitbucket-rest.html\n * https://docs.atlassian.com/bitbucket-server/rest/5.3.0/bitbucket-build-rest.html\n *\n * See following page for uptodate supported versions\n * https://confluence.atlassian.com/support/atlassian-support-end-of-life-policy-201851003.html#AtlassianSupportEndofLifePolicy-BitbucketServer\n */\n\nexport const id = 'bitbucket-server';\n\nlet config: BbsConfig = {} as any;\n\nconst bitbucketServerHttp = new BitbucketServerHttp();\n\nconst defaults: {\n endpoint?: string;\n hostType: string;\n version: string;\n} = {\n hostType: 'bitbucket-server',\n version: '0.0.0',\n};\n\n/* v8 ignore next */\nfunction updatePrVersion(pr: number, version: number): number {\n const res = Math.max(config.prVersions.get(pr) ?? 0, version);\n config.prVersions.set(pr, res);\n return res;\n}\n\nexport async function initPlatform({\n endpoint,\n token,\n username,\n password,\n gitAuthor,\n}: PlatformParams): Promise<PlatformResult> {\n if (!endpoint) {\n throw new Error('Init: You must configure a Bitbucket Server endpoint');\n }\n if (!(username && password) && !token) {\n throw new Error(\n 'Init: You must either configure a Bitbucket Server username/password or a HTTP access token',\n );\n } else if (password && token) {\n throw new Error(\n 'Init: You must configure either a Bitbucket Server password or a HTTP access token, not both',\n );\n }\n // TODO: Add a connection check that endpoint/username/password combination are valid (#9595)\n defaults.endpoint = ensureTrailingSlash(endpoint);\n setBaseUrl(defaults.endpoint);\n const platformConfig: PlatformResult = {\n endpoint: defaults.endpoint,\n };\n try {\n const env = getEnv();\n let bitbucketServerVersion = env.RENOVATE_X_PLATFORM_VERSION;\n const { body, headers } = await bitbucketServerHttp.getJsonUnchecked<{\n version: string;\n }>(`./rest/api/1.0/application-properties`, { ...(token && { token }) });\n\n bitbucketServerVersion ??= body.version;\n if (isNonEmptyStringAndNotWhitespace(headers['x-ausername']) && !username) {\n logger.debug(\n { 'x-ausername': headers['x-ausername'] },\n 'Platform: No username configured using headers[\"x-ausername\"]',\n );\n config.username = headers['x-ausername'];\n }\n logger.debug('Bitbucket Server version is: ' + bitbucketServerVersion);\n\n // v8 ignore else -- TODO: add test #40625\n if (semver.valid(bitbucketServerVersion)) {\n defaults.version = bitbucketServerVersion;\n }\n } catch (err) {\n logger.debug(\n { err },\n 'Error authenticating with Bitbucket. Check that your token includes \"api\" permissions',\n );\n }\n\n if (!gitAuthor && username) {\n logger.debug(`Attempting to confirm gitAuthor from username`);\n const options: HttpOptions = {\n memCache: false,\n };\n\n if (token) {\n options.token = token;\n } else {\n options.username = username;\n options.password = password;\n }\n\n try {\n const { displayName, emailAddress } = (\n await bitbucketServerHttp.getJson(\n `./rest/api/1.0/users/${username}`,\n options,\n User,\n )\n ).body;\n\n if (!emailAddress?.length) {\n throw new Error(`No email address configured for username ${username}`);\n }\n\n platformConfig.gitAuthor = `${displayName} <${emailAddress}>`;\n\n logger.debug(`Detected gitAuthor: ${platformConfig.gitAuthor}`);\n } catch (err) {\n logger.debug(\n { err },\n 'Failed to get user info, fallback gitAuthor will be used',\n );\n }\n }\n\n return platformConfig;\n}\n\n// Get all repositories that the user has access to\nexport async function getRepos(): Promise<string[]> {\n logger.debug('Autodiscovering Bitbucket Server repositories');\n try {\n const repos = (\n await bitbucketServerHttp.getJsonUnchecked<BbsRestRepo[]>(\n `./rest/api/1.0/repos?permission=REPO_WRITE&state=AVAILABLE`,\n { paginate: true },\n )\n ).body;\n\n const result = repos.map((repo) => `${repo.project.key}/${repo.slug}`);\n logger.debug({ result }, 'result of getRepos()');\n return result;\n } catch (err) /* v8 ignore next */ {\n logger.error({ err }, `bitbucket getRepos error`);\n throw err;\n }\n}\n\nexport async function getRawFile(\n fileName: string,\n repoName?: string,\n branchOrTag?: string,\n): Promise<string | null> {\n const repo = repoName ?? config.repository;\n const [project, slug] = repo.split('/');\n const fileUrl =\n `./rest/api/1.0/projects/${project}/repos/${slug}/browse/${fileName}?limit=20000` +\n (branchOrTag ? '&at=' + branchOrTag : '');\n const res = await bitbucketServerHttp.getJsonUnchecked<FileData>(fileUrl);\n const { isLastPage, lines, size } = res.body;\n if (isLastPage) {\n return lines.map(({ text }) => text).join('\\n');\n }\n logger.warn({ size }, 'The file is too big');\n throw new Error(`The file is too big (${size}B)`);\n}\n\nexport async function getJsonFile(\n fileName: string,\n repoName?: string,\n branchOrTag?: string,\n): Promise<any> {\n // TODO #22198\n const raw = await getRawFile(fileName, repoName, branchOrTag);\n return parseJson(raw, fileName);\n}\n\n// Initialize Bitbucket Server by getting base branch\nexport async function initRepo({\n repository,\n cloneSubmodules,\n cloneSubmodulesFilter,\n gitUrl,\n}: RepoParams): Promise<RepoResult> {\n logger.debug(`initRepo(\"${JSON.stringify({ repository }, null, 2)}\")`);\n const opts = hostRules.find({\n hostType: defaults.hostType,\n url: defaults.endpoint,\n });\n\n const [projectKey, repositorySlug] = repository.split('/');\n\n config = {\n projectKey,\n repositorySlug,\n repository,\n prVersions: new Map<number, number>(),\n username: opts.username,\n ignorePrAuthor: GlobalConfig.get('ignorePrAuthor', false),\n } as any;\n\n try {\n const info = (\n await bitbucketServerHttp.getJsonUnchecked<BbsRestRepo>(\n `./rest/api/1.0/projects/${config.projectKey}/repos/${config.repositorySlug}`,\n )\n ).body;\n config.owner = info.project.key;\n logger.debug(`${repository} owner = ${config.owner}`);\n const branchRes = await bitbucketServerHttp.getJsonUnchecked<BbsRestBranch>(\n `./rest/api/1.0/projects/${config.projectKey}/repos/${config.repositorySlug}/branches/default`,\n );\n\n // 204 means empty, 404 means repo not found or missing default branch. repo must exist here.\n if ([204, 404].includes(branchRes.statusCode)) {\n throw new Error(REPOSITORY_EMPTY);\n }\n\n const url = utils.getRepoGitUrl(\n config.repositorySlug,\n // TODO #22198\n defaults.endpoint!,\n gitUrl,\n info,\n opts,\n );\n\n await git.initRepo({\n ...config,\n url,\n extraCloneOpts: getExtraCloneOpts(opts),\n cloneSubmodules,\n cloneSubmodulesFilter,\n fullClone: semver.lte(defaults.version, '8.0.0'),\n });\n\n config.mergeMethod = 'merge';\n const repoConfig: RepoResult = {\n defaultBranch: branchRes.body.displayId,\n isFork: !!info.origin,\n repoFingerprint: repoFingerprint(info.id, defaults.endpoint),\n };\n\n return repoConfig;\n } catch (err) /* v8 ignore next */ {\n if (err.statusCode === 404) {\n throw new Error(REPOSITORY_NOT_FOUND);\n }\n if (err.message === REPOSITORY_EMPTY) {\n throw err;\n }\n\n logger.debug({ err }, 'Unknown Bitbucket initRepo error');\n throw err;\n }\n}\n\nexport async function getBranchForceRebase(\n _branchName: string,\n): Promise<boolean> {\n // https://docs.atlassian.com/bitbucket-server/rest/7.0.1/bitbucket-rest.html#idp342\n const res = await bitbucketServerHttp.getJsonUnchecked<{\n mergeConfig: { defaultStrategy: { id: string } };\n }>(\n `./rest/api/1.0/projects/${config.projectKey}/repos/${config.repositorySlug}/settings/pull-requests`,\n );\n\n // If the default merge strategy contains `ff-only` the PR can only be merged\n // if it is up to date with the base branch.\n // The current options for id are:\n // no-ff, ff, ff-only, rebase-no-ff, rebase-ff-only, squash, squash-ff-only\n return Boolean(\n res.body?.mergeConfig?.defaultStrategy?.id.includes('ff-only'),\n );\n}\n\n// Gets details for a PR\nexport async function getPr(\n prNo: number,\n refreshCache?: boolean,\n): Promise<BbsPr | null> {\n logger.debug(`getPr(${prNo})`);\n if (!prNo) {\n return null;\n }\n\n // Disables memCache (which is enabled by default) to be replaced by\n // memCacheProvider.\n const opts: HttpOptions = { memCache: false };\n // TODO: should refresh the cache rather than just ignore it\n if (!refreshCache) {\n opts.cacheProvider = memCacheProvider;\n }\n\n const res = await bitbucketServerHttp.getJsonUnchecked<BbsRestPr>(\n `./rest/api/1.0/projects/${config.projectKey}/repos/${config.repositorySlug}/pull-requests/${prNo}`,\n opts,\n );\n\n const pr: BbsPr = {\n ...utils.prInfo(res.body),\n reviewers: res.body.reviewers.map((r) => r.user.name),\n };\n // TODO #22198\n pr.version = updatePrVersion(pr.number, pr.version!);\n\n return pr;\n}\n\n// TODO: coverage (#40625)\n/* v8 ignore next */\nfunction matchesState(state: string, desiredState: string): boolean {\n if (desiredState === 'all') {\n return true;\n }\n if (desiredState.startsWith('!')) {\n return state !== desiredState.substring(1);\n }\n return state === desiredState;\n}\n\n// TODO: coverage (#40625)\n/* v8 ignore next */\nfunction isRelevantPr(\n branchName: string,\n prTitle: string | null | undefined,\n state: string,\n) {\n return (p: Pr): boolean =>\n p.sourceBranch === branchName &&\n (!prTitle || p.title.toUpperCase() === prTitle.toUpperCase()) &&\n matchesState(p.state, state);\n}\n\n// TODO: coverage (#40625)\nexport async function getPrList(): Promise<Pr[]> {\n logger.debug(`getPrList()`);\n return await BbsPrCache.getPrs(\n bitbucketServerHttp,\n config.projectKey,\n config.repositorySlug,\n config.ignorePrAuthor,\n config.username,\n );\n}\n\n// TODO: coverage (#40625)\n/* v8 ignore next */\nexport async function findPr({\n branchName,\n prTitle,\n state = 'all',\n includeOtherAuthors,\n}: FindPRConfig): Promise<Pr | null> {\n logger.debug(`findPr(${branchName}, \"${prTitle!}\", \"${state}\")`);\n\n if (includeOtherAuthors) {\n // PR might have been created by anyone, so don't use the cached Renovate PR list\n const searchParams: Record<string, string> = {\n state: 'OPEN',\n };\n searchParams.direction = 'outgoing';\n searchParams.at = `refs/heads/${branchName}`;\n\n const query = getQueryString(searchParams);\n const prs = (\n await bitbucketServerHttp.getJsonUnchecked<BbsRestPr[]>(\n `./rest/api/1.0/projects/${config.projectKey}/repos/${config.repositorySlug}/pull-requests?${query}`,\n {\n paginate: true,\n limit: 1, // only fetch the latest pr\n },\n )\n ).body;\n\n if (!prs.length) {\n logger.debug(`No PR found for branch ${branchName}`);\n return null;\n }\n\n return utils.prInfo(prs[0]);\n }\n\n const prList = await getPrList();\n const pr = prList.find(isRelevantPr(branchName, prTitle, state));\n if (pr) {\n logger.debug(`Found PR #${pr.number}`);\n } else {\n logger.debug(`Renovate did not find a PR for branch #${branchName}`);\n }\n return pr ?? null;\n}\n\n// Returns the Pull Request for a branch. Null if not exists.\nexport async function getBranchPr(branchName: string): Promise<BbsPr | null> {\n logger.debug(`getBranchPr(${branchName})`);\n const existingPr = await findPr({\n branchName,\n state: 'open',\n });\n return existingPr ? getPr(existingPr.number) : null;\n}\n\n/* v8 ignore next */\nexport async function refreshPr(number: number): Promise<void> {\n // wait for pr change propagation\n await setTimeout(1000);\n // refresh cache\n await getPr(number, true);\n}\n\nasync function getStatus(\n branchName: string,\n memCache = true,\n): Promise<utils.BitbucketCommitStatus> {\n const branchCommit = git.getBranchCommit(branchName);\n\n /* v8 ignore next: temporary code */\n const opts: HttpOptions = memCache\n ? { cacheProvider: memCacheProvider }\n : { memCache: false };\n\n return (\n await bitbucketServerHttp.getJsonUnchecked<utils.BitbucketCommitStatus>(\n // TODO: types (#22198)\n `./rest/build-status/1.0/commits/stats/${branchCommit!}`,\n opts,\n )\n ).body;\n}\n\n// Returns the combined status for a branch.\n// umbrella for status checks\n// https://docs.atlassian.com/bitbucket-server/rest/6.0.0/bitbucket-build-rest.html#idp2\nexport async function getBranchStatus(\n branchName: string,\n): Promise<BranchStatus> {\n logger.debug(`getBranchStatus(${branchName})`);\n\n if (!git.branchExists(branchName)) {\n logger.debug('Branch does not exist - cannot fetch status');\n throw new Error(REPOSITORY_CHANGED);\n }\n\n try {\n const commitStatus = await getStatus(branchName);\n\n logger.debug({ commitStatus }, 'branch status check result');\n\n if (commitStatus.failed > 0) {\n return 'red';\n }\n if (commitStatus.inProgress > 0) {\n return 'yellow';\n }\n return commitStatus.successful > 0 ? 'green' : 'yellow';\n } catch (err) {\n logger.warn({ err }, `Failed to get branch status`);\n return 'red';\n }\n}\n\nasync function getStatusCheck(\n branchName: string,\n memCache = true,\n): Promise<utils.BitbucketStatus[]> {\n const branchCommit = git.getBranchCommit(branchName);\n\n const opts: BitbucketServerHttpOptions = { paginate: true };\n /* v8 ignore next: temporary code */\n if (memCache) {\n opts.cacheProvider = memCacheProvider;\n } else {\n opts.memCache = false;\n }\n\n return (\n await bitbucketServerHttp.getJsonUnchecked<utils.BitbucketStatus[]>(\n `./rest/build-status/1.0/commits/${branchCommit!}`,\n opts,\n )\n ).body;\n}\n\n// https://docs.atlassian.com/bitbucket-server/rest/6.0.0/bitbucket-build-rest.html#idp2\nexport async function getBranchStatusCheck(\n branchName: string,\n context: string,\n): Promise<BranchStatus | null> {\n logger.debug(`getBranchStatusCheck(${branchName}, context=${context})`);\n\n try {\n const states = await getStatusCheck(branchName);\n\n for (const state of states) {\n if (state.key === context) {\n switch (state.state) {\n case 'SUCCESSFUL':\n return 'green';\n case 'INPROGRESS':\n return 'yellow';\n case 'FAILED':\n default:\n return 'red';\n }\n }\n }\n } catch (err) {\n logger.warn({ err }, `Failed to check branch status`);\n }\n return null;\n}\n\nexport async function setBranchStatus({\n branchName,\n context,\n description,\n state,\n url: targetUrl,\n}: BranchStatusConfig): Promise<void> {\n logger.debug(`setBranchStatus(${branchName})`);\n\n const existingStatus = await getBranchStatusCheck(branchName, context);\n if (existingStatus === state) {\n return;\n }\n logger.debug({ branch: branchName, context, state }, 'Setting branch status');\n\n const branchCommit = git.getBranchCommit(branchName);\n\n try {\n const body: any = {\n key: context,\n description,\n url: targetUrl ?? 'https://renovatebot.com',\n };\n\n switch (state) {\n case 'green':\n body.state = 'SUCCESSFUL';\n break;\n case 'yellow':\n body.state = 'INPROGRESS';\n break;\n case 'red':\n default:\n body.state = 'FAILED';\n break;\n }\n\n await bitbucketServerHttp.postJson(\n // TODO: types (#22198)\n `./rest/build-status/1.0/commits/${branchCommit!}`,\n { body },\n );\n\n // update status cache\n await getStatus(branchName, false);\n await getStatusCheck(branchName, false);\n } catch (err) {\n logger.warn({ err }, `Failed to set branch status`);\n }\n}\n\n// Issue\n\n/* v8 ignore next */\nexport function findIssue(title: string): Promise<Issue | null> {\n logger.debug(`findIssue(${title})`);\n // This is used by Renovate when creating its own issues,\n // e.g. for deprecated package warnings,\n // config error notifications, or \"dependencyDashboard\"\n //\n // Bitbucket Server does not have issues\n return Promise.resolve(null);\n}\n\n/* v8 ignore next */\nexport function ensureIssue({\n title,\n}: EnsureIssueConfig): Promise<EnsureIssueResult | null> {\n logger.warn({ title }, 'Cannot ensure issue');\n // This is used by Renovate when creating its own issues,\n // e.g. for deprecated package warnings,\n // config error notifications, or \"dependencyDashboard\"\n //\n // Bitbucket Server does not have issues\n return Promise.resolve(null);\n}\n\n/* v8 ignore next */\nexport function getIssueList(): Promise<Issue[]> {\n logger.debug(`getIssueList()`);\n // This is used by Renovate when creating its own issues,\n // e.g. for deprecated package warnings,\n // config error notifications, or \"dependencyDashboard\"\n //\n // Bitbucket Server does not have issues\n return Promise.resolve([]);\n}\n\n/* v8 ignore next */\nexport function ensureIssueClosing(title: string): Promise<void> {\n logger.debug(`ensureIssueClosing(${title})`);\n // This is used by Renovate when creating its own issues,\n // e.g. for deprecated package warnings,\n // config error notifications, or \"dependencyDashboard\"\n //\n // Bitbucket Server does not have issues\n return Promise.resolve();\n}\n\nexport function addAssignees(iid: number, assignees: string[]): Promise<void> {\n logger.debug(`addAssignees(${iid}, [${assignees.join(', ')}])`);\n // This is used by Renovate when creating its own issues,\n // e.g. for deprecated package warnings,\n // config error notifications, or \"dependencyDashboard\"\n //\n // Bitbucket Server does not have issues\n return Promise.resolve();\n}\n\nexport async function addReviewers(\n prNo: number,\n reviewers: string[],\n): Promise<void> {\n logger.debug(`Adding reviewers '${reviewers.join(', ')}' to #${prNo}`);\n\n const reviewerNames = new Set<string>();\n\n for (const entry of reviewers) {\n // If entry is an email address, resolve username\n if (isEmailAdress(entry)) {\n const names = await getUsernamesByEmail(entry);\n for (const name of names) {\n reviewerNames.add(name);\n }\n } else {\n reviewerNames.add(entry);\n }\n }\n\n await retry(updatePRAndAddReviewers, [prNo, Array.from(reviewerNames)], 3, [\n REPOSITORY_CHANGED,\n ]);\n}\n\n/**\n * Resolves Bitbucket users by email address,\n * restricted to users who have REPO_READ permission on the target repository.\n *\n * @param emailAddress - A string that could be the user's email address.\n * @returns List of usernames for active, matched users.\n */\nexport async function getUsernamesByEmail(\n emailAddress: string,\n): Promise<string[]> {\n try {\n const filterUrl =\n `./rest/api/1.0/users?filter=${emailAddress}` +\n `&permission.1=REPO_READ` +\n `&permission.1.projectKey=${config.projectKey}` +\n `&permission.1.repositorySlug=${config.repositorySlug}`;\n\n const users = await bitbucketServerHttp.getJson(\n filterUrl,\n { paginate: true, limit: 100 },\n Users,\n );\n\n if (users.body.length) {\n return users.body\n .filter((u) => u.active && u.emailAddress === emailAddress)\n .map((u) => u.name);\n }\n } catch (err) {\n logger.warn(\n { err, emailAddress },\n `Failed to resolve email address to username`,\n );\n throw err;\n }\n logger.debug({ userinfo: emailAddress }, 'No users found for email-address');\n return [];\n}\n\nasync function updatePRAndAddReviewers(\n prNo: number,\n reviewers: string[],\n): Promise<void> {\n try {\n const pr = await getPr(prNo);\n if (!pr) {\n throw new Error(REPOSITORY_NOT_FOUND);\n }\n\n // TODO: can `reviewers` be undefined? (#40625)\n const reviewersSet = new Set([...pr.reviewers!, ...reviewers]);\n\n await bitbucketServerHttp.putJson(\n `./rest/api/1.0/projects/${config.projectKey}/repos/${config.repositorySlug}/pull-requests/${prNo}`,\n {\n body: {\n title: pr.title,\n version: pr.version,\n reviewers: Array.from(reviewersSet).map((name) => ({\n user: { name },\n })),\n },\n },\n );\n await getPr(prNo, true);\n } catch (err) {\n logger.warn({ err, reviewers, prNo }, `Failed to add reviewers`);\n if (err.statusCode === 404) {\n throw new Error(REPOSITORY_NOT_FOUND);\n } else if (err.statusCode === 409) {\n if (utils.isInvalidReviewersResponse(err)) {\n // Retry again with invalid reviewers being removed\n const invalidReviewers = utils.getInvalidReviewers(err);\n const filteredReviewers = reviewers.filter(\n (name) => !invalidReviewers.includes(name),\n );\n if (filteredReviewers.length < reviewers.length) {\n await updatePRAndAddReviewers(prNo, filteredReviewers);\n } else {\n logger.warn(\n { invalidReviewers, reviewers },\n 'Could not filter invalid reviewers from list, aborting to prevent infinite recursion',\n );\n throw err;\n }\n } else {\n logger.debug(\n '409 response to adding reviewers - has repository changed?',\n );\n throw new Error(REPOSITORY_CHANGED);\n }\n } else {\n throw err;\n }\n }\n}\n\nasync function retry<T extends (...arg0: any[]) => Promise<any>>(\n fn: T,\n args: Parameters<T>,\n maxTries: number,\n retryErrorMessages: string[],\n): Promise<Awaited<ReturnType<T>>> {\n const maxAttempts = Math.max(maxTries, 1);\n let lastError: Error | undefined;\n for (let attempt = 0; attempt < maxAttempts; attempt++) {\n try {\n return await fn(...args);\n } catch (e) {\n lastError = e;\n if (\n retryErrorMessages.length !== 0 &&\n !retryErrorMessages.includes(e.message)\n ) {\n logger.debug(`Error not marked for retry`);\n throw e;\n }\n }\n }\n\n logger.debug(`All ${maxAttempts} retry attempts exhausted`);\n throw lastError!;\n}\n\nexport function deleteLabel(issueNo: number, label: string): Promise<void> {\n logger.debug(`deleteLabel(${issueNo}, ${label})`);\n // Only used for the \"request Renovate to rebase a PR using a label\" feature\n //\n // Bitbucket Server does not have issues\n return Promise.resolve();\n}\n\nasync function getComments(prNo: number): Promise<Comment[]> {\n // GET /rest/api/1.0/projects/{projectKey}/repos/{repositorySlug}/pull-requests/{pullRequestId}/activities\n const activities = (\n await bitbucketServerHttp.getJsonUnchecked<PullRequestActivity[]>(\n `./rest/api/1.0/projects/${config.projectKey}/repos/${config.repositorySlug}/pull-requests/${prNo}/activities`,\n { paginate: true },\n )\n ).body;\n\n const comments = activities\n .filter(\n (a): a is PullRequestCommentActivity =>\n a.action === 'COMMENTED' && 'comment' in a && 'commentAction' in a,\n )\n .filter((a) => a.commentAction === 'ADDED')\n .map((a) => a.comment);\n\n logger.debug(`Found ${comments.length} comments`);\n\n return comments;\n}\n\nasync function addComment(prNo: number, text: string): Promise<void> {\n // POST /rest/api/1.0/projects/{projectKey}/repos/{repositorySlug}/pull-requests/{pullRequestId}/comments\n await bitbucketServerHttp.postJson(\n `./rest/api/1.0/projects/${config.projectKey}/repos/${config.repositorySlug}/pull-requests/${prNo}/comments`,\n {\n body: { text },\n },\n );\n}\n\nasync function getCommentVersion(\n prNo: number,\n commentId: number,\n): Promise<number> {\n // GET /rest/api/1.0/projects/{projectKey}/repos/{repositorySlug}/pull-requests/{pullRequestId}/comments/{commentId}\n const { version } = (\n await bitbucketServerHttp.getJsonUnchecked<{ version: number }>(\n `./rest/api/1.0/projects/${config.projectKey}/repos/${config.repositorySlug}/pull-requests/${prNo}/comments/${commentId}`,\n )\n ).body;\n\n return version;\n}\n\nasync function editComment(\n prNo: number,\n commentId: number,\n text: string,\n): Promise<void> {\n const version = await getCommentVersion(prNo, commentId);\n\n // PUT /rest/api/1.0/projects/{projectKey}/repos/{repositorySlug}/pull-requests/{pullRequestId}/comments/{commentId}\n await bitbucketServerHttp.putJson(\n `./rest/api/1.0/projects/${config.projectKey}/repos/${config.repositorySlug}/pull-requests/${prNo}/comments/${commentId}`,\n {\n body: { text, version },\n },\n );\n}\n\nasync function deleteComment(prNo: number, commentId: number): Promise<void> {\n const version = await getCommentVersion(prNo, commentId);\n\n // DELETE /rest/api/1.0/projects/{projectKey}/repos/{repositorySlug}/pull-requests/{pullRequestId}/comments/{commentId}\n await bitbucketServerHttp.deleteJson(\n `./rest/api/1.0/projects/${config.projectKey}/repos/${config.repositorySlug}/pull-requests/${prNo}/comments/${commentId}?version=${version}`,\n );\n}\n\nexport async function ensureComment({\n number,\n topic,\n content,\n}: EnsureCommentConfig): Promise<boolean> {\n const sanitizedContent = sanitize(content);\n try {\n const comments = await getComments(number);\n let body: string;\n let commentId: number | undefined;\n let commentNeedsUpdating: boolean | undefined;\n if (topic) {\n logger.debug(`Ensuring comment \"${topic}\" in #${number}`);\n body = `### ${topic}\\n\\n${sanitizedContent}`;\n comments.forEach((comment) => {\n if (comment.text.startsWith(`### ${topic}\\n\\n`)) {\n commentId = comment.id;\n commentNeedsUpdating = comment.text !== body;\n }\n });\n } else {\n logger.debug(`Ensuring content-only comment in #${number}`);\n body = `${sanitizedContent}`;\n comments.forEach((comment) => {\n if (comment.text === body) {\n commentId = comment.id;\n commentNeedsUpdating = false;\n }\n });\n }\n if (!commentId) {\n await addComment(number, body);\n logger.info(\n { repository: config.repository, prNo: number, topic },\n 'Comment added',\n );\n } else if (commentNeedsUpdating) {\n await editComment(number, commentId, body);\n logger.debug(\n { repository: config.repository, prNo: number },\n 'Comment updated',\n );\n } else {\n logger.debug('Comment is already up-to-date');\n }\n return true;\n } catch (err) /* v8 ignore next */ {\n logger.warn({ err }, 'Error ensuring comment');\n return false;\n }\n}\n\nexport async function ensureCommentRemoval(\n deleteConfig: EnsureCommentRemovalConfig,\n): Promise<void> {\n try {\n const { number: prNo } = deleteConfig;\n const key =\n deleteConfig.type === 'by-topic'\n ? deleteConfig.topic\n : deleteConfig.content;\n logger.debug(`Ensuring comment \"${key}\" in #${prNo} is removed`);\n const comments = await getComments(prNo);\n\n let commentId: number | null | undefined = null;\n // v8 ignore else -- TODO: add test #40625\n if (deleteConfig.type === 'by-topic') {\n const byTopic = (comment: Comment): boolean =>\n comment.text.startsWith(`### ${deleteConfig.topic}\\n\\n`);\n commentId = comments.find(byTopic)?.id;\n } else if (deleteConfig.type === 'by-content') {\n const byContent = (comment: Comment): boolean =>\n comment.text.trim() === deleteConfig.content;\n commentId = comments.find(byContent)?.id;\n }\n\n if (commentId) {\n await deleteComment(prNo, commentId);\n }\n } catch (err) /* v8 ignore next */ {\n logger.warn({ err }, 'Error ensuring comment removal');\n }\n}\n\n// Pull Request\n\nconst escapeHash = (input: string): string =>\n input?.replace(regEx(/#/g), '%23');\n\nexport async function createPr({\n sourceBranch,\n targetBranch,\n prTitle: title,\n prBody: rawDescription,\n platformPrOptions,\n}: CreatePRConfig): Promise<Pr> {\n const description = sanitize(rawDescription);\n logger.debug(`createPr(${sourceBranch}, title=${title})`);\n const base = targetBranch;\n let reviewers: BbsRestUserRef[] = [];\n\n if (platformPrOptions?.bbUseDefaultReviewers) {\n logger.debug(`fetching default reviewers`);\n const { id } = (\n await bitbucketServerHttp.getJsonUnchecked<{ id: number }>(\n `./rest/api/1.0/projects/${config.projectKey}/repos/${config.repositorySlug}`,\n )\n ).body;\n\n const defReviewers = (\n await bitbucketServerHttp.getJsonUnchecked<{ name: string }[]>(\n `./rest/default-reviewers/1.0/projects/${config.projectKey}/repos/${\n config.repositorySlug\n }/reviewers?sourceRefId=refs/heads/${escapeHash(\n sourceBranch,\n )}&targetRefId=refs/heads/${base}&sourceRepoId=${id}&targetRepoId=${id}`,\n )\n ).body;\n\n reviewers = defReviewers.map((u) => ({\n user: { name: u.name },\n }));\n }\n\n const body: PartialDeep<BbsRestPr> = {\n title,\n description,\n fromRef: {\n id: `refs/heads/${sourceBranch}`,\n },\n toRef: {\n id: `refs/heads/${base}`,\n },\n reviewers,\n };\n let prInfoRes: HttpResponse<BbsRestPr>;\n try {\n prInfoRes = await bitbucketServerHttp.postJson<BbsRestPr>(\n `./rest/api/1.0/projects/${config.projectKey}/repos/${config.repositorySlug}/pull-requests`,\n { body },\n );\n } catch (err) /* v8 ignore next */ {\n if (\n err.body?.errors?.[0]?.exceptionName ===\n 'com.atlassian.bitbucket.pull.EmptyPullRequestException'\n ) {\n logger.debug(\n 'Empty pull request - deleting branch so it can be recreated next run',\n );\n await deleteBranch(sourceBranch);\n throw new Error(REPOSITORY_CHANGED);\n }\n throw err;\n }\n\n const pr: BbsPr = {\n ...utils.prInfo(prInfoRes.body),\n };\n\n // TODO #22198\n updatePrVersion(pr.number, pr.version!);\n await BbsPrCache.setPr(\n bitbucketServerHttp,\n config.projectKey,\n config.repositorySlug,\n config.ignorePrAuthor,\n config.username,\n pr,\n );\n\n if (platformPrOptions?.usePlatformAutomerge) {\n await tryPrAutomerge(pr.number, pr.version!);\n }\n\n return pr;\n}\n\nexport async function updatePr({\n number: prNo,\n prTitle: title,\n prBody: rawDescription,\n state,\n bitbucketInvalidReviewers,\n targetBranch,\n}: UpdatePrConfig & {\n bitbucketInvalidReviewers?: string[] | undefined;\n}): Promise<void> {\n const description = sanitize(rawDescription);\n logger.debug(`updatePr(${prNo}, title=${title})`);\n\n try {\n const pr = await getPr(prNo);\n if (!pr) {\n throw Object.assign(new Error(REPOSITORY_NOT_FOUND), { statusCode: 404 });\n }\n\n const body: any = {\n title,\n description,\n version: pr.version,\n reviewers: pr.reviewers\n ?.filter((name: string) => !bitbucketInvalidReviewers?.includes(name))\n .map((name: string) => ({ user: { name } })),\n };\n if (targetBranch) {\n body.toRef = {\n id: getNewBranchName(targetBranch),\n };\n }\n\n const { body: updatedPr } = await bitbucketServerHttp.putJson<\n BbsRestPr & {\n version: number;\n }\n >(\n `./rest/api/1.0/projects/${config.projectKey}/repos/${config.repositorySlug}/pull-requests/${prNo}`,\n { body },\n );\n\n updatePrVersion(prNo, updatedPr.version);\n\n const currentState = updatedPr.state;\n // TODO #22198\n const newState = {\n ['open']: 'OPEN',\n ['closed']: 'DECLINED',\n }[state!];\n\n let finalState: 'open' | 'closed' =\n currentState === 'OPEN' ? 'open' : 'closed';\n\n if (\n newState &&\n ['OPEN', 'DECLINED'].includes(currentState) &&\n currentState !== newState\n ) {\n const command = state === 'open' ? 'reopen' : 'decline';\n const { body: updatedStatePr } = await bitbucketServerHttp.postJson<{\n version: number;\n }>(\n `./rest/api/1.0/projects/${config.projectKey}/repos/${config.repositorySlug}/pull-requests/${pr.number}/${command}?version=${updatedPr.version}`,\n );\n\n finalState = state!;\n\n updatePrVersion(pr.number, updatedStatePr.version);\n }\n\n const bbsPr = utils.prInfo(updatedPr);\n await BbsPrCache.setPr(\n bitbucketServerHttp,\n config.projectKey,\n config.repositorySlug,\n config.ignorePrAuthor,\n config.username,\n { ...bbsPr, state: finalState },\n );\n } catch (err) {\n logger.debug({ err, prNo }, `Failed to update PR`);\n if (err.statusCode === 404) {\n throw new Error(REPOSITORY_NOT_FOUND);\n } else if (err.statusCode === 409) {\n if (utils.isInvalidReviewersResponse(err) && !bitbucketInvalidReviewers) {\n // Retry again with invalid reviewers being removed\n const invalidReviewers = utils.getInvalidReviewers(err);\n await updatePr({\n number: prNo,\n prTitle: title,\n prBody: rawDescription,\n state,\n bitbucketInvalidReviewers: invalidReviewers,\n });\n } else {\n throw new Error(REPOSITORY_CHANGED);\n }\n } else {\n throw err;\n }\n }\n}\n\n// https://docs.atlassian.com/bitbucket-server/rest/6.0.0/bitbucket-rest.html#idp261\nexport async function mergePr({\n branchName,\n id: prNo,\n}: MergePRConfig): Promise<boolean> {\n logger.debug(`mergePr(${prNo}, ${branchName!})`);\n // Used for \"automerge\" feature\n try {\n const pr = await getPr(prNo);\n if (!pr) {\n throw Object.assign(new Error(REPOSITORY_NOT_FOUND), { statusCode: 404 });\n }\n const { body } = await bitbucketServerHttp.postJson<{ version: number }>(\n // TODO: types (#22198)\n `./rest/api/1.0/projects/${config.projectKey}/repos/${\n config.repositorySlug\n }/pull-requests/${prNo}/merge?version=${pr.version!}`,\n );\n updatePrVersion(prNo, body.version);\n } catch (err) {\n if (err.statusCode === 404) {\n throw new Error(REPOSITORY_NOT_FOUND);\n } else if (err.statusCode === 409) {\n logger.warn({ err }, `Failed to merge PR`);\n return false;\n } else {\n logger.warn({ err }, `Failed to merge PR`);\n return false;\n }\n }\n\n logger.debug(`PR merged, PrNo:${prNo}`);\n return true;\n}\n\n/**\n * Enables Bitbucket Server-native automerge for the given PR.\n * https://confluence.atlassian.com/bitbucketserver094/merge-a-pull-request-1489802114.html#Mergeapullrequest-Auto-mergeapullrequest\n */\nasync function tryPrAutomerge(\n prNumber: number,\n prVersion: number,\n): Promise<void> {\n logger.debug(`automergePr(${prNumber})`);\n\n if (semver.lt(defaults.version, '8.15.0')) {\n logger.debug(\n { prNumber },\n 'Bitbucket Server-native automerge: not supported on this version of Bitbucket. Use 8.15.0 or newer.',\n );\n return;\n }\n\n try {\n const body: PullRequestMerge = { autoMerge: true };\n await bitbucketServerHttp.postJson(\n `./rest/api/1.0/projects/${config.projectKey}/repos/${config.repositorySlug}/pull-requests/${prNumber}/merge?version=${prVersion}`,\n { body },\n );\n\n // enabling auto-merge doesn't increase PR version, so we omit updating the cache\n logger.debug({ prNumber }, 'Bitbucket Server-native automerge: success');\n } catch (err) {\n logger.warn({ err, prNumber }, 'Bitbucket Server-native automerge: fail');\n }\n}\n\nexport async function expandGroupMembers(\n reviewers: string[],\n): Promise<string[]> {\n logger.debug(`expandGroupMembers(${reviewers.join(', ')})`);\n const expandedUsers: string[] = [];\n const reviewerGroupPrefix = '@reviewer-group/';\n\n for (const reviewer of reviewers) {\n const [baseEntry, modifier] = reviewer.split(':');\n\n if (baseEntry.startsWith(reviewerGroupPrefix)) {\n const groupName = baseEntry.replace(reviewerGroupPrefix, '');\n const groupUsers = await getUsersFromReviewerGroup(groupName);\n if (!groupUsers.length) {\n continue;\n }\n\n if (modifier) {\n const randomCount = parseModifier(modifier);\n if (randomCount) {\n expandedUsers.push(...sampleSize(groupUsers, randomCount));\n continue;\n }\n }\n\n expandedUsers.push(...groupUsers);\n } else {\n expandedUsers.push(baseEntry); // Add the user entry\n }\n }\n\n return [...new Set(expandedUsers)];\n}\n\nexport function extractRulesFromCodeOwnersLines(\n cleanedLines: string[],\n): FileOwnerRule[] {\n const results: FileOwnerRule[] = [];\n\n const reversedLines = cleanedLines\n .filter((line) => line.trim() !== '' && !line.trim().startsWith('#'))\n .reverse();\n\n for (const line of reversedLines) {\n const [pattern, ...entries] = splitEscapedSpaces(line);\n const matcher = ignore().add(pattern);\n results.push({\n pattern,\n usernames: [...new Set(entries)],\n score: pattern.length,\n match: (path: string) => matcher.ignores(path),\n });\n }\n\n return results;\n}\n\n// Gets active users by name, from a reviewer group\n// Returns an empty array if the group is not found or has no active users\n// As there is no direct API to get group by name, we get all reviewer groups per repo and filter them\n// This is not efficient, but it is the only way to get users from a group by name\n// Supports both repository-scoped and project-scoped groups following the BitBucket server logic described here:\n// https://confluence.atlassian.com/bitbucketserver/code-owners-1296171116.html#Codeowners-Whatifaprojectandrepositorycontainareviewergroupwiththesamename?\nasync function getUsersFromReviewerGroup(groupName: string): Promise<string[]> {\n const allGroups = [];\n\n try {\n const reviewerGroups = await bitbucketServerHttp.getJson(\n `./rest/api/1.0/projects/${config.projectKey}/repos/${config.repositorySlug}/settings/reviewer-groups`,\n { paginate: true },\n ReviewerGroups,\n );\n\n allGroups.push(...reviewerGroups.body);\n } catch (err) {\n logger.debug({ err, groupName }, 'Failed to get reviewer groups for repo');\n return [];\n }\n\n // First, try to find a repo-scoped group with this name\n const repoGroup = allGroups.find(\n (group) => group.name === groupName && group.scope?.type === 'REPOSITORY',\n );\n\n if (repoGroup) {\n return repoGroup.users\n .filter((user) => user.active)\n .map((user) => user.name);\n }\n\n // If no repo-level group, fall back to project-level group\n const projectGroup = allGroups.find(\n (group) => group.name === groupName && group.scope?.type === 'PROJECT',\n );\n\n if (projectGroup) {\n return projectGroup.users\n .filter((user) => user.active)\n .map((user) => user.name);\n }\n\n // Group not found at either level\n logger.warn(\n { groupName },\n 'Reviewer group not found at repo or project level',\n );\n return [];\n}\n\nexport function massageMarkdown(input: string): string {\n logger.debug(`massageMarkdown(${input.split(newlineRegex)[0]})`);\n // Remove any HTML we use\n return smartTruncate(input, maxBodyLength())\n .replace(\n 'you tick the rebase/retry checkbox',\n 'PR is renamed to start with \"rebase!\"',\n )\n .replace(\n 'checking the rebase/retry box above',\n 'renaming the PR to start with \"rebase!\"',\n )\n .replace(regEx(/<\\/?summary>/g), '**')\n .replace(regEx(/<\\/?details>/g), '')\n .replace(regEx(`\\n---\\n\\n.*?<!-- rebase-check -->.*?(\\n|$)`), '')\n .replace(regEx(/<!--.*?-->/gs), '')\n .replace(\n regEx(\n /(!\\[.+?\\]\\(https:\\/\\/developer\\.mend\\.io\\/api\\/mc\\/badges\\/.+?\\))/g,\n ),\n '$1{height=20}',\n );\n}\n\nexport function maxBodyLength(): number {\n return 30000;\n}\n"],"mappings":"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"}
@@ -85,7 +85,7 @@ async function cleanUpBranches(config, remainingBranches) {
85
85
  * @param config Renovate configuration
86
86
  */
87
87
  function calculateBaseBranchRegex(config) {
88
- if (!config.baseBranchPatterns?.length) return null;
88
+ if (!config.baseBranchPatterns?.length || !config.baseBranches?.length) return null;
89
89
  const branchPrefixes = [config.branchPrefix, config.branchPrefixOld].filter(isNonEmptyStringAndNotWhitespace).filter(uniqueStrings).map(escapeRegExp);
90
90
  const baseBranches = config.baseBranches.map(escapeRegExp);
91
91
  return regEx(`^(?:${branchPrefixes.join("|")})(${baseBranches.join("|")})-`);
@@ -1 +1 @@
1
- {"version":3,"file":"prune.js","names":[],"sources":["../../../../lib/workers/repository/finalize/prune.ts"],"sourcesContent":["import { isNonEmptyStringAndNotWhitespace } from '@sindresorhus/is';\nimport { GlobalConfig } from '../../../config/global.ts';\nimport type { RenovateConfig } from '../../../config/types.ts';\nimport { REPOSITORY_CHANGED } from '../../../constants/error-messages.ts';\nimport { logger } from '../../../logger/index.ts';\nimport { ensureComment } from '../../../modules/platform/comment.ts';\nimport { platform } from '../../../modules/platform/index.ts';\nimport { scm } from '../../../modules/platform/scm.ts';\nimport { getBranchList, setUserRepoConfig } from '../../../util/git/index.ts';\nimport { escapeRegExp, regEx } from '../../../util/regex.ts';\nimport { uniqueStrings } from '../../../util/string.ts';\nimport { getReconfigureBranchName } from '../reconfigure/utils.ts';\n\nasync function cleanUpBranches(\n config: RenovateConfig,\n remainingBranches: string[],\n): Promise<void> {\n if (!config.pruneStaleBranches) {\n logger.debug('Branch/PR pruning is disabled - skipping');\n return;\n }\n // set Git author in case the repository is not initialized yet\n setUserRepoConfig(config);\n\n // calculate regex to extract base branch from branch name\n const baseBranchRe = calculateBaseBranchRegex(config);\n\n for (const branchName of remainingBranches) {\n try {\n // get base branch from branch name if base branches are configured\n // use default branch if no base branches are configured\n // use defaul branch name if no match (can happen when base branches are configured later)\n const baseBranch =\n baseBranchRe?.exec(branchName)?.[1] ?? config.defaultBranch!;\n const pr = await platform.findPr({\n branchName,\n state: 'open',\n targetBranch: baseBranch,\n });\n const branchIsModified = await scm.isBranchModified(\n branchName,\n baseBranch,\n );\n if (pr) {\n if (branchIsModified) {\n logger.debug(\n { prNo: pr.number, prTitle: pr.title },\n 'Branch is modified - skipping PR autoclosing',\n );\n if (GlobalConfig.get('dryRun')) {\n logger.info(`DRY-RUN: Would update PR title and ensure comment.`);\n } else {\n if (!pr.title.endsWith('- abandoned')) {\n const newPrTitle = pr.title + ' - abandoned';\n await platform.updatePr({\n number: pr.number,\n prTitle: newPrTitle,\n state: 'open',\n });\n }\n\n await ensureComment({\n number: pr.number,\n topic: 'Autoclosing Skipped',\n content:\n 'This PR has been flagged for autoclosing. However, it is being skipped due to the branch being already modified. Please close/delete it manually or report a bug if you think this is in error.',\n });\n }\n } else if (GlobalConfig.get('dryRun')) {\n logger.info(\n { prNo: pr.number, prTitle: pr.title },\n `DRY-RUN: Would autoclose PR`,\n );\n } else {\n logger.info(\n { branchName, prNo: pr.number, prTitle: pr.title },\n 'Autoclosing PR',\n );\n let newPrTitle = pr.title;\n if (!pr.title.endsWith('- autoclosed')) {\n newPrTitle += ' - autoclosed';\n }\n await platform.updatePr({\n number: pr.number,\n prTitle: newPrTitle,\n state: 'closed',\n });\n await scm.deleteBranch(branchName);\n }\n } else if (branchIsModified) {\n logger.debug(\n { branch: branchName },\n 'Orphan Branch is modified - skipping branch deletion',\n );\n } else if (GlobalConfig.get('dryRun')) {\n logger.info(`DRY-RUN: Would delete orphan branch ${branchName}`);\n } else {\n logger.info({ branch: branchName }, `Deleting orphan branch`);\n await scm.deleteBranch(branchName);\n }\n } catch (err) /* istanbul ignore next */ {\n if (err.message === 'config-validation') {\n logger.debug(\n 'Cannot prune branch due to collision between tags and branch names',\n );\n } else if (err.message?.includes(\"bad revision 'origin/\")) {\n logger.debug(\n { branchName },\n 'Branch not found on origin when attempting to prune',\n );\n } else if (err.message !== REPOSITORY_CHANGED) {\n logger.warn({ err, branch: branchName }, 'Error pruning branch');\n }\n }\n }\n}\n\n/**\n * Calculates a {RegExp} to extract the base branch from a branch name if base branch patterns is configured.\n * @param config Renovate configuration\n */\nfunction calculateBaseBranchRegex(config: RenovateConfig): RegExp | null {\n if (!config.baseBranchPatterns?.length) {\n return null;\n }\n\n // calculate possible branch prefixes and escape for regex\n const branchPrefixes = [config.branchPrefix, config.branchPrefixOld]\n .filter(isNonEmptyStringAndNotWhitespace)\n .filter(uniqueStrings)\n .map(escapeRegExp);\n\n // calculate possible base branches and escape for regex\n // if baseBranchPatterns is configured, baseBranches will be defined too\n const baseBranches = config.baseBranches!.map(escapeRegExp);\n\n // create regex to extract base branche from branch name\n const baseBranchRe = regEx(\n `^(?:${branchPrefixes.join('|')})(${baseBranches.join('|')})-`,\n );\n\n return baseBranchRe;\n}\n\nexport async function pruneStaleBranches(\n config: RenovateConfig,\n branchList: string[] | null | undefined,\n): Promise<void> {\n logger.debug('Removing any stale branches');\n logger.trace({ config }, `pruneStaleBranches`);\n // TODO: types (#22198)\n logger.debug(`config.repoIsOnboarded=${config.repoIsOnboarded!}`);\n if (!branchList) {\n logger.debug('No branchList');\n return;\n }\n // TODO: types (#22198)\n let renovateBranches = getBranchList().filter(\n (branchName) =>\n branchName.startsWith(config.branchPrefix!) &&\n branchName !== getReconfigureBranchName(config.branchPrefix!),\n );\n if (!renovateBranches?.length) {\n logger.debug('No renovate branches found');\n return;\n }\n logger.debug(\n {\n branchList: branchList?.sort(),\n renovateBranches: renovateBranches?.sort(),\n },\n 'Branch lists',\n );\n // TODO: types (#22198)\n const lockFileBranch = `${config.branchPrefix!}lock-file-maintenance`;\n renovateBranches = renovateBranches.filter(\n (branch) => branch !== lockFileBranch,\n );\n const remainingBranches = renovateBranches.filter(\n (branch) => !branchList.includes(branch),\n );\n logger.debug(`remainingBranches=${String(remainingBranches)}`);\n if (remainingBranches.length === 0) {\n logger.debug('No branches to clean up');\n return;\n }\n\n await cleanUpBranches(config, remainingBranches);\n}\n"],"mappings":";;;;;;;;;;;;AAaA,eAAe,gBACb,QACA,mBACe;AACf,KAAI,CAAC,OAAO,oBAAoB;AAC9B,SAAO,MAAM,2CAA2C;AACxD;;AAGF,mBAAkB,OAAO;CAGzB,MAAM,eAAe,yBAAyB,OAAO;AAErD,MAAK,MAAM,cAAc,kBACvB,KAAI;EAIF,MAAM,aACJ,cAAc,KAAK,WAAW,GAAG,MAAM,OAAO;EAChD,MAAM,KAAK,MAAM,SAAS,OAAO;GAC/B;GACA,OAAO;GACP,cAAc;GACf,CAAC;EACF,MAAM,mBAAmB,MAAM,IAAI,iBACjC,YACA,WACD;AACD,MAAI,GACF,KAAI,kBAAkB;AACpB,UAAO,MACL;IAAE,MAAM,GAAG;IAAQ,SAAS,GAAG;IAAO,EACtC,+CACD;AACD,OAAI,aAAa,IAAI,SAAS,CAC5B,QAAO,KAAK,qDAAqD;QAC5D;AACL,QAAI,CAAC,GAAG,MAAM,SAAS,cAAc,EAAE;KACrC,MAAM,aAAa,GAAG,QAAQ;AAC9B,WAAM,SAAS,SAAS;MACtB,QAAQ,GAAG;MACX,SAAS;MACT,OAAO;MACR,CAAC;;AAGJ,UAAM,cAAc;KAClB,QAAQ,GAAG;KACX,OAAO;KACP,SACE;KACH,CAAC;;aAEK,aAAa,IAAI,SAAS,CACnC,QAAO,KACL;GAAE,MAAM,GAAG;GAAQ,SAAS,GAAG;GAAO,EACtC,8BACD;OACI;AACL,UAAO,KACL;IAAE;IAAY,MAAM,GAAG;IAAQ,SAAS,GAAG;IAAO,EAClD,iBACD;GACD,IAAI,aAAa,GAAG;AACpB,OAAI,CAAC,GAAG,MAAM,SAAS,eAAe,CACpC,eAAc;AAEhB,SAAM,SAAS,SAAS;IACtB,QAAQ,GAAG;IACX,SAAS;IACT,OAAO;IACR,CAAC;AACF,SAAM,IAAI,aAAa,WAAW;;WAE3B,iBACT,QAAO,MACL,EAAE,QAAQ,YAAY,EACtB,uDACD;WACQ,aAAa,IAAI,SAAS,CACnC,QAAO,KAAK,uCAAuC,aAAa;OAC3D;AACL,UAAO,KAAK,EAAE,QAAQ,YAAY,EAAE,yBAAyB;AAC7D,SAAM,IAAI,aAAa,WAAW;;UAE7B,iCAAgC;AACvC,MAAI,IAAI,YAAY,oBAClB,QAAO,MACL,qEACD;WACQ,IAAI,SAAS,SAAS,wBAAwB,CACvD,QAAO,MACL,EAAE,YAAY,EACd,sDACD;WACQ,IAAI,YAAA,qBACb,QAAO,KAAK;GAAE;GAAK,QAAQ;GAAY,EAAE,uBAAuB;;;;;;;AAUxE,SAAS,yBAAyB,QAAuC;AACvE,KAAI,CAAC,OAAO,oBAAoB,OAC9B,QAAO;CAIT,MAAM,iBAAiB,CAAC,OAAO,cAAc,OAAO,gBAAgB,CACjE,OAAO,iCAAiC,CACxC,OAAO,cAAc,CACrB,IAAI,aAAa;CAIpB,MAAM,eAAe,OAAO,aAAc,IAAI,aAAa;AAO3D,QAJqB,MACnB,OAAO,eAAe,KAAK,IAAI,CAAC,IAAI,aAAa,KAAK,IAAI,CAAC,IAC5D;;AAKH,eAAsB,mBACpB,QACA,YACe;AACf,QAAO,MAAM,8BAA8B;AAC3C,QAAO,MAAM,EAAE,QAAQ,EAAE,qBAAqB;AAE9C,QAAO,MAAM,0BAA0B,OAAO,kBAAmB;AACjE,KAAI,CAAC,YAAY;AACf,SAAO,MAAM,gBAAgB;AAC7B;;CAGF,IAAI,mBAAmB,eAAe,CAAC,QACpC,eACC,WAAW,WAAW,OAAO,aAAc,IAC3C,eAAe,yBAAyB,OAAO,aAAc,CAChE;AACD,KAAI,CAAC,kBAAkB,QAAQ;AAC7B,SAAO,MAAM,6BAA6B;AAC1C;;AAEF,QAAO,MACL;EACE,YAAY,YAAY,MAAM;EAC9B,kBAAkB,kBAAkB,MAAM;EAC3C,EACD,eACD;CAED,MAAM,iBAAiB,GAAG,OAAO,aAAc;AAC/C,oBAAmB,iBAAiB,QACjC,WAAW,WAAW,eACxB;CACD,MAAM,oBAAoB,iBAAiB,QACxC,WAAW,CAAC,WAAW,SAAS,OAAO,CACzC;AACD,QAAO,MAAM,qBAAqB,OAAO,kBAAkB,GAAG;AAC9D,KAAI,kBAAkB,WAAW,GAAG;AAClC,SAAO,MAAM,0BAA0B;AACvC;;AAGF,OAAM,gBAAgB,QAAQ,kBAAkB"}
1
+ {"version":3,"file":"prune.js","names":[],"sources":["../../../../lib/workers/repository/finalize/prune.ts"],"sourcesContent":["import { isNonEmptyStringAndNotWhitespace } from '@sindresorhus/is';\nimport { GlobalConfig } from '../../../config/global.ts';\nimport type { RenovateConfig } from '../../../config/types.ts';\nimport { REPOSITORY_CHANGED } from '../../../constants/error-messages.ts';\nimport { logger } from '../../../logger/index.ts';\nimport { ensureComment } from '../../../modules/platform/comment.ts';\nimport { platform } from '../../../modules/platform/index.ts';\nimport { scm } from '../../../modules/platform/scm.ts';\nimport { getBranchList, setUserRepoConfig } from '../../../util/git/index.ts';\nimport { escapeRegExp, regEx } from '../../../util/regex.ts';\nimport { uniqueStrings } from '../../../util/string.ts';\nimport { getReconfigureBranchName } from '../reconfigure/utils.ts';\n\nasync function cleanUpBranches(\n config: RenovateConfig,\n remainingBranches: string[],\n): Promise<void> {\n if (!config.pruneStaleBranches) {\n logger.debug('Branch/PR pruning is disabled - skipping');\n return;\n }\n // set Git author in case the repository is not initialized yet\n setUserRepoConfig(config);\n\n // calculate regex to extract base branch from branch name\n const baseBranchRe = calculateBaseBranchRegex(config);\n\n for (const branchName of remainingBranches) {\n try {\n // get base branch from branch name if base branches are configured\n // use default branch if no base branches are configured\n // use defaul branch name if no match (can happen when base branches are configured later)\n const baseBranch =\n baseBranchRe?.exec(branchName)?.[1] ?? config.defaultBranch!;\n const pr = await platform.findPr({\n branchName,\n state: 'open',\n targetBranch: baseBranch,\n });\n const branchIsModified = await scm.isBranchModified(\n branchName,\n baseBranch,\n );\n if (pr) {\n if (branchIsModified) {\n logger.debug(\n { prNo: pr.number, prTitle: pr.title },\n 'Branch is modified - skipping PR autoclosing',\n );\n if (GlobalConfig.get('dryRun')) {\n logger.info(`DRY-RUN: Would update PR title and ensure comment.`);\n } else {\n if (!pr.title.endsWith('- abandoned')) {\n const newPrTitle = pr.title + ' - abandoned';\n await platform.updatePr({\n number: pr.number,\n prTitle: newPrTitle,\n state: 'open',\n });\n }\n\n await ensureComment({\n number: pr.number,\n topic: 'Autoclosing Skipped',\n content:\n 'This PR has been flagged for autoclosing. However, it is being skipped due to the branch being already modified. Please close/delete it manually or report a bug if you think this is in error.',\n });\n }\n } else if (GlobalConfig.get('dryRun')) {\n logger.info(\n { prNo: pr.number, prTitle: pr.title },\n `DRY-RUN: Would autoclose PR`,\n );\n } else {\n logger.info(\n { branchName, prNo: pr.number, prTitle: pr.title },\n 'Autoclosing PR',\n );\n let newPrTitle = pr.title;\n if (!pr.title.endsWith('- autoclosed')) {\n newPrTitle += ' - autoclosed';\n }\n await platform.updatePr({\n number: pr.number,\n prTitle: newPrTitle,\n state: 'closed',\n });\n await scm.deleteBranch(branchName);\n }\n } else if (branchIsModified) {\n logger.debug(\n { branch: branchName },\n 'Orphan Branch is modified - skipping branch deletion',\n );\n } else if (GlobalConfig.get('dryRun')) {\n logger.info(`DRY-RUN: Would delete orphan branch ${branchName}`);\n } else {\n logger.info({ branch: branchName }, `Deleting orphan branch`);\n await scm.deleteBranch(branchName);\n }\n } catch (err) /* istanbul ignore next */ {\n if (err.message === 'config-validation') {\n logger.debug(\n 'Cannot prune branch due to collision between tags and branch names',\n );\n } else if (err.message?.includes(\"bad revision 'origin/\")) {\n logger.debug(\n { branchName },\n 'Branch not found on origin when attempting to prune',\n );\n } else if (err.message !== REPOSITORY_CHANGED) {\n logger.warn({ err, branch: branchName }, 'Error pruning branch');\n }\n }\n }\n}\n\n/**\n * Calculates a {RegExp} to extract the base branch from a branch name if base branch patterns is configured.\n * @param config Renovate configuration\n */\nfunction calculateBaseBranchRegex(config: RenovateConfig): RegExp | null {\n if (!config.baseBranchPatterns?.length || !config.baseBranches?.length) {\n return null;\n }\n\n // calculate possible branch prefixes and escape for regex\n const branchPrefixes = [config.branchPrefix, config.branchPrefixOld]\n .filter(isNonEmptyStringAndNotWhitespace)\n .filter(uniqueStrings)\n .map(escapeRegExp);\n\n const baseBranches = config.baseBranches.map(escapeRegExp);\n\n // create regex to extract base branche from branch name\n const baseBranchRe = regEx(\n `^(?:${branchPrefixes.join('|')})(${baseBranches.join('|')})-`,\n );\n\n return baseBranchRe;\n}\n\nexport async function pruneStaleBranches(\n config: RenovateConfig,\n branchList: string[] | null | undefined,\n): Promise<void> {\n logger.debug('Removing any stale branches');\n logger.trace({ config }, `pruneStaleBranches`);\n // TODO: types (#22198)\n logger.debug(`config.repoIsOnboarded=${config.repoIsOnboarded!}`);\n if (!branchList) {\n logger.debug('No branchList');\n return;\n }\n // TODO: types (#22198)\n let renovateBranches = getBranchList().filter(\n (branchName) =>\n branchName.startsWith(config.branchPrefix!) &&\n branchName !== getReconfigureBranchName(config.branchPrefix!),\n );\n if (!renovateBranches?.length) {\n logger.debug('No renovate branches found');\n return;\n }\n logger.debug(\n {\n branchList: branchList?.sort(),\n renovateBranches: renovateBranches?.sort(),\n },\n 'Branch lists',\n );\n // TODO: types (#22198)\n const lockFileBranch = `${config.branchPrefix!}lock-file-maintenance`;\n renovateBranches = renovateBranches.filter(\n (branch) => branch !== lockFileBranch,\n );\n const remainingBranches = renovateBranches.filter(\n (branch) => !branchList.includes(branch),\n );\n logger.debug(`remainingBranches=${String(remainingBranches)}`);\n if (remainingBranches.length === 0) {\n logger.debug('No branches to clean up');\n return;\n }\n\n await cleanUpBranches(config, remainingBranches);\n}\n"],"mappings":"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"}
@@ -4,7 +4,7 @@ import { noLeadingAtSymbol } from "../../../../util/common.js";
4
4
  import { sampleSize } from "../../../../util/sample.js";
5
5
  import { platform } from "../../../../modules/platform/index.js";
6
6
  import { codeOwnersForPr } from "./code-owners.js";
7
- import { isArray, isNumber } from "@sindresorhus/is";
7
+ import { isArray, isNonEmptyString, isNumber } from "@sindresorhus/is";
8
8
  //#region lib/workers/repository/update/pr/participants.ts
9
9
  async function addCodeOwners(config, assigneesOrReviewers, pr) {
10
10
  const codeOwners = await codeOwnersForPr(pr);
@@ -15,7 +15,7 @@ function filterUnavailableUsers(config, users) {
15
15
  return config.filterUnavailableUsers && platform.filterUnavailableUsers ? platform.filterUnavailableUsers(users) : Promise.resolve(users);
16
16
  }
17
17
  function prepareParticipants(config, usernames) {
18
- return filterUnavailableUsers(config, [...new Set(usernames.map(noLeadingAtSymbol))]);
18
+ return filterUnavailableUsers(config, [...new Set(usernames.map(noLeadingAtSymbol).filter(isNonEmptyString))]);
19
19
  }
20
20
  async function addParticipants(config, pr) {
21
21
  let assignees = config.assignees ?? [];
@@ -1 +1 @@
1
- {"version":3,"file":"participants.js","names":[],"sources":["../../../../../lib/workers/repository/update/pr/participants.ts"],"sourcesContent":["import { isArray, isNumber } from '@sindresorhus/is';\nimport { GlobalConfig } from '../../../../config/global.ts';\nimport type { RenovateConfig } from '../../../../config/types.ts';\nimport { logger } from '../../../../logger/index.ts';\nimport type { Pr } from '../../../../modules/platform/index.ts';\nimport { platform } from '../../../../modules/platform/index.ts';\nimport { noLeadingAtSymbol } from '../../../../util/common.ts';\nimport { sampleSize } from '../../../../util/sample.ts';\nimport { codeOwnersForPr } from './code-owners.ts';\n\nasync function addCodeOwners(\n config: RenovateConfig,\n assigneesOrReviewers: string[],\n pr: Pr,\n): Promise<string[]> {\n const codeOwners = await codeOwnersForPr(pr);\n\n const assignees =\n config.expandCodeOwnersGroups && platform.expandGroupMembers\n ? await platform.expandGroupMembers(codeOwners)\n : codeOwners;\n\n return [...new Set(assigneesOrReviewers.concat(assignees))];\n}\n\nfunction filterUnavailableUsers(\n config: RenovateConfig,\n users: string[],\n): Promise<string[]> {\n return config.filterUnavailableUsers && platform.filterUnavailableUsers\n ? platform.filterUnavailableUsers(users)\n : Promise.resolve(users);\n}\n\nfunction prepareParticipants(\n config: RenovateConfig,\n usernames: string[],\n): Promise<string[]> {\n const normalizedUsernames = [...new Set(usernames.map(noLeadingAtSymbol))];\n return filterUnavailableUsers(config, normalizedUsernames);\n}\n\nexport async function addParticipants(\n config: RenovateConfig,\n pr: Pr,\n): Promise<void> {\n let assignees = config.assignees ?? [];\n logger.debug(`addParticipants(pr=${pr?.number})`);\n if (config.assigneesFromCodeOwners) {\n assignees = await addCodeOwners(config, assignees, pr);\n }\n if (assignees.length > 0) {\n try {\n assignees = await prepareParticipants(config, assignees);\n if (isNumber(config.assigneesSampleSize)) {\n assignees = sampleSize(assignees, config.assigneesSampleSize);\n }\n if (assignees.length > 0) {\n if (GlobalConfig.get('dryRun')) {\n logger.info(`DRY-RUN: Would add assignees to PR #${pr.number}`);\n } else {\n await platform.addAssignees(pr.number, assignees);\n logger.debug({ assignees }, 'Added assignees');\n }\n }\n } catch (err) {\n logger.debug(\n { assignees: config.assignees, err },\n 'Failed to add assignees',\n );\n }\n }\n\n let reviewers = config.reviewers ?? [];\n if (config.reviewersFromCodeOwners) {\n reviewers = await addCodeOwners(config, reviewers, pr);\n logger.debug(\n `Reviewers from code owners: ${reviewers.map((reviewer) => `\"${reviewer}\"`).join(', ')}`,\n );\n }\n if (\n isArray(config.additionalReviewers) &&\n config.additionalReviewers.length > 0\n ) {\n logger.debug(\n `Additional reviewers: ${config.additionalReviewers.map((reviewer) => `\"${reviewer}\"`).join(', ')}`,\n );\n reviewers = reviewers.concat(config.additionalReviewers);\n }\n if (reviewers.length > 0) {\n try {\n reviewers = await prepareParticipants(config, reviewers);\n if (isNumber(config.reviewersSampleSize)) {\n logger.debug(\n `Sampling reviewersSampleSize=${config.reviewersSampleSize} reviewers`,\n );\n reviewers = sampleSize(reviewers, config.reviewersSampleSize);\n }\n if (reviewers.length > 0) {\n if (GlobalConfig.get('dryRun')) {\n logger.info(`DRY-RUN: Would add reviewers to PR #${pr.number}`);\n } else {\n await platform.addReviewers(pr.number, reviewers);\n logger.debug({ reviewers }, 'Added reviewers');\n }\n }\n } catch (err) {\n logger.debug(\n { reviewers: config.reviewers, err },\n 'Failed to add reviewers',\n );\n }\n }\n}\n"],"mappings":";;;;;;;;AAUA,eAAe,cACb,QACA,sBACA,IACmB;CACnB,MAAM,aAAa,MAAM,gBAAgB,GAAG;CAE5C,MAAM,YACJ,OAAO,0BAA0B,SAAS,qBACtC,MAAM,SAAS,mBAAmB,WAAW,GAC7C;AAEN,QAAO,CAAC,GAAG,IAAI,IAAI,qBAAqB,OAAO,UAAU,CAAC,CAAC;;AAG7D,SAAS,uBACP,QACA,OACmB;AACnB,QAAO,OAAO,0BAA0B,SAAS,yBAC7C,SAAS,uBAAuB,MAAM,GACtC,QAAQ,QAAQ,MAAM;;AAG5B,SAAS,oBACP,QACA,WACmB;AAEnB,QAAO,uBAAuB,QADF,CAAC,GAAG,IAAI,IAAI,UAAU,IAAI,kBAAkB,CAAC,CAAC,CAChB;;AAG5D,eAAsB,gBACpB,QACA,IACe;CACf,IAAI,YAAY,OAAO,aAAa,EAAE;AACtC,QAAO,MAAM,sBAAsB,IAAI,OAAO,GAAG;AACjD,KAAI,OAAO,wBACT,aAAY,MAAM,cAAc,QAAQ,WAAW,GAAG;AAExD,KAAI,UAAU,SAAS,EACrB,KAAI;AACF,cAAY,MAAM,oBAAoB,QAAQ,UAAU;AACxD,MAAI,SAAS,OAAO,oBAAoB,CACtC,aAAY,WAAW,WAAW,OAAO,oBAAoB;AAE/D,MAAI,UAAU,SAAS,EACrB,KAAI,aAAa,IAAI,SAAS,CAC5B,QAAO,KAAK,uCAAuC,GAAG,SAAS;OAC1D;AACL,SAAM,SAAS,aAAa,GAAG,QAAQ,UAAU;AACjD,UAAO,MAAM,EAAE,WAAW,EAAE,kBAAkB;;UAG3C,KAAK;AACZ,SAAO,MACL;GAAE,WAAW,OAAO;GAAW;GAAK,EACpC,0BACD;;CAIL,IAAI,YAAY,OAAO,aAAa,EAAE;AACtC,KAAI,OAAO,yBAAyB;AAClC,cAAY,MAAM,cAAc,QAAQ,WAAW,GAAG;AACtD,SAAO,MACL,+BAA+B,UAAU,KAAK,aAAa,IAAI,SAAS,GAAG,CAAC,KAAK,KAAK,GACvF;;AAEH,KACE,QAAQ,OAAO,oBAAoB,IACnC,OAAO,oBAAoB,SAAS,GACpC;AACA,SAAO,MACL,yBAAyB,OAAO,oBAAoB,KAAK,aAAa,IAAI,SAAS,GAAG,CAAC,KAAK,KAAK,GAClG;AACD,cAAY,UAAU,OAAO,OAAO,oBAAoB;;AAE1D,KAAI,UAAU,SAAS,EACrB,KAAI;AACF,cAAY,MAAM,oBAAoB,QAAQ,UAAU;AACxD,MAAI,SAAS,OAAO,oBAAoB,EAAE;AACxC,UAAO,MACL,gCAAgC,OAAO,oBAAoB,YAC5D;AACD,eAAY,WAAW,WAAW,OAAO,oBAAoB;;AAE/D,MAAI,UAAU,SAAS,EACrB,KAAI,aAAa,IAAI,SAAS,CAC5B,QAAO,KAAK,uCAAuC,GAAG,SAAS;OAC1D;AACL,SAAM,SAAS,aAAa,GAAG,QAAQ,UAAU;AACjD,UAAO,MAAM,EAAE,WAAW,EAAE,kBAAkB;;UAG3C,KAAK;AACZ,SAAO,MACL;GAAE,WAAW,OAAO;GAAW;GAAK,EACpC,0BACD"}
1
+ {"version":3,"file":"participants.js","names":[],"sources":["../../../../../lib/workers/repository/update/pr/participants.ts"],"sourcesContent":["import { isArray, isNonEmptyString, isNumber } from '@sindresorhus/is';\nimport { GlobalConfig } from '../../../../config/global.ts';\nimport type { RenovateConfig } from '../../../../config/types.ts';\nimport { logger } from '../../../../logger/index.ts';\nimport type { Pr } from '../../../../modules/platform/index.ts';\nimport { platform } from '../../../../modules/platform/index.ts';\nimport { noLeadingAtSymbol } from '../../../../util/common.ts';\nimport { sampleSize } from '../../../../util/sample.ts';\nimport { codeOwnersForPr } from './code-owners.ts';\n\nasync function addCodeOwners(\n config: RenovateConfig,\n assigneesOrReviewers: string[],\n pr: Pr,\n): Promise<string[]> {\n const codeOwners = await codeOwnersForPr(pr);\n\n const assignees =\n config.expandCodeOwnersGroups && platform.expandGroupMembers\n ? await platform.expandGroupMembers(codeOwners)\n : codeOwners;\n\n return [...new Set(assigneesOrReviewers.concat(assignees))];\n}\n\nfunction filterUnavailableUsers(\n config: RenovateConfig,\n users: string[],\n): Promise<string[]> {\n return config.filterUnavailableUsers && platform.filterUnavailableUsers\n ? platform.filterUnavailableUsers(users)\n : Promise.resolve(users);\n}\n\nfunction prepareParticipants(\n config: RenovateConfig,\n usernames: string[],\n): Promise<string[]> {\n const normalizedUsernames = [\n ...new Set(usernames.map(noLeadingAtSymbol).filter(isNonEmptyString)),\n ];\n return filterUnavailableUsers(config, normalizedUsernames);\n}\n\nexport async function addParticipants(\n config: RenovateConfig,\n pr: Pr,\n): Promise<void> {\n let assignees = config.assignees ?? [];\n logger.debug(`addParticipants(pr=${pr?.number})`);\n if (config.assigneesFromCodeOwners) {\n assignees = await addCodeOwners(config, assignees, pr);\n }\n if (assignees.length > 0) {\n try {\n assignees = await prepareParticipants(config, assignees);\n if (isNumber(config.assigneesSampleSize)) {\n assignees = sampleSize(assignees, config.assigneesSampleSize);\n }\n if (assignees.length > 0) {\n if (GlobalConfig.get('dryRun')) {\n logger.info(`DRY-RUN: Would add assignees to PR #${pr.number}`);\n } else {\n await platform.addAssignees(pr.number, assignees);\n logger.debug({ assignees }, 'Added assignees');\n }\n }\n } catch (err) {\n logger.debug(\n { assignees: config.assignees, err },\n 'Failed to add assignees',\n );\n }\n }\n\n let reviewers = config.reviewers ?? [];\n if (config.reviewersFromCodeOwners) {\n reviewers = await addCodeOwners(config, reviewers, pr);\n logger.debug(\n `Reviewers from code owners: ${reviewers.map((reviewer) => `\"${reviewer}\"`).join(', ')}`,\n );\n }\n if (\n isArray(config.additionalReviewers) &&\n config.additionalReviewers.length > 0\n ) {\n logger.debug(\n `Additional reviewers: ${config.additionalReviewers.map((reviewer) => `\"${reviewer}\"`).join(', ')}`,\n );\n reviewers = reviewers.concat(config.additionalReviewers);\n }\n if (reviewers.length > 0) {\n try {\n reviewers = await prepareParticipants(config, reviewers);\n if (isNumber(config.reviewersSampleSize)) {\n logger.debug(\n `Sampling reviewersSampleSize=${config.reviewersSampleSize} reviewers`,\n );\n reviewers = sampleSize(reviewers, config.reviewersSampleSize);\n }\n if (reviewers.length > 0) {\n if (GlobalConfig.get('dryRun')) {\n logger.info(`DRY-RUN: Would add reviewers to PR #${pr.number}`);\n } else {\n await platform.addReviewers(pr.number, reviewers);\n logger.debug({ reviewers }, 'Added reviewers');\n }\n }\n } catch (err) {\n logger.debug(\n { reviewers: config.reviewers, err },\n 'Failed to add reviewers',\n );\n }\n }\n}\n"],"mappings":";;;;;;;;AAUA,eAAe,cACb,QACA,sBACA,IACmB;CACnB,MAAM,aAAa,MAAM,gBAAgB,GAAG;CAE5C,MAAM,YACJ,OAAO,0BAA0B,SAAS,qBACtC,MAAM,SAAS,mBAAmB,WAAW,GAC7C;AAEN,QAAO,CAAC,GAAG,IAAI,IAAI,qBAAqB,OAAO,UAAU,CAAC,CAAC;;AAG7D,SAAS,uBACP,QACA,OACmB;AACnB,QAAO,OAAO,0BAA0B,SAAS,yBAC7C,SAAS,uBAAuB,MAAM,GACtC,QAAQ,QAAQ,MAAM;;AAG5B,SAAS,oBACP,QACA,WACmB;AAInB,QAAO,uBAAuB,QAHF,CAC1B,GAAG,IAAI,IAAI,UAAU,IAAI,kBAAkB,CAAC,OAAO,iBAAiB,CAAC,CACtE,CACyD;;AAG5D,eAAsB,gBACpB,QACA,IACe;CACf,IAAI,YAAY,OAAO,aAAa,EAAE;AACtC,QAAO,MAAM,sBAAsB,IAAI,OAAO,GAAG;AACjD,KAAI,OAAO,wBACT,aAAY,MAAM,cAAc,QAAQ,WAAW,GAAG;AAExD,KAAI,UAAU,SAAS,EACrB,KAAI;AACF,cAAY,MAAM,oBAAoB,QAAQ,UAAU;AACxD,MAAI,SAAS,OAAO,oBAAoB,CACtC,aAAY,WAAW,WAAW,OAAO,oBAAoB;AAE/D,MAAI,UAAU,SAAS,EACrB,KAAI,aAAa,IAAI,SAAS,CAC5B,QAAO,KAAK,uCAAuC,GAAG,SAAS;OAC1D;AACL,SAAM,SAAS,aAAa,GAAG,QAAQ,UAAU;AACjD,UAAO,MAAM,EAAE,WAAW,EAAE,kBAAkB;;UAG3C,KAAK;AACZ,SAAO,MACL;GAAE,WAAW,OAAO;GAAW;GAAK,EACpC,0BACD;;CAIL,IAAI,YAAY,OAAO,aAAa,EAAE;AACtC,KAAI,OAAO,yBAAyB;AAClC,cAAY,MAAM,cAAc,QAAQ,WAAW,GAAG;AACtD,SAAO,MACL,+BAA+B,UAAU,KAAK,aAAa,IAAI,SAAS,GAAG,CAAC,KAAK,KAAK,GACvF;;AAEH,KACE,QAAQ,OAAO,oBAAoB,IACnC,OAAO,oBAAoB,SAAS,GACpC;AACA,SAAO,MACL,yBAAyB,OAAO,oBAAoB,KAAK,aAAa,IAAI,SAAS,GAAG,CAAC,KAAK,KAAK,GAClG;AACD,cAAY,UAAU,OAAO,OAAO,oBAAoB;;AAE1D,KAAI,UAAU,SAAS,EACrB,KAAI;AACF,cAAY,MAAM,oBAAoB,QAAQ,UAAU;AACxD,MAAI,SAAS,OAAO,oBAAoB,EAAE;AACxC,UAAO,MACL,gCAAgC,OAAO,oBAAoB,YAC5D;AACD,eAAY,WAAW,WAAW,OAAO,oBAAoB;;AAE/D,MAAI,UAAU,SAAS,EACrB,KAAI,aAAa,IAAI,SAAS,CAC5B,QAAO,KAAK,uCAAuC,GAAG,SAAS;OAC1D;AACL,SAAM,SAAS,aAAa,GAAG,QAAQ,UAAU;AACjD,UAAO,MAAM,EAAE,WAAW,EAAE,kBAAkB;;UAG3C,KAAK;AACZ,SAAO,MACL;GAAE,WAAW,OAAO;GAAW;GAAK,EACpC,0BACD"}
package/package.json CHANGED
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "name": "renovate",
3
3
  "description": "Automated dependency updates. Flexible so you don't need to be.",
4
- "version": "43.76.0",
4
+ "version": "43.76.1",
5
5
  "type": "module",
6
6
  "bin": {
7
7
  "renovate": "dist/renovate.js",
@@ -1,8 +1,8 @@
1
1
  {
2
2
  "$id": "https://docs.renovatebot.com/renovate-schema.json",
3
- "title": "JSON schema for Renovate 43.76.0 config files (https://renovatebot.com/)",
3
+ "title": "JSON schema for Renovate 43.76.1 config files (https://renovatebot.com/)",
4
4
  "$schema": "http://json-schema.org/draft-07/schema#",
5
- "x-renovate-version": "43.76.0",
5
+ "x-renovate-version": "43.76.1",
6
6
  "allowComments": true,
7
7
  "type": "object",
8
8
  "properties": {