@backstage/plugin-catalog-backend-module-gitlab 0.1.14 → 0.2.0-next.0

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.
package/CHANGELOG.md CHANGED
@@ -1,5 +1,34 @@
1
1
  # @backstage/plugin-catalog-backend-module-gitlab
2
2
 
3
+ ## 0.2.0-next.0
4
+
5
+ ### Minor Changes
6
+
7
+ - f64345108a0: **BREAKING**: The configuration of the `GitlabDiscoveryEntityProvider` has changed as follows:
8
+
9
+ - The configuration key `branch` is now used to define the branch from which the catalog-info should be discovered.
10
+ - The old configuration key `branch` is now called `fallbackBranch`. This value specifies which branch should be used
11
+ if no default branch is defined on the project itself.
12
+
13
+ To migrate to the new configuration value, rename `branch` to `fallbackBranch`
14
+
15
+ ### Patch Changes
16
+
17
+ - 7b1b7bfdb7b: The gitlab org data integration now makes use of the GraphQL API to determine
18
+ the relationships between imported User and Group entities, effectively making
19
+ this integration usable without an administrator account's Personal Access
20
+ Token.
21
+ - Updated dependencies
22
+ - @backstage/backend-common@0.18.4-next.0
23
+ - @backstage/config@1.0.7
24
+ - @backstage/integration@1.4.3
25
+ - @backstage/backend-plugin-api@0.5.1-next.0
26
+ - @backstage/backend-tasks@0.5.1-next.0
27
+ - @backstage/catalog-model@1.2.1
28
+ - @backstage/errors@1.1.5
29
+ - @backstage/types@1.0.2
30
+ - @backstage/plugin-catalog-node@1.3.5-next.0
31
+
3
32
  ## 0.1.14
4
33
 
5
34
  ### Patch Changes
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@backstage/plugin-catalog-backend-module-gitlab",
3
- "version": "0.1.14",
3
+ "version": "0.2.0-next.0",
4
4
  "main": "../dist/alpha.cjs.js",
5
5
  "types": "../dist/alpha.d.ts"
6
6
  }
package/dist/alpha.cjs.js CHANGED
@@ -60,36 +60,59 @@ class GitLabClient {
60
60
  return this.pagedRequest(`/projects`, options);
61
61
  }
62
62
  async listUsers(options) {
63
- let requestOptions = options;
64
- if (!requestOptions) {
65
- requestOptions = {};
66
- }
67
- requestOptions.without_project_bots = true;
68
- requestOptions.exclude_internal = true;
69
- return this.pagedRequest(`/users?`, requestOptions);
63
+ return this.pagedRequest(`/users?`, {
64
+ ...options,
65
+ without_project_bots: true,
66
+ exclude_internal: true
67
+ });
70
68
  }
71
69
  async listGroups(options) {
72
70
  return this.pagedRequest(`/groups`, options);
73
71
  }
