diff --git a/docs/usage/self-hosted-configuration.md b/docs/usage/self-hosted-configuration.md index c7f0c0e6a7835948737a1645ca363cfb2ba7d58a..138d885a4d628a3d13faafa143d1d5a460feaa0b 100644 --- a/docs/usage/self-hosted-configuration.md +++ b/docs/usage/self-hosted-configuration.md @@ -213,6 +213,21 @@ For example: } ``` +## autodiscoverProjects + +You can use this option to filter the list of autodiscovered repositories by project names. +This feature is useful for users who want Renovate to only work on repositories within specific projects or exclude certain repositories from being processed. + +```json title="Example for Bitbucket" +{ + "platform": "bitbucket", + "autodiscoverProjects": ["a-group", "!another-group/some-subgroup"] +} +``` + +The `autodiscoverProjects` config option takes an array of minimatch-compatible globs or RE2-compatible regex strings. +For more details on this syntax see Renovate's [string pattern matching documentation](./string-pattern-matching.md). + ## autodiscoverTopics Some platforms allow you to add tags, or topics, to repositories and retrieve repository lists by specifying those diff --git a/lib/config/options/index.ts b/lib/config/options/index.ts index 447610d8a18f84ac3d2c836b1a4e4d6ff94c1e92..ca6a436fc8c186a3b0d9be181abc5c52655f4829 100644 --- a/lib/config/options/index.ts +++ b/lib/config/options/index.ts @@ -882,6 +882,17 @@ const options: RenovateOptions[] = [ globalOnly: true, supportedPlatforms: ['gitlab'], }, + { + name: 'autodiscoverProjects', + description: + 'Filter the list of autodiscovered repositories by project names.', + stage: 'global', + type: 'array', + subType: 'string', + default: null, + globalOnly: true, + supportedPlatforms: ['bitbucket'], + }, { name: 'autodiscoverTopics', description: 'Filter the list of autodiscovered repositories by topics.', diff --git a/lib/config/types.ts b/lib/config/types.ts index bb5882b981f613e70fcb2b687b71426a140bf845..ae3c2364edb46bd64a72571b77b55bc5989b43d5 100644 --- a/lib/config/types.ts +++ b/lib/config/types.ts @@ -100,6 +100,7 @@ export interface GlobalOnlyConfig { autodiscover?: boolean; autodiscoverFilter?: string[] | string; autodiscoverNamespaces?: string[]; + autodiscoverProjects?: string[]; autodiscoverTopics?: string[]; baseDir?: string; cacheDir?: string; diff --git a/lib/modules/platform/bitbucket/index.spec.ts b/lib/modules/platform/bitbucket/index.spec.ts index e958ea7c07d33965a56507618acde3aca1ddebf4..9d66ff1757b21bc6e0b62ee278355ac5f5658f7d 100644 --- a/lib/modules/platform/bitbucket/index.spec.ts +++ b/lib/modules/platform/bitbucket/index.spec.ts @@ -133,9 +133,37 @@ describe('modules/platform/bitbucket/index', () => { .reply(200, { values: [{ full_name: 'foo/bar' }, { full_name: 'some/repo' }], }); - const res = await bitbucket.getRepos(); + const res = await bitbucket.getRepos({}); expect(res).toEqual(['foo/bar', 'some/repo']); }); + + it('filters repos based on autodiscoverProjects patterns', async () => { + httpMock + .scope(baseUrl) + .get('/2.0/repositories?role=contributor&pagelen=100') + .reply(200, { + values: [ + { full_name: 'foo/bar', project: { name: 'ignore' } }, + { full_name: 'some/repo', project: { name: 'allow' } }, + ], + }); + const res = await bitbucket.getRepos({ projects: ['allow'] }); + expect(res).toEqual(['some/repo']); + }); + + it('filters repos based on autodiscoverProjects patterns with negation', async () => { + httpMock + .scope(baseUrl) + .get('/2.0/repositories?role=contributor&pagelen=100') + .reply(200, { + values: [ + { full_name: 'foo/bar', project: { name: 'ignore' } }, + { full_name: 'some/repo', project: { name: 'allow' } }, + ], + }); + const res = await bitbucket.getRepos({ projects: ['!ignore'] }); + expect(res).toEqual(['some/repo']); + }); }); describe('initRepo()', () => { diff --git a/lib/modules/platform/bitbucket/index.ts b/lib/modules/platform/bitbucket/index.ts index dc6301ae89d5a1daa507de82ca37ea19648cafc3..a2eef445a4d06717552a978623d1f6a2bd72f166 100644 --- a/lib/modules/platform/bitbucket/index.ts +++ b/lib/modules/platform/bitbucket/index.ts @@ -10,8 +10,9 @@ import { BitbucketHttp, setBaseUrl } from '../../../util/http/bitbucket'; import type { HttpOptions } from '../../../util/http/types'; import { regEx } from '../../../util/regex'; import { sanitize } from '../../../util/sanitize'; -import { UUIDRegex } from '../../../util/string-match'; +import { UUIDRegex, matchRegexOrGlobList } from '../../../util/string-match'; import type { + AutodiscoverConfig, BranchStatusConfig, CreatePRConfig, EnsureCommentConfig, @@ -113,10 +114,10 @@ export async function initPlatform({ } // Get all repositories that the user has access to -export async function getRepos(): Promise<string[]> { +export async function getRepos(config: AutodiscoverConfig): Promise<string[]> { logger.debug('Autodiscovering Bitbucket Cloud repositories'); try { - const repos = ( + let repos = ( await bitbucketHttp.getJson<PagedResult<RepoInfoBody>>( `/2.0/repositories/?role=contributor`, { @@ -124,6 +125,20 @@ export async function getRepos(): Promise<string[]> { }, ) ).body.values; + + // if autodiscoverProjects is configured + // filter the repos list + const autodiscoverProjects = config.projects; + if (is.nonEmptyArray(autodiscoverProjects)) { + logger.debug( + { autodiscoverProjects: config.projects }, + 'Applying autodiscoverProjects filter', + ); + repos = repos.filter((repo) => + matchRegexOrGlobList(repo.project.name, autodiscoverProjects), + ); + } + return repos.map((repo) => repo.full_name); } catch (err) /* istanbul ignore next */ { logger.error({ err }, `bitbucket getRepos error`); diff --git a/lib/modules/platform/bitbucket/types.ts b/lib/modules/platform/bitbucket/types.ts index 35fba3d1638a5e01c21ed723a83b4595b3a462b6..fb4638179e0106b6cffdf307fa43f03145ed25ca 100644 --- a/lib/modules/platform/bitbucket/types.ts +++ b/lib/modules/platform/bitbucket/types.ts @@ -66,6 +66,9 @@ export interface RepoInfoBody { uuid: string; full_name: string; is_private: boolean; + project: { + name: string; + }; } export interface PrResponse { diff --git a/lib/modules/platform/types.ts b/lib/modules/platform/types.ts index eff914fb7d4d0c76c659cc8bab40339cc546032d..b6f75f2a58588f95f87d87030a16d2a70bd27eab 100644 --- a/lib/modules/platform/types.ts +++ b/lib/modules/platform/types.ts @@ -178,6 +178,7 @@ export interface AutodiscoverConfig { topics?: string[]; includeMirrors?: boolean; namespaces?: string[]; + projects?: string[]; } export interface Platform { diff --git a/lib/workers/global/autodiscover.ts b/lib/workers/global/autodiscover.ts index ed74f350f3d95ecfdfd4ef576311f9946221f194..e6e633df551a9bd9402f88965303c5804ca09a37 100644 --- a/lib/workers/global/autodiscover.ts +++ b/lib/workers/global/autodiscover.ts @@ -40,6 +40,7 @@ export async function autodiscoverRepositories( topics: config.autodiscoverTopics, includeMirrors: config.includeMirrors, namespaces: config.autodiscoverNamespaces, + projects: config.autodiscoverProjects, }); if (!discovered?.length) { // Soft fail (no error thrown) if no accessible repositories