renovate 43.4.2 → 43.4.4

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.
@@ -47,7 +47,7 @@ var bitbucket_server_exports = /* @__PURE__ */ __exportAll({
47
47
  getPrList: () => getPrList,
48
48
  getRawFile: () => getRawFile,
49
49
  getRepos: () => getRepos,
50
- getUserSlugsByEmail: () => getUserSlugsByEmail,
50
+ getUsernamesByEmail: () => getUsernamesByEmail,
51
51
  id: () => id,
52
52
  initPlatform: () => initPlatform,
53
53
  initRepo: () => initRepo,
@@ -346,33 +346,33 @@ function addAssignees(iid, assignees) {
346
346
  }
347
347
  async function addReviewers(prNo, reviewers) {
348
348
  logger.debug(`Adding reviewers '${reviewers.join(", ")}' to #${prNo}`);
349
- const reviewerSlugs = /* @__PURE__ */ new Set();
349
+ const reviewerNames = /* @__PURE__ */ new Set();
350
350
  for (const entry of reviewers) if (isEmailAdress(entry)) {
351
- const slugs = await getUserSlugsByEmail(entry);
352
- for (const slug of slugs) reviewerSlugs.add(slug);
353
- } else reviewerSlugs.add(entry);
354
- await retry(updatePRAndAddReviewers, [prNo, Array.from(reviewerSlugs)], 3, [REPOSITORY_CHANGED]);
351
+ const names = await getUsernamesByEmail(entry);
352
+ for (const name of names) reviewerNames.add(name);
353
+ } else reviewerNames.add(entry);
354
+ await retry(updatePRAndAddReviewers, [prNo, Array.from(reviewerNames)], 3, [REPOSITORY_CHANGED]);
355
355
  }
356
356
  /**
357
357
  * Resolves Bitbucket users by email address,
358
358
  * restricted to users who have REPO_READ permission on the target repository.
359
359
  *
360
- * @param emailAddress - A string that could be the user's email-address.
361
- * @returns List of user slugs for active, matched users.
360
+ * @param emailAddress - A string that could be the user's email address.
361
+ * @returns List of usernames for active, matched users.
362
362
  */
363
- async function getUserSlugsByEmail(emailAddress) {
363
+ async function getUsernamesByEmail(emailAddress) {
364
364
  try {
365
365
  const filterUrl = `./rest/api/1.0/users?filter=${emailAddress}&permission.1=REPO_READ&permission.1.projectKey=${config.projectKey}&permission.1.repositorySlug=${config.repositorySlug}`;
366
366
  const users = await bitbucketServerHttp.getJson(filterUrl, {
367
367
  paginate: true,
368
368
  limit: 100
369
369
  }, Users);
370
- if (users.body.length) return users.body.filter((u) => u.active && u.emailAddress === emailAddress).map((u) => u.slug);
370
+ if (users.body.length) return users.body.filter((u) => u.active && u.emailAddress === emailAddress).map((u) => u.name);
371
371
  } catch (err) {
372
372
  logger.warn({
373
373
  err,
374
374
  emailAddress
375
- }, `Failed to resolve email address to user slug`);
375
+ }, `Failed to resolve email address to username`);
376
376
  throw err;
377
377
  }
378
378
  logger.debug({ userinfo: emailAddress }, "No users found for email-address");
@@ -686,9 +686,9 @@ async function getUsersFromReviewerGroup(groupName) {
686
686
  return [];
687
687
  }
688
688
  const repoGroup = allGroups.find((group) => group.name === groupName && group.scope?.type === "REPOSITORY");
689
- if (repoGroup) return repoGroup.users.filter((user) => user.active).map((user) => user.slug);
689
+ if (repoGroup) return repoGroup.users.filter((user) => user.active).map((user) => user.name);
690
690
  const projectGroup = allGroups.find((group) => group.name === groupName && group.scope?.type === "PROJECT");
691
- if (projectGroup) return projectGroup.users.filter((user) => user.active).map((user) => user.slug);
691
+ if (projectGroup) return projectGroup.users.filter((user) => user.active).map((user) => user.name);
692
692
  logger.warn({ groupName }, "Reviewer group not found at repo or project level");
693
693
  return [];
694
694
  }
