@backstage/plugin-catalog-backend-module-github 0.10.2-next.1 → 0.10.2

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,18 @@
1
1
  # @backstage/plugin-catalog-backend-module-github
2
2
 
3
+ ## 0.10.2
4
+
5
+ ### Patch Changes
6
+
7
+ - d421b16: This change introduces single user versions of the user group resolution code in the multi org provider that will not page through all membership when updating a single user.
8
+ - f6c64d1: Fix GitHub catalog entity provider branch matching on event processing.
9
+ - Updated dependencies
10
+ - @backstage/plugin-catalog-backend@3.0.1
11
+ - @backstage/catalog-client@1.11.0
12
+ - @backstage/plugin-catalog-node@1.18.0
13
+ - @backstage/backend-plugin-api@1.4.2
14
+ - @backstage/plugin-events-node@0.4.14
15
+
3
16
  ## 0.10.2-next.1
4
17
 
5
18
  ### Patch Changes
@@ -1 +1 @@
1
- {"version":3,"file":"alpha.cjs.js","sources":["../src/alpha.ts"],"sourcesContent":["/*\n * Copyright 2023 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 { default as feature } from './module';\n\n/** @alpha */\nconst _feature = feature;\nexport default _feature;\n"],"names":["feature"],"mappings":";;;;;;AAmBA,MAAM,QAAW,GAAAA;;;;"}
1
+ {"version":3,"file":"alpha.cjs.js","sources":["../src/alpha.ts"],"sourcesContent":["/*\n * Copyright 2023 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 { default as feature } from './module';\n\n/** @alpha */\nconst _feature = feature;\nexport default _feature;\n"],"names":["feature"],"mappings":";;;;;;AAmBA,MAAM,QAAA,GAAWA;;;;"}
@@ -1 +1 @@
1
- {"version":3,"file":"GithubLocationAnalyzer.cjs.js","sources":["../../src/analyzers/GithubLocationAnalyzer.ts"],"sourcesContent":["/*\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 DefaultGithubCredentialsProvider,\n GithubCredentialsProvider,\n ScmIntegrationRegistry,\n ScmIntegrations,\n} from '@backstage/integration';\nimport { Octokit } from '@octokit/rest';\nimport { isEmpty, trimEnd } from 'lodash';\nimport parseGitUrl from 'git-url-parse';\nimport {\n AnalyzeOptions,\n CatalogService,\n ScmLocationAnalyzer,\n} from '@backstage/plugin-catalog-node';\nimport { Config } from '@backstage/config';\nimport { AuthService } from '@backstage/backend-plugin-api';\nimport { extname } from 'path';\n\n/** @public */\nexport type GithubLocationAnalyzerOptions = {\n config: Config;\n auth: AuthService;\n githubCredentialsProvider?: GithubCredentialsProvider;\n catalog: CatalogService;\n};\n\n/** @public */\nexport class GithubLocationAnalyzer implements ScmLocationAnalyzer {\n private readonly catalogClient: CatalogService;\n private readonly githubCredentialsProvider: GithubCredentialsProvider;\n private readonly integrations: ScmIntegrationRegistry;\n private readonly auth: AuthService;\n\n constructor(options: GithubLocationAnalyzerOptions) {\n this.catalogClient = options.catalog;\n this.integrations = ScmIntegrations.fromConfig(options.config);\n this.githubCredentialsProvider =\n options.githubCredentialsProvider ||\n DefaultGithubCredentialsProvider.fromIntegrations(this.integrations);\n\n this.auth = options.auth;\n }\n\n supports(url: string) {\n const integration = this.integrations.byUrl(url);\n return integration?.type === 'github';\n }\n\n async analyze(options: AnalyzeOptions) {\n const { url, catalogFilename } = options;\n const { owner, name: repo } = parseGitUrl(url);\n\n const catalogFile = catalogFilename || 'catalog-info.yaml';\n const extension = extname(catalogFile);\n const extensionQuery = !isEmpty(extension)\n ? `extension:${extension.replace('.', '')}`\n : '';\n\n const query = `filename:${catalogFile} ${extensionQuery} repo:${owner}/${repo}`;\n\n const integration = this.integrations.github.byUrl(url);\n if (!integration) {\n throw new Error('Make sure you have a GitHub integration configured');\n }\n\n const { token: githubToken } =\n await this.githubCredentialsProvider.getCredentials({\n url,\n });\n\n const octokitClient = new Octokit({\n auth: githubToken,\n baseUrl: integration.config.apiBaseUrl,\n });\n\n const searchResult = await octokitClient.search\n .code({ q: query })\n .catch(e => {\n throw new Error(`Couldn't search repository for metadata file, ${e}`);\n });\n\n const exists = searchResult.data.total_count > 0;\n if (exists) {\n const repoInformation = await octokitClient.repos\n .get({ owner, repo })\n .catch(e => {\n throw new Error(`Couldn't fetch repo data, ${e}`);\n });\n const defaultBranch = repoInformation.data.default_branch;\n\n const result = await Promise.all(\n searchResult.data.items\n .map(i => `${trimEnd(url, '/')}/blob/${defaultBranch}/${i.path}`)\n .map(async target => {\n const addLocationResult = await this.catalogClient.addLocation(\n {\n type: 'url',\n target,\n dryRun: true,\n },\n { credentials: await this.auth.getOwnServiceCredentials() },\n );\n return addLocationResult.entities.map(e => ({\n location: { type: 'url', target },\n isRegistered: !!addLocationResult.exists,\n entity: e,\n }));\n }),\n );\n\n return { existing: result.flat() };\n }\n return { existing: [] };\n }\n}\n"],"names":["ScmIntegrations","DefaultGithubCredentialsProvider","parseGitUrl","extname","isEmpty","Octokit","trimEnd"],"mappings":"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"}
1
+ {"version":3,"file":"GithubLocationAnalyzer.cjs.js","sources":["../../src/analyzers/GithubLocationAnalyzer.ts"],"sourcesContent":["/*\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 DefaultGithubCredentialsProvider,\n GithubCredentialsProvider,\n ScmIntegrationRegistry,\n ScmIntegrations,\n} from '@backstage/integration';\nimport { Octokit } from '@octokit/rest';\nimport { isEmpty, trimEnd } from 'lodash';\nimport parseGitUrl from 'git-url-parse';\nimport {\n AnalyzeOptions,\n CatalogService,\n ScmLocationAnalyzer,\n} from '@backstage/plugin-catalog-node';\nimport { Config } from '@backstage/config';\nimport { AuthService } from '@backstage/backend-plugin-api';\nimport { extname } from 'path';\n\n/** @public */\nexport type GithubLocationAnalyzerOptions = {\n config: Config;\n auth: AuthService;\n githubCredentialsProvider?: GithubCredentialsProvider;\n catalog: CatalogService;\n};\n\n/** @public */\nexport class GithubLocationAnalyzer implements ScmLocationAnalyzer {\n private readonly catalogClient: CatalogService;\n private readonly githubCredentialsProvider: GithubCredentialsProvider;\n private readonly integrations: ScmIntegrationRegistry;\n private readonly auth: AuthService;\n\n constructor(options: GithubLocationAnalyzerOptions) {\n this.catalogClient = options.catalog;\n this.integrations = ScmIntegrations.fromConfig(options.config);\n this.githubCredentialsProvider =\n options.githubCredentialsProvider ||\n DefaultGithubCredentialsProvider.fromIntegrations(this.integrations);\n\n this.auth = options.auth;\n }\n\n supports(url: string) {\n const integration = this.integrations.byUrl(url);\n return integration?.type === 'github';\n }\n\n async analyze(options: AnalyzeOptions) {\n const { url, catalogFilename } = options;\n const { owner, name: repo } = parseGitUrl(url);\n\n const catalogFile = catalogFilename || 'catalog-info.yaml';\n const extension = extname(catalogFile);\n const extensionQuery = !isEmpty(extension)\n ? `extension:${extension.replace('.', '')}`\n : '';\n\n const query = `filename:${catalogFile} ${extensionQuery} repo:${owner}/${repo}`;\n\n const integration = this.integrations.github.byUrl(url);\n if (!integration) {\n throw new Error('Make sure you have a GitHub integration configured');\n }\n\n const { token: githubToken } =\n await this.githubCredentialsProvider.getCredentials({\n url,\n });\n\n const octokitClient = new Octokit({\n auth: githubToken,\n baseUrl: integration.config.apiBaseUrl,\n });\n\n const searchResult = await octokitClient.search\n .code({ q: query })\n .catch(e => {\n throw new Error(`Couldn't search repository for metadata file, ${e}`);\n });\n\n const exists = searchResult.data.total_count > 0;\n if (exists) {\n const repoInformation = await octokitClient.repos\n .get({ owner, repo })\n .catch(e => {\n throw new Error(`Couldn't fetch repo data, ${e}`);\n });\n const defaultBranch = repoInformation.data.default_branch;\n\n const result = await Promise.all(\n searchResult.data.items\n .map(i => `${trimEnd(url, '/')}/blob/${defaultBranch}/${i.path}`)\n .map(async target => {\n const addLocationResult = await this.catalogClient.addLocation(\n {\n type: 'url',\n target,\n dryRun: true,\n },\n { credentials: await this.auth.getOwnServiceCredentials() },\n );\n return addLocationResult.entities.map(e => ({\n location: { type: 'url', target },\n isRegistered: !!addLocationResult.exists,\n entity: e,\n }));\n }),\n );\n\n return { existing: result.flat() };\n }\n return { existing: [] };\n }\n}\n"],"names":["ScmIntegrations","DefaultGithubCredentialsProvider","parseGitUrl","extname","isEmpty","Octokit","trimEnd"],"mappings":"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"}
@@ -1 +1 @@
1
- {"version":3,"file":"deprecated.cjs.js","sources":["../src/deprecated.ts"],"sourcesContent":["/*\n * Copyright 2020 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 { Config } from '@backstage/config';\nimport {\n EntityProvider,\n EntityProviderConnection,\n} from '@backstage/plugin-catalog-node';\nimport { GithubEntityProvider } from './providers/GithubEntityProvider';\nimport {\n GithubOrgEntityProvider,\n GithubOrgEntityProviderOptions,\n} from './providers/GithubOrgEntityProvider';\nimport {\n LoggerService,\n SchedulerService,\n SchedulerServiceTaskRunner,\n} from '@backstage/backend-plugin-api';\n\n/**\n * @public\n * @deprecated Use {@link GithubOrgEntityProvider} instead.\n */\nexport class GitHubOrgEntityProvider extends GithubOrgEntityProvider {\n static fromConfig(config: Config, options: GitHubOrgEntityProviderOptions) {\n options.logger.warn(\n '[Deprecated] Use GithubOrgEntityProvider instead of GitHubOrgEntityProvider.',\n );\n return GithubOrgEntityProvider.fromConfig(\n config,\n options as GithubOrgEntityProviderOptions,\n ) as GitHubOrgEntityProvider;\n }\n}\n\n/**\n * @public\n * @deprecated Use {@link GithubOrgEntityProviderOptions} instead.\n */\nexport type GitHubOrgEntityProviderOptions = GithubOrgEntityProviderOptions;\n\n/**\n * @public\n * @deprecated Use {@link GithubEntityProvider} instead.\n */\nexport class GitHubEntityProvider implements EntityProvider {\n static fromConfig(\n config: Config,\n options: {\n logger: LoggerService;\n schedule?: SchedulerServiceTaskRunner;\n scheduler?: SchedulerService;\n },\n ): GitHubEntityProvider[] {\n options.logger.warn(\n '[Deprecated] Please use GithubEntityProvider instead of GitHubEntityProvider.',\n );\n return GithubEntityProvider.fromConfig(config, options).map(\n delegate => new GitHubEntityProvider(delegate),\n );\n }\n\n private constructor(private readonly delegate: GithubEntityProvider) {}\n\n connect(connection: EntityProviderConnection): Promise<void> {\n return this.delegate.connect(connection);\n }\n\n getProviderName(): string {\n return this.delegate.getProviderName();\n }\n\n refresh(logger: LoggerService): Promise<void> {\n return this.delegate.refresh(logger);\n }\n}\n"],"names":["GithubOrgEntityProvider","GithubEntityProvider"],"mappings":";;;;;AAoCO,MAAM,gCAAgCA,+CAAwB,CAAA;AAAA,EACnE,OAAO,UAAW,CAAA,MAAA,EAAgB,OAAyC,EAAA;AACzE,IAAA,OAAA,CAAQ,MAAO,CAAA,IAAA;AAAA,MACb;AAAA,KACF;AACA,IAAA,OAAOA,+CAAwB,CAAA,UAAA;AAAA,MAC7B,MAAA;AAAA,MACA;AAAA,KACF;AAAA;AAEJ;AAYO,MAAM,oBAA+C,CAAA;AAAA,EAiBlD,YAA6B,QAAgC,EAAA;AAAhC,IAAA,IAAA,CAAA,QAAA,GAAA,QAAA;AAAA;AAAiC,EAhBtE,OAAO,UACL,CAAA,MAAA,EACA,OAKwB,EAAA;AACxB,IAAA,OAAA,CAAQ,MAAO,CAAA,IAAA;AAAA,MACb;AAAA,KACF;AACA,IAAA,OAAOC,yCAAqB,CAAA,UAAA,CAAW,MAAQ,EAAA,OAAO,CAAE,CAAA,GAAA;AAAA,MACtD,CAAA,QAAA,KAAY,IAAI,oBAAA,CAAqB,QAAQ;AAAA,KAC/C;AAAA;AACF,EAIA,QAAQ,UAAqD,EAAA;AAC3D,IAAO,OAAA,IAAA,CAAK,QAAS,CAAA,OAAA,CAAQ,UAAU,CAAA;AAAA;AACzC,EAEA,eAA0B,GAAA;AACxB,IAAO,OAAA,IAAA,CAAK,SAAS,eAAgB,EAAA;AAAA;AACvC,EAEA,QAAQ,MAAsC,EAAA;AAC5C,IAAO,OAAA,IAAA,CAAK,QAAS,CAAA,OAAA,CAAQ,MAAM,CAAA;AAAA;AAEvC;;;;;"}
1
+ {"version":3,"file":"deprecated.cjs.js","sources":["../src/deprecated.ts"],"sourcesContent":["/*\n * Copyright 2020 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 { Config } from '@backstage/config';\nimport {\n EntityProvider,\n EntityProviderConnection,\n} from '@backstage/plugin-catalog-node';\nimport { GithubEntityProvider } from './providers/GithubEntityProvider';\nimport {\n GithubOrgEntityProvider,\n GithubOrgEntityProviderOptions,\n} from './providers/GithubOrgEntityProvider';\nimport {\n LoggerService,\n SchedulerService,\n SchedulerServiceTaskRunner,\n} from '@backstage/backend-plugin-api';\n\n/**\n * @public\n * @deprecated Use {@link GithubOrgEntityProvider} instead.\n */\nexport class GitHubOrgEntityProvider extends GithubOrgEntityProvider {\n static fromConfig(config: Config, options: GitHubOrgEntityProviderOptions) {\n options.logger.warn(\n '[Deprecated] Use GithubOrgEntityProvider instead of GitHubOrgEntityProvider.',\n );\n return GithubOrgEntityProvider.fromConfig(\n config,\n options as GithubOrgEntityProviderOptions,\n ) as GitHubOrgEntityProvider;\n }\n}\n\n/**\n * @public\n * @deprecated Use {@link GithubOrgEntityProviderOptions} instead.\n */\nexport type GitHubOrgEntityProviderOptions = GithubOrgEntityProviderOptions;\n\n/**\n * @public\n * @deprecated Use {@link GithubEntityProvider} instead.\n */\nexport class GitHubEntityProvider implements EntityProvider {\n static fromConfig(\n config: Config,\n options: {\n logger: LoggerService;\n schedule?: SchedulerServiceTaskRunner;\n scheduler?: SchedulerService;\n },\n ): GitHubEntityProvider[] {\n options.logger.warn(\n '[Deprecated] Please use GithubEntityProvider instead of GitHubEntityProvider.',\n );\n return GithubEntityProvider.fromConfig(config, options).map(\n delegate => new GitHubEntityProvider(delegate),\n );\n }\n\n private constructor(private readonly delegate: GithubEntityProvider) {}\n\n connect(connection: EntityProviderConnection): Promise<void> {\n return this.delegate.connect(connection);\n }\n\n getProviderName(): string {\n return this.delegate.getProviderName();\n }\n\n refresh(logger: LoggerService): Promise<void> {\n return this.delegate.refresh(logger);\n }\n}\n"],"names":["GithubOrgEntityProvider","GithubEntityProvider"],"mappings":";;;;;AAoCO,MAAM,gCAAgCA,+CAAA,CAAwB;AAAA,EACnE,OAAO,UAAA,CAAW,MAAA,EAAgB,OAAA,EAAyC;AACzE,IAAA,OAAA,CAAQ,MAAA,CAAO,IAAA;AAAA,MACb;AAAA,KACF;AACA,IAAA,OAAOA,+CAAA,CAAwB,UAAA;AAAA,MAC7B,MAAA;AAAA,MACA;AAAA,KACF;AAAA,EACF;AACF;AAYO,MAAM,oBAAA,CAA+C;AAAA,EAiBlD,YAA6B,QAAA,EAAgC;AAAhC,IAAA,IAAA,CAAA,QAAA,GAAA,QAAA;AAAA,EAAiC;AAAA,EAhBtE,OAAO,UAAA,CACL,MAAA,EACA,OAAA,EAKwB;AACxB,IAAA,OAAA,CAAQ,MAAA,CAAO,IAAA;AAAA,MACb;AAAA,KACF;AACA,IAAA,OAAOC,yCAAA,CAAqB,UAAA,CAAW,MAAA,EAAQ,OAAO,CAAA,CAAE,GAAA;AAAA,MACtD,CAAA,QAAA,KAAY,IAAI,oBAAA,CAAqB,QAAQ;AAAA,KAC/C;AAAA,EACF;AAAA,EAIA,QAAQ,UAAA,EAAqD;AAC3D,IAAA,OAAO,IAAA,CAAK,QAAA,CAAS,OAAA,CAAQ,UAAU,CAAA;AAAA,EACzC;AAAA,EAEA,eAAA,GAA0B;AACxB,IAAA,OAAO,IAAA,CAAK,SAAS,eAAA,EAAgB;AAAA,EACvC;AAAA,EAEA,QAAQ,MAAA,EAAsC;AAC5C,IAAA,OAAO,IAAA,CAAK,QAAA,CAAS,OAAA,CAAQ,MAAM,CAAA;AAAA,EACrC;AACF;;;;;"}
package/dist/index.d.ts CHANGED
@@ -3,7 +3,7 @@ import { AuthService, LoggerService, SchedulerServiceTaskRunner, SchedulerServic
3
3
  import * as _backstage_catalog_model from '@backstage/catalog-model';
4
4
  import { Entity, UserEntity } from '@backstage/catalog-model';
5
5
  import { GithubCredentialsProvider, ScmIntegrationRegistry, GithubIntegrationConfig } from '@backstage/integration';
6
- import { CatalogService, ScmLocationAnalyzer, AnalyzeOptions, CatalogProcessor, LocationSpec, CatalogProcessorEmit, EntityProvider, EntityProviderConnection } from '@backstage/plugin-catalog-node';
6
+ import { ScmLocationAnalyzer, CatalogService, AnalyzeOptions, CatalogProcessor, LocationSpec, CatalogProcessorEmit, EntityProvider, EntityProviderConnection } from '@backstage/plugin-catalog-node';
7
7
  import { Config } from '@backstage/config';
8
8
  import { graphql } from '@octokit/graphql';
9
9
  import { EventSubscriber, EventsService, EventParams } from '@backstage/plugin-events-node';
@@ -1 +1 @@
1
- {"version":3,"file":"annotation.cjs.js","sources":["../../src/lib/annotation.ts"],"sourcesContent":["/*\n * Copyright 2020 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\n/**\n * The value of this annotation is the so-called login that identifies a user on\n * [GitHub](https://github.com) (either the public one, or a private GitHub\n * Enterprise installation) that is related to this entity. It is on the format\n * `<username>`, and is the same as can be seen in the URL location bar of the\n * browser when viewing that user.\n *\n * @public\n */\nexport const ANNOTATION_GITHUB_USER_LOGIN = 'github.com/user-login';\n\n/**\n * The value of this annotation is the so-called slug that identifies a team on\n * [GitHub](https://github.com) (either the public one, or a private GitHub\n * Enterprise installation) that is related to this entity. It is on the format\n * `<organization>/<team>`, and is the same as can be seen in the URL location\n * bar of the browser when viewing that team.\n *\n * @public\n */\nexport const ANNOTATION_GITHUB_TEAM_SLUG = 'github.com/team-slug';\n"],"names":[],"mappings":";;AAyBO,MAAM,4BAA+B,GAAA;AAWrC,MAAM,2BAA8B,GAAA;;;;;"}
1
+ {"version":3,"file":"annotation.cjs.js","sources":["../../src/lib/annotation.ts"],"sourcesContent":["/*\n * Copyright 2020 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\n/**\n * The value of this annotation is the so-called login that identifies a user on\n * [GitHub](https://github.com) (either the public one, or a private GitHub\n * Enterprise installation) that is related to this entity. It is on the format\n * `<username>`, and is the same as can be seen in the URL location bar of the\n * browser when viewing that user.\n *\n * @public\n */\nexport const ANNOTATION_GITHUB_USER_LOGIN = 'github.com/user-login';\n\n/**\n * The value of this annotation is the so-called slug that identifies a team on\n * [GitHub](https://github.com) (either the public one, or a private GitHub\n * Enterprise installation) that is related to this entity. It is on the format\n * `<organization>/<team>`, and is the same as can be seen in the URL location\n * bar of the browser when viewing that team.\n *\n * @public\n */\nexport const ANNOTATION_GITHUB_TEAM_SLUG = 'github.com/team-slug';\n"],"names":[],"mappings":";;AAyBO,MAAM,4BAAA,GAA+B;AAWrC,MAAM,2BAAA,GAA8B;;;;;"}
@@ -1 +1 @@
1
- {"version":3,"file":"config.cjs.js","sources":["../../src/lib/config.ts"],"sourcesContent":["/*\n * Copyright 2020 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 { Config } from '@backstage/config';\n\n/**\n * The configuration parameters for a multi-org GitHub processor.\n * @public\n */\nexport type GithubMultiOrgConfig = Array<{\n /**\n * The name of the GitHub org to process.\n */\n name: string;\n /**\n * The namespace of the group created for this org.\n */\n groupNamespace: string;\n /**\n * The namespace of the users created for this org. If not specified defaults to undefined.\n */\n userNamespace: string | undefined;\n}>;\n\nexport function readGithubMultiOrgConfig(config: Config): GithubMultiOrgConfig {\n const orgConfigs = config.getOptionalConfigArray('orgs') ?? [];\n return orgConfigs.map(c => ({\n name: c.getString('name'),\n groupNamespace: (\n c.getOptionalString('groupNamespace') ?? c.getString('name')\n ).toLowerCase(),\n userNamespace: c.getOptionalString('userNamespace') ?? undefined,\n }));\n}\n"],"names":[],"mappings":";;AAqCO,SAAS,yBAAyB,MAAsC,EAAA;AAC7E,EAAA,MAAM,UAAa,GAAA,MAAA,CAAO,sBAAuB,CAAA,MAAM,KAAK,EAAC;AAC7D,EAAO,OAAA,UAAA,CAAW,IAAI,CAAM,CAAA,MAAA;AAAA,IAC1B,IAAA,EAAM,CAAE,CAAA,SAAA,CAAU,MAAM,CAAA;AAAA,IACxB,cAAA,EAAA,CACE,EAAE,iBAAkB,CAAA,gBAAgB,KAAK,CAAE,CAAA,SAAA,CAAU,MAAM,CAAA,EAC3D,WAAY,EAAA;AAAA,IACd,aAAe,EAAA,CAAA,CAAE,iBAAkB,CAAA,eAAe,CAAK,IAAA,KAAA;AAAA,GACvD,CAAA,CAAA;AACJ;;;;"}
1
+ {"version":3,"file":"config.cjs.js","sources":["../../src/lib/config.ts"],"sourcesContent":["/*\n * Copyright 2020 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 { Config } from '@backstage/config';\n\n/**\n * The configuration parameters for a multi-org GitHub processor.\n * @public\n */\nexport type GithubMultiOrgConfig = Array<{\n /**\n * The name of the GitHub org to process.\n */\n name: string;\n /**\n * The namespace of the group created for this org.\n */\n groupNamespace: string;\n /**\n * The namespace of the users created for this org. If not specified defaults to undefined.\n */\n userNamespace: string | undefined;\n}>;\n\nexport function readGithubMultiOrgConfig(config: Config): GithubMultiOrgConfig {\n const orgConfigs = config.getOptionalConfigArray('orgs') ?? [];\n return orgConfigs.map(c => ({\n name: c.getString('name'),\n groupNamespace: (\n c.getOptionalString('groupNamespace') ?? c.getString('name')\n ).toLowerCase(),\n userNamespace: c.getOptionalString('userNamespace') ?? undefined,\n }));\n}\n"],"names":[],"mappings":";;AAqCO,SAAS,yBAAyB,MAAA,EAAsC;AAC7E,EAAA,MAAM,UAAA,GAAa,MAAA,CAAO,sBAAA,CAAuB,MAAM,KAAK,EAAC;AAC7D,EAAA,OAAO,UAAA,CAAW,IAAI,CAAA,CAAA,MAAM;AAAA,IAC1B,IAAA,EAAM,CAAA,CAAE,SAAA,CAAU,MAAM,CAAA;AAAA,IACxB,cAAA,EAAA,CACE,EAAE,iBAAA,CAAkB,gBAAgB,KAAK,CAAA,CAAE,SAAA,CAAU,MAAM,CAAA,EAC3D,WAAA,EAAY;AAAA,IACd,aAAA,EAAe,CAAA,CAAE,iBAAA,CAAkB,eAAe,CAAA,IAAK;AAAA,GACzD,CAAE,CAAA;AACJ;;;;"}
@@ -1 +1 @@
1
- {"version":3,"file":"defaultTransformers.cjs.js","sources":["../../src/lib/defaultTransformers.ts"],"sourcesContent":["/*\n * Copyright 2020 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 { Entity, GroupEntity, UserEntity } from '@backstage/catalog-model';\nimport { graphql } from '@octokit/graphql';\nimport {\n ANNOTATION_GITHUB_TEAM_SLUG,\n ANNOTATION_GITHUB_USER_LOGIN,\n} from './annotation';\nimport { GithubTeam, GithubUser } from './github';\n\n/**\n * Context passed to Transformers\n *\n * @public\n */\nexport interface TransformerContext {\n client: typeof graphql;\n query: string;\n org: string;\n}\n\n/**\n * Transformer for GitHub users to an Entity\n *\n * @public\n */\nexport type UserTransformer = (\n item: GithubUser,\n ctx: TransformerContext,\n) => Promise<Entity | undefined>;\n\n/**\n * Transformer for GitHub Team to an Entity\n *\n * @public\n */\nexport type TeamTransformer = (\n item: GithubTeam,\n ctx: TransformerContext,\n) => Promise<Entity | undefined>;\n\n/**\n * Default transformer for GitHub users to UserEntity\n *\n * @public\n */\nexport const defaultUserTransformer = async (\n item: GithubUser,\n _ctx: TransformerContext,\n): Promise<UserEntity | undefined> => {\n const entity: UserEntity = {\n apiVersion: 'backstage.io/v1alpha1',\n kind: 'User',\n metadata: {\n name: item.login,\n annotations: {\n [ANNOTATION_GITHUB_USER_LOGIN]: item.login,\n },\n },\n spec: {\n profile: {},\n memberOf: [],\n },\n };\n\n if (item.bio) entity.metadata.description = item.bio;\n if (item.name) entity.spec.profile!.displayName = item.name;\n if (item.email) entity.spec.profile!.email = item.email;\n if (item.avatarUrl) entity.spec.profile!.picture = item.avatarUrl;\n return entity;\n};\n\n/**\n * Default transformer for GitHub Team to GroupEntity\n *\n * @public\n */\nexport const defaultOrganizationTeamTransformer: TeamTransformer =\n async team => {\n const annotations: { [annotationName: string]: string } = {\n [ANNOTATION_GITHUB_TEAM_SLUG]: team.combinedSlug,\n };\n\n if (team.editTeamUrl) {\n annotations['backstage.io/edit-url'] = team.editTeamUrl;\n }\n\n const entity: GroupEntity = {\n apiVersion: 'backstage.io/v1alpha1',\n kind: 'Group',\n metadata: {\n name: team.slug,\n annotations,\n },\n spec: {\n type: 'team',\n profile: {},\n children: [],\n },\n };\n\n if (team.description) {\n entity.metadata.description = team.description;\n }\n if (team.name) {\n entity.spec.profile!.displayName = team.name;\n }\n if (team.avatarUrl) {\n entity.spec.profile!.picture = team.avatarUrl;\n }\n if (team.parentTeam) {\n entity.spec.parent = team.parentTeam.slug;\n }\n\n entity.spec.members = team.members.map(user => user.login);\n\n return entity;\n };\n"],"names":["ANNOTATION_GITHUB_USER_LOGIN","ANNOTATION_GITHUB_TEAM_SLUG"],"mappings":";;;;AA4Da,MAAA,sBAAA,GAAyB,OACpC,IAAA,EACA,IACoC,KAAA;AACpC,EAAA,MAAM,MAAqB,GAAA;AAAA,IACzB,UAAY,EAAA,uBAAA;AAAA,IACZ,IAAM,EAAA,MAAA;AAAA,IACN,QAAU,EAAA;AAAA,MACR,MAAM,IAAK,CAAA,KAAA;AAAA,MACX,WAAa,EAAA;AAAA,QACX,CAACA,uCAA4B,GAAG,IAAK,CAAA;AAAA;AACvC,KACF;AAAA,IACA,IAAM,EAAA;AAAA,MACJ,SAAS,EAAC;AAAA,MACV,UAAU;AAAC;AACb,GACF;AAEA,EAAA,IAAI,IAAK,CAAA,GAAA,EAAY,MAAA,CAAA,QAAA,CAAS,cAAc,IAAK,CAAA,GAAA;AACjD,EAAA,IAAI,KAAK,IAAM,EAAA,MAAA,CAAO,IAAK,CAAA,OAAA,CAAS,cAAc,IAAK,CAAA,IAAA;AACvD,EAAA,IAAI,KAAK,KAAO,EAAA,MAAA,CAAO,IAAK,CAAA,OAAA,CAAS,QAAQ,IAAK,CAAA,KAAA;AAClD,EAAA,IAAI,KAAK,SAAW,EAAA,MAAA,CAAO,IAAK,CAAA,OAAA,CAAS,UAAU,IAAK,CAAA,SAAA;AACxD,EAAO,OAAA,MAAA;AACT;AAOa,MAAA,kCAAA,GACX,OAAM,IAAQ,KAAA;AACZ,EAAA,MAAM,WAAoD,GAAA;AAAA,IACxD,CAACC,sCAA2B,GAAG,IAAK,CAAA;AAAA,GACtC;AAEA,EAAA,IAAI,KAAK,WAAa,EAAA;AACpB,IAAY,WAAA,CAAA,uBAAuB,IAAI,IAAK,CAAA,WAAA;AAAA;AAG9C,EAAA,MAAM,MAAsB,GAAA;AAAA,IAC1B,UAAY,EAAA,uBAAA;AAAA,IACZ,IAAM,EAAA,OAAA;AAAA,IACN,QAAU,EAAA;AAAA,MACR,MAAM,IAAK,CAAA,IAAA;AAAA,MACX;AAAA,KACF;AAAA,IACA,IAAM,EAAA;AAAA,MACJ,IAAM,EAAA,MAAA;AAAA,MACN,SAAS,EAAC;AAAA,MACV,UAAU;AAAC;AACb,GACF;AAEA,EAAA,IAAI,KAAK,WAAa,EAAA;AACpB,IAAO,MAAA,CAAA,QAAA,CAAS,cAAc,IAAK,CAAA,WAAA;AAAA;AAErC,EAAA,IAAI,KAAK,IAAM,EAAA;AACb,IAAO,MAAA,CAAA,IAAA,CAAK,OAAS,CAAA,WAAA,GAAc,IAAK,CAAA,IAAA;AAAA;AAE1C,EAAA,IAAI,KAAK,SAAW,EAAA;AAClB,IAAO,MAAA,CAAA,IAAA,CAAK,OAAS,CAAA,OAAA,GAAU,IAAK,CAAA,SAAA;AAAA;AAEtC,EAAA,IAAI,KAAK,UAAY,EAAA;AACnB,IAAO,MAAA,CAAA,IAAA,CAAK,MAAS,GAAA,IAAA,CAAK,UAAW,CAAA,IAAA;AAAA;AAGvC,EAAA,MAAA,CAAO,KAAK,OAAU,GAAA,IAAA,CAAK,QAAQ,GAAI,CAAA,CAAA,IAAA,KAAQ,KAAK,KAAK,CAAA;AAEzD,EAAO,OAAA,MAAA;AACT;;;;;"}
1
+ {"version":3,"file":"defaultTransformers.cjs.js","sources":["../../src/lib/defaultTransformers.ts"],"sourcesContent":["/*\n * Copyright 2020 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 { Entity, GroupEntity, UserEntity } from '@backstage/catalog-model';\nimport { graphql } from '@octokit/graphql';\nimport {\n ANNOTATION_GITHUB_TEAM_SLUG,\n ANNOTATION_GITHUB_USER_LOGIN,\n} from './annotation';\nimport { GithubTeam, GithubUser } from './github';\n\n/**\n * Context passed to Transformers\n *\n * @public\n */\nexport interface TransformerContext {\n client: typeof graphql;\n query: string;\n org: string;\n}\n\n/**\n * Transformer for GitHub users to an Entity\n *\n * @public\n */\nexport type UserTransformer = (\n item: GithubUser,\n ctx: TransformerContext,\n) => Promise<Entity | undefined>;\n\n/**\n * Transformer for GitHub Team to an Entity\n *\n * @public\n */\nexport type TeamTransformer = (\n item: GithubTeam,\n ctx: TransformerContext,\n) => Promise<Entity | undefined>;\n\n/**\n * Default transformer for GitHub users to UserEntity\n *\n * @public\n */\nexport const defaultUserTransformer = async (\n item: GithubUser,\n _ctx: TransformerContext,\n): Promise<UserEntity | undefined> => {\n const entity: UserEntity = {\n apiVersion: 'backstage.io/v1alpha1',\n kind: 'User',\n metadata: {\n name: item.login,\n annotations: {\n [ANNOTATION_GITHUB_USER_LOGIN]: item.login,\n },\n },\n spec: {\n profile: {},\n memberOf: [],\n },\n };\n\n if (item.bio) entity.metadata.description = item.bio;\n if (item.name) entity.spec.profile!.displayName = item.name;\n if (item.email) entity.spec.profile!.email = item.email;\n if (item.avatarUrl) entity.spec.profile!.picture = item.avatarUrl;\n return entity;\n};\n\n/**\n * Default transformer for GitHub Team to GroupEntity\n *\n * @public\n */\nexport const defaultOrganizationTeamTransformer: TeamTransformer =\n async team => {\n const annotations: { [annotationName: string]: string } = {\n [ANNOTATION_GITHUB_TEAM_SLUG]: team.combinedSlug,\n };\n\n if (team.editTeamUrl) {\n annotations['backstage.io/edit-url'] = team.editTeamUrl;\n }\n\n const entity: GroupEntity = {\n apiVersion: 'backstage.io/v1alpha1',\n kind: 'Group',\n metadata: {\n name: team.slug,\n annotations,\n },\n spec: {\n type: 'team',\n profile: {},\n children: [],\n },\n };\n\n if (team.description) {\n entity.metadata.description = team.description;\n }\n if (team.name) {\n entity.spec.profile!.displayName = team.name;\n }\n if (team.avatarUrl) {\n entity.spec.profile!.picture = team.avatarUrl;\n }\n if (team.parentTeam) {\n entity.spec.parent = team.parentTeam.slug;\n }\n\n entity.spec.members = team.members.map(user => user.login);\n\n return entity;\n };\n"],"names":["ANNOTATION_GITHUB_USER_LOGIN","ANNOTATION_GITHUB_TEAM_SLUG"],"mappings":";;;;AA4DO,MAAM,sBAAA,GAAyB,OACpC,IAAA,EACA,IAAA,KACoC;AACpC,EAAA,MAAM,MAAA,GAAqB;AAAA,IACzB,UAAA,EAAY,uBAAA;AAAA,IACZ,IAAA,EAAM,MAAA;AAAA,IACN,QAAA,EAAU;AAAA,MACR,MAAM,IAAA,CAAK,KAAA;AAAA,MACX,WAAA,EAAa;AAAA,QACX,CAACA,uCAA4B,GAAG,IAAA,CAAK;AAAA;AACvC,KACF;AAAA,IACA,IAAA,EAAM;AAAA,MACJ,SAAS,EAAC;AAAA,MACV,UAAU;AAAC;AACb,GACF;AAEA,EAAA,IAAI,IAAA,CAAK,GAAA,EAAK,MAAA,CAAO,QAAA,CAAS,cAAc,IAAA,CAAK,GAAA;AACjD,EAAA,IAAI,KAAK,IAAA,EAAM,MAAA,CAAO,IAAA,CAAK,OAAA,CAAS,cAAc,IAAA,CAAK,IAAA;AACvD,EAAA,IAAI,KAAK,KAAA,EAAO,MAAA,CAAO,IAAA,CAAK,OAAA,CAAS,QAAQ,IAAA,CAAK,KAAA;AAClD,EAAA,IAAI,KAAK,SAAA,EAAW,MAAA,CAAO,IAAA,CAAK,OAAA,CAAS,UAAU,IAAA,CAAK,SAAA;AACxD,EAAA,OAAO,MAAA;AACT;AAOO,MAAM,kCAAA,GACX,OAAM,IAAA,KAAQ;AACZ,EAAA,MAAM,WAAA,GAAoD;AAAA,IACxD,CAACC,sCAA2B,GAAG,IAAA,CAAK;AAAA,GACtC;AAEA,EAAA,IAAI,KAAK,WAAA,EAAa;AACpB,IAAA,WAAA,CAAY,uBAAuB,IAAI,IAAA,CAAK,WAAA;AAAA,EAC9C;AAEA,EAAA,MAAM,MAAA,GAAsB;AAAA,IAC1B,UAAA,EAAY,uBAAA;AAAA,IACZ,IAAA,EAAM,OAAA;AAAA,IACN,QAAA,EAAU;AAAA,MACR,MAAM,IAAA,CAAK,IAAA;AAAA,MACX;AAAA,KACF;AAAA,IACA,IAAA,EAAM;AAAA,MACJ,IAAA,EAAM,MAAA;AAAA,MACN,SAAS,EAAC;AAAA,MACV,UAAU;AAAC;AACb,GACF;AAEA,EAAA,IAAI,KAAK,WAAA,EAAa;AACpB,IAAA,MAAA,CAAO,QAAA,CAAS,cAAc,IAAA,CAAK,WAAA;AAAA,EACrC;AACA,EAAA,IAAI,KAAK,IAAA,EAAM;AACb,IAAA,MAAA,CAAO,IAAA,CAAK,OAAA,CAAS,WAAA,GAAc,IAAA,CAAK,IAAA;AAAA,EAC1C;AACA,EAAA,IAAI,KAAK,SAAA,EAAW;AAClB,IAAA,MAAA,CAAO,IAAA,CAAK,OAAA,CAAS,OAAA,GAAU,IAAA,CAAK,SAAA;AAAA,EACtC;AACA,EAAA,IAAI,KAAK,UAAA,EAAY;AACnB,IAAA,MAAA,CAAO,IAAA,CAAK,MAAA,GAAS,IAAA,CAAK,UAAA,CAAW,IAAA;AAAA,EACvC;AAEA,EAAA,MAAA,CAAO,KAAK,OAAA,GAAU,IAAA,CAAK,QAAQ,GAAA,CAAI,CAAA,IAAA,KAAQ,KAAK,KAAK,CAAA;AAEzD,EAAA,OAAO,MAAA;AACT;;;;;"}
@@ -1 +1 @@
1
- {"version":3,"file":"github.cjs.js","sources":["../../src/lib/github.ts"],"sourcesContent":["/*\n * Copyright 2020 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 { Entity } from '@backstage/catalog-model';\nimport { GithubCredentialType } from '@backstage/integration';\nimport { graphql } from '@octokit/graphql';\nimport {\n defaultOrganizationTeamTransformer,\n defaultUserTransformer,\n TeamTransformer,\n TransformerContext,\n UserTransformer,\n} from './defaultTransformers';\nimport { withLocations } from './withLocations';\n\nimport { DeferredEntity } from '@backstage/plugin-catalog-node';\nimport { Octokit } from '@octokit/core';\nimport { LoggerService } from '@backstage/backend-plugin-api';\nimport { throttling } from '@octokit/plugin-throttling';\n// Graphql types\n\nexport type QueryResponse = {\n organization?: OrganizationResponse;\n repositoryOwner?: RepositoryOwnerResponse;\n user?: UserResponse;\n};\n\ntype RepositoryOwnerResponse = {\n repositories?: Connection<RepositoryResponse>;\n repository?: RepositoryResponse;\n};\n\nexport type OrganizationResponse = {\n membersWithRole?: Connection<GithubUser>;\n team?: GithubTeamResponse;\n teams?: Connection<GithubTeamResponse>;\n repositories?: Connection<RepositoryResponse>;\n};\n\nexport type UserResponse = {\n organizations?: Connection<GithubOrg>;\n};\n\nexport type PageInfo = {\n hasNextPage: boolean;\n endCursor?: string;\n};\n\nexport type GithubOrg = {\n login: string;\n};\n\n/**\n * Github User\n *\n * @public\n */\nexport type GithubUser = {\n login: string;\n bio?: string;\n avatarUrl?: string;\n email?: string;\n name?: string;\n organizationVerifiedDomainEmails?: string[];\n};\n\n/**\n * Github Team\n *\n * @public\n */\nexport type GithubTeam = {\n slug: string;\n combinedSlug: string;\n name?: string;\n description?: string;\n avatarUrl?: string;\n editTeamUrl?: string;\n parentTeam?: GithubTeam;\n members: GithubUser[];\n};\n\nexport type GithubTeamResponse = Omit<GithubTeam, 'members'> & {\n members: Connection<GithubUser>;\n};\n\nexport type RepositoryResponse = {\n name: string;\n url: string;\n isArchived: boolean;\n isFork: boolean;\n repositoryTopics: RepositoryTopics;\n defaultBranchRef: {\n name: string;\n } | null;\n catalogInfoFile: {\n __typename: string;\n id: string;\n text: string;\n } | null;\n visibility: string;\n};\n\ntype RepositoryTopics = {\n nodes: TopicNodes[];\n};\n\ntype TopicNodes = {\n topic: {\n name: string;\n };\n};\n\nexport type Connection<T> = {\n pageInfo: PageInfo;\n nodes: T[];\n};\n\n/**\n * Gets all the users out of a Github organization.\n *\n * Note that the users will not have their memberships filled in.\n *\n * @param client - An octokit graphql client\n * @param org - The slug of the org to read\n */\nexport async function getOrganizationUsers(\n client: typeof graphql,\n org: string,\n tokenType: GithubCredentialType,\n userTransformer: UserTransformer = defaultUserTransformer,\n): Promise<{ users: Entity[] }> {\n const query = `\n query users($org: String!, $email: Boolean!, $cursor: String) {\n organization(login: $org) {\n membersWithRole(first: 100, after: $cursor) {\n pageInfo { hasNextPage, endCursor }\n nodes {\n avatarUrl,\n bio,\n email @include(if: $email),\n login,\n name,\n organizationVerifiedDomainEmails(login: $org)\n }\n }\n }\n }`;\n\n // There is no user -> teams edge, so we leave the memberships empty for\n // now and let the team iteration handle it instead\n\n const users = await queryWithPaging(\n client,\n query,\n org,\n r => r.organization?.membersWithRole,\n userTransformer,\n {\n org,\n email: tokenType === 'token',\n },\n );\n\n return { users };\n}\n\n/**\n * Gets all the teams out of a Github organization.\n *\n * Note that the teams will not have any relations apart from parent filled in.\n *\n * @param client - An octokit graphql client\n * @param org - The slug of the org to read\n */\nexport async function getOrganizationTeams(\n client: typeof graphql,\n org: string,\n teamTransformer: TeamTransformer = defaultOrganizationTeamTransformer,\n): Promise<{\n teams: Entity[];\n}> {\n const query = `\n query teams($org: String!, $cursor: String) {\n organization(login: $org) {\n teams(first: 50, after: $cursor) {\n pageInfo { hasNextPage, endCursor }\n nodes {\n slug\n combinedSlug\n name\n description\n avatarUrl\n editTeamUrl\n parentTeam { slug }\n members(first: 100, membership: IMMEDIATE) {\n pageInfo { hasNextPage }\n nodes {\n avatarUrl,\n bio,\n email,\n login,\n name,\n organizationVerifiedDomainEmails(login: $org)\n }\n }\n }\n }\n }\n }`;\n\n const materialisedTeams = async (\n item: GithubTeamResponse,\n ctx: TransformerContext,\n ): Promise<Entity | undefined> => {\n const memberNames: GithubUser[] = [];\n\n if (!item.members.pageInfo.hasNextPage) {\n // We got all the members in one go, run the fast path\n for (const user of item.members.nodes) {\n memberNames.push(user);\n }\n } else {\n // There were more than a hundred immediate members - run the slow\n // path of fetching them explicitly\n const { members } = await getTeamMembers(ctx.client, ctx.org, item.slug);\n for (const userLogin of members) {\n memberNames.push(userLogin);\n }\n }\n\n const team: GithubTeam = {\n ...item,\n members: memberNames,\n };\n\n return await teamTransformer(team, ctx);\n };\n\n const teams = await queryWithPaging(\n client,\n query,\n org,\n r => r.organization?.teams,\n materialisedTeams,\n { org },\n );\n\n return { teams };\n}\n\nexport async function getOrganizationTeamsFromUsers(\n client: typeof graphql,\n org: string,\n userLogins: string[],\n teamTransformer: TeamTransformer = defaultOrganizationTeamTransformer,\n): Promise<{\n teams: Entity[];\n}> {\n const query = `\n query teams($org: String!, $cursor: String, $userLogins: [String!] = \"\") {\n organization(login: $org) {\n teams(first: 100, after: $cursor, userLogins: $userLogins) {\n pageInfo {\n hasNextPage\n endCursor\n }\n nodes {\n slug\n combinedSlug\n name\n description\n avatarUrl\n editTeamUrl\n parentTeam {\n slug\n }\n members(first: 100, membership: IMMEDIATE) {\n pageInfo {\n hasNextPage\n }\n nodes {\n avatarUrl,\n bio,\n email,\n login,\n name,\n organizationVerifiedDomainEmails(login: $org)\n }\n }\n }\n }\n }\n}`;\n\n const materialisedTeams = async (\n item: GithubTeamResponse,\n ctx: TransformerContext,\n ): Promise<Entity | undefined> => {\n const memberNames: GithubUser[] = [];\n\n if (!item.members.pageInfo.hasNextPage) {\n // We got all the members in one go, run the fast path\n for (const user of item.members.nodes) {\n memberNames.push(user);\n }\n } else {\n // There were more than a hundred immediate members - run the slow\n // path of fetching them explicitly\n const { members } = await getTeamMembers(ctx.client, ctx.org, item.slug);\n for (const userLogin of members) {\n memberNames.push(userLogin);\n }\n }\n\n const team: GithubTeam = {\n ...item,\n members: memberNames,\n };\n\n return await teamTransformer(team, ctx);\n };\n\n const teams = await queryWithPaging(\n client,\n query,\n org,\n r => r.organization?.teams,\n materialisedTeams,\n { org, userLogins },\n );\n\n return { teams };\n}\n\nexport async function getOrganizationTeamsForUser(\n client: typeof graphql,\n org: string,\n userLogin: string,\n teamTransformer: TeamTransformer,\n): Promise<{ teams: Entity[] }> {\n const query = `\n query teams($org: String!, $cursor: String, $userLogins: [String!] = \"\") {\n organization(login: $org) {\n teams(first: 100, after: $cursor, userLogins: $userLogins) {\n pageInfo {\n hasNextPage\n endCursor\n }\n nodes {\n slug\n combinedSlug\n name\n description\n avatarUrl\n editTeamUrl\n parentTeam {\n slug\n }\n }\n }\n }\n}`;\n\n const materialisedTeams = async (\n item: GithubTeamResponse,\n ctx: TransformerContext,\n ): Promise<Entity | undefined> => {\n const team: GithubTeam = {\n ...item,\n members: [{ login: userLogin }],\n };\n\n return await teamTransformer(team, ctx);\n };\n\n const teams = await queryWithPaging(\n client,\n query,\n org,\n r => r.organization?.teams,\n materialisedTeams,\n { org, userLogins: [userLogin] },\n );\n\n return { teams };\n}\n\nexport async function getOrganizationsFromUser(\n client: typeof graphql,\n user: string,\n): Promise<{\n orgs: string[];\n}> {\n const query = `\n query orgs($user: String!) {\n user(login: $user) {\n organizations(first: 100) {\n nodes { login }\n pageInfo { hasNextPage, endCursor }\n }\n }\n }`;\n\n const orgs = await queryWithPaging(\n client,\n query,\n '',\n r => r.user?.organizations,\n async o => o.login,\n { user },\n );\n\n return { orgs };\n}\n\nexport async function getOrganizationTeam(\n client: typeof graphql,\n org: string,\n teamSlug: string,\n teamTransformer: TeamTransformer = defaultOrganizationTeamTransformer,\n): Promise<{\n team: Entity;\n}> {\n const query = `\n query teams($org: String!, $teamSlug: String!) {\n organization(login: $org) {\n team(slug:$teamSlug) {\n slug\n combinedSlug\n name\n description\n avatarUrl\n editTeamUrl\n parentTeam { slug }\n members(first: 100, membership: IMMEDIATE) {\n pageInfo { hasNextPage }\n nodes { login }\n }\n }\n }\n }`;\n\n const materialisedTeam = async (\n item: GithubTeamResponse,\n ctx: TransformerContext,\n ): Promise<Entity | undefined> => {\n const memberNames: GithubUser[] = [];\n\n if (!item.members.pageInfo.hasNextPage) {\n // We got all the members in one go, run the fast path\n for (const user of item.members.nodes) {\n memberNames.push(user);\n }\n } else {\n // There were more than a hundred immediate members - run the slow\n // path of fetching them explicitly\n const { members } = await getTeamMembers(ctx.client, ctx.org, item.slug);\n for (const userLogin of members) {\n memberNames.push(userLogin);\n }\n }\n\n const team: GithubTeam = {\n ...item,\n members: memberNames,\n };\n\n return await teamTransformer(team, ctx);\n };\n\n const response: QueryResponse = await client(query, {\n org,\n teamSlug,\n });\n\n if (!response.organization?.team)\n throw new Error(`Found no match for team ${teamSlug}`);\n\n const team = await materialisedTeam(response.organization?.team, {\n query,\n client,\n org,\n });\n\n if (!team) throw new Error(`Can't transform for team ${teamSlug}`);\n\n return { team };\n}\n\nexport async function getOrganizationRepositories(\n client: typeof graphql,\n org: string,\n catalogPath: string,\n): Promise<{ repositories: RepositoryResponse[] }> {\n let relativeCatalogPathRef: string;\n // We must strip the leading slash or the query for objects does not work\n if (catalogPath.startsWith('/')) {\n relativeCatalogPathRef = catalogPath.substring(1);\n } else {\n relativeCatalogPathRef = catalogPath;\n }\n const catalogPathRef = `HEAD:${relativeCatalogPathRef}`;\n const query = `\n query repositories($org: String!, $catalogPathRef: String!, $cursor: String) {\n repositoryOwner(login: $org) {\n login\n repositories(first: 50, after: $cursor) {\n nodes {\n name\n catalogInfoFile: object(expression: $catalogPathRef) {\n __typename\n ... on Blob {\n id\n text\n }\n }\n url\n isArchived\n isFork\n visibility\n repositoryTopics(first: 100) {\n nodes {\n ... on RepositoryTopic {\n topic {\n name\n }\n }\n }\n }\n defaultBranchRef {\n name\n }\n }\n pageInfo {\n hasNextPage\n endCursor\n }\n }\n }\n }`;\n\n const repositories = await queryWithPaging(\n client,\n query,\n org,\n r => r.repositoryOwner?.repositories,\n async x => x,\n { org, catalogPathRef },\n );\n\n return { repositories };\n}\n\nexport async function getOrganizationRepository(\n client: typeof graphql,\n org: string,\n repoName: string,\n catalogPath: string,\n): Promise<RepositoryResponse | null> {\n let relativeCatalogPathRef: string;\n // We must strip the leading slash or the query for objects does not work\n if (catalogPath.startsWith('/')) {\n relativeCatalogPathRef = catalogPath.substring(1);\n } else {\n relativeCatalogPathRef = catalogPath;\n }\n const catalogPathRef = `HEAD:${relativeCatalogPathRef}`;\n const query = `\n query repository($org: String!, $repoName: String!, $catalogPathRef: String!) {\n repositoryOwner(login: $org) {\n repository(name: $repoName) {\n name\n catalogInfoFile: object(expression: $catalogPathRef) {\n __typename\n ... on Blob {\n id\n text\n }\n }\n url\n isArchived\n isFork\n visibility\n repositoryTopics(first: 100) {\n nodes {\n ... on RepositoryTopic {\n topic {\n name\n }\n }\n }\n }\n defaultBranchRef {\n name\n }\n }\n }\n }`;\n\n const response: QueryResponse = await client(query, {\n org,\n repoName,\n catalogPathRef,\n });\n\n return response.repositoryOwner?.repository || null;\n}\n\n/**\n * Gets all the users out of a Github organization.\n *\n * Note that the users will not have their memberships filled in.\n *\n * @param client - An octokit graphql client\n * @param org - The slug of the org to read\n * @param teamSlug - The slug of the team to read\n */\nexport async function getTeamMembers(\n client: typeof graphql,\n org: string,\n teamSlug: string,\n): Promise<{ members: GithubUser[] }> {\n const query = `\n query members($org: String!, $teamSlug: String!, $cursor: String) {\n organization(login: $org) {\n team(slug: $teamSlug) {\n members(first: 100, after: $cursor, membership: IMMEDIATE) {\n pageInfo { hasNextPage, endCursor }\n nodes { login }\n }\n }\n }\n }`;\n\n const members = await queryWithPaging(\n client,\n query,\n org,\n r => r.organization?.team?.members,\n async user => user,\n { org, teamSlug },\n );\n\n return { members };\n}\n\n//\n// Helpers\n//\n\n/**\n * Assists in repeatedly executing a query with a paged response.\n *\n * Requires that the query accepts a $cursor variable.\n *\n * @param client - The octokit client\n * @param query - The query to execute\n * @param org - The slug of the org to read\n * @param connection - A function that, given the response, picks out the actual\n * Connection object that's being iterated\n * @param transformer - A function that, given one of the nodes in the Connection,\n * returns the model mapped form of it\n * @param variables - The variable values that the query needs, minus the cursor\n */\nexport async function queryWithPaging<\n GraphqlType,\n OutputType,\n Variables extends {},\n Response = QueryResponse,\n>(\n client: typeof graphql,\n query: string,\n org: string,\n connection: (response: Response) => Connection<GraphqlType> | undefined,\n transformer: (\n item: GraphqlType,\n ctx: TransformerContext,\n ) => Promise<OutputType | undefined>,\n variables: Variables,\n): Promise<OutputType[]> {\n const result: OutputType[] = [];\n const sleep = (ms: number) => new Promise(r => setTimeout(r, ms));\n\n let cursor: string | undefined = undefined;\n for (let j = 0; j < 1000 /* just for sanity */; ++j) {\n const response: Response = await client(query, {\n ...variables,\n cursor,\n });\n\n const conn = connection(response);\n if (!conn) {\n throw new Error(`Found no match for ${JSON.stringify(variables)}`);\n }\n\n for (const node of conn.nodes) {\n const transformedNode = await transformer(node, {\n client,\n query,\n org,\n });\n\n if (transformedNode) {\n result.push(transformedNode);\n }\n }\n\n if (!conn.pageInfo.hasNextPage) {\n break;\n } else {\n await sleep(1000);\n cursor = conn.pageInfo.endCursor;\n }\n }\n\n return result;\n}\n\nexport type DeferredEntitiesBuilder = (\n org: string,\n entities: Entity[],\n) => { added: DeferredEntity[]; removed: DeferredEntity[] };\n\nexport const createAddEntitiesOperation =\n (id: string, host: string) => (org: string, entities: Entity[]) => ({\n removed: [],\n added: entities.map(entity => ({\n locationKey: `github-org-provider:${id}`,\n entity: withLocations(`https://${host}`, org, entity),\n })),\n });\n\nexport const createRemoveEntitiesOperation =\n (id: string, host: string) => (org: string, entities: Entity[]) => ({\n added: [],\n removed: entities.map(entity => ({\n locationKey: `github-org-provider:${id}`,\n entity: withLocations(`https://${host}`, org, entity),\n })),\n });\n\nexport const createReplaceEntitiesOperation =\n (id: string, host: string) => (org: string, entities: Entity[]) => {\n const entitiesToReplace = entities.map(entity => ({\n locationKey: `github-org-provider:${id}`,\n entity: withLocations(`https://${host}`, org, entity),\n }));\n\n return {\n removed: entitiesToReplace,\n added: entitiesToReplace,\n };\n };\n\n/**\n * Creates a GraphQL Client with Throttling\n */\nexport const createGraphqlClient = (args: {\n headers:\n | {\n [name: string]: string;\n }\n | undefined;\n baseUrl: string;\n logger: LoggerService;\n}): typeof graphql => {\n const { headers, baseUrl, logger } = args;\n const ThrottledOctokit = Octokit.plugin(throttling);\n const octokit = new ThrottledOctokit({\n throttle: {\n onRateLimit: (retryAfter, rateLimitData, _, retryCount) => {\n logger.warn(\n `Request quota exhausted for request ${rateLimitData?.method} ${rateLimitData?.url}`,\n );\n\n if (retryCount < 2) {\n logger.warn(\n `Retrying after ${retryAfter} seconds for the ${retryCount} time due to Rate Limit!`,\n );\n return true;\n }\n\n return false;\n },\n onSecondaryRateLimit: (retryAfter, rateLimitData, _, retryCount) => {\n logger.warn(\n `Secondary Rate Limit Exhausted for request ${rateLimitData?.method} ${rateLimitData?.url}`,\n );\n\n if (retryCount < 2) {\n logger.warn(\n `Retrying after ${retryAfter} seconds for the ${retryCount} time due to Secondary Rate Limit!`,\n );\n return true;\n }\n\n return false;\n },\n },\n });\n\n const client = octokit.graphql.defaults({\n headers,\n baseUrl,\n });\n\n return client;\n};\n"],"names":["defaultUserTransformer","defaultOrganizationTeamTransformer","team","withLocations","Octokit","throttling"],"mappings":"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"}
1
+ {"version":3,"file":"github.cjs.js","sources":["../../src/lib/github.ts"],"sourcesContent":["/*\n * Copyright 2020 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 { Entity } from '@backstage/catalog-model';\nimport { GithubCredentialType } from '@backstage/integration';\nimport { graphql } from '@octokit/graphql';\nimport {\n defaultOrganizationTeamTransformer,\n defaultUserTransformer,\n TeamTransformer,\n TransformerContext,\n UserTransformer,\n} from './defaultTransformers';\nimport { withLocations } from './withLocations';\n\nimport { DeferredEntity } from '@backstage/plugin-catalog-node';\nimport { Octokit } from '@octokit/core';\nimport { LoggerService } from '@backstage/backend-plugin-api';\nimport { throttling } from '@octokit/plugin-throttling';\n// Graphql types\n\nexport type QueryResponse = {\n organization?: OrganizationResponse;\n repositoryOwner?: RepositoryOwnerResponse;\n user?: UserResponse;\n};\n\ntype RepositoryOwnerResponse = {\n repositories?: Connection<RepositoryResponse>;\n repository?: RepositoryResponse;\n};\n\nexport type OrganizationResponse = {\n membersWithRole?: Connection<GithubUser>;\n team?: GithubTeamResponse;\n teams?: Connection<GithubTeamResponse>;\n repositories?: Connection<RepositoryResponse>;\n};\n\nexport type UserResponse = {\n organizations?: Connection<GithubOrg>;\n};\n\nexport type PageInfo = {\n hasNextPage: boolean;\n endCursor?: string;\n};\n\nexport type GithubOrg = {\n login: string;\n};\n\n/**\n * Github User\n *\n * @public\n */\nexport type GithubUser = {\n login: string;\n bio?: string;\n avatarUrl?: string;\n email?: string;\n name?: string;\n organizationVerifiedDomainEmails?: string[];\n};\n\n/**\n * Github Team\n *\n * @public\n */\nexport type GithubTeam = {\n slug: string;\n combinedSlug: string;\n name?: string;\n description?: string;\n avatarUrl?: string;\n editTeamUrl?: string;\n parentTeam?: GithubTeam;\n members: GithubUser[];\n};\n\nexport type GithubTeamResponse = Omit<GithubTeam, 'members'> & {\n members: Connection<GithubUser>;\n};\n\nexport type RepositoryResponse = {\n name: string;\n url: string;\n isArchived: boolean;\n isFork: boolean;\n repositoryTopics: RepositoryTopics;\n defaultBranchRef: {\n name: string;\n } | null;\n catalogInfoFile: {\n __typename: string;\n id: string;\n text: string;\n } | null;\n visibility: string;\n};\n\ntype RepositoryTopics = {\n nodes: TopicNodes[];\n};\n\ntype TopicNodes = {\n topic: {\n name: string;\n };\n};\n\nexport type Connection<T> = {\n pageInfo: PageInfo;\n nodes: T[];\n};\n\n/**\n * Gets all the users out of a Github organization.\n *\n * Note that the users will not have their memberships filled in.\n *\n * @param client - An octokit graphql client\n * @param org - The slug of the org to read\n */\nexport async function getOrganizationUsers(\n client: typeof graphql,\n org: string,\n tokenType: GithubCredentialType,\n userTransformer: UserTransformer = defaultUserTransformer,\n): Promise<{ users: Entity[] }> {\n const query = `\n query users($org: String!, $email: Boolean!, $cursor: String) {\n organization(login: $org) {\n membersWithRole(first: 100, after: $cursor) {\n pageInfo { hasNextPage, endCursor }\n nodes {\n avatarUrl,\n bio,\n email @include(if: $email),\n login,\n name,\n organizationVerifiedDomainEmails(login: $org)\n }\n }\n }\n }`;\n\n // There is no user -> teams edge, so we leave the memberships empty for\n // now and let the team iteration handle it instead\n\n const users = await queryWithPaging(\n client,\n query,\n org,\n r => r.organization?.membersWithRole,\n userTransformer,\n {\n org,\n email: tokenType === 'token',\n },\n );\n\n return { users };\n}\n\n/**\n * Gets all the teams out of a Github organization.\n *\n * Note that the teams will not have any relations apart from parent filled in.\n *\n * @param client - An octokit graphql client\n * @param org - The slug of the org to read\n */\nexport async function getOrganizationTeams(\n client: typeof graphql,\n org: string,\n teamTransformer: TeamTransformer = defaultOrganizationTeamTransformer,\n): Promise<{\n teams: Entity[];\n}> {\n const query = `\n query teams($org: String!, $cursor: String) {\n organization(login: $org) {\n teams(first: 50, after: $cursor) {\n pageInfo { hasNextPage, endCursor }\n nodes {\n slug\n combinedSlug\n name\n description\n avatarUrl\n editTeamUrl\n parentTeam { slug }\n members(first: 100, membership: IMMEDIATE) {\n pageInfo { hasNextPage }\n nodes {\n avatarUrl,\n bio,\n email,\n login,\n name,\n organizationVerifiedDomainEmails(login: $org)\n }\n }\n }\n }\n }\n }`;\n\n const materialisedTeams = async (\n item: GithubTeamResponse,\n ctx: TransformerContext,\n ): Promise<Entity | undefined> => {\n const memberNames: GithubUser[] = [];\n\n if (!item.members.pageInfo.hasNextPage) {\n // We got all the members in one go, run the fast path\n for (const user of item.members.nodes) {\n memberNames.push(user);\n }\n } else {\n // There were more than a hundred immediate members - run the slow\n // path of fetching them explicitly\n const { members } = await getTeamMembers(ctx.client, ctx.org, item.slug);\n for (const userLogin of members) {\n memberNames.push(userLogin);\n }\n }\n\n const team: GithubTeam = {\n ...item,\n members: memberNames,\n };\n\n return await teamTransformer(team, ctx);\n };\n\n const teams = await queryWithPaging(\n client,\n query,\n org,\n r => r.organization?.teams,\n materialisedTeams,\n { org },\n );\n\n return { teams };\n}\n\nexport async function getOrganizationTeamsFromUsers(\n client: typeof graphql,\n org: string,\n userLogins: string[],\n teamTransformer: TeamTransformer = defaultOrganizationTeamTransformer,\n): Promise<{\n teams: Entity[];\n}> {\n const query = `\n query teams($org: String!, $cursor: String, $userLogins: [String!] = \"\") {\n organization(login: $org) {\n teams(first: 100, after: $cursor, userLogins: $userLogins) {\n pageInfo {\n hasNextPage\n endCursor\n }\n nodes {\n slug\n combinedSlug\n name\n description\n avatarUrl\n editTeamUrl\n parentTeam {\n slug\n }\n members(first: 100, membership: IMMEDIATE) {\n pageInfo {\n hasNextPage\n }\n nodes {\n avatarUrl,\n bio,\n email,\n login,\n name,\n organizationVerifiedDomainEmails(login: $org)\n }\n }\n }\n }\n }\n}`;\n\n const materialisedTeams = async (\n item: GithubTeamResponse,\n ctx: TransformerContext,\n ): Promise<Entity | undefined> => {\n const memberNames: GithubUser[] = [];\n\n if (!item.members.pageInfo.hasNextPage) {\n // We got all the members in one go, run the fast path\n for (const user of item.members.nodes) {\n memberNames.push(user);\n }\n } else {\n // There were more than a hundred immediate members - run the slow\n // path of fetching them explicitly\n const { members } = await getTeamMembers(ctx.client, ctx.org, item.slug);\n for (const userLogin of members) {\n memberNames.push(userLogin);\n }\n }\n\n const team: GithubTeam = {\n ...item,\n members: memberNames,\n };\n\n return await teamTransformer(team, ctx);\n };\n\n const teams = await queryWithPaging(\n client,\n query,\n org,\n r => r.organization?.teams,\n materialisedTeams,\n { org, userLogins },\n );\n\n return { teams };\n}\n\nexport async function getOrganizationTeamsForUser(\n client: typeof graphql,\n org: string,\n userLogin: string,\n teamTransformer: TeamTransformer,\n): Promise<{ teams: Entity[] }> {\n const query = `\n query teams($org: String!, $cursor: String, $userLogins: [String!] = \"\") {\n organization(login: $org) {\n teams(first: 100, after: $cursor, userLogins: $userLogins) {\n pageInfo {\n hasNextPage\n endCursor\n }\n nodes {\n slug\n combinedSlug\n name\n description\n avatarUrl\n editTeamUrl\n parentTeam {\n slug\n }\n }\n }\n }\n}`;\n\n const materialisedTeams = async (\n item: GithubTeamResponse,\n ctx: TransformerContext,\n ): Promise<Entity | undefined> => {\n const team: GithubTeam = {\n ...item,\n members: [{ login: userLogin }],\n };\n\n return await teamTransformer(team, ctx);\n };\n\n const teams = await queryWithPaging(\n client,\n query,\n org,\n r => r.organization?.teams,\n materialisedTeams,\n { org, userLogins: [userLogin] },\n );\n\n return { teams };\n}\n\nexport async function getOrganizationsFromUser(\n client: typeof graphql,\n user: string,\n): Promise<{\n orgs: string[];\n}> {\n const query = `\n query orgs($user: String!) {\n user(login: $user) {\n organizations(first: 100) {\n nodes { login }\n pageInfo { hasNextPage, endCursor }\n }\n }\n }`;\n\n const orgs = await queryWithPaging(\n client,\n query,\n '',\n r => r.user?.organizations,\n async o => o.login,\n { user },\n );\n\n return { orgs };\n}\n\nexport async function getOrganizationTeam(\n client: typeof graphql,\n org: string,\n teamSlug: string,\n teamTransformer: TeamTransformer = defaultOrganizationTeamTransformer,\n): Promise<{\n team: Entity;\n}> {\n const query = `\n query teams($org: String!, $teamSlug: String!) {\n organization(login: $org) {\n team(slug:$teamSlug) {\n slug\n combinedSlug\n name\n description\n avatarUrl\n editTeamUrl\n parentTeam { slug }\n members(first: 100, membership: IMMEDIATE) {\n pageInfo { hasNextPage }\n nodes { login }\n }\n }\n }\n }`;\n\n const materialisedTeam = async (\n item: GithubTeamResponse,\n ctx: TransformerContext,\n ): Promise<Entity | undefined> => {\n const memberNames: GithubUser[] = [];\n\n if (!item.members.pageInfo.hasNextPage) {\n // We got all the members in one go, run the fast path\n for (const user of item.members.nodes) {\n memberNames.push(user);\n }\n } else {\n // There were more than a hundred immediate members - run the slow\n // path of fetching them explicitly\n const { members } = await getTeamMembers(ctx.client, ctx.org, item.slug);\n for (const userLogin of members) {\n memberNames.push(userLogin);\n }\n }\n\n const team: GithubTeam = {\n ...item,\n members: memberNames,\n };\n\n return await teamTransformer(team, ctx);\n };\n\n const response: QueryResponse = await client(query, {\n org,\n teamSlug,\n });\n\n if (!response.organization?.team)\n throw new Error(`Found no match for team ${teamSlug}`);\n\n const team = await materialisedTeam(response.organization?.team, {\n query,\n client,\n org,\n });\n\n if (!team) throw new Error(`Can't transform for team ${teamSlug}`);\n\n return { team };\n}\n\nexport async function getOrganizationRepositories(\n client: typeof graphql,\n org: string,\n catalogPath: string,\n): Promise<{ repositories: RepositoryResponse[] }> {\n let relativeCatalogPathRef: string;\n // We must strip the leading slash or the query for objects does not work\n if (catalogPath.startsWith('/')) {\n relativeCatalogPathRef = catalogPath.substring(1);\n } else {\n relativeCatalogPathRef = catalogPath;\n }\n const catalogPathRef = `HEAD:${relativeCatalogPathRef}`;\n const query = `\n query repositories($org: String!, $catalogPathRef: String!, $cursor: String) {\n repositoryOwner(login: $org) {\n login\n repositories(first: 50, after: $cursor) {\n nodes {\n name\n catalogInfoFile: object(expression: $catalogPathRef) {\n __typename\n ... on Blob {\n id\n text\n }\n }\n url\n isArchived\n isFork\n visibility\n repositoryTopics(first: 100) {\n nodes {\n ... on RepositoryTopic {\n topic {\n name\n }\n }\n }\n }\n defaultBranchRef {\n name\n }\n }\n pageInfo {\n hasNextPage\n endCursor\n }\n }\n }\n }`;\n\n const repositories = await queryWithPaging(\n client,\n query,\n org,\n r => r.repositoryOwner?.repositories,\n async x => x,\n { org, catalogPathRef },\n );\n\n return { repositories };\n}\n\nexport async function getOrganizationRepository(\n client: typeof graphql,\n org: string,\n repoName: string,\n catalogPath: string,\n): Promise<RepositoryResponse | null> {\n let relativeCatalogPathRef: string;\n // We must strip the leading slash or the query for objects does not work\n if (catalogPath.startsWith('/')) {\n relativeCatalogPathRef = catalogPath.substring(1);\n } else {\n relativeCatalogPathRef = catalogPath;\n }\n const catalogPathRef = `HEAD:${relativeCatalogPathRef}`;\n const query = `\n query repository($org: String!, $repoName: String!, $catalogPathRef: String!) {\n repositoryOwner(login: $org) {\n repository(name: $repoName) {\n name\n catalogInfoFile: object(expression: $catalogPathRef) {\n __typename\n ... on Blob {\n id\n text\n }\n }\n url\n isArchived\n isFork\n visibility\n repositoryTopics(first: 100) {\n nodes {\n ... on RepositoryTopic {\n topic {\n name\n }\n }\n }\n }\n defaultBranchRef {\n name\n }\n }\n }\n }`;\n\n const response: QueryResponse = await client(query, {\n org,\n repoName,\n catalogPathRef,\n });\n\n return response.repositoryOwner?.repository || null;\n}\n\n/**\n * Gets all the users out of a Github organization.\n *\n * Note that the users will not have their memberships filled in.\n *\n * @param client - An octokit graphql client\n * @param org - The slug of the org to read\n * @param teamSlug - The slug of the team to read\n */\nexport async function getTeamMembers(\n client: typeof graphql,\n org: string,\n teamSlug: string,\n): Promise<{ members: GithubUser[] }> {\n const query = `\n query members($org: String!, $teamSlug: String!, $cursor: String) {\n organization(login: $org) {\n team(slug: $teamSlug) {\n members(first: 100, after: $cursor, membership: IMMEDIATE) {\n pageInfo { hasNextPage, endCursor }\n nodes { login }\n }\n }\n }\n }`;\n\n const members = await queryWithPaging(\n client,\n query,\n org,\n r => r.organization?.team?.members,\n async user => user,\n { org, teamSlug },\n );\n\n return { members };\n}\n\n//\n// Helpers\n//\n\n/**\n * Assists in repeatedly executing a query with a paged response.\n *\n * Requires that the query accepts a $cursor variable.\n *\n * @param client - The octokit client\n * @param query - The query to execute\n * @param org - The slug of the org to read\n * @param connection - A function that, given the response, picks out the actual\n * Connection object that's being iterated\n * @param transformer - A function that, given one of the nodes in the Connection,\n * returns the model mapped form of it\n * @param variables - The variable values that the query needs, minus the cursor\n */\nexport async function queryWithPaging<\n GraphqlType,\n OutputType,\n Variables extends {},\n Response = QueryResponse,\n>(\n client: typeof graphql,\n query: string,\n org: string,\n connection: (response: Response) => Connection<GraphqlType> | undefined,\n transformer: (\n item: GraphqlType,\n ctx: TransformerContext,\n ) => Promise<OutputType | undefined>,\n variables: Variables,\n): Promise<OutputType[]> {\n const result: OutputType[] = [];\n const sleep = (ms: number) => new Promise(r => setTimeout(r, ms));\n\n let cursor: string | undefined = undefined;\n for (let j = 0; j < 1000 /* just for sanity */; ++j) {\n const response: Response = await client(query, {\n ...variables,\n cursor,\n });\n\n const conn = connection(response);\n if (!conn) {\n throw new Error(`Found no match for ${JSON.stringify(variables)}`);\n }\n\n for (const node of conn.nodes) {\n const transformedNode = await transformer(node, {\n client,\n query,\n org,\n });\n\n if (transformedNode) {\n result.push(transformedNode);\n }\n }\n\n if (!conn.pageInfo.hasNextPage) {\n break;\n } else {\n await sleep(1000);\n cursor = conn.pageInfo.endCursor;\n }\n }\n\n return result;\n}\n\nexport type DeferredEntitiesBuilder = (\n org: string,\n entities: Entity[],\n) => { added: DeferredEntity[]; removed: DeferredEntity[] };\n\nexport const createAddEntitiesOperation =\n (id: string, host: string) => (org: string, entities: Entity[]) => ({\n removed: [],\n added: entities.map(entity => ({\n locationKey: `github-org-provider:${id}`,\n entity: withLocations(`https://${host}`, org, entity),\n })),\n });\n\nexport const createRemoveEntitiesOperation =\n (id: string, host: string) => (org: string, entities: Entity[]) => ({\n added: [],\n removed: entities.map(entity => ({\n locationKey: `github-org-provider:${id}`,\n entity: withLocations(`https://${host}`, org, entity),\n })),\n });\n\nexport const createReplaceEntitiesOperation =\n (id: string, host: string) => (org: string, entities: Entity[]) => {\n const entitiesToReplace = entities.map(entity => ({\n locationKey: `github-org-provider:${id}`,\n entity: withLocations(`https://${host}`, org, entity),\n }));\n\n return {\n removed: entitiesToReplace,\n added: entitiesToReplace,\n };\n };\n\n/**\n * Creates a GraphQL Client with Throttling\n */\nexport const createGraphqlClient = (args: {\n headers:\n | {\n [name: string]: string;\n }\n | undefined;\n baseUrl: string;\n logger: LoggerService;\n}): typeof graphql => {\n const { headers, baseUrl, logger } = args;\n const ThrottledOctokit = Octokit.plugin(throttling);\n const octokit = new ThrottledOctokit({\n throttle: {\n onRateLimit: (retryAfter, rateLimitData, _, retryCount) => {\n logger.warn(\n `Request quota exhausted for request ${rateLimitData?.method} ${rateLimitData?.url}`,\n );\n\n if (retryCount < 2) {\n logger.warn(\n `Retrying after ${retryAfter} seconds for the ${retryCount} time due to Rate Limit!`,\n );\n return true;\n }\n\n return false;\n },\n onSecondaryRateLimit: (retryAfter, rateLimitData, _, retryCount) => {\n logger.warn(\n `Secondary Rate Limit Exhausted for request ${rateLimitData?.method} ${rateLimitData?.url}`,\n );\n\n if (retryCount < 2) {\n logger.warn(\n `Retrying after ${retryAfter} seconds for the ${retryCount} time due to Secondary Rate Limit!`,\n );\n return true;\n }\n\n return false;\n },\n },\n });\n\n const client = octokit.graphql.defaults({\n headers,\n baseUrl,\n });\n\n return client;\n};\n"],"names":["defaultUserTransformer","defaultOrganizationTeamTransformer","team","withLocations","Octokit","throttling"],"mappings":"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"}
@@ -1 +1 @@
1
- {"version":3,"file":"guards.cjs.js","sources":["../../src/lib/guards.ts"],"sourcesContent":["/*\n * Copyright 2023 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 */\nimport {\n Entity,\n GroupEntity,\n UserEntity,\n isGroupEntity,\n isUserEntity,\n} from '@backstage/catalog-model';\n\nexport function areGroupEntities(\n entities: Entity[],\n): entities is GroupEntity[] {\n return entities.every(e => isGroupEntity(e));\n}\n\nexport function areUserEntities(entities: Entity[]): entities is UserEntity[] {\n return entities.every(e => isUserEntity(e));\n}\n"],"names":["isGroupEntity","isUserEntity"],"mappings":";;;;AAuBO,SAAS,iBACd,QAC2B,EAAA;AAC3B,EAAA,OAAO,QAAS,CAAA,KAAA,CAAM,CAAK,CAAA,KAAAA,0BAAA,CAAc,CAAC,CAAC,CAAA;AAC7C;AAEO,SAAS,gBAAgB,QAA8C,EAAA;AAC5E,EAAA,OAAO,QAAS,CAAA,KAAA,CAAM,CAAK,CAAA,KAAAC,yBAAA,CAAa,CAAC,CAAC,CAAA;AAC5C;;;;;"}
1
+ {"version":3,"file":"guards.cjs.js","sources":["../../src/lib/guards.ts"],"sourcesContent":["/*\n * Copyright 2023 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 */\nimport {\n Entity,\n GroupEntity,\n UserEntity,\n isGroupEntity,\n isUserEntity,\n} from '@backstage/catalog-model';\n\nexport function areGroupEntities(\n entities: Entity[],\n): entities is GroupEntity[] {\n return entities.every(e => isGroupEntity(e));\n}\n\nexport function areUserEntities(entities: Entity[]): entities is UserEntity[] {\n return entities.every(e => isUserEntity(e));\n}\n"],"names":["isGroupEntity","isUserEntity"],"mappings":";;;;AAuBO,SAAS,iBACd,QAAA,EAC2B;AAC3B,EAAA,OAAO,QAAA,CAAS,KAAA,CAAM,CAAA,CAAA,KAAKA,0BAAA,CAAc,CAAC,CAAC,CAAA;AAC7C;AAEO,SAAS,gBAAgB,QAAA,EAA8C;AAC5E,EAAA,OAAO,QAAA,CAAS,KAAA,CAAM,CAAA,CAAA,KAAKC,yBAAA,CAAa,CAAC,CAAC,CAAA;AAC5C;;;;;"}
@@ -1 +1 @@
1
- {"version":3,"file":"org.cjs.js","sources":["../../src/lib/org.ts"],"sourcesContent":["/*\n * Copyright 2020 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 DEFAULT_NAMESPACE,\n GroupEntity,\n parseEntityRef,\n stringifyEntityRef,\n UserEntity,\n} from '@backstage/catalog-model';\n\nexport function buildOrgHierarchy(groups: GroupEntity[]) {\n const groupsByName = new Map(groups.map(g => [g.metadata.name, g]));\n\n //\n // Make sure that g.parent.children contain g\n //\n\n for (const group of groups) {\n const selfName = group.metadata.name;\n const parentName = group.spec.parent;\n if (parentName) {\n const parent = groupsByName.get(parentName);\n if (parent && !parent.spec.children.includes(selfName)) {\n parent.spec.children.push(selfName);\n }\n }\n }\n\n //\n // Make sure that g.children.parent is g\n //\n\n for (const group of groups) {\n const selfName = group.metadata.name;\n for (const childName of group.spec.children) {\n const child = groupsByName.get(childName);\n if (child && !child.spec.parent) {\n child.spec.parent = selfName;\n }\n }\n }\n}\n\n// Ensure that users have their direct group memberships.\nexport function assignGroupsToUsers(\n users: UserEntity[],\n groups: GroupEntity[],\n) {\n const groupMemberUsers = new Map(\n groups.map(group => {\n const groupKey =\n group.metadata.namespace &&\n group.metadata.namespace !== DEFAULT_NAMESPACE\n ? `${group.metadata.namespace}/${group.metadata.name}`\n : group.metadata.name;\n // Fully qualify member refs so they can be keyed off of since they may contain namespace prefixes\n return [\n groupKey,\n group.spec.members?.map(m =>\n stringifyEntityRef(parseEntityRef(m, { defaultKind: 'user' })),\n ) || [],\n ];\n }),\n );\n\n const usersByRef = new Map(users.map(u => [stringifyEntityRef(u), u]));\n for (const [groupName, userRefs] of groupMemberUsers.entries()) {\n for (const ref of userRefs) {\n const user = usersByRef.get(ref);\n if (user && !user.spec.memberOf?.includes(groupName)) {\n if (!user.spec.memberOf) {\n user.spec.memberOf = [];\n }\n user.spec.memberOf.push(groupName);\n }\n }\n }\n}\n\n// Assign all relevant groups to a single user if the user is a member of each group.\nexport function assignGroupsToUser(user: UserEntity, groups: GroupEntity[]) {\n const userRef = stringifyEntityRef(user);\n for (const group of groups) {\n const groupKey =\n group.metadata.namespace && group.metadata.namespace !== DEFAULT_NAMESPACE\n ? `${group.metadata.namespace}/${group.metadata.name}`\n : group.metadata.name;\n const memberRefs =\n group.spec.members?.map(m =>\n stringifyEntityRef(parseEntityRef(m, { defaultKind: 'user' })),\n ) || [];\n if (memberRefs.includes(userRef)) {\n if (!user.spec.memberOf) {\n user.spec.memberOf = [];\n }\n if (!user.spec.memberOf.includes(groupKey)) {\n user.spec.memberOf.push(groupKey);\n }\n }\n }\n}\n\n// Ensure that users have their transitive group memberships. Requires that\n// the groups were previously processed with buildOrgHierarchy()\nexport function buildMemberOf(groups: GroupEntity[], users: UserEntity[]) {\n const groupsByName = new Map(groups.map(g => [g.metadata.name, g]));\n\n users.forEach(user => {\n const transitiveMemberOf = new Set<string>();\n\n const todo = [\n ...(user.spec.memberOf ?? []),\n ...groups\n .filter(g => g.spec.members?.includes(user.metadata.name))\n .map(g => g.metadata.name),\n ];\n\n for (;;) {\n const current = todo.pop();\n if (!current) {\n break;\n }\n\n if (!transitiveMemberOf.has(current)) {\n transitiveMemberOf.add(current);\n const group = groupsByName.get(current);\n if (group?.spec.parent) {\n todo.push(group.spec.parent);\n }\n }\n }\n\n user.spec.memberOf = [...transitiveMemberOf];\n });\n}\n"],"names":["DEFAULT_NAMESPACE","stringifyEntityRef","parseEntityRef"],"mappings":"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"}
1
+ {"version":3,"file":"org.cjs.js","sources":["../../src/lib/org.ts"],"sourcesContent":["/*\n * Copyright 2020 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 DEFAULT_NAMESPACE,\n GroupEntity,\n parseEntityRef,\n stringifyEntityRef,\n UserEntity,\n} from '@backstage/catalog-model';\n\nexport function buildOrgHierarchy(groups: GroupEntity[]) {\n const groupsByName = new Map(groups.map(g => [g.metadata.name, g]));\n\n //\n // Make sure that g.parent.children contain g\n //\n\n for (const group of groups) {\n const selfName = group.metadata.name;\n const parentName = group.spec.parent;\n if (parentName) {\n const parent = groupsByName.get(parentName);\n if (parent && !parent.spec.children.includes(selfName)) {\n parent.spec.children.push(selfName);\n }\n }\n }\n\n //\n // Make sure that g.children.parent is g\n //\n\n for (const group of groups) {\n const selfName = group.metadata.name;\n for (const childName of group.spec.children) {\n const child = groupsByName.get(childName);\n if (child && !child.spec.parent) {\n child.spec.parent = selfName;\n }\n }\n }\n}\n\n// Ensure that users have their direct group memberships.\nexport function assignGroupsToUsers(\n users: UserEntity[],\n groups: GroupEntity[],\n) {\n const groupMemberUsers = new Map(\n groups.map(group => {\n const groupKey =\n group.metadata.namespace &&\n group.metadata.namespace !== DEFAULT_NAMESPACE\n ? `${group.metadata.namespace}/${group.metadata.name}`\n : group.metadata.name;\n // Fully qualify member refs so they can be keyed off of since they may contain namespace prefixes\n return [\n groupKey,\n group.spec.members?.map(m =>\n stringifyEntityRef(parseEntityRef(m, { defaultKind: 'user' })),\n ) || [],\n ];\n }),\n );\n\n const usersByRef = new Map(users.map(u => [stringifyEntityRef(u), u]));\n for (const [groupName, userRefs] of groupMemberUsers.entries()) {\n for (const ref of userRefs) {\n const user = usersByRef.get(ref);\n if (user && !user.spec.memberOf?.includes(groupName)) {\n if (!user.spec.memberOf) {\n user.spec.memberOf = [];\n }\n user.spec.memberOf.push(groupName);\n }\n }\n }\n}\n\n// Assign all relevant groups to a single user if the user is a member of each group.\nexport function assignGroupsToUser(user: UserEntity, groups: GroupEntity[]) {\n const userRef = stringifyEntityRef(user);\n for (const group of groups) {\n const groupKey =\n group.metadata.namespace && group.metadata.namespace !== DEFAULT_NAMESPACE\n ? `${group.metadata.namespace}/${group.metadata.name}`\n : group.metadata.name;\n const memberRefs =\n group.spec.members?.map(m =>\n stringifyEntityRef(parseEntityRef(m, { defaultKind: 'user' })),\n ) || [];\n if (memberRefs.includes(userRef)) {\n if (!user.spec.memberOf) {\n user.spec.memberOf = [];\n }\n if (!user.spec.memberOf.includes(groupKey)) {\n user.spec.memberOf.push(groupKey);\n }\n }\n }\n}\n\n// Ensure that users have their transitive group memberships. Requires that\n// the groups were previously processed with buildOrgHierarchy()\nexport function buildMemberOf(groups: GroupEntity[], users: UserEntity[]) {\n const groupsByName = new Map(groups.map(g => [g.metadata.name, g]));\n\n users.forEach(user => {\n const transitiveMemberOf = new Set<string>();\n\n const todo = [\n ...(user.spec.memberOf ?? []),\n ...groups\n .filter(g => g.spec.members?.includes(user.metadata.name))\n .map(g => g.metadata.name),\n ];\n\n for (;;) {\n const current = todo.pop();\n if (!current) {\n break;\n }\n\n if (!transitiveMemberOf.has(current)) {\n transitiveMemberOf.add(current);\n const group = groupsByName.get(current);\n if (group?.spec.parent) {\n todo.push(group.spec.parent);\n }\n }\n }\n\n user.spec.memberOf = [...transitiveMemberOf];\n });\n}\n"],"names":["DEFAULT_NAMESPACE","stringifyEntityRef","parseEntityRef"],"mappings":"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"}
@@ -1 +1 @@
1
- {"version":3,"file":"util.cjs.js","sources":["../../src/lib/util.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 { GithubTopicFilters } from '../providers/GithubEntityProviderConfig';\n\nexport function parseGithubOrgUrl(urlString: string): { org: string } {\n const path = new URL(urlString).pathname.slice(1).split('/');\n\n // /backstage\n if (path.length === 1 && path[0].length) {\n return { org: decodeURIComponent(path[0]) };\n }\n\n throw new Error(`Expected a URL pointing to /<org>`);\n}\n\nexport function satisfiesTopicFilter(\n topics: string[],\n topicFilter: GithubTopicFilters | undefined,\n): Boolean {\n // We don't want to do anything if a filter is not configured (or configured but empty)\n if (!topicFilter) return true;\n if (!topicFilter.include && !topicFilter.exclude) return true;\n if (!topicFilter.include?.length && !topicFilter.exclude?.length) return true;\n // If topic.include is in use, a topic MUST be set that matches the inclusion\n // filter in order for a repository to be ingested\n if (topicFilter.include?.length && !topicFilter.exclude) {\n for (const topic of topics) {\n if (topicFilter.include.includes(topic)) return true;\n }\n return false;\n }\n // If topic.exclude is in use, all topics are included by default\n // with anything matching the filter being discarded. It is technically\n // possible for the filter to be explicitly empty meaning all repositories\n // are ingested although doing so would be pointless.\n if (!topicFilter.include && topicFilter.exclude?.length) {\n if (!topics.length) return true;\n for (const topic of topics) {\n if (topicFilter.exclude.includes(topic)) return false;\n }\n return true;\n }\n // Now the tricky part is where we have both include and exclude configured.\n // In this case, exclude wins out so we take everything matching the initial\n // inclusion filter and from that group, we further discard anything matching\n // the exclusion filter. If an item were to have a topic set in both filters,\n // we expect it to be discarded in the end. The use case here is that is that\n // you may want to retrieve all repos with the topic of team-a while excluding\n // matching repos that are marked as experiments.\n if (topicFilter.include && topicFilter.exclude) {\n const matchesInclude = satisfiesTopicFilter(topics, {\n include: topicFilter.include,\n });\n const matchesExclude = !satisfiesTopicFilter(topics, {\n exclude: topicFilter.exclude,\n });\n if (matchesExclude) return false;\n return matchesInclude;\n }\n\n // If the topic filter is somehow ever in a state that is not covered here, we\n // will fail \"open\" so that Backstage is still usable as if the filter was\n // not configured at all.\n return true;\n}\n\nexport function satisfiesForkFilter(\n allowForks: boolean | true,\n isFork: boolean | false,\n): Boolean {\n // we don't want to include forks if forks are not allowed\n if (!allowForks && isFork) return false;\n\n // if forks are allowed, allow all repos through\n return true;\n}\n\n// Given the github organisation team slug, returns a tuple containing [organisation, team]\nexport function splitTeamSlug(slug: string): [string, string] {\n const parts = slug.split('/');\n if (parts.length !== 2) {\n throw new Error(\n `Github team slug '${slug}' was not in the expected format <organisation>/<team>`,\n );\n }\n return [parts[0], parts[1]];\n}\n\nexport function satisfiesVisibilityFilter(\n visibilities: string[],\n visibility: string,\n): Boolean {\n if (!visibilities.length) {\n return true;\n }\n const lowerCaseVisibilities = visibilities.map(v =>\n v.toLocaleLowerCase('en-US'),\n );\n const lowerCaseVisibility = visibility.toLocaleLowerCase('en-US');\n\n return lowerCaseVisibilities.includes(lowerCaseVisibility);\n}\n"],"names":[],"mappings":"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"}
1
+ {"version":3,"file":"util.cjs.js","sources":["../../src/lib/util.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 { GithubTopicFilters } from '../providers/GithubEntityProviderConfig';\n\nexport function parseGithubOrgUrl(urlString: string): { org: string } {\n const path = new URL(urlString).pathname.slice(1).split('/');\n\n // /backstage\n if (path.length === 1 && path[0].length) {\n return { org: decodeURIComponent(path[0]) };\n }\n\n throw new Error(`Expected a URL pointing to /<org>`);\n}\n\nexport function satisfiesTopicFilter(\n topics: string[],\n topicFilter: GithubTopicFilters | undefined,\n): Boolean {\n // We don't want to do anything if a filter is not configured (or configured but empty)\n if (!topicFilter) return true;\n if (!topicFilter.include && !topicFilter.exclude) return true;\n if (!topicFilter.include?.length && !topicFilter.exclude?.length) return true;\n // If topic.include is in use, a topic MUST be set that matches the inclusion\n // filter in order for a repository to be ingested\n if (topicFilter.include?.length && !topicFilter.exclude) {\n for (const topic of topics) {\n if (topicFilter.include.includes(topic)) return true;\n }\n return false;\n }\n // If topic.exclude is in use, all topics are included by default\n // with anything matching the filter being discarded. It is technically\n // possible for the filter to be explicitly empty meaning all repositories\n // are ingested although doing so would be pointless.\n if (!topicFilter.include && topicFilter.exclude?.length) {\n if (!topics.length) return true;\n for (const topic of topics) {\n if (topicFilter.exclude.includes(topic)) return false;\n }\n return true;\n }\n // Now the tricky part is where we have both include and exclude configured.\n // In this case, exclude wins out so we take everything matching the initial\n // inclusion filter and from that group, we further discard anything matching\n // the exclusion filter. If an item were to have a topic set in both filters,\n // we expect it to be discarded in the end. The use case here is that is that\n // you may want to retrieve all repos with the topic of team-a while excluding\n // matching repos that are marked as experiments.\n if (topicFilter.include && topicFilter.exclude) {\n const matchesInclude = satisfiesTopicFilter(topics, {\n include: topicFilter.include,\n });\n const matchesExclude = !satisfiesTopicFilter(topics, {\n exclude: topicFilter.exclude,\n });\n if (matchesExclude) return false;\n return matchesInclude;\n }\n\n // If the topic filter is somehow ever in a state that is not covered here, we\n // will fail \"open\" so that Backstage is still usable as if the filter was\n // not configured at all.\n return true;\n}\n\nexport function satisfiesForkFilter(\n allowForks: boolean | true,\n isFork: boolean | false,\n): Boolean {\n // we don't want to include forks if forks are not allowed\n if (!allowForks && isFork) return false;\n\n // if forks are allowed, allow all repos through\n return true;\n}\n\n// Given the github organisation team slug, returns a tuple containing [organisation, team]\nexport function splitTeamSlug(slug: string): [string, string] {\n const parts = slug.split('/');\n if (parts.length !== 2) {\n throw new Error(\n `Github team slug '${slug}' was not in the expected format <organisation>/<team>`,\n );\n }\n return [parts[0], parts[1]];\n}\n\nexport function satisfiesVisibilityFilter(\n visibilities: string[],\n visibility: string,\n): Boolean {\n if (!visibilities.length) {\n return true;\n }\n const lowerCaseVisibilities = visibilities.map(v =>\n v.toLocaleLowerCase('en-US'),\n );\n const lowerCaseVisibility = visibility.toLocaleLowerCase('en-US');\n\n return lowerCaseVisibilities.includes(lowerCaseVisibility);\n}\n"],"names":[],"mappings":"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"}
@@ -1 +1 @@
1
- {"version":3,"file":"withLocations.cjs.js","sources":["../../src/lib/withLocations.ts"],"sourcesContent":["/*\n * Copyright 2023 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 ANNOTATION_LOCATION,\n ANNOTATION_ORIGIN_LOCATION,\n Entity,\n} from '@backstage/catalog-model';\nimport { merge } from 'lodash';\nimport {\n ANNOTATION_GITHUB_TEAM_SLUG,\n ANNOTATION_GITHUB_USER_LOGIN,\n} from './annotation';\nimport { splitTeamSlug } from './util';\n\n// Makes sure that emitted entities have a proper location\nexport function withLocations(\n baseUrl: string,\n org: string,\n entity: Entity,\n): Entity {\n const login =\n entity.metadata.annotations?.[ANNOTATION_GITHUB_USER_LOGIN] ||\n entity.metadata.name;\n\n let team = entity.metadata.name;\n const slug = entity.metadata.annotations?.[ANNOTATION_GITHUB_TEAM_SLUG];\n if (slug) {\n const [_, slugTeam] = splitTeamSlug(slug);\n team = slugTeam;\n }\n\n const location =\n entity.kind === 'Group'\n ? `url:${baseUrl}/orgs/${org}/teams/${team}`\n : `url:${baseUrl}/${login}`;\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"],"names":["ANNOTATION_GITHUB_USER_LOGIN","ANNOTATION_GITHUB_TEAM_SLUG","splitTeamSlug","merge","ANNOTATION_LOCATION","ANNOTATION_ORIGIN_LOCATION"],"mappings":";;;;;;;AA6BgB,SAAA,aAAA,CACd,OACA,EAAA,GAAA,EACA,MACQ,EAAA;AACR,EAAA,MAAM,QACJ,MAAO,CAAA,QAAA,CAAS,cAAcA,uCAA4B,CAAA,IAC1D,OAAO,QAAS,CAAA,IAAA;AAElB,EAAI,IAAA,IAAA,GAAO,OAAO,QAAS,CAAA,IAAA;AAC3B,EAAA,MAAM,IAAO,GAAA,MAAA,CAAO,QAAS,CAAA,WAAA,GAAcC,sCAA2B,CAAA;AACtE,EAAA,IAAI,IAAM,EAAA;AACR,IAAA,MAAM,CAAC,CAAA,EAAG,QAAQ,CAAA,GAAIC,mBAAc,IAAI,CAAA;AACxC,IAAO,IAAA,GAAA,QAAA;AAAA;AAGT,EAAA,MAAM,QACJ,GAAA,MAAA,CAAO,IAAS,KAAA,OAAA,GACZ,OAAO,OAAO,CAAA,MAAA,EAAS,GAAG,CAAA,OAAA,EAAU,IAAI,CAAA,CAAA,GACxC,CAAO,IAAA,EAAA,OAAO,IAAI,KAAK,CAAA,CAAA;AAC7B,EAAO,OAAAC,YAAA;AAAA,IACL;AAAA,MACE,QAAU,EAAA;AAAA,QACR,WAAa,EAAA;AAAA,UACX,CAACC,gCAAmB,GAAG,QAAA;AAAA,UACvB,CAACC,uCAA0B,GAAG;AAAA;AAChC;AACF,KACF;AAAA,IACA;AAAA,GACF;AACF;;;;"}
1
+ {"version":3,"file":"withLocations.cjs.js","sources":["../../src/lib/withLocations.ts"],"sourcesContent":["/*\n * Copyright 2023 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 ANNOTATION_LOCATION,\n ANNOTATION_ORIGIN_LOCATION,\n Entity,\n} from '@backstage/catalog-model';\nimport { merge } from 'lodash';\nimport {\n ANNOTATION_GITHUB_TEAM_SLUG,\n ANNOTATION_GITHUB_USER_LOGIN,\n} from './annotation';\nimport { splitTeamSlug } from './util';\n\n// Makes sure that emitted entities have a proper location\nexport function withLocations(\n baseUrl: string,\n org: string,\n entity: Entity,\n): Entity {\n const login =\n entity.metadata.annotations?.[ANNOTATION_GITHUB_USER_LOGIN] ||\n entity.metadata.name;\n\n let team = entity.metadata.name;\n const slug = entity.metadata.annotations?.[ANNOTATION_GITHUB_TEAM_SLUG];\n if (slug) {\n const [_, slugTeam] = splitTeamSlug(slug);\n team = slugTeam;\n }\n\n const location =\n entity.kind === 'Group'\n ? `url:${baseUrl}/orgs/${org}/teams/${team}`\n : `url:${baseUrl}/${login}`;\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"],"names":["ANNOTATION_GITHUB_USER_LOGIN","ANNOTATION_GITHUB_TEAM_SLUG","splitTeamSlug","merge","ANNOTATION_LOCATION","ANNOTATION_ORIGIN_LOCATION"],"mappings":";;;;;;;AA6BO,SAAS,aAAA,CACd,OAAA,EACA,GAAA,EACA,MAAA,EACQ;AACR,EAAA,MAAM,QACJ,MAAA,CAAO,QAAA,CAAS,cAAcA,uCAA4B,CAAA,IAC1D,OAAO,QAAA,CAAS,IAAA;AAElB,EAAA,IAAI,IAAA,GAAO,OAAO,QAAA,CAAS,IAAA;AAC3B,EAAA,MAAM,IAAA,GAAO,MAAA,CAAO,QAAA,CAAS,WAAA,GAAcC,sCAA2B,CAAA;AACtE,EAAA,IAAI,IAAA,EAAM;AACR,IAAA,MAAM,CAAC,CAAA,EAAG,QAAQ,CAAA,GAAIC,mBAAc,IAAI,CAAA;AACxC,IAAA,IAAA,GAAO,QAAA;AAAA,EACT;AAEA,EAAA,MAAM,QAAA,GACJ,MAAA,CAAO,IAAA,KAAS,OAAA,GACZ,OAAO,OAAO,CAAA,MAAA,EAAS,GAAG,CAAA,OAAA,EAAU,IAAI,CAAA,CAAA,GACxC,CAAA,IAAA,EAAO,OAAO,IAAI,KAAK,CAAA,CAAA;AAC7B,EAAA,OAAOC,YAAA;AAAA,IACL;AAAA,MACE,QAAA,EAAU;AAAA,QACR,WAAA,EAAa;AAAA,UACX,CAACC,gCAAmB,GAAG,QAAA;AAAA,UACvB,CAACC,uCAA0B,GAAG;AAAA;AAChC;AACF,KACF;AAAA,IACA;AAAA,GACF;AACF;;;;"}
@@ -1 +1 @@
1
- {"version":3,"file":"githubCatalogModule.cjs.js","sources":["../../src/module/githubCatalogModule.ts"],"sourcesContent":["/*\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 coreServices,\n createBackendModule,\n} from '@backstage/backend-plugin-api';\nimport {\n catalogAnalysisExtensionPoint,\n catalogProcessingExtensionPoint,\n} from '@backstage/plugin-catalog-node/alpha';\nimport { catalogServiceRef } from '@backstage/plugin-catalog-node';\nimport { eventsServiceRef } from '@backstage/plugin-events-node';\nimport { GithubEntityProvider } from '../providers/GithubEntityProvider';\nimport { GithubLocationAnalyzer } from '../analyzers/GithubLocationAnalyzer';\n\n/**\n * Registers the `GithubEntityProvider` with the catalog processing extension point.\n *\n * @public\n */\nexport const githubCatalogModule = createBackendModule({\n pluginId: 'catalog',\n moduleId: 'github',\n register(env) {\n env.registerInit({\n deps: {\n catalogAnalyzers: catalogAnalysisExtensionPoint,\n auth: coreServices.auth,\n catalogProcessing: catalogProcessingExtensionPoint,\n config: coreServices.rootConfig,\n events: eventsServiceRef,\n logger: coreServices.logger,\n scheduler: coreServices.scheduler,\n catalog: catalogServiceRef,\n },\n async init({\n catalogProcessing,\n config,\n events,\n logger,\n scheduler,\n catalogAnalyzers,\n auth,\n catalog,\n }) {\n catalogAnalyzers.addScmLocationAnalyzer(\n new GithubLocationAnalyzer({\n config,\n auth,\n catalog,\n }),\n );\n\n catalogProcessing.addEntityProvider(\n GithubEntityProvider.fromConfig(config, {\n events,\n logger,\n scheduler,\n }),\n );\n },\n });\n },\n});\n"],"names":["createBackendModule","catalogAnalysisExtensionPoint","coreServices","catalogProcessingExtensionPoint","eventsServiceRef","catalogServiceRef","GithubLocationAnalyzer","GithubEntityProvider"],"mappings":";;;;;;;;;AAkCO,MAAM,sBAAsBA,oCAAoB,CAAA;AAAA,EACrD,QAAU,EAAA,SAAA;AAAA,EACV,QAAU,EAAA,QAAA;AAAA,EACV,SAAS,GAAK,EAAA;AACZ,IAAA,GAAA,CAAI,YAAa,CAAA;AAAA,MACf,IAAM,EAAA;AAAA,QACJ,gBAAkB,EAAAC,mCAAA;AAAA,QAClB,MAAMC,6BAAa,CAAA,IAAA;AAAA,QACnB,iBAAmB,EAAAC,qCAAA;AAAA,QACnB,QAAQD,6BAAa,CAAA,UAAA;AAAA,QACrB,MAAQ,EAAAE,iCAAA;AAAA,QACR,QAAQF,6BAAa,CAAA,MAAA;AAAA,QACrB,WAAWA,6BAAa,CAAA,SAAA;AAAA,QACxB,OAAS,EAAAG;AAAA,OACX;AAAA,MACA,MAAM,IAAK,CAAA;AAAA,QACT,iBAAA;AAAA,QACA,MAAA;AAAA,QACA,MAAA;AAAA,QACA,MAAA;AAAA,QACA,SAAA;AAAA,QACA,gBAAA;AAAA,QACA,IAAA;AAAA,QACA;AAAA,OACC,EAAA;AACD,QAAiB,gBAAA,CAAA,sBAAA;AAAA,UACf,IAAIC,6CAAuB,CAAA;AAAA,YACzB,MAAA;AAAA,YACA,IAAA;AAAA,YACA;AAAA,WACD;AAAA,SACH;AAEA,QAAkB,iBAAA,CAAA,iBAAA;AAAA,UAChBC,yCAAA,CAAqB,WAAW,MAAQ,EAAA;AAAA,YACtC,MAAA;AAAA,YACA,MAAA;AAAA,YACA;AAAA,WACD;AAAA,SACH;AAAA;AACF,KACD,CAAA;AAAA;AAEL,CAAC;;;;"}
1
+ {"version":3,"file":"githubCatalogModule.cjs.js","sources":["../../src/module/githubCatalogModule.ts"],"sourcesContent":["/*\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 coreServices,\n createBackendModule,\n} from '@backstage/backend-plugin-api';\nimport {\n catalogAnalysisExtensionPoint,\n catalogProcessingExtensionPoint,\n} from '@backstage/plugin-catalog-node/alpha';\nimport { catalogServiceRef } from '@backstage/plugin-catalog-node';\nimport { eventsServiceRef } from '@backstage/plugin-events-node';\nimport { GithubEntityProvider } from '../providers/GithubEntityProvider';\nimport { GithubLocationAnalyzer } from '../analyzers/GithubLocationAnalyzer';\n\n/**\n * Registers the `GithubEntityProvider` with the catalog processing extension point.\n *\n * @public\n */\nexport const githubCatalogModule = createBackendModule({\n pluginId: 'catalog',\n moduleId: 'github',\n register(env) {\n env.registerInit({\n deps: {\n catalogAnalyzers: catalogAnalysisExtensionPoint,\n auth: coreServices.auth,\n catalogProcessing: catalogProcessingExtensionPoint,\n config: coreServices.rootConfig,\n events: eventsServiceRef,\n logger: coreServices.logger,\n scheduler: coreServices.scheduler,\n catalog: catalogServiceRef,\n },\n async init({\n catalogProcessing,\n config,\n events,\n logger,\n scheduler,\n catalogAnalyzers,\n auth,\n catalog,\n }) {\n catalogAnalyzers.addScmLocationAnalyzer(\n new GithubLocationAnalyzer({\n config,\n auth,\n catalog,\n }),\n );\n\n catalogProcessing.addEntityProvider(\n GithubEntityProvider.fromConfig(config, {\n events,\n logger,\n scheduler,\n }),\n );\n },\n });\n },\n});\n"],"names":["createBackendModule","catalogAnalysisExtensionPoint","coreServices","catalogProcessingExtensionPoint","eventsServiceRef","catalogServiceRef","GithubLocationAnalyzer","GithubEntityProvider"],"mappings":";;;;;;;;;AAkCO,MAAM,sBAAsBA,oCAAA,CAAoB;AAAA,EACrD,QAAA,EAAU,SAAA;AAAA,EACV,QAAA,EAAU,QAAA;AAAA,EACV,SAAS,GAAA,EAAK;AACZ,IAAA,GAAA,CAAI,YAAA,CAAa;AAAA,MACf,IAAA,EAAM;AAAA,QACJ,gBAAA,EAAkBC,mCAAA;AAAA,QAClB,MAAMC,6BAAA,CAAa,IAAA;AAAA,QACnB,iBAAA,EAAmBC,qCAAA;AAAA,QACnB,QAAQD,6BAAA,CAAa,UAAA;AAAA,QACrB,MAAA,EAAQE,iCAAA;AAAA,QACR,QAAQF,6BAAA,CAAa,MAAA;AAAA,QACrB,WAAWA,6BAAA,CAAa,SAAA;AAAA,QACxB,OAAA,EAASG;AAAA,OACX;AAAA,MACA,MAAM,IAAA,CAAK;AAAA,QACT,iBAAA;AAAA,QACA,MAAA;AAAA,QACA,MAAA;AAAA,QACA,MAAA;AAAA,QACA,SAAA;AAAA,QACA,gBAAA;AAAA,QACA,IAAA;AAAA,QACA;AAAA,OACF,EAAG;AACD,QAAA,gBAAA,CAAiB,sBAAA;AAAA,UACf,IAAIC,6CAAA,CAAuB;AAAA,YACzB,MAAA;AAAA,YACA,IAAA;AAAA,YACA;AAAA,WACD;AAAA,SACH;AAEA,QAAA,iBAAA,CAAkB,iBAAA;AAAA,UAChBC,yCAAA,CAAqB,WAAW,MAAA,EAAQ;AAAA,YACtC,MAAA;AAAA,YACA,MAAA;AAAA,YACA;AAAA,WACD;AAAA,SACH;AAAA,MACF;AAAA,KACD,CAAA;AAAA,EACH;AACF,CAAC;;;;"}
@@ -1 +1 @@
1
- {"version":3,"file":"GithubDiscoveryProcessor.cjs.js","sources":["../../src/processors/GithubDiscoveryProcessor.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 { Config } from '@backstage/config';\nimport {\n DefaultGithubCredentialsProvider,\n GithubCredentialsProvider,\n ScmIntegrationRegistry,\n ScmIntegrations,\n} from '@backstage/integration';\nimport {\n CatalogProcessor,\n CatalogProcessorEmit,\n LocationSpec,\n processingResult,\n} from '@backstage/plugin-catalog-node';\nimport { graphql } from '@octokit/graphql';\nimport { getOrganizationRepositories } from '../lib';\nimport { LoggerService } from '@backstage/backend-plugin-api';\n\n/**\n * Extracts repositories out of a GitHub org.\n *\n * The following will create locations for all projects which have a catalog-info.yaml\n * on the default branch. The first is shorthand for the second.\n *\n * target: \"https://github.com/backstage\"\n * or\n * target: https://github.com/backstage/*\\/blob/-/catalog-info.yaml\n *\n * You may also explicitly specify the source branch:\n *\n * target: https://github.com/backstage/*\\/blob/main/catalog-info.yaml\n *\n * @public\n */\nexport class GithubDiscoveryProcessor implements CatalogProcessor {\n private readonly integrations: ScmIntegrationRegistry;\n private readonly logger: LoggerService;\n private readonly githubCredentialsProvider: GithubCredentialsProvider;\n\n static fromConfig(\n config: Config,\n options: {\n logger: LoggerService;\n githubCredentialsProvider?: GithubCredentialsProvider;\n },\n ) {\n const integrations = ScmIntegrations.fromConfig(config);\n\n return new GithubDiscoveryProcessor({\n ...options,\n integrations,\n });\n }\n\n constructor(options: {\n integrations: ScmIntegrationRegistry;\n logger: LoggerService;\n githubCredentialsProvider?: GithubCredentialsProvider;\n }) {\n this.integrations = options.integrations;\n this.logger = options.logger;\n this.githubCredentialsProvider =\n options.githubCredentialsProvider ||\n DefaultGithubCredentialsProvider.fromIntegrations(this.integrations);\n }\n getProcessorName(): string {\n return 'GithubDiscoveryProcessor';\n }\n\n async readLocation(\n location: LocationSpec,\n _optional: boolean,\n emit: CatalogProcessorEmit,\n ): Promise<boolean> {\n if (location.type !== 'github-discovery') {\n return false;\n }\n\n const gitHubConfig = this.integrations.github.byUrl(\n location.target,\n )?.config;\n if (!gitHubConfig) {\n throw new Error(\n `There is no GitHub integration that matches ${location.target}. Please add a configuration entry for it under integrations.github`,\n );\n }\n\n const { org, repoSearchPath, catalogPath, branch, host } = parseUrl(\n location.target,\n );\n\n // Building the org url here so that the github creds provider doesn't need to know\n // about how to handle the wild card which is special for this processor.\n const orgUrl = `https://${host}/${org}`;\n\n const { headers } = await this.githubCredentialsProvider.getCredentials({\n url: orgUrl,\n });\n\n const client = graphql.defaults({\n baseUrl: gitHubConfig.apiBaseUrl,\n headers,\n });\n\n // Read out all of the raw data\n const startTimestamp = Date.now();\n this.logger.info(`Reading GitHub repositories from ${location.target}`);\n\n const { repositories } = await getOrganizationRepositories(\n client,\n org,\n catalogPath,\n );\n const matching = repositories.filter(\n r => !r.isArchived && repoSearchPath.test(r.name),\n );\n\n const duration = ((Date.now() - startTimestamp) / 1000).toFixed(1);\n this.logger.debug(\n `Read ${repositories.length} GitHub repositories (${matching.length} matching the pattern) in ${duration} seconds`,\n );\n\n for (const repository of matching) {\n const branchName =\n branch === '-' ? repository.defaultBranchRef?.name : branch;\n\n if (!branchName) {\n this.logger.info(\n `the repository ${repository.url} does not have a default branch, skipping`,\n );\n continue;\n }\n\n const path = `/blob/${branchName}${catalogPath}`;\n\n emit(\n processingResult.location({\n type: 'url',\n target: `${repository.url}${path}`,\n // Not all locations may actually exist, since the user defined them as a wildcard pattern.\n // Thus, we emit them as optional and let the downstream processor find them while not outputting\n // an error if it couldn't.\n presence: 'optional',\n }),\n );\n }\n\n return true;\n }\n}\n\n/*\n * Helpers\n */\n\nexport function parseUrl(urlString: string): {\n org: string;\n repoSearchPath: RegExp;\n catalogPath: string;\n branch: string;\n host: string;\n} {\n const url = new URL(urlString);\n const path = url.pathname.slice(1).split('/');\n\n // /backstage/techdocs-*/blob/master/catalog-info.yaml\n // can also be\n // /backstage\n if (path.length > 2 && path[0].length && path[1].length) {\n return {\n org: decodeURIComponent(path[0]),\n repoSearchPath: escapeRegExp(decodeURIComponent(path[1])),\n branch: decodeURIComponent(path[3]),\n catalogPath: `/${decodeURIComponent(path.slice(4).join('/'))}`,\n host: url.host,\n };\n } else if (path.length === 1 && path[0].length) {\n return {\n org: decodeURIComponent(path[0]),\n host: url.host,\n repoSearchPath: escapeRegExp('*'),\n catalogPath: '/catalog-info.yaml',\n branch: '-',\n };\n }\n\n throw new Error(`Failed to parse ${urlString}`);\n}\n\nexport function escapeRegExp(str: string): RegExp {\n return new RegExp(`^${str.replace(/\\*/g, '.*')}$`);\n}\n"],"names":["ScmIntegrations","DefaultGithubCredentialsProvider","graphql","getOrganizationRepositories","processingResult"],"mappings":"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"}
1
+ {"version":3,"file":"GithubDiscoveryProcessor.cjs.js","sources":["../../src/processors/GithubDiscoveryProcessor.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 { Config } from '@backstage/config';\nimport {\n DefaultGithubCredentialsProvider,\n GithubCredentialsProvider,\n ScmIntegrationRegistry,\n ScmIntegrations,\n} from '@backstage/integration';\nimport {\n CatalogProcessor,\n CatalogProcessorEmit,\n LocationSpec,\n processingResult,\n} from '@backstage/plugin-catalog-node';\nimport { graphql } from '@octokit/graphql';\nimport { getOrganizationRepositories } from '../lib';\nimport { LoggerService } from '@backstage/backend-plugin-api';\n\n/**\n * Extracts repositories out of a GitHub org.\n *\n * The following will create locations for all projects which have a catalog-info.yaml\n * on the default branch. The first is shorthand for the second.\n *\n * target: \"https://github.com/backstage\"\n * or\n * target: https://github.com/backstage/*\\/blob/-/catalog-info.yaml\n *\n * You may also explicitly specify the source branch:\n *\n * target: https://github.com/backstage/*\\/blob/main/catalog-info.yaml\n *\n * @public\n */\nexport class GithubDiscoveryProcessor implements CatalogProcessor {\n private readonly integrations: ScmIntegrationRegistry;\n private readonly logger: LoggerService;\n private readonly githubCredentialsProvider: GithubCredentialsProvider;\n\n static fromConfig(\n config: Config,\n options: {\n logger: LoggerService;\n githubCredentialsProvider?: GithubCredentialsProvider;\n },\n ) {\n const integrations = ScmIntegrations.fromConfig(config);\n\n return new GithubDiscoveryProcessor({\n ...options,\n integrations,\n });\n }\n\n constructor(options: {\n integrations: ScmIntegrationRegistry;\n logger: LoggerService;\n githubCredentialsProvider?: GithubCredentialsProvider;\n }) {\n this.integrations = options.integrations;\n this.logger = options.logger;\n this.githubCredentialsProvider =\n options.githubCredentialsProvider ||\n DefaultGithubCredentialsProvider.fromIntegrations(this.integrations);\n }\n getProcessorName(): string {\n return 'GithubDiscoveryProcessor';\n }\n\n async readLocation(\n location: LocationSpec,\n _optional: boolean,\n emit: CatalogProcessorEmit,\n ): Promise<boolean> {\n if (location.type !== 'github-discovery') {\n return false;\n }\n\n const gitHubConfig = this.integrations.github.byUrl(\n location.target,\n )?.config;\n if (!gitHubConfig) {\n throw new Error(\n `There is no GitHub integration that matches ${location.target}. Please add a configuration entry for it under integrations.github`,\n );\n }\n\n const { org, repoSearchPath, catalogPath, branch, host } = parseUrl(\n location.target,\n );\n\n // Building the org url here so that the github creds provider doesn't need to know\n // about how to handle the wild card which is special for this processor.\n const orgUrl = `https://${host}/${org}`;\n\n const { headers } = await this.githubCredentialsProvider.getCredentials({\n url: orgUrl,\n });\n\n const client = graphql.defaults({\n baseUrl: gitHubConfig.apiBaseUrl,\n headers,\n });\n\n // Read out all of the raw data\n const startTimestamp = Date.now();\n this.logger.info(`Reading GitHub repositories from ${location.target}`);\n\n const { repositories } = await getOrganizationRepositories(\n client,\n org,\n catalogPath,\n );\n const matching = repositories.filter(\n r => !r.isArchived && repoSearchPath.test(r.name),\n );\n\n const duration = ((Date.now() - startTimestamp) / 1000).toFixed(1);\n this.logger.debug(\n `Read ${repositories.length} GitHub repositories (${matching.length} matching the pattern) in ${duration} seconds`,\n );\n\n for (const repository of matching) {\n const branchName =\n branch === '-' ? repository.defaultBranchRef?.name : branch;\n\n if (!branchName) {\n this.logger.info(\n `the repository ${repository.url} does not have a default branch, skipping`,\n );\n continue;\n }\n\n const path = `/blob/${branchName}${catalogPath}`;\n\n emit(\n processingResult.location({\n type: 'url',\n target: `${repository.url}${path}`,\n // Not all locations may actually exist, since the user defined them as a wildcard pattern.\n // Thus, we emit them as optional and let the downstream processor find them while not outputting\n // an error if it couldn't.\n presence: 'optional',\n }),\n );\n }\n\n return true;\n }\n}\n\n/*\n * Helpers\n */\n\nexport function parseUrl(urlString: string): {\n org: string;\n repoSearchPath: RegExp;\n catalogPath: string;\n branch: string;\n host: string;\n} {\n const url = new URL(urlString);\n const path = url.pathname.slice(1).split('/');\n\n // /backstage/techdocs-*/blob/master/catalog-info.yaml\n // can also be\n // /backstage\n if (path.length > 2 && path[0].length && path[1].length) {\n return {\n org: decodeURIComponent(path[0]),\n repoSearchPath: escapeRegExp(decodeURIComponent(path[1])),\n branch: decodeURIComponent(path[3]),\n catalogPath: `/${decodeURIComponent(path.slice(4).join('/'))}`,\n host: url.host,\n };\n } else if (path.length === 1 && path[0].length) {\n return {\n org: decodeURIComponent(path[0]),\n host: url.host,\n repoSearchPath: escapeRegExp('*'),\n catalogPath: '/catalog-info.yaml',\n branch: '-',\n };\n }\n\n throw new Error(`Failed to parse ${urlString}`);\n}\n\nexport function escapeRegExp(str: string): RegExp {\n return new RegExp(`^${str.replace(/\\*/g, '.*')}$`);\n}\n"],"names":["ScmIntegrations","DefaultGithubCredentialsProvider","graphql","getOrganizationRepositories","processingResult"],"mappings":"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"}