74
- async getUserMemberships(userId) {
75
- const endpoint = `/users/${encodeURIComponent(userId)}/memberships`;
76
- const request = new URL(`${this.config.apiBaseUrl}${endpoint}`);
77
- request.searchParams.append("per_page", "100");
78
- const response = await fetch__default["default"](request.toString(), {
79
- headers: integration.getGitLabRequestOptions(this.config).headers,
80
- method: "GET"
81
- });
82
- if (!response.ok) {
83
- if (response.status >= 500) {
84
- this.logger.debug(
85
- `Unexpected response when fetching ${request.toString()}. Expected 200 but got ${response.status} - ${response.statusText}`
86
- );
72
+ async getGroupMembers(groupPath) {
73
+ const memberIds = [];
74
+ let hasNextPage = false;
75
+ let endCursor = null;
76
+ do {
77
+ const response = await fetch__default["default"](
78
+ `${this.config.baseUrl}/api/graphql`,
79
+ {
80
+ method: "POST",
81
+ headers: {
82
+ ...integration.getGitLabRequestOptions(this.config).headers,
83
+ ["Content-Type"]: "application/json"
84
+ },
85
+ body: JSON.stringify({
86
+ variables: { group: groupPath, endCursor },
87
+ query: `query($group: ID!, $endCursor: String) {
88
+ group(fullPath: $group) {
89
+ groupMembers(first: 100, relations: [DIRECT], after: $endCursor) {
90
+ nodes {
91
+ user {
92
+ id
93
+ }
94
+ }
95
+ pageInfo {
96
+ endCursor
97
+ hasNextPage
98
+ }
99
+ }
100
+ }
101
+ }`
102
+ })
103
+ }
104
+ ).then((r) => r.json());
105
+ if (response.errors) {
106
+ throw new Error(`GraphQL errors: ${JSON.stringify(response.errors)}`);
87
107
  }
88
- return [];
89
- }
90
- return response.json().then((items) => {
91
- return items;
92
- });
108
+ memberIds.push(
109
+ ...response.data.group.groupMembers.nodes.map(
110
+ (node) => Number(node.user.id.replace(/^gid:\/\/gitlab\/User\//, ""))
111
+ )
112
+ );
113
+ ({ hasNextPage, endCursor } = response.data.group.groupMembers.pageInfo);
114
+ } while (hasNextPage);
115
+ return memberIds;
93
116
  }
94
117
  /**
95
118
  * General existence check.
@@ -167,32 +190,28 @@ async function* paginated(request, options) {
167
190
  } while (res.nextPage);
168
191
  }
169
192
 
170
- function readGitlabConfig(id, config, logger) {
171
- var _a, _b, _c, _d, _e, _f, _g, _h;
193
+ function readGitlabConfig(id, config) {
194
+ var _a, _b, _c, _d, _e, _f, _g;
172
195
  const group = (_a = config.getOptionalString("group")) != null ? _a : "";
173
196
  const host = config.getString("host");
174
197
  const branch = config.getOptionalString("branch");
175
- if (branch) {
176
- logger.warn(
177
- "The configuration key `branch` has been deprecated in favor of the configuration key `fallbackBranch`."
178
- );
179
- }
180
- const fallbackBranch = (_c = (_b = config.getOptionalString("fallbackBranch")) != null ? _b : branch) != null ? _c : "master";
181
- const catalogFile = (_d = config.getOptionalString("entityFilename")) != null ? _d : "catalog-info.yaml";
198
+ const fallbackBranch = (_b = config.getOptionalString("fallbackBranch")) != null ? _b : "master";
199
+ const catalogFile = (_c = config.getOptionalString("entityFilename")) != null ? _c : "catalog-info.yaml";
182
200
  const projectPattern = new RegExp(
183
- (_e = config.getOptionalString("projectPattern")) != null ? _e : /[\s\S]*/
201
+ (_d = config.getOptionalString("projectPattern")) != null ? _d : /[\s\S]*/
184
202
  );
185
203
  const userPattern = new RegExp(
186
- (_f = config.getOptionalString("userPattern")) != null ? _f : /[\s\S]*/
204
+ (_e = config.getOptionalString("userPattern")) != null ? _e : /[\s\S]*/
187
205
  );
188
206
  const groupPattern = new RegExp(
189
- (_g = config.getOptionalString("groupPattern")) != null ? _g : /[\s\S]*/
207
+ (_f = config.getOptionalString("groupPattern")) != null ? _f : /[\s\S]*/
190
208
  );
191
- const orgEnabled = (_h = config.getOptionalBoolean("orgEnabled")) != null ? _h : false;
209
+ const orgEnabled = (_g = config.getOptionalBoolean("orgEnabled")) != null ? _g : false;
192
210
  const schedule = config.has("schedule") ? backendTasks.readTaskScheduleDefinitionFromConfig(config.getConfig("schedule")) : void 0;
193
211
  return {
194
212
  id,
195
213
  group,
214
+ branch,
196
215
  fallbackBranch,
197
216
  host,
198
217
  catalogFile,
@@ -203,20 +222,14 @@ function readGitlabConfig(id, config, logger) {
203
222
  orgEnabled
204
223
  };
205
224
  }
206
- function readGitlabConfigs(config, logger) {
225
+ function readGitlabConfigs(config) {
207
226
  const configs = [];
208
227
  const providerConfigs = config.getOptionalConfig("catalog.providers.gitlab");
209
228
  if (!providerConfigs) {
210
229
  return configs;
211
230
  }
212
231
  for (const id of providerConfigs.keys()) {
213
- configs.push(
214
- readGitlabConfig(
215
- id,
216
- providerConfigs.getConfig(id),
217
- logger.child({ target: id })
218
- )
219
- );
232
+ configs.push(readGitlabConfig(id, providerConfigs.getConfig(id)));
220
233
  }
221
234
  return configs;
222
235
  }
@@ -226,10 +239,7 @@ class GitlabDiscoveryEntityProvider {
226
239
  if (!options.schedule && !options.scheduler) {
227
240
  throw new Error("Either schedule or scheduler must be provided.");
228
241
  }
229
- const providerConfigs = readGitlabConfigs(
230
- config,
231
- options.logger.child({ target: "GitlabDiscoveryEntityProvider" })
232
- );
242
+ const providerConfigs = readGitlabConfigs(config);
233
243
  const integrations = integration.ScmIntegrations.fromConfig(config).gitlab;
234
244
  const providers = [];
235
245
  providerConfigs.forEach((providerConfig) => {
@@ -293,7 +303,7 @@ class GitlabDiscoveryEntityProvider {
293
303
  };
294
304
  }
295
305
  async refresh(logger) {
296
- var _a, _b, _c;
306
+ var _a, _b, _c, _d;
297
307
  if (!this.connection) {
298
308
  throw new Error(
299
309
  `Gitlab discovery connection not initialized for ${this.getProviderName()}`
@@ -323,12 +333,12 @@ class GitlabDiscoveryEntityProvider {
323
333
  if (project.archived) {
324
334
  continue;
325
335
  }
326
- if (this.config.fallbackBranch === "*" && project.default_branch === void 0) {
336
+ if (!this.config.branch && this.config.fallbackBranch === "*" && project.default_branch === void 0) {
327
337
  continue;
328
338
  }
329
- const project_branch = (_b = project.default_branch) != null ? _b : this.config.fallbackBranch;
339
+ const project_branch = (_c = (_b = this.config.branch) != null ? _b : project.default_branch) != null ? _c : this.config.fallbackBranch;
330
340
  const projectHasFile = await client.hasFile(
331
- (_c = project.path_with_namespace) != null ? _c : "",
341
+ (_d = project.path_with_namespace) != null ? _d : "",
332
342
  project_branch,
333
343
  this.config.catalogFile
334
344
  );
@@ -346,8 +356,8 @@ class GitlabDiscoveryEntityProvider {
346
356
  });
347
357
  }
348
358
  createLocationSpec(project) {
349
- var _a;
350
- const project_branch = (_a = project.default_branch) != null ? _a : this.config.fallbackBranch;
359
+ var _a, _b;
360
+ const project_branch = (_b = (_a = this.config.branch) != null ? _a : project.default_branch) != null ? _b : this.config.fallbackBranch;
351
361
  return {
352
362
  type: "url",
353
363
  target: `${project.web_url}/-/blob/${project_branch}/${this.config.catalogFile}`,
@@ -1 +1 @@
1
- {"version":3,"file":"alpha.cjs.js","sources":["../src/lib/client.ts","../src/providers/config.ts","../src/providers/GitlabDiscoveryEntityProvider.ts","../src/module/catalogModuleGitlabDiscoveryEntityProvider.ts"],"sourcesContent":["/*\n * Copyright 2021 The Backstage Authors\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nimport fetch from 'node-fetch';\nimport {\n getGitLabRequestOptions,\n GitLabIntegrationConfig,\n} from '@backstage/integration';\nimport { Logger } from 'winston';\nimport { GitLabGroup, GitLabMembership, GitLabUser } from './types';\n\nexport type CommonListOptions = {\n [key: string]: string | number | boolean | undefined;\n per_page?: number | undefined;\n page?: number | undefined;\n active?: boolean;\n};\n\ninterface ListProjectOptions extends CommonListOptions {\n group?: string;\n}\n\ninterface UserListOptions extends CommonListOptions {\n without_project_bots?: boolean | undefined;\n exclude_internal?: boolean | undefined;\n}\n\nexport type PagedResponse<T> = {\n items: T[];\n nextPage?: number;\n};\n\nexport class GitLabClient {\n private readonly config: GitLabIntegrationConfig;\n private readonly logger: Logger;\n\n constructor(options: { config: GitLabIntegrationConfig; logger: Logger }) {\n this.config = options.config;\n this.logger = options.logger;\n }\n\n /**\n * Indicates whether the client is for a SaaS or self managed GitLab instance.\n */\n isSelfManaged(): boolean {\n return this.config.host !== 'gitlab.com';\n }\n\n async listProjects(\n options?: ListProjectOptions,\n ): Promise<PagedResponse<any>> {\n if (options?.group) {\n return this.pagedRequest(\n `/groups/${encodeURIComponent(options?.group)}/projects`,\n {\n ...options,\n include_subgroups: true,\n },\n );\n }\n\n return this.pagedRequest(`/projects`, options);\n }\n\n async listUsers(\n options?: UserListOptions,\n ): Promise<PagedResponse<GitLabUser>> {\n let requestOptions = options;\n\n if (!requestOptions) {\n requestOptions = {};\n }\n\n requestOptions.without_project_bots = true;\n requestOptions.exclude_internal = true;\n\n return this.pagedRequest(`/users?`, requestOptions);\n }\n\n async listGroups(\n options?: CommonListOptions,\n ): Promise<PagedResponse<GitLabGroup>> {\n return this.pagedRequest(`/groups`, options);\n }\n\n async getUserMemberships(userId: number): Promise<GitLabMembership[]> {\n const endpoint: string = `/users/${encodeURIComponent(userId)}/memberships`;\n const request = new URL(`${this.config.apiBaseUrl}${endpoint}`);\n request.searchParams.append('per_page', '100');\n\n const response = await fetch(request.toString(), {\n headers: getGitLabRequestOptions(this.config).headers,\n method: 'GET',\n });\n\n if (!response.ok) {\n if (response.status >= 500) {\n this.logger.debug(\n `Unexpected response when fetching ${request.toString()}. Expected 200 but got ${\n response.status\n } - ${response.statusText}`,\n );\n }\n return [];\n }\n\n return response.json().then(items => {\n return items as GitLabMembership[];\n });\n }\n\n /**\n * General existence check.\n *\n * @param projectPath - The path to the project\n * @param branch - The branch used to search\n * @param filePath - The path to the file\n */\n async hasFile(\n projectPath: string,\n branch: string,\n filePath: string,\n ): Promise<boolean> {\n const endpoint: string = `/projects/${encodeURIComponent(\n projectPath,\n )}/repository/files/${encodeURIComponent(filePath)}`;\n const request = new URL(`${this.config.apiBaseUrl}${endpoint}`);\n request.searchParams.append('ref', branch);\n\n const response = await fetch(request.toString(), {\n headers: getGitLabRequestOptions(this.config).headers,\n method: 'HEAD',\n });\n\n if (!response.ok) {\n if (response.status >= 500) {\n this.logger.debug(\n `Unexpected response when fetching ${request.toString()}. Expected 200 but got ${\n response.status\n } - ${response.statusText}`,\n );\n }\n return false;\n }\n\n return true;\n }\n\n /**\n * Performs a request against a given paginated GitLab endpoint.\n *\n * This method may be used to perform authenticated REST calls against any\n * paginated GitLab endpoint which uses X-NEXT-PAGE headers. The return value\n * can be be used with the {@link paginated} async-generator function to yield\n * each item from the paged request.\n *\n * @see {@link paginated}\n * @param endpoint - The request endpoint, e.g. /projects.\n * @param options - Request queryString options which may also include page variables.\n */\n async pagedRequest<T = any>(\n endpoint: string,\n options?: CommonListOptions,\n ): Promise<PagedResponse<T>> {\n const request = new URL(`${this.config.apiBaseUrl}${endpoint}`);\n for (const key in options) {\n if (options[key]) {\n request.searchParams.append(key, options[key]!.toString());\n }\n }\n\n this.logger.debug(`Fetching: ${request.toString()}`);\n const response = await fetch(\n request.toString(),\n getGitLabRequestOptions(this.config),\n );\n if (!response.ok) {\n throw new Error(\n `Unexpected response when fetching ${request.toString()}. Expected 200 but got ${\n response.status\n } - ${response.statusText}`,\n );\n }\n return response.json().then(items => {\n const nextPage = response.headers.get('x-next-page');\n\n return {\n items,\n nextPage: nextPage ? Number(nextPage) : null,\n } as PagedResponse<any>;\n });\n }\n}\n\n/**\n * Advances through each page and provides each item from a paginated request.\n *\n * The async generator function yields each item from repeated calls to the\n * provided request function. The generator walks through each available page by\n * setting the page key in the options passed into the request function and\n * making repeated calls until there are no more pages.\n *\n * @see {@link pagedRequest}\n * @param request - Function which returns a PagedResponse to walk through.\n * @param options - Initial ListOptions for the request function.\n */\nexport async function* paginated<T = any>(\n request: (options: CommonListOptions) => Promise<PagedResponse<T>>,\n options: CommonListOptions,\n) {\n let res;\n do {\n res = await request(options);\n options.page = res.nextPage;\n for (const item of res.items) {\n yield item;\n }\n } while (res.nextPage);\n}\n","/*\n * Copyright 2022 The Backstage Authors\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nimport { readTaskScheduleDefinitionFromConfig } from '@backstage/backend-tasks';\nimport { Config } from '@backstage/config';\nimport { GitlabProviderConfig } from '../lib';\nimport { Logger } from 'winston';\n\n/**\n * Extracts the gitlab config from a config object\n *\n * @public\n *\n * @param id - The provider key\n * @param config - The config object to extract from\n * @param logger - The logger\n */\nfunction readGitlabConfig(\n id: string,\n config: Config,\n logger: Logger,\n): GitlabProviderConfig {\n const group = config.getOptionalString('group') ?? '';\n const host = config.getString('host');\n const branch = config.getOptionalString('branch');\n\n if (branch) {\n logger.warn(\n 'The configuration key `branch` has been deprecated in favor of the configuration key `fallbackBranch`.',\n );\n }\n\n const fallbackBranch =\n config.getOptionalString('fallbackBranch') ?? branch ?? 'master';\n const catalogFile =\n config.getOptionalString('entityFilename') ?? 'catalog-info.yaml';\n const projectPattern = new RegExp(\n config.getOptionalString('projectPattern') ?? /[\\s\\S]*/,\n );\n const userPattern = new RegExp(\n config.getOptionalString('userPattern') ?? /[\\s\\S]*/,\n );\n const groupPattern = new RegExp(\n config.getOptionalString('groupPattern') ?? /[\\s\\S]*/,\n );\n const orgEnabled: boolean = config.getOptionalBoolean('orgEnabled') ?? false;\n\n const schedule = config.has('schedule')\n ? readTaskScheduleDefinitionFromConfig(config.getConfig('schedule'))\n : undefined;\n\n return {\n id,\n group,\n fallbackBranch,\n host,\n catalogFile,\n projectPattern,\n userPattern,\n groupPattern,\n schedule,\n orgEnabled,\n };\n}\n\n/**\n * Extracts the gitlab config from a config object array\n *\n * @public\n *\n * @param config - The config object to extract from\n * @param logger - The logger\n */\nexport function readGitlabConfigs(\n config: Config,\n logger: Logger,\n): GitlabProviderConfig[] {\n const configs: GitlabProviderConfig[] = [];\n\n const providerConfigs = config.getOptionalConfig('catalog.providers.gitlab');\n\n if (!providerConfigs) {\n return configs;\n }\n\n for (const id of providerConfigs.keys()) {\n configs.push(\n readGitlabConfig(\n id,\n providerConfigs.getConfig(id),\n logger.child({ target: id }),\n ),\n );\n }\n\n return configs;\n}\n","/*\n * Copyright 2021 The Backstage Authors\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nimport { PluginTaskScheduler, TaskRunner } from '@backstage/backend-tasks';\nimport { Config } from '@backstage/config';\nimport { GitLabIntegration, ScmIntegrations } from '@backstage/integration';\nimport {\n EntityProvider,\n EntityProviderConnection,\n LocationSpec,\n locationSpecToLocationEntity,\n} from '@backstage/plugin-catalog-node';\nimport * as uuid from 'uuid';\nimport { Logger } from 'winston';\nimport {\n GitLabClient,\n GitLabProject,\n GitlabProviderConfig,\n paginated,\n readGitlabConfigs,\n} from '../lib';\n\ntype Result = {\n scanned: number;\n matches: GitLabProject[];\n};\n\n/**\n * Discovers entity definition files in the groups of a Gitlab instance.\n * @public\n */\nexport class GitlabDiscoveryEntityProvider implements EntityProvider {\n private readonly config: GitlabProviderConfig;\n private readonly integration: GitLabIntegration;\n private readonly logger: Logger;\n private readonly scheduleFn: () => Promise<void>;\n private connection?: EntityProviderConnection;\n\n static fromConfig(\n config: Config,\n options: {\n logger: Logger;\n schedule?: TaskRunner;\n scheduler?: PluginTaskScheduler;\n },\n ): GitlabDiscoveryEntityProvider[] {\n if (!options.schedule && !options.scheduler) {\n throw new Error('Either schedule or scheduler must be provided.');\n }\n\n const providerConfigs = readGitlabConfigs(\n config,\n options.logger.child({ target: 'GitlabDiscoveryEntityProvider' }),\n );\n const integrations = ScmIntegrations.fromConfig(config).gitlab;\n const providers: GitlabDiscoveryEntityProvider[] = [];\n\n providerConfigs.forEach(providerConfig => {\n const integration = integrations.byHost(providerConfig.host);\n if (!integration) {\n throw new Error(\n `No gitlab integration found that matches host ${providerConfig.host}`,\n );\n }\n\n if (!options.schedule && !providerConfig.schedule) {\n throw new Error(\n `No schedule provided neither via code nor config for GitlabDiscoveryEntityProvider:${providerConfig.id}.`,\n );\n }\n\n const taskRunner =\n options.schedule ??\n options.scheduler!.createScheduledTaskRunner(providerConfig.schedule!);\n\n providers.push(\n new GitlabDiscoveryEntityProvider({\n ...options,\n config: providerConfig,\n integration,\n taskRunner,\n }),\n );\n });\n return providers;\n }\n\n private constructor(options: {\n config: GitlabProviderConfig;\n integration: GitLabIntegration;\n logger: Logger;\n taskRunner: TaskRunner;\n }) {\n this.config = options.config;\n this.integration = options.integration;\n this.logger = options.logger.child({\n target: this.getProviderName(),\n });\n this.scheduleFn = this.createScheduleFn(options.taskRunner);\n }\n\n getProviderName(): string {\n return `GitlabDiscoveryEntityProvider:${this.config.id}`;\n }\n\n async connect(connection: EntityProviderConnection): Promise<void> {\n this.connection = connection;\n await this.scheduleFn();\n }\n\n private createScheduleFn(taskRunner: TaskRunner): () => Promise<void> {\n return async () => {\n const taskId = `${this.getProviderName()}:refresh`;\n return taskRunner.run({\n id: taskId,\n fn: async () => {\n const logger = this.logger.child({\n class: GitlabDiscoveryEntityProvider.prototype.constructor.name,\n taskId,\n taskInstanceId: uuid.v4(),\n });\n\n try {\n await this.refresh(logger);\n } catch (error) {\n logger.error(`${this.getProviderName()} refresh failed`, error);\n }\n },\n });\n };\n }\n\n async refresh(logger: Logger): Promise<void> {\n if (!this.connection) {\n throw new Error(\n `Gitlab discovery connection not initialized for ${this.getProviderName()}`,\n );\n }\n\n const client = new GitLabClient({\n config: this.integration.config,\n logger: logger,\n });\n\n const projects = paginated<GitLabProject>(\n options => client.listProjects(options),\n {\n group: this.config.group,\n page: 1,\n per_page: 50,\n },\n );\n\n const res: Result = {\n scanned: 0,\n matches: [],\n };\n\n for await (const project of projects) {\n if (!this.config.projectPattern.test(project.path_with_namespace ?? '')) {\n continue;\n }\n\n res.scanned++;\n\n if (project.archived) {\n continue;\n }\n\n if (\n this.config.fallbackBranch === '*' &&\n project.default_branch === undefined\n ) {\n continue;\n }\n\n const project_branch =\n project.default_branch ?? this.config.fallbackBranch;\n\n const projectHasFile: boolean = await client.hasFile(\n project.path_with_namespace ?? '',\n project_branch,\n this.config.catalogFile,\n );\n if (projectHasFile) {\n res.matches.push(project);\n }\n }\n\n const locations = res.matches.map(p => this.createLocationSpec(p));\n await this.connection.applyMutation({\n type: 'full',\n entities: locations.map(location => ({\n locationKey: this.getProviderName(),\n entity: locationSpecToLocationEntity({ location }),\n })),\n });\n }\n\n private createLocationSpec(project: GitLabProject): LocationSpec {\n const project_branch = project.default_branch ?? this.config.fallbackBranch;\n return {\n type: 'url',\n target: `${project.web_url}/-/blob/${project_branch}/${this.config.catalogFile}`,\n presence: 'optional',\n };\n }\n}\n","/*\n * Copyright 2022 The Backstage Authors\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nimport {\n createBackendModule,\n coreServices,\n} from '@backstage/backend-plugin-api';\nimport { loggerToWinstonLogger } from '@backstage/backend-common';\nimport { catalogProcessingExtensionPoint } from '@backstage/plugin-catalog-node/alpha';\nimport { GitlabDiscoveryEntityProvider } from '../providers';\n\n/**\n * Registers the GitlabDiscoveryEntityProvider with the catalog processing extension point.\n *\n * @alpha\n */\nexport const catalogModuleGitlabDiscoveryEntityProvider = createBackendModule({\n pluginId: 'catalog',\n moduleId: 'gitlabDiscoveryEntityProvider',\n register(env) {\n env.registerInit({\n deps: {\n config: coreServices.config,\n catalog: catalogProcessingExtensionPoint,\n logger: coreServices.logger,\n scheduler: coreServices.scheduler,\n },\n async init({ config, catalog, logger, scheduler }) {\n catalog.addEntityProvider(\n GitlabDiscoveryEntityProvider.fromConfig(config, {\n logger: loggerToWinstonLogger(logger),\n scheduler,\n }),\n );\n },\n });\n },\n});\n"],"names":["fetch","getGitLabRequestOptions","readTaskScheduleDefinitionFromConfig","ScmIntegrations","uuid","locationSpecToLocationEntity","createBackendModule","coreServices","catalogProcessingExtensionPoint","loggerToWinstonLogger"],"mappings":"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"}
1
+ {"version":3,"file":"alpha.cjs.js","sources":["../src/lib/client.ts","../src/providers/config.ts","../src/providers/GitlabDiscoveryEntityProvider.ts","../src/module/catalogModuleGitlabDiscoveryEntityProvider.ts"],"sourcesContent":["/*\n * Copyright 2021 The Backstage Authors\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nimport fetch from 'node-fetch';\nimport {\n getGitLabRequestOptions,\n GitLabIntegrationConfig,\n} from '@backstage/integration';\nimport { Logger } from 'winston';\nimport { GitLabGroup, GitLabGroupMembersResponse, GitLabUser } from './types';\n\nexport type CommonListOptions = {\n [key: string]: string | number | boolean | undefined;\n per_page?: number | undefined;\n page?: number | undefined;\n active?: boolean;\n};\n\ninterface ListProjectOptions extends CommonListOptions {\n group?: string;\n}\n\ninterface UserListOptions extends CommonListOptions {\n without_project_bots?: boolean | undefined;\n exclude_internal?: boolean | undefined;\n}\n\nexport type PagedResponse<T> = {\n items: T[];\n nextPage?: number;\n};\n\nexport class GitLabClient {\n private readonly config: GitLabIntegrationConfig;\n private readonly logger: Logger;\n\n constructor(options: { config: GitLabIntegrationConfig; logger: Logger }) {\n this.config = options.config;\n this.logger = options.logger;\n }\n\n /**\n * Indicates whether the client is for a SaaS or self managed GitLab instance.\n */\n isSelfManaged(): boolean {\n return this.config.host !== 'gitlab.com';\n }\n\n async listProjects(\n options?: ListProjectOptions,\n ): Promise<PagedResponse<any>> {\n if (options?.group) {\n return this.pagedRequest(\n `/groups/${encodeURIComponent(options?.group)}/projects`,\n {\n ...options,\n include_subgroups: true,\n },\n );\n }\n\n return this.pagedRequest(`/projects`, options);\n }\n\n async listUsers(\n options?: UserListOptions,\n ): Promise<PagedResponse<GitLabUser>> {\n return this.pagedRequest(`/users?`, {\n ...options,\n without_project_bots: true,\n exclude_internal: true,\n });\n }\n\n async listGroups(\n options?: CommonListOptions,\n ): Promise<PagedResponse<GitLabGroup>> {\n return this.pagedRequest(`/groups`, options);\n }\n\n async getGroupMembers(groupPath: string): Promise<number[]> {\n const memberIds = [];\n let hasNextPage: boolean = false;\n let endCursor: string | null = null;\n do {\n const response: GitLabGroupMembersResponse = await fetch(\n `${this.config.baseUrl}/api/graphql`,\n {\n method: 'POST',\n headers: {\n ...getGitLabRequestOptions(this.config).headers,\n ['Content-Type']: 'application/json',\n },\n body: JSON.stringify({\n variables: { group: groupPath, endCursor },\n query: `query($group: ID!, $endCursor: String) {\n group(fullPath: $group) {\n groupMembers(first: 100, relations: [DIRECT], after: $endCursor) {\n nodes {\n user {\n id\n }\n }\n pageInfo {\n endCursor\n hasNextPage\n }\n }\n }\n }`,\n }),\n },\n ).then(r => r.json());\n if (response.errors) {\n throw new Error(`GraphQL errors: ${JSON.stringify(response.errors)}`);\n }\n memberIds.push(\n ...response.data.group.groupMembers.nodes.map(\n (node: { user: { id: string } }) =>\n Number(node.user.id.replace(/^gid:\\/\\/gitlab\\/User\\//, '')),\n ),\n );\n ({ hasNextPage, endCursor } = response.data.group.groupMembers.pageInfo);\n } while (hasNextPage);\n return memberIds;\n }\n\n /**\n * General existence check.\n *\n * @param projectPath - The path to the project\n * @param branch - The branch used to search\n * @param filePath - The path to the file\n */\n async hasFile(\n projectPath: string,\n branch: string,\n filePath: string,\n ): Promise<boolean> {\n const endpoint: string = `/projects/${encodeURIComponent(\n projectPath,\n )}/repository/files/${encodeURIComponent(filePath)}`;\n const request = new URL(`${this.config.apiBaseUrl}${endpoint}`);\n request.searchParams.append('ref', branch);\n\n const response = await fetch(request.toString(), {\n headers: getGitLabRequestOptions(this.config).headers,\n method: 'HEAD',\n });\n\n if (!response.ok) {\n if (response.status >= 500) {\n this.logger.debug(\n `Unexpected response when fetching ${request.toString()}. Expected 200 but got ${\n response.status\n } - ${response.statusText}`,\n );\n }\n return false;\n }\n\n return true;\n }\n\n /**\n * Performs a request against a given paginated GitLab endpoint.\n *\n * This method may be used to perform authenticated REST calls against any\n * paginated GitLab endpoint which uses X-NEXT-PAGE headers. The return value\n * can be be used with the {@link paginated} async-generator function to yield\n * each item from the paged request.\n *\n * @see {@link paginated}\n * @param endpoint - The request endpoint, e.g. /projects.\n * @param options - Request queryString options which may also include page variables.\n */\n async pagedRequest<T = any>(\n endpoint: string,\n options?: CommonListOptions,\n ): Promise<PagedResponse<T>> {\n const request = new URL(`${this.config.apiBaseUrl}${endpoint}`);\n for (const key in options) {\n if (options[key]) {\n request.searchParams.append(key, options[key]!.toString());\n }\n }\n\n this.logger.debug(`Fetching: ${request.toString()}`);\n const response = await fetch(\n request.toString(),\n getGitLabRequestOptions(this.config),\n );\n if (!response.ok) {\n throw new Error(\n `Unexpected response when fetching ${request.toString()}. Expected 200 but got ${\n response.status\n } - ${response.statusText}`,\n );\n }\n return response.json().then(items => {\n const nextPage = response.headers.get('x-next-page');\n\n return {\n items,\n nextPage: nextPage ? Number(nextPage) : null,\n } as PagedResponse<any>;\n });\n }\n}\n\n/**\n * Advances through each page and provides each item from a paginated request.\n *\n * The async generator function yields each item from repeated calls to the\n * provided request function. The generator walks through each available page by\n * setting the page key in the options passed into the request function and\n * making repeated calls until there are no more pages.\n *\n * @see {@link pagedRequest}\n * @param request - Function which returns a PagedResponse to walk through.\n * @param options - Initial ListOptions for the request function.\n */\nexport async function* paginated<T = any>(\n request: (options: CommonListOptions) => Promise<PagedResponse<T>>,\n options: CommonListOptions,\n) {\n let res;\n do {\n res = await request(options);\n options.page = res.nextPage;\n for (const item of res.items) {\n yield item;\n }\n } while (res.nextPage);\n}\n","/*\n * Copyright 2022 The Backstage Authors\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nimport { readTaskScheduleDefinitionFromConfig } from '@backstage/backend-tasks';\nimport { Config } from '@backstage/config';\nimport { GitlabProviderConfig } from '../lib';\n\n/**\n * Extracts the gitlab config from a config object\n *\n * @public\n *\n * @param id - The provider key\n * @param config - The config object to extract from\n */\nfunction readGitlabConfig(id: string, config: Config): GitlabProviderConfig {\n const group = config.getOptionalString('group') ?? '';\n const host = config.getString('host');\n const branch = config.getOptionalString('branch');\n const fallbackBranch = config.getOptionalString('fallbackBranch') ?? 'master';\n const catalogFile =\n config.getOptionalString('entityFilename') ?? 'catalog-info.yaml';\n const projectPattern = new RegExp(\n config.getOptionalString('projectPattern') ?? /[\\s\\S]*/,\n );\n const userPattern = new RegExp(\n config.getOptionalString('userPattern') ?? /[\\s\\S]*/,\n );\n const groupPattern = new RegExp(\n config.getOptionalString('groupPattern') ?? /[\\s\\S]*/,\n );\n const orgEnabled: boolean = config.getOptionalBoolean('orgEnabled') ?? false;\n\n const schedule = config.has('schedule')\n ? readTaskScheduleDefinitionFromConfig(config.getConfig('schedule'))\n : undefined;\n\n return {\n id,\n group,\n branch,\n fallbackBranch,\n host,\n catalogFile,\n projectPattern,\n userPattern,\n groupPattern,\n schedule,\n orgEnabled,\n };\n}\n\n/**\n * Extracts the gitlab config from a config object array\n *\n * @public\n *\n * @param config - The config object to extract from\n */\nexport function readGitlabConfigs(config: Config): GitlabProviderConfig[] {\n const configs: GitlabProviderConfig[] = [];\n\n const providerConfigs = config.getOptionalConfig('catalog.providers.gitlab');\n\n if (!providerConfigs) {\n return configs;\n }\n\n for (const id of providerConfigs.keys()) {\n configs.push(readGitlabConfig(id, providerConfigs.getConfig(id)));\n }\n\n return configs;\n}\n","/*\n * Copyright 2021 The Backstage Authors\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nimport { PluginTaskScheduler, TaskRunner } from '@backstage/backend-tasks';\nimport { Config } from '@backstage/config';\nimport { GitLabIntegration, ScmIntegrations } from '@backstage/integration';\nimport {\n EntityProvider,\n EntityProviderConnection,\n LocationSpec,\n locationSpecToLocationEntity,\n} from '@backstage/plugin-catalog-node';\nimport * as uuid from 'uuid';\nimport { Logger } from 'winston';\nimport {\n GitLabClient,\n GitLabProject,\n GitlabProviderConfig,\n paginated,\n readGitlabConfigs,\n} from '../lib';\n\ntype Result = {\n scanned: number;\n matches: GitLabProject[];\n};\n\n/**\n * Discovers entity definition files in the groups of a Gitlab instance.\n * @public\n */\nexport class GitlabDiscoveryEntityProvider implements EntityProvider {\n private readonly config: GitlabProviderConfig;\n private readonly integration: GitLabIntegration;\n private readonly logger: Logger;\n private readonly scheduleFn: () => Promise<void>;\n private connection?: EntityProviderConnection;\n\n static fromConfig(\n config: Config,\n options: {\n logger: Logger;\n schedule?: TaskRunner;\n scheduler?: PluginTaskScheduler;\n },\n ): GitlabDiscoveryEntityProvider[] {\n if (!options.schedule && !options.scheduler) {\n throw new Error('Either schedule or scheduler must be provided.');\n }\n\n const providerConfigs = readGitlabConfigs(config);\n const integrations = ScmIntegrations.fromConfig(config).gitlab;\n const providers: GitlabDiscoveryEntityProvider[] = [];\n\n providerConfigs.forEach(providerConfig => {\n const integration = integrations.byHost(providerConfig.host);\n if (!integration) {\n throw new Error(\n `No gitlab integration found that matches host ${providerConfig.host}`,\n );\n }\n\n if (!options.schedule && !providerConfig.schedule) {\n throw new Error(\n `No schedule provided neither via code nor config for GitlabDiscoveryEntityProvider:${providerConfig.id}.`,\n );\n }\n\n const taskRunner =\n options.schedule ??\n options.scheduler!.createScheduledTaskRunner(providerConfig.schedule!);\n\n providers.push(\n new GitlabDiscoveryEntityProvider({\n ...options,\n config: providerConfig,\n integration,\n taskRunner,\n }),\n );\n });\n return providers;\n }\n\n private constructor(options: {\n config: GitlabProviderConfig;\n integration: GitLabIntegration;\n logger: Logger;\n taskRunner: TaskRunner;\n }) {\n this.config = options.config;\n this.integration = options.integration;\n this.logger = options.logger.child({\n target: this.getProviderName(),\n });\n this.scheduleFn = this.createScheduleFn(options.taskRunner);\n }\n\n getProviderName(): string {\n return `GitlabDiscoveryEntityProvider:${this.config.id}`;\n }\n\n async connect(connection: EntityProviderConnection): Promise<void> {\n this.connection = connection;\n await this.scheduleFn();\n }\n\n private createScheduleFn(taskRunner: TaskRunner): () => Promise<void> {\n return async () => {\n const taskId = `${this.getProviderName()}:refresh`;\n return taskRunner.run({\n id: taskId,\n fn: async () => {\n const logger = this.logger.child({\n class: GitlabDiscoveryEntityProvider.prototype.constructor.name,\n taskId,\n taskInstanceId: uuid.v4(),\n });\n\n try {\n await this.refresh(logger);\n } catch (error) {\n logger.error(`${this.getProviderName()} refresh failed`, error);\n }\n },\n });\n };\n }\n\n async refresh(logger: Logger): Promise<void> {\n if (!this.connection) {\n throw new Error(\n `Gitlab discovery connection not initialized for ${this.getProviderName()}`,\n );\n }\n\n const client = new GitLabClient({\n config: this.integration.config,\n logger: logger,\n });\n\n const projects = paginated<GitLabProject>(\n options => client.listProjects(options),\n {\n group: this.config.group,\n page: 1,\n per_page: 50,\n },\n );\n\n const res: Result = {\n scanned: 0,\n matches: [],\n };\n\n for await (const project of projects) {\n if (!this.config.projectPattern.test(project.path_with_namespace ?? '')) {\n continue;\n }\n\n res.scanned++;\n\n if (project.archived) {\n continue;\n }\n\n if (\n !this.config.branch &&\n this.config.fallbackBranch === '*' &&\n project.default_branch === undefined\n ) {\n continue;\n }\n\n const project_branch =\n this.config.branch ??\n project.default_branch ??\n this.config.fallbackBranch;\n\n const projectHasFile: boolean = await client.hasFile(\n project.path_with_namespace ?? '',\n project_branch,\n this.config.catalogFile,\n );\n if (projectHasFile) {\n res.matches.push(project);\n }\n }\n\n const locations = res.matches.map(p => this.createLocationSpec(p));\n await this.connection.applyMutation({\n type: 'full',\n entities: locations.map(location => ({\n locationKey: this.getProviderName(),\n entity: locationSpecToLocationEntity({ location }),\n })),\n });\n }\n\n private createLocationSpec(project: GitLabProject): LocationSpec {\n const project_branch =\n this.config.branch ??\n project.default_branch ??\n this.config.fallbackBranch;\n return {\n type: 'url',\n target: `${project.web_url}/-/blob/${project_branch}/${this.config.catalogFile}`,\n presence: 'optional',\n };\n }\n}\n","/*\n * Copyright 2022 The Backstage Authors\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nimport {\n createBackendModule,\n coreServices,\n} from '@backstage/backend-plugin-api';\nimport { loggerToWinstonLogger } from '@backstage/backend-common';\nimport { catalogProcessingExtensionPoint } from '@backstage/plugin-catalog-node/alpha';\nimport { GitlabDiscoveryEntityProvider } from '../providers';\n\n/**\n * Registers the GitlabDiscoveryEntityProvider with the catalog processing extension point.\n *\n * @alpha\n */\nexport const catalogModuleGitlabDiscoveryEntityProvider = createBackendModule({\n pluginId: 'catalog',\n moduleId: 'gitlabDiscoveryEntityProvider',\n register(env) {\n env.registerInit({\n deps: {\n config: coreServices.config,\n catalog: catalogProcessingExtensionPoint,\n logger: coreServices.logger,\n scheduler: coreServices.scheduler,\n },\n async init({ config, catalog, logger, scheduler }) {\n catalog.addEntityProvider(\n GitlabDiscoveryEntityProvider.fromConfig(config, {\n logger: loggerToWinstonLogger(logger),\n scheduler,\n }),\n );\n },\n });\n },\n});\n"],"names":["fetch","getGitLabRequestOptions","readTaskScheduleDefinitionFromConfig","ScmIntegrations","uuid","locationSpecToLocationEntity","createBackendModule","coreServices","catalogProcessingExtensionPoint","loggerToWinstonLogger"],"mappings":"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"}
package/dist/index.cjs.js CHANGED
@@ -58,36 +58,59 @@ class GitLabClient {
58
58
  return this.pagedRequest(`/projects`, options);
59
59
  }
60
60
  async listUsers(options) {
61
- let requestOptions = options;
62
- if (!requestOptions) {
63
- requestOptions = {};
64
- }
65
- requestOptions.without_project_bots = true;
66
- requestOptions.exclude_internal = true;
67
- return this.pagedRequest(`/users?`, requestOptions);
61
+ return this.pagedRequest(`/users?`, {
62
+ ...options,
63
+ without_project_bots: true,
64
+ exclude_internal: true
65
+ });
68
66
  }
69
67
  async listGroups(options) {
70
68
  return this.pagedRequest(`/groups`, options);
71
69
  }
72
- async getUserMemberships(userId) {
73
- const endpoint = `/users/${encodeURIComponent(userId)}/memberships`;
74
- const request = new URL(`${this.config.apiBaseUrl}${endpoint}`);
75
- request.searchParams.append("per_page", "100");
76
- const response = await fetch__default["default"](request.toString(), {
77
- headers: integration.getGitLabRequestOptions(this.config).headers,
78
- method: "GET"
79
- });
80
- if (!response.ok) {
81
- if (response.status >= 500) {
82
- this.logger.debug(
83
- `Unexpected response when fetching ${request.toString()}. Expected 200 but got ${response.status} - ${response.statusText}`
84
- );
70
+ async getGroupMembers(groupPath) {
71
+ const memberIds = [];
72
+ let hasNextPage = false;
73
+ let endCursor = null;
74
+ do {
75
+ const response = await fetch__default["default"](
76
+ `${this.config.baseUrl}/api/graphql`,
77
+ {
78
+ method: "POST",
79
+ headers: {
80
+ ...integration.getGitLabRequestOptions(this.config).headers,
81
+ ["Content-Type"]: "application/json"
82
+ },
83
+ body: JSON.stringify({
84
+ variables: { group: groupPath, endCursor },
85
+ query: `query($group: ID!, $endCursor: String) {
86
+ group(fullPath: $group) {
87
+ groupMembers(first: 100, relations: [DIRECT], after: $endCursor) {
88
+ nodes {
89
+ user {
90
+ id
91
+ }
92
+ }
93
+ pageInfo {
94
+ endCursor
95
+ hasNextPage
96
+ }
97
+ }
98
+ }
99
+ }`
100
+ })
101
+ }
102
+ ).then((r) => r.json());
103
+ if (response.errors) {
104
+ throw new Error(`GraphQL errors: ${JSON.stringify(response.errors)}`);
85
105
  }
86
- return [];
87
- }
88
- return response.json().then((items) => {
89
- return items;
90
- });
106
+ memberIds.push(
107
+ ...response.data.group.groupMembers.nodes.map(
108
+ (node) => Number(node.user.id.replace(/^gid:\/\/gitlab\/User\//, ""))
109
+ )
110
+ );
111
+ ({ hasNextPage, endCursor } = response.data.group.groupMembers.pageInfo);
112
+ } while (hasNextPage);
113
+ return memberIds;
91
114
  }
92
115
  /**
93
116
  * General existence check.
@@ -165,32 +188,28 @@ async function* paginated(request, options) {
165
188
  } while (res.nextPage);
166
189
  }
167
190
 
168
- function readGitlabConfig(id, config, logger) {
169
- var _a, _b, _c, _d, _e, _f, _g, _h;
191
+ function readGitlabConfig(id, config) {
192
+ var _a, _b, _c, _d, _e, _f, _g;
170
193
  const group = (_a = config.getOptionalString("group")) != null ? _a : "";
171
194
  const host = config.getString("host");
172
195
  const branch = config.getOptionalString("branch");
173
- if (branch) {
174
- logger.warn(
175
- "The configuration key `branch` has been deprecated in favor of the configuration key `fallbackBranch`."
176
- );
177
- }
178
- const fallbackBranch = (_c = (_b = config.getOptionalString("fallbackBranch")) != null ? _b : branch) != null ? _c : "master";
179
- const catalogFile = (_d = config.getOptionalString("entityFilename")) != null ? _d : "catalog-info.yaml";
196
+ const fallbackBranch = (_b = config.getOptionalString("fallbackBranch")) != null ? _b : "master";
197
+ const catalogFile = (_c = config.getOptionalString("entityFilename")) != null ? _c : "catalog-info.yaml";
180
198
  const projectPattern = new RegExp(
181
- (_e = config.getOptionalString("projectPattern")) != null ? _e : /[\s\S]*/
199
+ (_d = config.getOptionalString("projectPattern")) != null ? _d : /[\s\S]*/
182
200
  );
183
201
  const userPattern = new RegExp(
184
- (_f = config.getOptionalString("userPattern")) != null ? _f : /[\s\S]*/
202
+ (_e = config.getOptionalString("userPattern")) != null ? _e : /[\s\S]*/
185
203
  );
186
204
  const groupPattern = new RegExp(
187
- (_g = config.getOptionalString("groupPattern")) != null ? _g : /[\s\S]*/
205
+ (_f = config.getOptionalString("groupPattern")) != null ? _f : /[\s\S]*/
188
206
  );
189
- const orgEnabled = (_h = config.getOptionalBoolean("orgEnabled")) != null ? _h : false;
207
+ const orgEnabled = (_g = config.getOptionalBoolean("orgEnabled")) != null ? _g : false;
190
208
  const schedule = config.has("schedule") ? backendTasks.readTaskScheduleDefinitionFromConfig(config.getConfig("schedule")) : void 0;
191
209
  return {
192
210
  id,
193
211
  group,
212
+ branch,
194
213
  fallbackBranch,
195
214
  host,
196
215
  catalogFile,
@@ -201,20 +220,14 @@ function readGitlabConfig(id, config, logger) {
201
220
  orgEnabled
202
221
  };
203
222
  }
204
- function readGitlabConfigs(config, logger) {
223
+ function readGitlabConfigs(config) {
205
224
  const configs = [];
206
225
  const providerConfigs = config.getOptionalConfig("catalog.providers.gitlab");
207
226
  if (!providerConfigs) {
208
227
  return configs;
209
228
  }
210
229
  for (const id of providerConfigs.keys()) {
211
- configs.push(
212
- readGitlabConfig(
213
- id,
214
- providerConfigs.getConfig(id),
215
- logger.child({ target: id })
216
- )
217
- );
230
+ configs.push(readGitlabConfig(id, providerConfigs.getConfig(id)));
218
231
  }
219
232
  return configs;
220
233
  }
@@ -337,10 +350,7 @@ class GitlabDiscoveryEntityProvider {
337
350
  if (!options.schedule && !options.scheduler) {
338
351
  throw new Error("Either schedule or scheduler must be provided.");
339
352
  }
340
- const providerConfigs = readGitlabConfigs(
341
- config,
342
- options.logger.child({ target: "GitlabDiscoveryEntityProvider" })
343
- );
353
+ const providerConfigs = readGitlabConfigs(config);
344
354
  const integrations = integration.ScmIntegrations.fromConfig(config).gitlab;
345
355
  const providers = [];
346
356
  providerConfigs.forEach((providerConfig) => {
@@ -404,7 +414,7 @@ class GitlabDiscoveryEntityProvider {
404
414
  };
405
415
  }
406
416
  async refresh(logger) {
407
- var _a, _b, _c;
417
+ var _a, _b, _c, _d;
408
418
  if (!this.connection) {
409
419
  throw new Error(
410
420
  `Gitlab discovery connection not initialized for ${this.getProviderName()}`
@@ -434,12 +444,12 @@ class GitlabDiscoveryEntityProvider {
434
444
  if (project.archived) {
435
445
  continue;
436
446
  }
437
- if (this.config.fallbackBranch === "*" && project.default_branch === void 0) {
447
+ if (!this.config.branch && this.config.fallbackBranch === "*" && project.default_branch === void 0) {
438
448
  continue;
439
449
  }
440
- const project_branch = (_b = project.default_branch) != null ? _b : this.config.fallbackBranch;
450
+ const project_branch = (_c = (_b = this.config.branch) != null ? _b : project.default_branch) != null ? _c : this.config.fallbackBranch;
441
451
  const projectHasFile = await client.hasFile(
442
- (_c = project.path_with_namespace) != null ? _c : "",
452
+ (_d = project.path_with_namespace) != null ? _d : "",
443
453
  project_branch,
444
454
  this.config.catalogFile
445
455
  );
@@ -457,8 +467,8 @@ class GitlabDiscoveryEntityProvider {
457
467
  });
458
468
  }
459
469
  createLocationSpec(project) {
460
- var _a;
461
- const project_branch = (_a = project.default_branch) != null ? _a : this.config.fallbackBranch;
470
+ var _a, _b;
471
+ const project_branch = (_b = (_a = this.config.branch) != null ? _a : project.default_branch) != null ? _b : this.config.fallbackBranch;
462
472
  return {
463
473
  type: "url",
464
474
  target: `${project.web_url}/-/blob/${project_branch}/${this.config.catalogFile}`,
@@ -472,10 +482,7 @@ class GitlabOrgDiscoveryEntityProvider {
472
482
  if (!options.schedule && !options.scheduler) {
473
483
  throw new Error("Either schedule or scheduler must be provided.");
474
484
  }
475
- const providerConfigs = readGitlabConfigs(
476
- config,
477
- options.logger.child({ target: "GitlabOrgDiscoveryEntityProvider" })
478
- );
485
+ const providerConfigs = readGitlabConfigs(config);
479
486
  const integrations = integration.ScmIntegrations.fromConfig(config).gitlab;
480
487
  const providers = [];
481
488
  providerConfigs.forEach((providerConfig) => {
@@ -542,7 +549,7 @@ class GitlabOrgDiscoveryEntityProvider {
542
549
  };
543
550
  }
544
551
  async refresh(logger) {
545
- var _a, _b, _c;
552
+ var _a, _b, _c, _d;
546
553
  if (!this.connection) {
547
554
  throw new Error(
548
555
  `Gitlab discovery connection not initialized for ${this.getProviderName()}`
@@ -564,7 +571,7 @@ class GitlabOrgDiscoveryEntityProvider {
564
571
  per_page: 100
565
572
  }
566
573
  );
567
- const idMappedGroup = {};
574
+ const idMappedUser = {};
568
575
  const res = {
569
576
  scanned: 0,
570
577
  matches: []
@@ -573,34 +580,32 @@ class GitlabOrgDiscoveryEntityProvider {
573
580
  scanned: 0,
574
581
  matches: []
575
582
  };
576
- for await (const group of groups) {
577
- if (!this.config.groupPattern.test((_a = group.full_path) != null ? _a : "")) {
583
+ for await (const user of users) {
584
+ if (!this.config.userPattern.test((_b = (_a = user.email) != null ? _a : user.username) != null ? _b : "")) {
578
585
  continue;
579
586
  }
580
- if (this.config.group && !group.full_path.startsWith(`${this.config.group}/`)) {
587
+ res.scanned++;
588
+ if (user.state !== "active") {
581
589
  continue;
582
590
  }
583
- groupRes.scanned++;
584
- groupRes.matches.push(group);
585
- idMappedGroup[group.id] = group;
591
+ idMappedUser[user.id] = user;
592
+ res.matches.push(user);
586
593
  }
587
- for await (const user of users) {
588
- if (!this.config.userPattern.test((_c = (_b = user.email) != null ? _b : user.username) != null ? _c : "")) {
594
+ for await (const group of groups) {
595
+ if (!this.config.groupPattern.test((_c = group.full_path) != null ? _c : "")) {
589
596
  continue;
590
597
  }
591
- res.scanned++;
592
- if (user.state !== "active") {
598
+ if (this.config.group && !group.full_path.startsWith(`${this.config.group}/`)) {
593
599
  continue;
594
600
  }
595
- const memberships = await client.getUserMemberships(user.id);
596
- const userGroups = [];
597
- for (const i of memberships) {
598
- if (i.source_type === "Namespace" && idMappedGroup.hasOwnProperty(i.source_id)) {
599
- userGroups.push(idMappedGroup[i.source_id]);
601
+ groupRes.scanned++;
602
+ groupRes.matches.push(group);
603
+ for (const id of await client.getGroupMembers(group.full_path)) {
604
+ const user = idMappedUser[id];
605
+ if (user) {
606
+ user.groups = ((_d = user.groups) != null ? _d : []).concat(group);
600
607
  }
601
608
  }
602
- user.groups = userGroups;
603
- res.matches.push(user);
604
609
  }
605
610
  const groupsWithUsers = groupRes.matches.filter((group) => {
606
611
  return res.matches.filter((x) => {
@@ -1 +1 @@
1
- {"version":3,"file":"index.cjs.js","sources":["../src/lib/client.ts","../src/providers/config.ts","../src/GitLabDiscoveryProcessor.ts","../src/providers/GitlabDiscoveryEntityProvider.ts","../src/providers/GitlabOrgDiscoveryEntityProvider.ts"],"sourcesContent":["/*\n * Copyright 2021 The Backstage Authors\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nimport fetch from 'node-fetch';\nimport {\n getGitLabRequestOptions,\n GitLabIntegrationConfig,\n} from '@backstage/integration';\nimport { Logger } from 'winston';\nimport { GitLabGroup, GitLabMembership, GitLabUser } from './types';\n\nexport type CommonListOptions = {\n [key: string]: string | number | boolean | undefined;\n per_page?: number | undefined;\n page?: number | undefined;\n active?: boolean;\n};\n\ninterface ListProjectOptions extends CommonListOptions {\n group?: string;\n}\n\ninterface UserListOptions extends CommonListOptions {\n without_project_bots?: boolean | undefined;\n exclude_internal?: boolean | undefined;\n}\n\nexport type PagedResponse<T> = {\n items: T[];\n nextPage?: number;\n};\n\nexport class GitLabClient {\n private readonly config: GitLabIntegrationConfig;\n private readonly logger: Logger;\n\n constructor(options: { config: GitLabIntegrationConfig; logger: Logger }) {\n this.config = options.config;\n this.logger = options.logger;\n }\n\n /**\n * Indicates whether the client is for a SaaS or self managed GitLab instance.\n */\n isSelfManaged(): boolean {\n return this.config.host !== 'gitlab.com';\n }\n\n async listProjects(\n options?: ListProjectOptions,\n ): Promise<PagedResponse<any>> {\n if (options?.group) {\n return this.pagedRequest(\n `/groups/${encodeURIComponent(options?.group)}/projects`,\n {\n ...options,\n include_subgroups: true,\n },\n );\n }\n\n return this.pagedRequest(`/projects`, options);\n }\n\n async listUsers(\n options?: UserListOptions,\n ): Promise<PagedResponse<GitLabUser>> {\n let requestOptions = options;\n\n if (!requestOptions) {\n requestOptions = {};\n }\n\n requestOptions.without_project_bots = true;\n requestOptions.exclude_internal = true;\n\n return this.pagedRequest(`/users?`, requestOptions);\n }\n\n async listGroups(\n options?: CommonListOptions,\n ): Promise<PagedResponse<GitLabGroup>> {\n return this.pagedRequest(`/groups`, options);\n }\n\n async getUserMemberships(userId: number): Promise<GitLabMembership[]> {\n const endpoint: string = `/users/${encodeURIComponent(userId)}/memberships`;\n const request = new URL(`${this.config.apiBaseUrl}${endpoint}`);\n request.searchParams.append('per_page', '100');\n\n const response = await fetch(request.toString(), {\n headers: getGitLabRequestOptions(this.config).headers,\n method: 'GET',\n });\n\n if (!response.ok) {\n if (response.status >= 500) {\n this.logger.debug(\n `Unexpected response when fetching ${request.toString()}. Expected 200 but got ${\n response.status\n } - ${response.statusText}`,\n );\n }\n return [];\n }\n\n return response.json().then(items => {\n return items as GitLabMembership[];\n });\n }\n\n /**\n * General existence check.\n *\n * @param projectPath - The path to the project\n * @param branch - The branch used to search\n * @param filePath - The path to the file\n */\n async hasFile(\n projectPath: string,\n branch: string,\n filePath: string,\n ): Promise<boolean> {\n const endpoint: string = `/projects/${encodeURIComponent(\n projectPath,\n )}/repository/files/${encodeURIComponent(filePath)}`;\n const request = new URL(`${this.config.apiBaseUrl}${endpoint}`);\n request.searchParams.append('ref', branch);\n\n const response = await fetch(request.toString(), {\n headers: getGitLabRequestOptions(this.config).headers,\n method: 'HEAD',\n });\n\n if (!response.ok) {\n if (response.status >= 500) {\n this.logger.debug(\n `Unexpected response when fetching ${request.toString()}. Expected 200 but got ${\n response.status\n } - ${response.statusText}`,\n );\n }\n return false;\n }\n\n return true;\n }\n\n /**\n * Performs a request against a given paginated GitLab endpoint.\n *\n * This method may be used to perform authenticated REST calls against any\n * paginated GitLab endpoint which uses X-NEXT-PAGE headers. The return value\n * can be be used with the {@link paginated} async-generator function to yield\n * each item from the paged request.\n *\n * @see {@link paginated}\n * @param endpoint - The request endpoint, e.g. /projects.\n * @param options - Request queryString options which may also include page variables.\n */\n async pagedRequest<T = any>(\n endpoint: string,\n options?: CommonListOptions,\n ): Promise<PagedResponse<T>> {\n const request = new URL(`${this.config.apiBaseUrl}${endpoint}`);\n for (const key in options) {\n if (options[key]) {\n request.searchParams.append(key, options[key]!.toString());\n }\n }\n\n this.logger.debug(`Fetching: ${request.toString()}`);\n const response = await fetch(\n request.toString(),\n getGitLabRequestOptions(this.config),\n );\n if (!response.ok) {\n throw new Error(\n `Unexpected response when fetching ${request.toString()}. Expected 200 but got ${\n response.status\n } - ${response.statusText}`,\n );\n }\n return response.json().then(items => {\n const nextPage = response.headers.get('x-next-page');\n\n return {\n items,\n nextPage: nextPage ? Number(nextPage) : null,\n } as PagedResponse<any>;\n });\n }\n}\n\n/**\n * Advances through each page and provides each item from a paginated request.\n *\n * The async generator function yields each item from repeated calls to the\n * provided request function. The generator walks through each available page by\n * setting the page key in the options passed into the request function and\n * making repeated calls until there are no more pages.\n *\n * @see {@link pagedRequest}\n * @param request - Function which returns a PagedResponse to walk through.\n * @param options - Initial ListOptions for the request function.\n */\nexport async function* paginated<T = any>(\n request: (options: CommonListOptions) => Promise<PagedResponse<T>>,\n options: CommonListOptions,\n) {\n let res;\n do {\n res = await request(options);\n options.page = res.nextPage;\n for (const item of res.items) {\n yield item;\n }\n } while (res.nextPage);\n}\n","/*\n * Copyright 2022 The Backstage Authors\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nimport { readTaskScheduleDefinitionFromConfig } from '@backstage/backend-tasks';\nimport { Config } from '@backstage/config';\nimport { GitlabProviderConfig } from '../lib';\nimport { Logger } from 'winston';\n\n/**\n * Extracts the gitlab config from a config object\n *\n * @public\n *\n * @param id - The provider key\n * @param config - The config object to extract from\n * @param logger - The logger\n */\nfunction readGitlabConfig(\n id: string,\n config: Config,\n logger: Logger,\n): GitlabProviderConfig {\n const group = config.getOptionalString('group') ?? '';\n const host = config.getString('host');\n const branch = config.getOptionalString('branch');\n\n if (branch) {\n logger.warn(\n 'The configuration key `branch` has been deprecated in favor of the configuration key `fallbackBranch`.',\n );\n }\n\n const fallbackBranch =\n config.getOptionalString('fallbackBranch') ?? branch ?? 'master';\n const catalogFile =\n config.getOptionalString('entityFilename') ?? 'catalog-info.yaml';\n const projectPattern = new RegExp(\n config.getOptionalString('projectPattern') ?? /[\\s\\S]*/,\n );\n const userPattern = new RegExp(\n config.getOptionalString('userPattern') ?? /[\\s\\S]*/,\n );\n const groupPattern = new RegExp(\n config.getOptionalString('groupPattern') ?? /[\\s\\S]*/,\n );\n const orgEnabled: boolean = config.getOptionalBoolean('orgEnabled') ?? false;\n\n const schedule = config.has('schedule')\n ? readTaskScheduleDefinitionFromConfig(config.getConfig('schedule'))\n : undefined;\n\n return {\n id,\n group,\n fallbackBranch,\n host,\n catalogFile,\n projectPattern,\n userPattern,\n groupPattern,\n schedule,\n orgEnabled,\n };\n}\n\n/**\n * Extracts the gitlab config from a config object array\n *\n * @public\n *\n * @param config - The config object to extract from\n * @param logger - The logger\n */\nexport function readGitlabConfigs(\n config: Config,\n logger: Logger,\n): GitlabProviderConfig[] {\n const configs: GitlabProviderConfig[] = [];\n\n const providerConfigs = config.getOptionalConfig('catalog.providers.gitlab');\n\n if (!providerConfigs) {\n return configs;\n }\n\n for (const id of providerConfigs.keys()) {\n configs.push(\n readGitlabConfig(\n id,\n providerConfigs.getConfig(id),\n logger.child({ target: id }),\n ),\n );\n }\n\n return configs;\n}\n","/*\n * Copyright 2021 The Backstage Authors\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nimport {\n CacheClient,\n CacheManager,\n PluginCacheManager,\n} from '@backstage/backend-common';\nimport { Config } from '@backstage/config';\nimport {\n ScmIntegrationRegistry,\n ScmIntegrations,\n} from '@backstage/integration';\nimport {\n CatalogProcessor,\n CatalogProcessorEmit,\n LocationSpec,\n processingResult,\n} from '@backstage/plugin-catalog-node';\nimport { Logger } from 'winston';\nimport { GitLabClient, GitLabProject, paginated } from './lib';\n\n/**\n * Extracts repositories out of an GitLab instance.\n * @public\n */\nexport class GitLabDiscoveryProcessor implements CatalogProcessor {\n private readonly integrations: ScmIntegrationRegistry;\n private readonly logger: Logger;\n private readonly cache: CacheClient;\n private readonly skipReposWithoutExactFileMatch: boolean;\n\n static fromConfig(\n config: Config,\n options: { logger: Logger; skipReposWithoutExactFileMatch?: boolean },\n ): GitLabDiscoveryProcessor {\n const integrations = ScmIntegrations.fromConfig(config);\n const pluginCache =\n CacheManager.fromConfig(config).forPlugin('gitlab-discovery');\n\n return new GitLabDiscoveryProcessor({\n ...options,\n integrations,\n pluginCache,\n });\n }\n\n private constructor(options: {\n integrations: ScmIntegrationRegistry;\n pluginCache: PluginCacheManager;\n logger: Logger;\n skipReposWithoutExactFileMatch?: boolean;\n }) {\n this.integrations = options.integrations;\n this.cache = options.pluginCache.getClient();\n this.logger = options.logger;\n this.skipReposWithoutExactFileMatch =\n options.skipReposWithoutExactFileMatch || false;\n }\n\n getProcessorName(): string {\n return 'GitLabDiscoveryProcessor';\n }\n\n async readLocation(\n location: LocationSpec,\n _optional: boolean,\n emit: CatalogProcessorEmit,\n ): Promise<boolean> {\n if (location.type !== 'gitlab-discovery') {\n return false;\n }\n\n const startTime = new Date();\n const { group, host, branch, catalogPath } = parseUrl(location.target);\n\n const integration = this.integrations.gitlab.byUrl(`https://${host}`);\n if (!integration) {\n throw new Error(\n `There is no GitLab integration that matches ${host}. Please add a configuration entry for it under integrations.gitlab`,\n );\n }\n\n const client = new GitLabClient({\n config: integration.config,\n logger: this.logger,\n });\n this.logger.debug(`Reading GitLab projects from ${location.target}`);\n\n const lastActivity = (await this.cache.get(this.getCacheKey())) as string;\n const opts = {\n group,\n page: 1,\n // We check for the existence of lastActivity and only set it if it's present to ensure\n // that the options doesn't include the key so that the API doesn't receive an empty query parameter.\n ...(lastActivity && { last_activity_after: lastActivity }),\n };\n\n const projects = paginated(options => client.listProjects(options), opts);\n\n const res: Result = {\n scanned: 0,\n matches: [],\n };\n for await (const project of projects) {\n res.scanned++;\n\n if (project.archived) {\n continue;\n }\n\n if (branch === '*' && project.default_branch === undefined) {\n continue;\n }\n\n if (this.skipReposWithoutExactFileMatch) {\n const project_branch = branch === '*' ? project.default_branch : branch;\n\n const projectHasFile: boolean = await client.hasFile(\n project.path_with_namespace,\n project_branch,\n catalogPath,\n );\n\n if (!projectHasFile) {\n continue;\n }\n }\n\n res.matches.push(project);\n }\n\n for (const project of res.matches) {\n const project_branch = branch === '*' ? project.default_branch : branch;\n\n emit(\n processingResult.location({\n type: 'url',\n // The format expected by the GitLabUrlReader:\n // https://gitlab.com/groupA/teams/teamA/subgroupA/repoA/-/blob/branch/filepath\n //\n // This unfortunately will trigger another API call in `getGitLabFileFetchUrl` to get the project ID.\n // The alternative is using the `buildRawUrl` function, which does not support subgroups, so providing a raw\n // URL here won't work either.\n target: `${project.web_url}/-/blob/${project_branch}/${catalogPath}`,\n presence: 'optional',\n }),\n );\n }\n\n // Save an ISO formatted string in the cache as that's what GitLab expects in the API request.\n await this.cache.set(this.getCacheKey(), startTime.toISOString());\n\n const duration = ((Date.now() - startTime.getTime()) / 1000).toFixed(1);\n this.logger.debug(\n `Read ${res.scanned} GitLab repositories in ${duration} seconds`,\n );\n\n return true;\n }\n\n private getCacheKey(): string {\n return `processors/${this.getProcessorName()}/last-activity`;\n }\n}\n\ntype Result = {\n scanned: number;\n matches: GitLabProject[];\n};\n\n/*\n * Helpers\n */\n\nexport function parseUrl(urlString: string): {\n group?: string;\n host: string;\n branch: string;\n catalogPath: string;\n} {\n const url = new URL(urlString);\n const path = url.pathname.slice(1).split('/');\n\n // (/group/subgroup)/blob/branch|*/filepath\n const blobIndex = path.findIndex(p => p === 'blob');\n if (blobIndex !== -1 && path.length > blobIndex + 2) {\n const group =\n blobIndex > 0 ? path.slice(0, blobIndex).join('/') : undefined;\n\n return {\n group,\n host: url.host,\n branch: decodeURIComponent(path[blobIndex + 1]),\n catalogPath: decodeURIComponent(path.slice(blobIndex + 2).join('/')),\n };\n }\n\n throw new Error(`Failed to parse ${urlString}`);\n}\n","/*\n * Copyright 2021 The Backstage Authors\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nimport { PluginTaskScheduler, TaskRunner } from '@backstage/backend-tasks';\nimport { Config } from '@backstage/config';\nimport { GitLabIntegration, ScmIntegrations } from '@backstage/integration';\nimport {\n EntityProvider,\n EntityProviderConnection,\n LocationSpec,\n locationSpecToLocationEntity,\n} from '@backstage/plugin-catalog-node';\nimport * as uuid from 'uuid';\nimport { Logger } from 'winston';\nimport {\n GitLabClient,\n GitLabProject,\n GitlabProviderConfig,\n paginated,\n readGitlabConfigs,\n} from '../lib';\n\ntype Result = {\n scanned: number;\n matches: GitLabProject[];\n};\n\n/**\n * Discovers entity definition files in the groups of a Gitlab instance.\n * @public\n */\nexport class GitlabDiscoveryEntityProvider implements EntityProvider {\n private readonly config: GitlabProviderConfig;\n private readonly integration: GitLabIntegration;\n private readonly logger: Logger;\n private readonly scheduleFn: () => Promise<void>;\n private connection?: EntityProviderConnection;\n\n static fromConfig(\n config: Config,\n options: {\n logger: Logger;\n schedule?: TaskRunner;\n scheduler?: PluginTaskScheduler;\n },\n ): GitlabDiscoveryEntityProvider[] {\n if (!options.schedule && !options.scheduler) {\n throw new Error('Either schedule or scheduler must be provided.');\n }\n\n const providerConfigs = readGitlabConfigs(\n config,\n options.logger.child({ target: 'GitlabDiscoveryEntityProvider' }),\n );\n const integrations = ScmIntegrations.fromConfig(config).gitlab;\n const providers: GitlabDiscoveryEntityProvider[] = [];\n\n providerConfigs.forEach(providerConfig => {\n const integration = integrations.byHost(providerConfig.host);\n if (!integration) {\n throw new Error(\n `No gitlab integration found that matches host ${providerConfig.host}`,\n );\n }\n\n if (!options.schedule && !providerConfig.schedule) {\n throw new Error(\n `No schedule provided neither via code nor config for GitlabDiscoveryEntityProvider:${providerConfig.id}.`,\n );\n }\n\n const taskRunner =\n options.schedule ??\n options.scheduler!.createScheduledTaskRunner(providerConfig.schedule!);\n\n providers.push(\n new GitlabDiscoveryEntityProvider({\n ...options,\n config: providerConfig,\n integration,\n taskRunner,\n }),\n );\n });\n return providers;\n }\n\n private constructor(options: {\n config: GitlabProviderConfig;\n integration: GitLabIntegration;\n logger: Logger;\n taskRunner: TaskRunner;\n }) {\n this.config = options.config;\n this.integration = options.integration;\n this.logger = options.logger.child({\n target: this.getProviderName(),\n });\n this.scheduleFn = this.createScheduleFn(options.taskRunner);\n }\n\n getProviderName(): string {\n return `GitlabDiscoveryEntityProvider:${this.config.id}`;\n }\n\n async connect(connection: EntityProviderConnection): Promise<void> {\n this.connection = connection;\n await this.scheduleFn();\n }\n\n private createScheduleFn(taskRunner: TaskRunner): () => Promise<void> {\n return async () => {\n const taskId = `${this.getProviderName()}:refresh`;\n return taskRunner.run({\n id: taskId,\n fn: async () => {\n const logger = this.logger.child({\n class: GitlabDiscoveryEntityProvider.prototype.constructor.name,\n taskId,\n taskInstanceId: uuid.v4(),\n });\n\n try {\n await this.refresh(logger);\n } catch (error) {\n logger.error(`${this.getProviderName()} refresh failed`, error);\n }\n },\n });\n };\n }\n\n async refresh(logger: Logger): Promise<void> {\n if (!this.connection) {\n throw new Error(\n `Gitlab discovery connection not initialized for ${this.getProviderName()}`,\n );\n }\n\n const client = new GitLabClient({\n config: this.integration.config,\n logger: logger,\n });\n\n const projects = paginated<GitLabProject>(\n options => client.listProjects(options),\n {\n group: this.config.group,\n page: 1,\n per_page: 50,\n },\n );\n\n const res: Result = {\n scanned: 0,\n matches: [],\n };\n\n for await (const project of projects) {\n if (!this.config.projectPattern.test(project.path_with_namespace ?? '')) {\n continue;\n }\n\n res.scanned++;\n\n if (project.archived) {\n continue;\n }\n\n if (\n this.config.fallbackBranch === '*' &&\n project.default_branch === undefined\n ) {\n continue;\n }\n\n const project_branch =\n project.default_branch ?? this.config.fallbackBranch;\n\n const projectHasFile: boolean = await client.hasFile(\n project.path_with_namespace ?? '',\n project_branch,\n this.config.catalogFile,\n );\n if (projectHasFile) {\n res.matches.push(project);\n }\n }\n\n const locations = res.matches.map(p => this.createLocationSpec(p));\n await this.connection.applyMutation({\n type: 'full',\n entities: locations.map(location => ({\n locationKey: this.getProviderName(),\n entity: locationSpecToLocationEntity({ location }),\n })),\n });\n }\n\n private createLocationSpec(project: GitLabProject): LocationSpec {\n const project_branch = project.default_branch ?? this.config.fallbackBranch;\n return {\n type: 'url',\n target: `${project.web_url}/-/blob/${project_branch}/${this.config.catalogFile}`,\n presence: 'optional',\n };\n }\n}\n","/*\n * Copyright 2021 The Backstage Authors\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nimport { PluginTaskScheduler, TaskRunner } from '@backstage/backend-tasks';\nimport { Config } from '@backstage/config';\nimport { GitLabIntegration, ScmIntegrations } from '@backstage/integration';\nimport {\n EntityProvider,\n EntityProviderConnection,\n} from '@backstage/plugin-catalog-node';\nimport * as uuid from 'uuid';\nimport { Logger } from 'winston';\nimport {\n GitLabClient,\n GitlabProviderConfig,\n paginated,\n readGitlabConfigs,\n} from '../lib';\nimport { GitLabGroup, GitLabUser } from '../lib/types';\nimport {\n ANNOTATION_LOCATION,\n ANNOTATION_ORIGIN_LOCATION,\n Entity,\n UserEntity,\n GroupEntity,\n} from '@backstage/catalog-model';\nimport { merge } from 'lodash';\n\ntype Result = {\n scanned: number;\n matches: GitLabUser[];\n};\n\ntype GroupResult = {\n scanned: number;\n matches: GitLabGroup[];\n};\n\n/**\n * Discovers users and groups from a Gitlab instance.\n * @public\n */\nexport class GitlabOrgDiscoveryEntityProvider implements EntityProvider {\n private readonly config: GitlabProviderConfig;\n private readonly integration: GitLabIntegration;\n private readonly logger: Logger;\n private readonly scheduleFn: () => Promise<void>;\n private connection?: EntityProviderConnection;\n\n static fromConfig(\n config: Config,\n options: {\n logger: Logger;\n schedule?: TaskRunner;\n scheduler?: PluginTaskScheduler;\n },\n ): GitlabOrgDiscoveryEntityProvider[] {\n if (!options.schedule && !options.scheduler) {\n throw new Error('Either schedule or scheduler must be provided.');\n }\n\n const providerConfigs = readGitlabConfigs(\n config,\n options.logger.child({ target: 'GitlabOrgDiscoveryEntityProvider' }),\n );\n const integrations = ScmIntegrations.fromConfig(config).gitlab;\n const providers: GitlabOrgDiscoveryEntityProvider[] = [];\n\n providerConfigs.forEach(providerConfig => {\n const integration = integrations.byHost(providerConfig.host);\n\n if (!providerConfig.orgEnabled) {\n return;\n }\n\n if (!integration) {\n throw new Error(\n `No gitlab integration found that matches host ${providerConfig.host}`,\n );\n }\n\n if (!options.schedule && !providerConfig.schedule) {\n throw new Error(\n `No schedule provided neither via code nor config for GitlabOrgDiscoveryEntityProvider:${providerConfig.id}.`,\n );\n }\n\n const taskRunner =\n options.schedule ??\n options.scheduler!.createScheduledTaskRunner(providerConfig.schedule!);\n\n providers.push(\n new GitlabOrgDiscoveryEntityProvider({\n ...options,\n config: providerConfig,\n integration,\n taskRunner,\n }),\n );\n });\n return providers;\n }\n\n private constructor(options: {\n config: GitlabProviderConfig;\n integration: GitLabIntegration;\n logger: Logger;\n taskRunner: TaskRunner;\n }) {\n this.config = options.config;\n this.integration = options.integration;\n this.logger = options.logger.child({\n target: this.getProviderName(),\n });\n this.scheduleFn = this.createScheduleFn(options.taskRunner);\n }\n\n getProviderName(): string {\n return `GitlabOrgDiscoveryEntityProvider:${this.config.id}`;\n }\n\n async connect(connection: EntityProviderConnection): Promise<void> {\n this.connection = connection;\n await this.scheduleFn();\n }\n\n private createScheduleFn(taskRunner: TaskRunner): () => Promise<void> {\n return async () => {\n const taskId = `${this.getProviderName()}:refresh`;\n return taskRunner.run({\n id: taskId,\n fn: async () => {\n const logger = this.logger.child({\n class: GitlabOrgDiscoveryEntityProvider.prototype.constructor.name,\n taskId,\n taskInstanceId: uuid.v4(),\n });\n\n try {\n await this.refresh(logger);\n } catch (error) {\n logger.error(`${this.getProviderName()} refresh failed`, error);\n }\n },\n });\n };\n }\n\n private async refresh(logger: Logger): Promise<void> {\n if (!this.connection) {\n throw new Error(\n `Gitlab discovery connection not initialized for ${this.getProviderName()}`,\n );\n }\n\n const client = new GitLabClient({\n config: this.integration.config,\n logger: logger,\n });\n\n const users = paginated<GitLabUser>(options => client.listUsers(options), {\n page: 1,\n per_page: 100,\n active: true,\n });\n\n const groups = paginated<GitLabGroup>(\n options => client.listGroups(options),\n {\n page: 1,\n per_page: 100,\n },\n );\n\n const idMappedGroup: { [groupId: number]: GitLabGroup } = {};\n\n const res: Result = {\n scanned: 0,\n matches: [],\n };\n\n const groupRes: GroupResult = {\n scanned: 0,\n matches: [],\n };\n\n for await (const group of groups) {\n if (!this.config.groupPattern.test(group.full_path ?? '')) {\n continue;\n }\n\n if (\n this.config.group &&\n !group.full_path.startsWith(`${this.config.group}/`)\n ) {\n continue;\n }\n\n groupRes.scanned++;\n groupRes.matches.push(group);\n\n idMappedGroup[group.id] = group;\n }\n\n for await (const user of users) {\n if (!this.config.userPattern.test(user.email ?? user.username ?? '')) {\n continue;\n }\n\n res.scanned++;\n\n if (user.state !== 'active') {\n continue;\n }\n\n const memberships = await client.getUserMemberships(user.id);\n const userGroups: GitLabGroup[] = [];\n\n for (const i of memberships) {\n if (\n i.source_type === 'Namespace' &&\n idMappedGroup.hasOwnProperty(i.source_id)\n ) {\n userGroups.push(idMappedGroup[i.source_id]);\n }\n }\n\n user.groups = userGroups;\n\n res.matches.push(user);\n }\n\n const groupsWithUsers = groupRes.matches.filter(group => {\n return (\n res.matches.filter(x => {\n return !!x.groups?.find(y => y.id === group.id);\n }).length > 0\n );\n });\n\n const userEntities = res.matches.map(p =>\n this.createUserEntity(p, this.integration.config.host),\n );\n const groupEntities = this.createGroupEntities(\n groupsWithUsers,\n this.integration.config.host,\n );\n\n await this.connection.applyMutation({\n type: 'full',\n entities: [...userEntities, ...groupEntities].map(entity => ({\n locationKey: this.getProviderName(),\n entity: this.withLocations(\n this.integration.config.host,\n this.integration.config.baseUrl,\n entity,\n ),\n })),\n });\n }\n\n private createGroupEntities(\n groupResult: GitLabGroup[],\n host: string,\n ): GroupEntity[] {\n const idMapped: { [groupId: number]: GitLabGroup } = {};\n const entities: GroupEntity[] = [];\n\n for (const group of groupResult) {\n idMapped[group.id] = group;\n }\n\n for (const group of groupResult) {\n const entity = this.createGroupEntity(group, host);\n\n if (group.parent_id && idMapped.hasOwnProperty(group.parent_id)) {\n entity.spec.parent = this.groupName(\n idMapped[group.parent_id].full_path,\n );\n }\n\n entities.push(entity);\n }\n\n return entities;\n }\n\n private withLocations(host: string, baseUrl: string, entity: Entity): Entity {\n const location =\n entity.kind === 'Group'\n ? `url:${baseUrl}/${entity.metadata.annotations?.[`${host}/team-path`]}`\n : `url:${baseUrl}/${entity.metadata.name}`;\n return merge(\n {\n metadata: {\n annotations: {\n [ANNOTATION_LOCATION]: location,\n [ANNOTATION_ORIGIN_LOCATION]: location,\n },\n },\n },\n entity,\n ) as Entity;\n }\n\n private createUserEntity(user: GitLabUser, host: string): UserEntity {\n const annotations: { [annotationName: string]: string } = {};\n\n annotations[`${host}/user-login`] = user.web_url;\n\n const entity: UserEntity = {\n apiVersion: 'backstage.io/v1alpha1',\n kind: 'User',\n metadata: {\n name: user.username,\n annotations: annotations,\n },\n spec: {\n profile: {\n displayName: user.name,\n picture: user.avatar_url,\n },\n memberOf: [],\n },\n };\n\n if (user.email) {\n if (!entity.spec) {\n entity.spec = {};\n }\n\n if (!entity.spec.profile) {\n entity.spec.profile = {};\n }\n\n entity.spec.profile.email = user.email;\n }\n\n if (user.groups) {\n for (const group of user.groups) {\n if (!entity.spec.memberOf) {\n entity.spec.memberOf = [];\n }\n entity.spec.memberOf.push(this.groupName(group.full_path));\n }\n }\n\n return entity;\n }\n\n private groupName(full_path: string): string {\n if (this.config.group && full_path.startsWith(`${this.config.group}/`)) {\n return full_path\n .replace(`${this.config.group}/`, '')\n .replaceAll('/', '-');\n }\n return full_path.replaceAll('/', '-');\n }\n\n private createGroupEntity(group: GitLabGroup, host: string): GroupEntity {\n const annotations: { [annotationName: string]: string } = {};\n\n annotations[`${host}/team-path`] = group.full_path;\n\n const entity: GroupEntity = {\n apiVersion: 'backstage.io/v1alpha1',\n kind: 'Group',\n metadata: {\n name: this.groupName(group.full_path),\n annotations: annotations,\n },\n spec: {\n type: 'team',\n children: [],\n profile: {\n displayName: group.name,\n },\n },\n };\n\n if (group.description) {\n entity.metadata.description = group.description;\n }\n\n return entity;\n }\n}\n"],"names":["fetch","getGitLabRequestOptions","readTaskScheduleDefinitionFromConfig","ScmIntegrations","CacheManager","processingResult","uuid","locationSpecToLocationEntity","_a","merge","ANNOTATION_LOCATION","ANNOTATION_ORIGIN_LOCATION"],"mappings":"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"}
1
+ {"version":3,"file":"index.cjs.js","sources":["../src/lib/client.ts","../src/providers/config.ts","../src/GitLabDiscoveryProcessor.ts","../src/providers/GitlabDiscoveryEntityProvider.ts","../src/providers/GitlabOrgDiscoveryEntityProvider.ts"],"sourcesContent":["/*\n * Copyright 2021 The Backstage Authors\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nimport fetch from 'node-fetch';\nimport {\n getGitLabRequestOptions,\n GitLabIntegrationConfig,\n} from '@backstage/integration';\nimport { Logger } from 'winston';\nimport { GitLabGroup, GitLabGroupMembersResponse, GitLabUser } from './types';\n\nexport type CommonListOptions = {\n [key: string]: string | number | boolean | undefined;\n per_page?: number | undefined;\n page?: number | undefined;\n active?: boolean;\n};\n\ninterface ListProjectOptions extends CommonListOptions {\n group?: string;\n}\n\ninterface UserListOptions extends CommonListOptions {\n without_project_bots?: boolean | undefined;\n exclude_internal?: boolean | undefined;\n}\n\nexport type PagedResponse<T> = {\n items: T[];\n nextPage?: number;\n};\n\nexport class GitLabClient {\n private readonly config: GitLabIntegrationConfig;\n private readonly logger: Logger;\n\n constructor(options: { config: GitLabIntegrationConfig; logger: Logger }) {\n this.config = options.config;\n this.logger = options.logger;\n }\n\n /**\n * Indicates whether the client is for a SaaS or self managed GitLab instance.\n */\n isSelfManaged(): boolean {\n return this.config.host !== 'gitlab.com';\n }\n\n async listProjects(\n options?: ListProjectOptions,\n ): Promise<PagedResponse<any>> {\n if (options?.group) {\n return this.pagedRequest(\n `/groups/${encodeURIComponent(options?.group)}/projects`,\n {\n ...options,\n include_subgroups: true,\n },\n );\n }\n\n return this.pagedRequest(`/projects`, options);\n }\n\n async listUsers(\n options?: UserListOptions,\n ): Promise<PagedResponse<GitLabUser>> {\n return this.pagedRequest(`/users?`, {\n ...options,\n without_project_bots: true,\n exclude_internal: true,\n });\n }\n\n async listGroups(\n options?: CommonListOptions,\n ): Promise<PagedResponse<GitLabGroup>> {\n return this.pagedRequest(`/groups`, options);\n }\n\n async getGroupMembers(groupPath: string): Promise<number[]> {\n const memberIds = [];\n let hasNextPage: boolean = false;\n let endCursor: string | null = null;\n do {\n const response: GitLabGroupMembersResponse = await fetch(\n `${this.config.baseUrl}/api/graphql`,\n {\n method: 'POST',\n headers: {\n ...getGitLabRequestOptions(this.config).headers,\n ['Content-Type']: 'application/json',\n },\n body: JSON.stringify({\n variables: { group: groupPath, endCursor },\n query: `query($group: ID!, $endCursor: String) {\n group(fullPath: $group) {\n groupMembers(first: 100, relations: [DIRECT], after: $endCursor) {\n nodes {\n user {\n id\n }\n }\n pageInfo {\n endCursor\n hasNextPage\n }\n }\n }\n }`,\n }),\n },\n ).then(r => r.json());\n if (response.errors) {\n throw new Error(`GraphQL errors: ${JSON.stringify(response.errors)}`);\n }\n memberIds.push(\n ...response.data.group.groupMembers.nodes.map(\n (node: { user: { id: string } }) =>\n Number(node.user.id.replace(/^gid:\\/\\/gitlab\\/User\\//, '')),\n ),\n );\n ({ hasNextPage, endCursor } = response.data.group.groupMembers.pageInfo);\n } while (hasNextPage);\n return memberIds;\n }\n\n /**\n * General existence check.\n *\n * @param projectPath - The path to the project\n * @param branch - The branch used to search\n * @param filePath - The path to the file\n */\n async hasFile(\n projectPath: string,\n branch: string,\n filePath: string,\n ): Promise<boolean> {\n const endpoint: string = `/projects/${encodeURIComponent(\n projectPath,\n )}/repository/files/${encodeURIComponent(filePath)}`;\n const request = new URL(`${this.config.apiBaseUrl}${endpoint}`);\n request.searchParams.append('ref', branch);\n\n const response = await fetch(request.toString(), {\n headers: getGitLabRequestOptions(this.config).headers,\n method: 'HEAD',\n });\n\n if (!response.ok) {\n if (response.status >= 500) {\n this.logger.debug(\n `Unexpected response when fetching ${request.toString()}. Expected 200 but got ${\n response.status\n } - ${response.statusText}`,\n );\n }\n return false;\n }\n\n return true;\n }\n\n /**\n * Performs a request against a given paginated GitLab endpoint.\n *\n * This method may be used to perform authenticated REST calls against any\n * paginated GitLab endpoint which uses X-NEXT-PAGE headers. The return value\n * can be be used with the {@link paginated} async-generator function to yield\n * each item from the paged request.\n *\n * @see {@link paginated}\n * @param endpoint - The request endpoint, e.g. /projects.\n * @param options - Request queryString options which may also include page variables.\n */\n async pagedRequest<T = any>(\n endpoint: string,\n options?: CommonListOptions,\n ): Promise<PagedResponse<T>> {\n const request = new URL(`${this.config.apiBaseUrl}${endpoint}`);\n for (const key in options) {\n if (options[key]) {\n request.searchParams.append(key, options[key]!.toString());\n }\n }\n\n this.logger.debug(`Fetching: ${request.toString()}`);\n const response = await fetch(\n request.toString(),\n getGitLabRequestOptions(this.config),\n );\n if (!response.ok) {\n throw new Error(\n `Unexpected response when fetching ${request.toString()}. Expected 200 but got ${\n response.status\n } - ${response.statusText}`,\n );\n }\n return response.json().then(items => {\n const nextPage = response.headers.get('x-next-page');\n\n return {\n items,\n nextPage: nextPage ? Number(nextPage) : null,\n } as PagedResponse<any>;\n });\n }\n}\n\n/**\n * Advances through each page and provides each item from a paginated request.\n *\n * The async generator function yields each item from repeated calls to the\n * provided request function. The generator walks through each available page by\n * setting the page key in the options passed into the request function and\n * making repeated calls until there are no more pages.\n *\n * @see {@link pagedRequest}\n * @param request - Function which returns a PagedResponse to walk through.\n * @param options - Initial ListOptions for the request function.\n */\nexport async function* paginated<T = any>(\n request: (options: CommonListOptions) => Promise<PagedResponse<T>>,\n options: CommonListOptions,\n) {\n let res;\n do {\n res = await request(options);\n options.page = res.nextPage;\n for (const item of res.items) {\n yield item;\n }\n } while (res.nextPage);\n}\n","/*\n * Copyright 2022 The Backstage Authors\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nimport { readTaskScheduleDefinitionFromConfig } from '@backstage/backend-tasks';\nimport { Config } from '@backstage/config';\nimport { GitlabProviderConfig } from '../lib';\n\n/**\n * Extracts the gitlab config from a config object\n *\n * @public\n *\n * @param id - The provider key\n * @param config - The config object to extract from\n */\nfunction readGitlabConfig(id: string, config: Config): GitlabProviderConfig {\n const group = config.getOptionalString('group') ?? '';\n const host = config.getString('host');\n const branch = config.getOptionalString('branch');\n const fallbackBranch = config.getOptionalString('fallbackBranch') ?? 'master';\n const catalogFile =\n config.getOptionalString('entityFilename') ?? 'catalog-info.yaml';\n const projectPattern = new RegExp(\n config.getOptionalString('projectPattern') ?? /[\\s\\S]*/,\n );\n const userPattern = new RegExp(\n config.getOptionalString('userPattern') ?? /[\\s\\S]*/,\n );\n const groupPattern = new RegExp(\n config.getOptionalString('groupPattern') ?? /[\\s\\S]*/,\n );\n const orgEnabled: boolean = config.getOptionalBoolean('orgEnabled') ?? false;\n\n const schedule = config.has('schedule')\n ? readTaskScheduleDefinitionFromConfig(config.getConfig('schedule'))\n : undefined;\n\n return {\n id,\n group,\n branch,\n fallbackBranch,\n host,\n catalogFile,\n projectPattern,\n userPattern,\n groupPattern,\n schedule,\n orgEnabled,\n };\n}\n\n/**\n * Extracts the gitlab config from a config object array\n *\n * @public\n *\n * @param config - The config object to extract from\n */\nexport function readGitlabConfigs(config: Config): GitlabProviderConfig[] {\n const configs: GitlabProviderConfig[] = [];\n\n const providerConfigs = config.getOptionalConfig('catalog.providers.gitlab');\n\n if (!providerConfigs) {\n return configs;\n }\n\n for (const id of providerConfigs.keys()) {\n configs.push(readGitlabConfig(id, providerConfigs.getConfig(id)));\n }\n\n return configs;\n}\n","/*\n * Copyright 2021 The Backstage Authors\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nimport {\n CacheClient,\n CacheManager,\n PluginCacheManager,\n} from '@backstage/backend-common';\nimport { Config } from '@backstage/config';\nimport {\n ScmIntegrationRegistry,\n ScmIntegrations,\n} from '@backstage/integration';\nimport {\n CatalogProcessor,\n CatalogProcessorEmit,\n LocationSpec,\n processingResult,\n} from '@backstage/plugin-catalog-node';\nimport { Logger } from 'winston';\nimport { GitLabClient, GitLabProject, paginated } from './lib';\n\n/**\n * Extracts repositories out of an GitLab instance.\n * @public\n */\nexport class GitLabDiscoveryProcessor implements CatalogProcessor {\n private readonly integrations: ScmIntegrationRegistry;\n private readonly logger: Logger;\n private readonly cache: CacheClient;\n private readonly skipReposWithoutExactFileMatch: boolean;\n\n static fromConfig(\n config: Config,\n options: { logger: Logger; skipReposWithoutExactFileMatch?: boolean },\n ): GitLabDiscoveryProcessor {\n const integrations = ScmIntegrations.fromConfig(config);\n const pluginCache =\n CacheManager.fromConfig(config).forPlugin('gitlab-discovery');\n\n return new GitLabDiscoveryProcessor({\n ...options,\n integrations,\n pluginCache,\n });\n }\n\n private constructor(options: {\n integrations: ScmIntegrationRegistry;\n pluginCache: PluginCacheManager;\n logger: Logger;\n skipReposWithoutExactFileMatch?: boolean;\n }) {\n this.integrations = options.integrations;\n this.cache = options.pluginCache.getClient();\n this.logger = options.logger;\n this.skipReposWithoutExactFileMatch =\n options.skipReposWithoutExactFileMatch || false;\n }\n\n getProcessorName(): string {\n return 'GitLabDiscoveryProcessor';\n }\n\n async readLocation(\n location: LocationSpec,\n _optional: boolean,\n emit: CatalogProcessorEmit,\n ): Promise<boolean> {\n if (location.type !== 'gitlab-discovery') {\n return false;\n }\n\n const startTime = new Date();\n const { group, host, branch, catalogPath } = parseUrl(location.target);\n\n const integration = this.integrations.gitlab.byUrl(`https://${host}`);\n if (!integration) {\n throw new Error(\n `There is no GitLab integration that matches ${host}. Please add a configuration entry for it under integrations.gitlab`,\n );\n }\n\n const client = new GitLabClient({\n config: integration.config,\n logger: this.logger,\n });\n this.logger.debug(`Reading GitLab projects from ${location.target}`);\n\n const lastActivity = (await this.cache.get(this.getCacheKey())) as string;\n const opts = {\n group,\n page: 1,\n // We check for the existence of lastActivity and only set it if it's present to ensure\n // that the options doesn't include the key so that the API doesn't receive an empty query parameter.\n ...(lastActivity && { last_activity_after: lastActivity }),\n };\n\n const projects = paginated(options => client.listProjects(options), opts);\n\n const res: Result = {\n scanned: 0,\n matches: [],\n };\n for await (const project of projects) {\n res.scanned++;\n\n if (project.archived) {\n continue;\n }\n\n if (branch === '*' && project.default_branch === undefined) {\n continue;\n }\n\n if (this.skipReposWithoutExactFileMatch) {\n const project_branch = branch === '*' ? project.default_branch : branch;\n\n const projectHasFile: boolean = await client.hasFile(\n project.path_with_namespace,\n project_branch,\n catalogPath,\n );\n\n if (!projectHasFile) {\n continue;\n }\n }\n\n res.matches.push(project);\n }\n\n for (const project of res.matches) {\n const project_branch = branch === '*' ? project.default_branch : branch;\n\n emit(\n processingResult.location({\n type: 'url',\n // The format expected by the GitLabUrlReader:\n // https://gitlab.com/groupA/teams/teamA/subgroupA/repoA/-/blob/branch/filepath\n //\n // This unfortunately will trigger another API call in `getGitLabFileFetchUrl` to get the project ID.\n // The alternative is using the `buildRawUrl` function, which does not support subgroups, so providing a raw\n // URL here won't work either.\n target: `${project.web_url}/-/blob/${project_branch}/${catalogPath}`,\n presence: 'optional',\n }),\n );\n }\n\n // Save an ISO formatted string in the cache as that's what GitLab expects in the API request.\n await this.cache.set(this.getCacheKey(), startTime.toISOString());\n\n const duration = ((Date.now() - startTime.getTime()) / 1000).toFixed(1);\n this.logger.debug(\n `Read ${res.scanned} GitLab repositories in ${duration} seconds`,\n );\n\n return true;\n }\n\n private getCacheKey(): string {\n return `processors/${this.getProcessorName()}/last-activity`;\n }\n}\n\ntype Result = {\n scanned: number;\n matches: GitLabProject[];\n};\n\n/*\n * Helpers\n */\n\nexport function parseUrl(urlString: string): {\n group?: string;\n host: string;\n branch: string;\n catalogPath: string;\n} {\n const url = new URL(urlString);\n const path = url.pathname.slice(1).split('/');\n\n // (/group/subgroup)/blob/branch|*/filepath\n const blobIndex = path.findIndex(p => p === 'blob');\n if (blobIndex !== -1 && path.length > blobIndex + 2) {\n const group =\n blobIndex > 0 ? path.slice(0, blobIndex).join('/') : undefined;\n\n return {\n group,\n host: url.host,\n branch: decodeURIComponent(path[blobIndex + 1]),\n catalogPath: decodeURIComponent(path.slice(blobIndex + 2).join('/')),\n };\n }\n\n throw new Error(`Failed to parse ${urlString}`);\n}\n","/*\n * Copyright 2021 The Backstage Authors\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nimport { PluginTaskScheduler, TaskRunner } from '@backstage/backend-tasks';\nimport { Config } from '@backstage/config';\nimport { GitLabIntegration, ScmIntegrations } from '@backstage/integration';\nimport {\n EntityProvider,\n EntityProviderConnection,\n LocationSpec,\n locationSpecToLocationEntity,\n} from '@backstage/plugin-catalog-node';\nimport * as uuid from 'uuid';\nimport { Logger } from 'winston';\nimport {\n GitLabClient,\n GitLabProject,\n GitlabProviderConfig,\n paginated,\n readGitlabConfigs,\n} from '../lib';\n\ntype Result = {\n scanned: number;\n matches: GitLabProject[];\n};\n\n/**\n * Discovers entity definition files in the groups of a Gitlab instance.\n * @public\n */\nexport class GitlabDiscoveryEntityProvider implements EntityProvider {\n private readonly config: GitlabProviderConfig;\n private readonly integration: GitLabIntegration;\n private readonly logger: Logger;\n private readonly scheduleFn: () => Promise<void>;\n private connection?: EntityProviderConnection;\n\n static fromConfig(\n config: Config,\n options: {\n logger: Logger;\n schedule?: TaskRunner;\n scheduler?: PluginTaskScheduler;\n },\n ): GitlabDiscoveryEntityProvider[] {\n if (!options.schedule && !options.scheduler) {\n throw new Error('Either schedule or scheduler must be provided.');\n }\n\n const providerConfigs = readGitlabConfigs(config);\n const integrations = ScmIntegrations.fromConfig(config).gitlab;\n const providers: GitlabDiscoveryEntityProvider[] = [];\n\n providerConfigs.forEach(providerConfig => {\n const integration = integrations.byHost(providerConfig.host);\n if (!integration) {\n throw new Error(\n `No gitlab integration found that matches host ${providerConfig.host}`,\n );\n }\n\n if (!options.schedule && !providerConfig.schedule) {\n throw new Error(\n `No schedule provided neither via code nor config for GitlabDiscoveryEntityProvider:${providerConfig.id}.`,\n );\n }\n\n const taskRunner =\n options.schedule ??\n options.scheduler!.createScheduledTaskRunner(providerConfig.schedule!);\n\n providers.push(\n new GitlabDiscoveryEntityProvider({\n ...options,\n config: providerConfig,\n integration,\n taskRunner,\n }),\n );\n });\n return providers;\n }\n\n private constructor(options: {\n config: GitlabProviderConfig;\n integration: GitLabIntegration;\n logger: Logger;\n taskRunner: TaskRunner;\n }) {\n this.config = options.config;\n this.integration = options.integration;\n this.logger = options.logger.child({\n target: this.getProviderName(),\n });\n this.scheduleFn = this.createScheduleFn(options.taskRunner);\n }\n\n getProviderName(): string {\n return `GitlabDiscoveryEntityProvider:${this.config.id}`;\n }\n\n async connect(connection: EntityProviderConnection): Promise<void> {\n this.connection = connection;\n await this.scheduleFn();\n }\n\n private createScheduleFn(taskRunner: TaskRunner): () => Promise<void> {\n return async () => {\n const taskId = `${this.getProviderName()}:refresh`;\n return taskRunner.run({\n id: taskId,\n fn: async () => {\n const logger = this.logger.child({\n class: GitlabDiscoveryEntityProvider.prototype.constructor.name,\n taskId,\n taskInstanceId: uuid.v4(),\n });\n\n try {\n await this.refresh(logger);\n } catch (error) {\n logger.error(`${this.getProviderName()} refresh failed`, error);\n }\n },\n });\n };\n }\n\n async refresh(logger: Logger): Promise<void> {\n if (!this.connection) {\n throw new Error(\n `Gitlab discovery connection not initialized for ${this.getProviderName()}`,\n );\n }\n\n const client = new GitLabClient({\n config: this.integration.config,\n logger: logger,\n });\n\n const projects = paginated<GitLabProject>(\n options => client.listProjects(options),\n {\n group: this.config.group,\n page: 1,\n per_page: 50,\n },\n );\n\n const res: Result = {\n scanned: 0,\n matches: [],\n };\n\n for await (const project of projects) {\n if (!this.config.projectPattern.test(project.path_with_namespace ?? '')) {\n continue;\n }\n\n res.scanned++;\n\n if (project.archived) {\n continue;\n }\n\n if (\n !this.config.branch &&\n this.config.fallbackBranch === '*' &&\n project.default_branch === undefined\n ) {\n continue;\n }\n\n const project_branch =\n this.config.branch ??\n project.default_branch ??\n this.config.fallbackBranch;\n\n const projectHasFile: boolean = await client.hasFile(\n project.path_with_namespace ?? '',\n project_branch,\n this.config.catalogFile,\n );\n if (projectHasFile) {\n res.matches.push(project);\n }\n }\n\n const locations = res.matches.map(p => this.createLocationSpec(p));\n await this.connection.applyMutation({\n type: 'full',\n entities: locations.map(location => ({\n locationKey: this.getProviderName(),\n entity: locationSpecToLocationEntity({ location }),\n })),\n });\n }\n\n private createLocationSpec(project: GitLabProject): LocationSpec {\n const project_branch =\n this.config.branch ??\n project.default_branch ??\n this.config.fallbackBranch;\n return {\n type: 'url',\n target: `${project.web_url}/-/blob/${project_branch}/${this.config.catalogFile}`,\n presence: 'optional',\n };\n }\n}\n","/*\n * Copyright 2021 The Backstage Authors\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nimport { PluginTaskScheduler, TaskRunner } from '@backstage/backend-tasks';\nimport { Config } from '@backstage/config';\nimport { GitLabIntegration, ScmIntegrations } from '@backstage/integration';\nimport {\n EntityProvider,\n EntityProviderConnection,\n} from '@backstage/plugin-catalog-node';\nimport * as uuid from 'uuid';\nimport { Logger } from 'winston';\nimport {\n GitLabClient,\n GitlabProviderConfig,\n paginated,\n readGitlabConfigs,\n} from '../lib';\nimport { GitLabGroup, GitLabUser } from '../lib/types';\nimport {\n ANNOTATION_LOCATION,\n ANNOTATION_ORIGIN_LOCATION,\n Entity,\n UserEntity,\n GroupEntity,\n} from '@backstage/catalog-model';\nimport { merge } from 'lodash';\n\ntype Result = {\n scanned: number;\n matches: GitLabUser[];\n};\n\ntype GroupResult = {\n scanned: number;\n matches: GitLabGroup[];\n};\n\n/**\n * Discovers users and groups from a Gitlab instance.\n * @public\n */\nexport class GitlabOrgDiscoveryEntityProvider implements EntityProvider {\n private readonly config: GitlabProviderConfig;\n private readonly integration: GitLabIntegration;\n private readonly logger: Logger;\n private readonly scheduleFn: () => Promise<void>;\n private connection?: EntityProviderConnection;\n\n static fromConfig(\n config: Config,\n options: {\n logger: Logger;\n schedule?: TaskRunner;\n scheduler?: PluginTaskScheduler;\n },\n ): GitlabOrgDiscoveryEntityProvider[] {\n if (!options.schedule && !options.scheduler) {\n throw new Error('Either schedule or scheduler must be provided.');\n }\n\n const providerConfigs = readGitlabConfigs(config);\n const integrations = ScmIntegrations.fromConfig(config).gitlab;\n const providers: GitlabOrgDiscoveryEntityProvider[] = [];\n\n providerConfigs.forEach(providerConfig => {\n const integration = integrations.byHost(providerConfig.host);\n\n if (!providerConfig.orgEnabled) {\n return;\n }\n\n if (!integration) {\n throw new Error(\n `No gitlab integration found that matches host ${providerConfig.host}`,\n );\n }\n\n if (!options.schedule && !providerConfig.schedule) {\n throw new Error(\n `No schedule provided neither via code nor config for GitlabOrgDiscoveryEntityProvider:${providerConfig.id}.`,\n );\n }\n\n const taskRunner =\n options.schedule ??\n options.scheduler!.createScheduledTaskRunner(providerConfig.schedule!);\n\n providers.push(\n new GitlabOrgDiscoveryEntityProvider({\n ...options,\n config: providerConfig,\n integration,\n taskRunner,\n }),\n );\n });\n return providers;\n }\n\n private constructor(options: {\n config: GitlabProviderConfig;\n integration: GitLabIntegration;\n logger: Logger;\n taskRunner: TaskRunner;\n }) {\n this.config = options.config;\n this.integration = options.integration;\n this.logger = options.logger.child({\n target: this.getProviderName(),\n });\n this.scheduleFn = this.createScheduleFn(options.taskRunner);\n }\n\n getProviderName(): string {\n return `GitlabOrgDiscoveryEntityProvider:${this.config.id}`;\n }\n\n async connect(connection: EntityProviderConnection): Promise<void> {\n this.connection = connection;\n await this.scheduleFn();\n }\n\n private createScheduleFn(taskRunner: TaskRunner): () => Promise<void> {\n return async () => {\n const taskId = `${this.getProviderName()}:refresh`;\n return taskRunner.run({\n id: taskId,\n fn: async () => {\n const logger = this.logger.child({\n class: GitlabOrgDiscoveryEntityProvider.prototype.constructor.name,\n taskId,\n taskInstanceId: uuid.v4(),\n });\n\n try {\n await this.refresh(logger);\n } catch (error) {\n logger.error(`${this.getProviderName()} refresh failed`, error);\n }\n },\n });\n };\n }\n\n private async refresh(logger: Logger): Promise<void> {\n if (!this.connection) {\n throw new Error(\n `Gitlab discovery connection not initialized for ${this.getProviderName()}`,\n );\n }\n\n const client = new GitLabClient({\n config: this.integration.config,\n logger: logger,\n });\n\n const users = paginated<GitLabUser>(options => client.listUsers(options), {\n page: 1,\n per_page: 100,\n active: true,\n });\n\n const groups = paginated<GitLabGroup>(\n options => client.listGroups(options),\n {\n page: 1,\n per_page: 100,\n },\n );\n\n const idMappedUser: { [userId: number]: GitLabUser } = {};\n\n const res: Result = {\n scanned: 0,\n matches: [],\n };\n\n const groupRes: GroupResult = {\n scanned: 0,\n matches: [],\n };\n\n for await (const user of users) {\n if (!this.config.userPattern.test(user.email ?? user.username ?? '')) {\n continue;\n }\n\n res.scanned++;\n\n if (user.state !== 'active') {\n continue;\n }\n\n idMappedUser[user.id] = user;\n res.matches.push(user);\n }\n\n for await (const group of groups) {\n if (!this.config.groupPattern.test(group.full_path ?? '')) {\n continue;\n }\n\n if (\n this.config.group &&\n !group.full_path.startsWith(`${this.config.group}/`)\n ) {\n continue;\n }\n\n groupRes.scanned++;\n groupRes.matches.push(group);\n\n for (const id of await client.getGroupMembers(group.full_path)) {\n const user = idMappedUser[id];\n if (user) {\n user.groups = (user.groups ?? []).concat(group);\n }\n }\n }\n\n const groupsWithUsers = groupRes.matches.filter(group => {\n return (\n res.matches.filter(x => {\n return !!x.groups?.find(y => y.id === group.id);\n }).length > 0\n );\n });\n\n const userEntities = res.matches.map(p =>\n this.createUserEntity(p, this.integration.config.host),\n );\n const groupEntities = this.createGroupEntities(\n groupsWithUsers,\n this.integration.config.host,\n );\n\n await this.connection.applyMutation({\n type: 'full',\n entities: [...userEntities, ...groupEntities].map(entity => ({\n locationKey: this.getProviderName(),\n entity: this.withLocations(\n this.integration.config.host,\n this.integration.config.baseUrl,\n entity,\n ),\n })),\n });\n }\n\n private createGroupEntities(\n groupResult: GitLabGroup[],\n host: string,\n ): GroupEntity[] {\n const idMapped: { [groupId: number]: GitLabGroup } = {};\n const entities: GroupEntity[] = [];\n\n for (const group of groupResult) {\n idMapped[group.id] = group;\n }\n\n for (const group of groupResult) {\n const entity = this.createGroupEntity(group, host);\n\n if (group.parent_id && idMapped.hasOwnProperty(group.parent_id)) {\n entity.spec.parent = this.groupName(\n idMapped[group.parent_id].full_path,\n );\n }\n\n entities.push(entity);\n }\n\n return entities;\n }\n\n private withLocations(host: string, baseUrl: string, entity: Entity): Entity {\n const location =\n entity.kind === 'Group'\n ? `url:${baseUrl}/${entity.metadata.annotations?.[`${host}/team-path`]}`\n : `url:${baseUrl}/${entity.metadata.name}`;\n return merge(\n {\n metadata: {\n annotations: {\n [ANNOTATION_LOCATION]: location,\n [ANNOTATION_ORIGIN_LOCATION]: location,\n },\n },\n },\n entity,\n ) as Entity;\n }\n\n private createUserEntity(user: GitLabUser, host: string): UserEntity {\n const annotations: { [annotationName: string]: string } = {};\n\n annotations[`${host}/user-login`] = user.web_url;\n\n const entity: UserEntity = {\n apiVersion: 'backstage.io/v1alpha1',\n kind: 'User',\n metadata: {\n name: user.username,\n annotations: annotations,\n },\n spec: {\n profile: {\n displayName: user.name,\n picture: user.avatar_url,\n },\n memberOf: [],\n },\n };\n\n if (user.email) {\n if (!entity.spec) {\n entity.spec = {};\n }\n\n if (!entity.spec.profile) {\n entity.spec.profile = {};\n }\n\n entity.spec.profile.email = user.email;\n }\n\n if (user.groups) {\n for (const group of user.groups) {\n if (!entity.spec.memberOf) {\n entity.spec.memberOf = [];\n }\n entity.spec.memberOf.push(this.groupName(group.full_path));\n }\n }\n\n return entity;\n }\n\n private groupName(full_path: string): string {\n if (this.config.group && full_path.startsWith(`${this.config.group}/`)) {\n return full_path\n .replace(`${this.config.group}/`, '')\n .replaceAll('/', '-');\n }\n return full_path.replaceAll('/', '-');\n }\n\n private createGroupEntity(group: GitLabGroup, host: string): GroupEntity {\n const annotations: { [annotationName: string]: string } = {};\n\n annotations[`${host}/team-path`] = group.full_path;\n\n const entity: GroupEntity = {\n apiVersion: 'backstage.io/v1alpha1',\n kind: 'Group',\n metadata: {\n name: this.groupName(group.full_path),\n annotations: annotations,\n },\n spec: {\n type: 'team',\n children: [],\n profile: {\n displayName: group.name,\n },\n },\n };\n\n if (group.description) {\n entity.metadata.description = group.description;\n }\n\n return entity;\n }\n}\n"],"names":["fetch","getGitLabRequestOptions","readTaskScheduleDefinitionFromConfig","ScmIntegrations","CacheManager","processingResult","uuid","locationSpecToLocationEntity","_a","merge","ANNOTATION_LOCATION","ANNOTATION_ORIGIN_LOCATION"],"mappings":"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"}
package/package.json CHANGED
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "name": "@backstage/plugin-catalog-backend-module-gitlab",
3
3
  "description": "A Backstage catalog backend module that helps integrate towards GitLab",
4
- "version": "0.1.14",
4
+ "version": "0.2.0-next.0",
5
5
  "main": "./dist/index.cjs.js",
6
6
  "types": "./dist/index.d.ts",
7
7
  "license": "Apache-2.0",
@@ -43,14 +43,14 @@
43
43
  "clean": "backstage-cli package clean"
44
44
  },
45
45
  "dependencies": {
46
- "@backstage/backend-common": "^0.18.3",
47
- "@backstage/backend-plugin-api": "^0.5.0",
48
- "@backstage/backend-tasks": "^0.5.0",
46
+ "@backstage/backend-common": "^0.18.4-next.0",
47
+ "@backstage/backend-plugin-api": "^0.5.1-next.0",
48
+ "@backstage/backend-tasks": "^0.5.1-next.0",
49
49
  "@backstage/catalog-model": "^1.2.1",
50
50
  "@backstage/config": "^1.0.7",
51
51
  "@backstage/errors": "^1.1.5",
52
52
  "@backstage/integration": "^1.4.3",
53
- "@backstage/plugin-catalog-node": "^1.3.4",
53
+ "@backstage/plugin-catalog-node": "^1.3.5-next.0",
54
54
  "@backstage/types": "^1.0.2",
55
55
  "lodash": "^4.17.21",
56
56
  "node-fetch": "^2.6.7",
@@ -58,8 +58,8 @@
58
58
  "winston": "^3.2.1"
59
59
  },
60
60
  "devDependencies": {
61
- "@backstage/backend-test-utils": "^0.1.35",
62
- "@backstage/cli": "^0.22.4",
61
+ "@backstage/backend-test-utils": "^0.1.36-next.0",
62
+ "@backstage/cli": "^0.22.6-next.0",
63
63
  "@types/lodash": "^4.14.151",
64
64
  "@types/uuid": "^8.0.0",
65
65
  "luxon": "^3.0.0",