@@ -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 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 (#9624)\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 (#9624)\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 (#9624)\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 (#9624)\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 reviewerSlugs = new Set<string>();\n\n for (const entry of reviewers) {\n // If entry is an email-address, resolve userslugs\n if (isEmailAdress(entry)) {\n const slugs = await getUserSlugsByEmail(entry);\n for (const slug of slugs) {\n reviewerSlugs.add(slug);\n }\n } else {\n reviewerSlugs.add(entry);\n }\n }\n\n await retry(updatePRAndAddReviewers, [prNo, Array.from(reviewerSlugs)], 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 user slugs for active, matched users.\n */\nexport async function getUserSlugsByEmail(\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.slug);\n }\n } catch (err) {\n logger.warn(\n { err, emailAddress },\n `Failed to resolve email address to user slug`,\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? (#22198)\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 update-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 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.slug);\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.slug);\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 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 (#9624)\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 (#9624)\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 (#9624)\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 (#9624)\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? (#22198)\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 update-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 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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"}
@@ -3,10 +3,10 @@ import { z } from "zod";
3
3
 
4
4
  //#region lib/modules/platform/bitbucket-server/schema.ts
5
5
  const User = z.object({
6
+ name: z.string(),
6
7
  displayName: z.string(),
7
8
  emailAddress: EmailAddress.catch(""),
8
- active: z.boolean(),
9
- slug: z.string()
9
+ active: z.boolean()
10
10
  });
11
11
  const Users = z.array(User);
12
12
  const Files = z.array(z.string());
@@ -1 +1 @@
1
- {"version":3,"file":"schema.js","names":[],"sources":["../../../../lib/modules/platform/bitbucket-server/schema.ts"],"sourcesContent":["import { z } from 'zod';\nimport { EmailAddress } from '../../../util/schema-utils/index.ts';\n\nexport const User = z.object({\n displayName: z.string(),\n emailAddress: EmailAddress.catch(''),\n active: z.boolean(),\n slug: z.string(),\n});\n\nexport const Users = z.array(User);\n\nexport const Files = z.array(z.string());\n\nexport const Comment = z.object({\n text: z.string(),\n id: z.number(),\n});\n\nexport type Comment = z.infer<typeof Comment>;\n\nexport const PullRequestMerge = z.object({\n autoMerge: z.boolean().optional(),\n});\n\nexport type PullRequestMerge = z.infer<typeof PullRequestMerge>;\n\nexport const PullRequestCommentActivity = z.object({\n action: z.literal('COMMENTED'),\n commentAction: z.string(),\n comment: Comment,\n});\n\nexport type PullRequestCommentActivity = z.infer<\n typeof PullRequestCommentActivity\n>;\n\nexport const PullRequestActivity = z.union([\n z.object({ action: z.string() }),\n PullRequestCommentActivity,\n]);\n\nexport type PullRequestActivity = z.infer<typeof PullRequestActivity>;\n\nexport const ReviewerGroup = z.object({\n name: z.string(),\n users: z.array(User),\n scope: z.object({\n type: z.union([z.literal('REPOSITORY'), z.literal('PROJECT')]),\n }),\n});\nexport const ReviewerGroups = z.array(ReviewerGroup);\n"],"mappings":";;;;AAGA,MAAa,OAAO,EAAE,OAAO;CAC3B,aAAa,EAAE,QAAQ;CACvB,cAAc,aAAa,MAAM,GAAG;CACpC,QAAQ,EAAE,SAAS;CACnB,MAAM,EAAE,QAAQ;CACjB,CAAC;AAEF,MAAa,QAAQ,EAAE,MAAM,KAAK;AAElC,MAAa,QAAQ,EAAE,MAAM,EAAE,QAAQ,CAAC;AAExC,MAAa,UAAU,EAAE,OAAO;CAC9B,MAAM,EAAE,QAAQ;CAChB,IAAI,EAAE,QAAQ;CACf,CAAC;AAIF,MAAa,mBAAmB,EAAE,OAAO,EACvC,WAAW,EAAE,SAAS,CAAC,UAAU,EAClC,CAAC;AAIF,MAAa,6BAA6B,EAAE,OAAO;CACjD,QAAQ,EAAE,QAAQ,YAAY;CAC9B,eAAe,EAAE,QAAQ;CACzB,SAAS;CACV,CAAC;AAMF,MAAa,sBAAsB,EAAE,MAAM,CACzC,EAAE,OAAO,EAAE,QAAQ,EAAE,QAAQ,EAAE,CAAC,EAChC,2BACD,CAAC;AAIF,MAAa,gBAAgB,EAAE,OAAO;CACpC,MAAM,EAAE,QAAQ;CAChB,OAAO,EAAE,MAAM,KAAK;CACpB,OAAO,EAAE,OAAO,EACd,MAAM,EAAE,MAAM,CAAC,EAAE,QAAQ,aAAa,EAAE,EAAE,QAAQ,UAAU,CAAC,CAAC,EAC/D,CAAC;CACH,CAAC;AACF,MAAa,iBAAiB,EAAE,MAAM,cAAc"}
1
+ {"version":3,"file":"schema.js","names":[],"sources":["../../../../lib/modules/platform/bitbucket-server/schema.ts"],"sourcesContent":["import { z } from 'zod';\nimport { EmailAddress } from '../../../util/schema-utils/index.ts';\n\nexport const User = z.object({\n name: z.string(),\n displayName: z.string(),\n emailAddress: EmailAddress.catch(''),\n active: z.boolean(),\n});\n\nexport const Users = z.array(User);\n\nexport const Files = z.array(z.string());\n\nexport const Comment = z.object({\n text: z.string(),\n id: z.number(),\n});\n\nexport type Comment = z.infer<typeof Comment>;\n\nexport const PullRequestMerge = z.object({\n autoMerge: z.boolean().optional(),\n});\n\nexport type PullRequestMerge = z.infer<typeof PullRequestMerge>;\n\nexport const PullRequestCommentActivity = z.object({\n action: z.literal('COMMENTED'),\n commentAction: z.string(),\n comment: Comment,\n});\n\nexport type PullRequestCommentActivity = z.infer<\n typeof PullRequestCommentActivity\n>;\n\nexport const PullRequestActivity = z.union([\n z.object({ action: z.string() }),\n PullRequestCommentActivity,\n]);\n\nexport type PullRequestActivity = z.infer<typeof PullRequestActivity>;\n\nexport const ReviewerGroup = z.object({\n name: z.string(),\n users: z.array(User),\n scope: z.object({\n type: z.union([z.literal('REPOSITORY'), z.literal('PROJECT')]),\n }),\n});\nexport const ReviewerGroups = z.array(ReviewerGroup);\n"],"mappings":";;;;AAGA,MAAa,OAAO,EAAE,OAAO;CAC3B,MAAM,EAAE,QAAQ;CAChB,aAAa,EAAE,QAAQ;CACvB,cAAc,aAAa,MAAM,GAAG;CACpC,QAAQ,EAAE,SAAS;CACpB,CAAC;AAEF,MAAa,QAAQ,EAAE,MAAM,KAAK;AAElC,MAAa,QAAQ,EAAE,MAAM,EAAE,QAAQ,CAAC;AAExC,MAAa,UAAU,EAAE,OAAO;CAC9B,MAAM,EAAE,QAAQ;CAChB,IAAI,EAAE,QAAQ;CACf,CAAC;AAIF,MAAa,mBAAmB,EAAE,OAAO,EACvC,WAAW,EAAE,SAAS,CAAC,UAAU,EAClC,CAAC;AAIF,MAAa,6BAA6B,EAAE,OAAO;CACjD,QAAQ,EAAE,QAAQ,YAAY;CAC9B,eAAe,EAAE,QAAQ;CACzB,SAAS;CACV,CAAC;AAMF,MAAa,sBAAsB,EAAE,MAAM,CACzC,EAAE,OAAO,EAAE,QAAQ,EAAE,QAAQ,EAAE,CAAC,EAChC,2BACD,CAAC;AAIF,MAAa,gBAAgB,EAAE,OAAO;CACpC,MAAM,EAAE,QAAQ;CAChB,OAAO,EAAE,MAAM,KAAK;CACpB,OAAO,EAAE,OAAO,EACd,MAAM,EAAE,MAAM,CAAC,EAAE,QAAQ,aAAa,EAAE,EAAE,QAAQ,UAAU,CAAC,CAAC,EAC/D,CAAC;CACH,CAAC;AACF,MAAa,iBAAiB,EAAE,MAAM,cAAc"}
@@ -73,7 +73,8 @@ function exec(commandArgument, opts) {
73
73
  const cp = execa(cmd, args, {
74
74
  ...opts,
75
75
  detached: process.platform !== "win32",
76
- shell
76
+ shell,
77
+ extendEnv: false
77
78
  });
78
79
  const [stdout, stderr] = initStreamListeners(cp, {
79
80
  ...opts,
@@ -1 +1 @@
1
- {"version":3,"file":"common.js","names":[],"sources":["../../../lib/util/exec/common.ts"],"sourcesContent":["import type { ChildProcess } from 'node:child_process';\nimport type { Readable } from 'node:stream';\nimport { isNullOrUndefined } from '@sindresorhus/is';\nimport { execa } from 'execa';\nimport { join, split } from 'shlex';\nimport { instrument } from '../../instrumentation/index.ts';\nimport { logger } from '../../logger/index.ts';\nimport { getEnv } from '../env.ts';\nimport { sanitize } from '../sanitize.ts';\nimport type { ExecErrorData } from './exec-error.ts';\nimport { ExecError } from './exec-error.ts';\nimport type {\n CommandWithOptions,\n DataListener,\n ExecResult,\n RawExecOptions,\n} from './types.ts';\nimport { asRawCommand, isCommandWithOptions } from './utils.ts';\n\n// https://man7.org/linux/man-pages/man7/signal.7.html#NAME\n// Non TERM/CORE signals\n// The following is step 3. in https://github.com/renovatebot/renovate/issues/16197#issuecomment-1171423890\nconst NONTERM = [\n 'SIGCHLD',\n 'SIGCLD',\n 'SIGCONT',\n 'SIGSTOP',\n 'SIGTSTP',\n 'SIGTTIN',\n 'SIGTTOU',\n 'SIGURG',\n 'SIGWINCH',\n];\n\nconst encoding = 'utf8';\n\nfunction stringify(list: Buffer[]): string {\n return Buffer.concat(list).toString(encoding);\n}\n\nfunction initStreamListeners(\n cp: ChildProcess,\n opts: RawExecOptions & { maxBuffer: number },\n): [Buffer[], Buffer[]] {\n const stdout: Buffer[] = [];\n const stderr: Buffer[] = [];\n let stdoutLen = 0;\n let stderrLen = 0;\n\n registerDataListeners(cp.stdout, opts.outputListeners?.stdout);\n registerDataListeners(cp.stderr, opts.outputListeners?.stderr);\n\n cp.stdout?.on('data', (chunk: Buffer) => {\n // process.stdout.write(data.toString());\n const len = Buffer.byteLength(chunk, encoding);\n stdoutLen += len;\n if (stdoutLen > opts.maxBuffer) {\n cp.emit('error', new Error('stdout maxBuffer exceeded'));\n } else {\n stdout.push(chunk);\n }\n });\n\n cp.stderr?.on('data', (chunk: Buffer) => {\n // process.stderr.write(data.toString());\n const len = Buffer.byteLength(chunk, encoding);\n stderrLen += len;\n if (stderrLen > opts.maxBuffer) {\n cp.emit('error', new Error('stderr maxBuffer exceeded'));\n } else {\n stderr.push(chunk);\n }\n });\n return [stdout, stderr];\n}\n\nfunction registerDataListeners(\n readable: Readable | null,\n dataListeners: DataListener[] | undefined,\n): void {\n if (isNullOrUndefined(readable) || isNullOrUndefined(dataListeners)) {\n return;\n }\n\n for (const listener of dataListeners) {\n readable.on('data', listener);\n }\n}\n\nexport function exec(\n commandArgument: string | CommandWithOptions,\n opts: RawExecOptions,\n): Promise<ExecResult> {\n let theCmd = commandArgument;\n let ignoreFailure = false;\n if (isCommandWithOptions(commandArgument)) {\n theCmd = join(commandArgument.command);\n if (commandArgument.ignoreFailure !== undefined) {\n ignoreFailure = commandArgument.ignoreFailure;\n }\n }\n\n return new Promise((resolve, reject) => {\n let cmd = asRawCommand(theCmd);\n let args: string[] = [];\n const maxBuffer = opts.maxBuffer ?? 10 * 1024 * 1024; // Set default max buffer size to 10MB\n\n // don't use shell by default, as it leads to potential security issues\n let shell = opts.shell ?? false;\n if (\n isCommandWithOptions(commandArgument) &&\n commandArgument.shell !== undefined\n ) {\n shell = commandArgument.shell;\n }\n\n // if we're not in shell mode, we need to provide the command and arguments\n if (shell === false) {\n const parts = split(cmd);\n // v8 ignore else -- TODO: add test #40625\n if (parts) {\n cmd = parts[0];\n args = parts.slice(1);\n }\n }\n\n const cp = execa(cmd, args, {\n ...opts,\n // force detached on non WIN platforms\n // https://github.com/nodejs/node/issues/21825#issuecomment-611328888\n detached: process.platform !== 'win32',\n shell,\n });\n\n // handle streams\n const [stdout, stderr] = initStreamListeners(cp, {\n ...opts,\n maxBuffer,\n });\n\n // handle process events\n void cp.on('error', (error) => {\n kill(cp, 'SIGTERM');\n // rethrowing, use originally emitted error message\n reject(new ExecError(error.message, rejectInfo(), error));\n });\n\n void cp.on('exit', (code: number, signal: NodeJS.Signals) => {\n if (NONTERM.includes(signal)) {\n return;\n }\n if (signal) {\n kill(cp, signal);\n reject(\n new ExecError(\n `Command failed: ${cp.spawnargs.join(' ')}\\nInterrupted by ${signal}`,\n {\n ...rejectInfo(),\n signal,\n },\n ),\n );\n return;\n }\n if (code !== 0) {\n if (ignoreFailure === undefined || ignoreFailure === false) {\n reject(\n new ExecError(\n `Command failed: ${cp.spawnargs.join(' ')}\\n${stringify(stderr)}`,\n {\n ...rejectInfo(),\n exitCode: code,\n },\n ),\n );\n return;\n }\n\n logger.once.debug(\n {\n command: cp.spawnargs.join(' '),\n stdout: stringify(stdout),\n stderr: stringify(stderr),\n exitCode: code,\n },\n `Ignoring failure to execute comamnd \\`${cp.spawnargs.join(' ')}\\`, as ignoreFailure=true is set`,\n );\n\n resolve({\n stderr: stringify(stderr),\n stdout: stringify(stdout),\n exitCode: code,\n });\n return;\n }\n resolve({\n stderr: stringify(stderr),\n stdout: stringify(stdout),\n });\n });\n\n function rejectInfo(): ExecErrorData {\n return {\n cmd: cp.spawnargs.join(' '),\n options: opts,\n stdout: stringify(stdout),\n stderr: stringify(stderr),\n };\n }\n });\n}\n\nfunction kill(cp: ChildProcess, signal: NodeJS.Signals): boolean {\n try {\n if (cp.pid && getEnv().RENOVATE_X_EXEC_GPID_HANDLE) {\n /**\n * If `pid` is negative, but not `-1`, signal shall be sent to all processes\n * (excluding an unspecified set of system processes),\n * whose process group ID (pgid) is equal to the absolute value of pid,\n * and for which the process has permission to send a signal.\n */\n return process.kill(-cp.pid, signal);\n } else {\n // destroying stdio is needed for unref to work\n // https://nodejs.org/api/child_process.html#subprocessunref\n // https://github.com/nodejs/node/blob/4d5ff25a813fd18939c9f76b17e36291e3ea15c3/lib/child_process.js#L412-L426\n cp.stderr?.destroy();\n cp.stdout?.destroy();\n cp.unref();\n return cp.kill(signal);\n }\n } catch {\n // cp is a single node tree, therefore -pid is invalid as there is no such pgid,\n return false;\n }\n}\n\nexport const rawExec: (\n cmd: string | CommandWithOptions,\n opts: RawExecOptions,\n) => Promise<ExecResult> = (\n cmd: string | CommandWithOptions,\n opts: RawExecOptions,\n) =>\n instrument(`rawExec: ${sanitize(asRawCommand(cmd))}`, () => exec(cmd, opts));\n"],"mappings":"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"}
1
+ {"version":3,"file":"common.js","names":[],"sources":["../../../lib/util/exec/common.ts"],"sourcesContent":["import type { ChildProcess } from 'node:child_process';\nimport type { Readable } from 'node:stream';\nimport { isNullOrUndefined } from '@sindresorhus/is';\nimport { execa } from 'execa';\nimport { join, split } from 'shlex';\nimport { instrument } from '../../instrumentation/index.ts';\nimport { logger } from '../../logger/index.ts';\nimport { getEnv } from '../env.ts';\nimport { sanitize } from '../sanitize.ts';\nimport type { ExecErrorData } from './exec-error.ts';\nimport { ExecError } from './exec-error.ts';\nimport type {\n CommandWithOptions,\n DataListener,\n ExecResult,\n RawExecOptions,\n} from './types.ts';\nimport { asRawCommand, isCommandWithOptions } from './utils.ts';\n\n// https://man7.org/linux/man-pages/man7/signal.7.html#NAME\n// Non TERM/CORE signals\n// The following is step 3. in https://github.com/renovatebot/renovate/issues/16197#issuecomment-1171423890\nconst NONTERM = [\n 'SIGCHLD',\n 'SIGCLD',\n 'SIGCONT',\n 'SIGSTOP',\n 'SIGTSTP',\n 'SIGTTIN',\n 'SIGTTOU',\n 'SIGURG',\n 'SIGWINCH',\n];\n\nconst encoding = 'utf8';\n\nfunction stringify(list: Buffer[]): string {\n return Buffer.concat(list).toString(encoding);\n}\n\nfunction initStreamListeners(\n cp: ChildProcess,\n opts: RawExecOptions & { maxBuffer: number },\n): [Buffer[], Buffer[]] {\n const stdout: Buffer[] = [];\n const stderr: Buffer[] = [];\n let stdoutLen = 0;\n let stderrLen = 0;\n\n registerDataListeners(cp.stdout, opts.outputListeners?.stdout);\n registerDataListeners(cp.stderr, opts.outputListeners?.stderr);\n\n cp.stdout?.on('data', (chunk: Buffer) => {\n // process.stdout.write(data.toString());\n const len = Buffer.byteLength(chunk, encoding);\n stdoutLen += len;\n if (stdoutLen > opts.maxBuffer) {\n cp.emit('error', new Error('stdout maxBuffer exceeded'));\n } else {\n stdout.push(chunk);\n }\n });\n\n cp.stderr?.on('data', (chunk: Buffer) => {\n // process.stderr.write(data.toString());\n const len = Buffer.byteLength(chunk, encoding);\n stderrLen += len;\n if (stderrLen > opts.maxBuffer) {\n cp.emit('error', new Error('stderr maxBuffer exceeded'));\n } else {\n stderr.push(chunk);\n }\n });\n return [stdout, stderr];\n}\n\nfunction registerDataListeners(\n readable: Readable | null,\n dataListeners: DataListener[] | undefined,\n): void {\n if (isNullOrUndefined(readable) || isNullOrUndefined(dataListeners)) {\n return;\n }\n\n for (const listener of dataListeners) {\n readable.on('data', listener);\n }\n}\n\nexport function exec(\n commandArgument: string | CommandWithOptions,\n opts: RawExecOptions,\n): Promise<ExecResult> {\n let theCmd = commandArgument;\n let ignoreFailure = false;\n if (isCommandWithOptions(commandArgument)) {\n theCmd = join(commandArgument.command);\n if (commandArgument.ignoreFailure !== undefined) {\n ignoreFailure = commandArgument.ignoreFailure;\n }\n }\n\n return new Promise((resolve, reject) => {\n let cmd = asRawCommand(theCmd);\n let args: string[] = [];\n const maxBuffer = opts.maxBuffer ?? 10 * 1024 * 1024; // Set default max buffer size to 10MB\n\n // don't use shell by default, as it leads to potential security issues\n let shell = opts.shell ?? false;\n if (\n isCommandWithOptions(commandArgument) &&\n commandArgument.shell !== undefined\n ) {\n shell = commandArgument.shell;\n }\n\n // if we're not in shell mode, we need to provide the command and arguments\n if (shell === false) {\n const parts = split(cmd);\n // v8 ignore else -- TODO: add test #40625\n if (parts) {\n cmd = parts[0];\n args = parts.slice(1);\n }\n }\n\n const cp = execa(cmd, args, {\n ...opts,\n // force detached on non WIN platforms\n // https://github.com/nodejs/node/issues/21825#issuecomment-611328888\n detached: process.platform !== 'win32',\n shell,\n extendEnv: false,\n });\n\n // handle streams\n const [stdout, stderr] = initStreamListeners(cp, {\n ...opts,\n maxBuffer,\n });\n\n // handle process events\n void cp.on('error', (error) => {\n kill(cp, 'SIGTERM');\n // rethrowing, use originally emitted error message\n reject(new ExecError(error.message, rejectInfo(), error));\n });\n\n void cp.on('exit', (code: number, signal: NodeJS.Signals) => {\n if (NONTERM.includes(signal)) {\n return;\n }\n if (signal) {\n kill(cp, signal);\n reject(\n new ExecError(\n `Command failed: ${cp.spawnargs.join(' ')}\\nInterrupted by ${signal}`,\n {\n ...rejectInfo(),\n signal,\n },\n ),\n );\n return;\n }\n if (code !== 0) {\n if (ignoreFailure === undefined || ignoreFailure === false) {\n reject(\n new ExecError(\n `Command failed: ${cp.spawnargs.join(' ')}\\n${stringify(stderr)}`,\n {\n ...rejectInfo(),\n exitCode: code,\n },\n ),\n );\n return;\n }\n\n logger.once.debug(\n {\n command: cp.spawnargs.join(' '),\n stdout: stringify(stdout),\n stderr: stringify(stderr),\n exitCode: code,\n },\n `Ignoring failure to execute comamnd \\`${cp.spawnargs.join(' ')}\\`, as ignoreFailure=true is set`,\n );\n\n resolve({\n stderr: stringify(stderr),\n stdout: stringify(stdout),\n exitCode: code,\n });\n return;\n }\n resolve({\n stderr: stringify(stderr),\n stdout: stringify(stdout),\n });\n });\n\n function rejectInfo(): ExecErrorData {\n return {\n cmd: cp.spawnargs.join(' '),\n options: opts,\n stdout: stringify(stdout),\n stderr: stringify(stderr),\n };\n }\n });\n}\n\nfunction kill(cp: ChildProcess, signal: NodeJS.Signals): boolean {\n try {\n if (cp.pid && getEnv().RENOVATE_X_EXEC_GPID_HANDLE) {\n /**\n * If `pid` is negative, but not `-1`, signal shall be sent to all processes\n * (excluding an unspecified set of system processes),\n * whose process group ID (pgid) is equal to the absolute value of pid,\n * and for which the process has permission to send a signal.\n */\n return process.kill(-cp.pid, signal);\n } else {\n // destroying stdio is needed for unref to work\n // https://nodejs.org/api/child_process.html#subprocessunref\n // https://github.com/nodejs/node/blob/4d5ff25a813fd18939c9f76b17e36291e3ea15c3/lib/child_process.js#L412-L426\n cp.stderr?.destroy();\n cp.stdout?.destroy();\n cp.unref();\n return cp.kill(signal);\n }\n } catch {\n // cp is a single node tree, therefore -pid is invalid as there is no such pgid,\n return false;\n }\n}\n\nexport const rawExec: (\n cmd: string | CommandWithOptions,\n opts: RawExecOptions,\n) => Promise<ExecResult> = (\n cmd: string | CommandWithOptions,\n opts: RawExecOptions,\n) =>\n instrument(`rawExec: ${sanitize(asRawCommand(cmd))}`, () => exec(cmd, opts));\n"],"mappings":";;;;;;;;;;;AAsBA,MAAM,UAAU;CACd;CACA;CACA;CACA;CACA;CACA;CACA;CACA;CACA;CACD;AAED,MAAM,WAAW;AAEjB,SAAS,UAAU,MAAwB;AACzC,QAAO,OAAO,OAAO,KAAK,CAAC,SAAS,SAAS;;AAG/C,SAAS,oBACP,IACA,MACsB;CACtB,MAAM,SAAmB,EAAE;CAC3B,MAAM,SAAmB,EAAE;CAC3B,IAAI,YAAY;CAChB,IAAI,YAAY;AAEhB,uBAAsB,GAAG,QAAQ,KAAK,iBAAiB,OAAO;AAC9D,uBAAsB,GAAG,QAAQ,KAAK,iBAAiB,OAAO;AAE9D,IAAG,QAAQ,GAAG,SAAS,UAAkB;EAEvC,MAAM,MAAM,OAAO,WAAW,OAAO,SAAS;AAC9C,eAAa;AACb,MAAI,YAAY,KAAK,UACnB,IAAG,KAAK,yBAAS,IAAI,MAAM,4BAA4B,CAAC;MAExD,QAAO,KAAK,MAAM;GAEpB;AAEF,IAAG,QAAQ,GAAG,SAAS,UAAkB;EAEvC,MAAM,MAAM,OAAO,WAAW,OAAO,SAAS;AAC9C,eAAa;AACb,MAAI,YAAY,KAAK,UACnB,IAAG,KAAK,yBAAS,IAAI,MAAM,4BAA4B,CAAC;MAExD,QAAO,KAAK,MAAM;GAEpB;AACF,QAAO,CAAC,QAAQ,OAAO;;AAGzB,SAAS,sBACP,UACA,eACM;AACN,KAAI,kBAAkB,SAAS,IAAI,kBAAkB,cAAc,CACjE;AAGF,MAAK,MAAM,YAAY,cACrB,UAAS,GAAG,QAAQ,SAAS;;AAIjC,SAAgB,KACd,iBACA,MACqB;CACrB,IAAI,SAAS;CACb,IAAI,gBAAgB;AACpB,KAAI,qBAAqB,gBAAgB,EAAE;AACzC,WAAS,KAAK,gBAAgB,QAAQ;AACtC,MAAI,gBAAgB,kBAAkB,OACpC,iBAAgB,gBAAgB;;AAIpC,QAAO,IAAI,SAAS,SAAS,WAAW;EACtC,IAAI,MAAM,aAAa,OAAO;EAC9B,IAAI,OAAiB,EAAE;EACvB,MAAM,YAAY,KAAK,aAAa,KAAK,OAAO;EAGhD,IAAI,QAAQ,KAAK,SAAS;AAC1B,MACE,qBAAqB,gBAAgB,IACrC,gBAAgB,UAAU,OAE1B,SAAQ,gBAAgB;AAI1B,MAAI,UAAU,OAAO;GACnB,MAAM,QAAQ,MAAM,IAAI;;AAExB,OAAI,OAAO;AACT,UAAM,MAAM;AACZ,WAAO,MAAM,MAAM,EAAE;;;EAIzB,MAAM,KAAK,MAAM,KAAK,MAAM;GAC1B,GAAG;GAGH,UAAU,QAAQ,aAAa;GAC/B;GACA,WAAW;GACZ,CAAC;EAGF,MAAM,CAAC,QAAQ,UAAU,oBAAoB,IAAI;GAC/C,GAAG;GACH;GACD,CAAC;AAGF,EAAK,GAAG,GAAG,UAAU,UAAU;AAC7B,QAAK,IAAI,UAAU;AAEnB,UAAO,IAAI,UAAU,MAAM,SAAS,YAAY,EAAE,MAAM,CAAC;IACzD;AAEF,EAAK,GAAG,GAAG,SAAS,MAAc,WAA2B;AAC3D,OAAI,QAAQ,SAAS,OAAO,CAC1B;AAEF,OAAI,QAAQ;AACV,SAAK,IAAI,OAAO;AAChB,WACE,IAAI,UACF,mBAAmB,GAAG,UAAU,KAAK,IAAI,CAAC,mBAAmB,UAC7D;KACE,GAAG,YAAY;KACf;KACD,CACF,CACF;AACD;;AAEF,OAAI,SAAS,GAAG;AACd,QAAI,kBAAkB,UAAa,kBAAkB,OAAO;AAC1D,YACE,IAAI,UACF,mBAAmB,GAAG,UAAU,KAAK,IAAI,CAAC,IAAI,UAAU,OAAO,IAC/D;MACE,GAAG,YAAY;MACf,UAAU;MACX,CACF,CACF;AACD;;AAGF,WAAO,KAAK,MACV;KACE,SAAS,GAAG,UAAU,KAAK,IAAI;KAC/B,QAAQ,UAAU,OAAO;KACzB,QAAQ,UAAU,OAAO;KACzB,UAAU;KACX,EACD,yCAAyC,GAAG,UAAU,KAAK,IAAI,CAAC,kCACjE;AAED,YAAQ;KACN,QAAQ,UAAU,OAAO;KACzB,QAAQ,UAAU,OAAO;KACzB,UAAU;KACX,CAAC;AACF;;AAEF,WAAQ;IACN,QAAQ,UAAU,OAAO;IACzB,QAAQ,UAAU,OAAO;IAC1B,CAAC;IACF;EAEF,SAAS,aAA4B;AACnC,UAAO;IACL,KAAK,GAAG,UAAU,KAAK,IAAI;IAC3B,SAAS;IACT,QAAQ,UAAU,OAAO;IACzB,QAAQ,UAAU,OAAO;IAC1B;;GAEH;;AAGJ,SAAS,KAAK,IAAkB,QAAiC;AAC/D,KAAI;AACF,MAAI,GAAG,OAAO,QAAQ,CAAC;;;;;;;AAOrB,SAAO,QAAQ,KAAK,CAAC,GAAG,KAAK,OAAO;OAC/B;AAIL,MAAG,QAAQ,SAAS;AACpB,MAAG,QAAQ,SAAS;AACpB,MAAG,OAAO;AACV,UAAO,GAAG,KAAK,OAAO;;SAElB;AAEN,SAAO;;;AAIX,MAAa,WAIX,KACA,SAEA,WAAW,YAAY,SAAS,aAAa,IAAI,CAAC,UAAU,KAAK,KAAK,KAAK,CAAC"}
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.4.2",
4
+ "version": "43.4.4",
5
5
  "type": "module",
6
6
  "bin": {
7
7
  "renovate": "dist/renovate.js",
@@ -218,9 +218,9 @@
218
218
  "devDependencies": {
219
219
  "@biomejs/biome": "2.3.13",
220
220
  "@commander-js/extra-typings": "14.0.0",
221
- "@containerbase/eslint-plugin": "1.1.30",
222
- "@containerbase/istanbul-reports-html": "1.1.28",
223
- "@containerbase/semantic-release-pnpm": "1.3.18",
221
+ "@containerbase/eslint-plugin": "1.1.31",
222
+ "@containerbase/istanbul-reports-html": "1.1.30",
223
+ "@containerbase/semantic-release-pnpm": "1.3.20",
224
224
  "@eslint/js": "9.39.2",
225
225
  "@hyrious/marshal": "0.3.3",
226
226
  "@ls-lint/ls-lint": "2.3.1",
@@ -281,7 +281,7 @@
281
281
  "eslint-plugin-oxlint": "1.42.0",
282
282
  "eslint-plugin-promise": "7.2.1",
283
283
  "expect-more-jest": "5.5.0",
284
- "globals": "17.2.0",
284
+ "globals": "17.3.0",
285
285
  "graphql": "16.12.0",
286
286
  "husky": "9.1.7",
287
287
  "jest-extended": "7.0.0",
@@ -299,7 +299,7 @@
299
299
  "tmp-promise": "3.0.3",
300
300
  "tsdown": "0.20.1",
301
301
  "tsx": "4.21.0",
302
- "type-fest": "5.4.2",
302
+ "type-fest": "5.4.3",
303
303
  "typescript": "5.9.3",
304
304
  "typescript-eslint": "8.54.0",
305
305
  "unified": "11.0.5",
@@ -1,7 +1,7 @@
1
1
  {
2
- "title": "JSON schema for Renovate 43.4.2 config files (https://renovatebot.com/)",
2
+ "title": "JSON schema for Renovate 43.4.4 config files (https://renovatebot.com/)",
3
3
  "$schema": "http://json-schema.org/draft-07/schema#",
4
- "x-renovate-version": "43.4.2",
4
+ "x-renovate-version": "43.4.4",
5
5
  "allowComments": true,
6
6
  "type": "object",
7
7
  "properties": {