@backstage/plugin-catalog-backend-module-gcp 0.3.11-next.0 → 0.3.12-next.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/CHANGELOG.md CHANGED
@@ -1,5 +1,23 @@
1
1
  # @backstage/plugin-catalog-backend-module-gcp
2
2
 
3
+ ## 0.3.12-next.0
4
+
5
+ ### Patch Changes
6
+
7
+ - Updated dependencies
8
+ - @backstage/backend-plugin-api@1.4.3-next.0
9
+ - @backstage/plugin-catalog-node@1.18.1-next.0
10
+
11
+ ## 0.3.11
12
+
13
+ ### Patch Changes
14
+
15
+ - cb36a86: Added support for Google Service account credentials config used in GkeEntityProvider.
16
+ Added support for additional metadata `authProvider` and `owner` to be set for the GKE cluster entities.
17
+ - Updated dependencies
18
+ - @backstage/plugin-catalog-node@1.18.0
19
+ - @backstage/backend-plugin-api@1.4.2
20
+
3
21
  ## 0.3.11-next.0
4
22
 
5
23
  ### Patch Changes
package/README.md CHANGED
@@ -2,6 +2,15 @@
2
2
 
3
3
  This is an extension module to the plugin-catalog-backend plugin, containing catalog processors and providers to ingest GCP resources as `Resource` kind entities.
4
4
 
5
+ ## Authentication
6
+
7
+ The GKE Entity Provider supports two authentication methods:
8
+
9
+ 1. **Service Account Credentials** (recommended for production): Provide Google Service Account credentials directly in the configuration
10
+ 2. **Application Default Credentials**: If no credentials are provided, the provider falls back to:
11
+ - `GOOGLE_APPLICATION_CREDENTIALS` environment variable pointing to a service account key file
12
+ - Google Cloud SDK default credentials (when running on Google Cloud Platform)
13
+
5
14
  ## installation
6
15
 
7
16
  Register the plugin in `catalog.ts``
@@ -38,4 +47,26 @@ catalog:
38
47
  frequency: { minutes: 30 }
39
48
  # supports ISO duration, "human duration" as used in code
40
49
  timeout: { minutes: 3 }
50
+ # Optional: Google Service Account credentials for authentication
51
+ # If not provided, falls back to Application Default Credentials or GOOGLE_APPLICATION_CREDENTIALS
52
+ googleServiceAccountCredentials: |
53
+ {
54
+ "type": "service_account",
55
+ "project_id": "your-project-id",
56
+ "private_key_id": "key-id",
57
+ "private_key": "-----BEGIN PRIVATE KEY-----\n...\n-----END PRIVATE KEY-----\n",
58
+ "client_email": "your-service-account@your-project.iam.gserviceaccount.com",
59
+ "client_id": "client-id",
60
+ "auth_uri": "https://accounts.google.com/o/oauth2/auth",
61
+ "token_uri": "https://oauth2.googleapis.com/token",
62
+ "auth_provider_x509_cert_url": "https://www.googleapis.com/oauth2/v1/certs",
63
+ "client_x509_cert_url": "https://www.googleapis.com/robot/v1/metadata/x509/..."
64
+ }
65
+ # Optional: Authentication provider for Kubernetes clusters
66
+ # Defaults to 'google' if not specified
67
+ # Common values: 'google', 'googleServiceAccount'
68
+ authProvider: googleServiceAccount
69
+ # Optional: Owner of the discovered GKE clusters
70
+ # Defaults to 'unknown' if not specified
71
+ owner: platform-team
41
72
  ```
package/config.d.ts CHANGED
@@ -38,6 +38,22 @@ export interface Config {
38
38
  * (Optional) TaskScheduleDefinition for the refresh.
39
39
  */
40
40
  schedule: SchedulerServiceTaskScheduleDefinitionConfig;
41
+ /**
42
+ * (Optional) Google Service Account credentials for authentication
43
+ * JSON string containing the service account key
44
+ * @visibility secret
45
+ */
46
+ googleServiceAccountCredentials?: string;
47
+ /**
48
+ * (Optional) Authentication provider to use for Kubernetes clusters
49
+ * Defaults to 'google' for backward compatibility
50
+ */
51
+ authProvider?: string;
52
+ /**
53
+ * (Optional) Owner of the discovered clusters
54
+ * Defaults to 'unknown' if not specified
55
+ */
56
+ owner?: string;
41
57
  };
42
58
  };
43
59
  };
package/dist/index.d.ts CHANGED
@@ -13,6 +13,8 @@ declare class GkeEntityProvider implements EntityProvider {
13
13
  private readonly logger;
14
14
  private readonly scheduleFn;
15
15
  private readonly gkeParents;
16
+ private readonly gkeAuthProvider;
17
+ private readonly gkeOwner;
16
18
  private readonly clusterManagerClient;
17
19
  private connection?;
18
20
  private constructor();
@@ -1 +1 @@
1
- {"version":3,"file":"catalogModuleGcpGkeEntityProvider.cjs.js","sources":["../../src/module/catalogModuleGcpGkeEntityProvider.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 { catalogProcessingExtensionPoint } from '@backstage/plugin-catalog-node/alpha';\nimport { GkeEntityProvider } from '../providers/GkeEntityProvider';\n\n/**\n * Registers the GcpGkeEntityProvider with the catalog processing extension point.\n *\n * @public\n */\nexport const catalogModuleGcpGkeEntityProvider = createBackendModule({\n pluginId: 'catalog',\n moduleId: 'gcp-gke-entity-provider',\n register(env) {\n env.registerInit({\n deps: {\n config: coreServices.rootConfig,\n catalog: catalogProcessingExtensionPoint,\n logger: coreServices.logger,\n scheduler: coreServices.scheduler,\n },\n async init({ config, catalog, logger, scheduler }) {\n catalog.addEntityProvider(\n GkeEntityProvider.fromConfig({\n logger,\n scheduler,\n config,\n }),\n );\n },\n });\n },\n});\n"],"names":["createBackendModule","coreServices","catalogProcessingExtensionPoint","GkeEntityProvider"],"mappings":";;;;;;AA4BO,MAAM,oCAAoCA,oCAAoB,CAAA;AAAA,EACnE,QAAU,EAAA,SAAA;AAAA,EACV,QAAU,EAAA,yBAAA;AAAA,EACV,SAAS,GAAK,EAAA;AACZ,IAAA,GAAA,CAAI,YAAa,CAAA;AAAA,MACf,IAAM,EAAA;AAAA,QACJ,QAAQC,6BAAa,CAAA,UAAA;AAAA,QACrB,OAAS,EAAAC,qCAAA;AAAA,QACT,QAAQD,6BAAa,CAAA,MAAA;AAAA,QACrB,WAAWA,6BAAa,CAAA;AAAA,OAC1B;AAAA,MACA,MAAM,IAAK,CAAA,EAAE,QAAQ,OAAS,EAAA,MAAA,EAAQ,WAAa,EAAA;AACjD,QAAQ,OAAA,CAAA,iBAAA;AAAA,UACNE,oCAAkB,UAAW,CAAA;AAAA,YAC3B,MAAA;AAAA,YACA,SAAA;AAAA,YACA;AAAA,WACD;AAAA,SACH;AAAA;AACF,KACD,CAAA;AAAA;AAEL,CAAC;;;;"}
1
+ {"version":3,"file":"catalogModuleGcpGkeEntityProvider.cjs.js","sources":["../../src/module/catalogModuleGcpGkeEntityProvider.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 { catalogProcessingExtensionPoint } from '@backstage/plugin-catalog-node/alpha';\nimport { GkeEntityProvider } from '../providers/GkeEntityProvider';\n\n/**\n * Registers the GcpGkeEntityProvider with the catalog processing extension point.\n *\n * @public\n */\nexport const catalogModuleGcpGkeEntityProvider = createBackendModule({\n pluginId: 'catalog',\n moduleId: 'gcp-gke-entity-provider',\n register(env) {\n env.registerInit({\n deps: {\n config: coreServices.rootConfig,\n catalog: catalogProcessingExtensionPoint,\n logger: coreServices.logger,\n scheduler: coreServices.scheduler,\n },\n async init({ config, catalog, logger, scheduler }) {\n catalog.addEntityProvider(\n GkeEntityProvider.fromConfig({\n logger,\n scheduler,\n config,\n }),\n );\n },\n });\n },\n});\n"],"names":["createBackendModule","coreServices","catalogProcessingExtensionPoint","GkeEntityProvider"],"mappings":";;;;;;AA4BO,MAAM,oCAAoCA,oCAAA,CAAoB;AAAA,EACnE,QAAA,EAAU,SAAA;AAAA,EACV,QAAA,EAAU,yBAAA;AAAA,EACV,SAAS,GAAA,EAAK;AACZ,IAAA,GAAA,CAAI,YAAA,CAAa;AAAA,MACf,IAAA,EAAM;AAAA,QACJ,QAAQC,6BAAA,CAAa,UAAA;AAAA,QACrB,OAAA,EAASC,qCAAA;AAAA,QACT,QAAQD,6BAAA,CAAa,MAAA;AAAA,QACrB,WAAWA,6BAAA,CAAa;AAAA,OAC1B;AAAA,MACA,MAAM,IAAA,CAAK,EAAE,QAAQ,OAAA,EAAS,MAAA,EAAQ,WAAU,EAAG;AACjD,QAAA,OAAA,CAAQ,iBAAA;AAAA,UACNE,oCAAkB,UAAA,CAAW;AAAA,YAC3B,MAAA;AAAA,YACA,SAAA;AAAA,YACA;AAAA,WACD;AAAA,SACH;AAAA,MACF;AAAA,KACD,CAAA;AAAA,EACH;AACF,CAAC;;;;"}
@@ -29,12 +29,16 @@ class GkeEntityProvider {
29
29
  logger;
30
30
  scheduleFn;
31
31
  gkeParents;
32
+ gkeAuthProvider;
33
+ gkeOwner;
32
34
  clusterManagerClient;
33
35
  connection;
34
- constructor(logger, taskRunner, gkeParents, clusterManagerClient) {
36
+ constructor(logger, taskRunner, gkeParents, gkeAuthProvider, gkeOwner, clusterManagerClient) {
35
37
  this.logger = logger;
36
38
  this.scheduleFn = this.createScheduleFn(taskRunner);
37
39
  this.gkeParents = gkeParents;
40
+ this.gkeAuthProvider = gkeAuthProvider;
41
+ this.gkeOwner = gkeOwner;
38
42
  this.clusterManagerClient = clusterManagerClient;
39
43
  }
40
44
  static fromConfig({
@@ -42,11 +46,30 @@ class GkeEntityProvider {
42
46
  scheduler,
43
47
  config
44
48
  }) {
49
+ const gkeProviderConfig = config.getConfig("catalog.providers.gcp.gke");
50
+ const credentials = gkeProviderConfig.getOptionalString(
51
+ "googleServiceAccountCredentials"
52
+ );
53
+ let clusterManagerClient;
54
+ if (credentials && credentials.trim()) {
55
+ try {
56
+ const credentialsObject = JSON.parse(credentials);
57
+ clusterManagerClient = new container__namespace.v1.ClusterManagerClient({
58
+ credentials: credentialsObject
59
+ });
60
+ } catch (error) {
61
+ throw new Error(
62
+ `Failed to parse Google Service Account credentials from config: ${error instanceof Error ? error.message : "Invalid JSON"}`
63
+ );
64
+ }
65
+ } else {
66
+ clusterManagerClient = new container__namespace.v1.ClusterManagerClient();
67
+ }
45
68
  return GkeEntityProvider.fromConfigWithClient({
46
69
  logger,
47
70
  scheduler,
48
71
  config,
49
- clusterManagerClient: new container__namespace.v1.ClusterManagerClient()
72
+ clusterManagerClient
50
73
  });
51
74
  }
52
75
  static fromConfigWithClient({
@@ -63,6 +86,8 @@ class GkeEntityProvider {
63
86
  logger,
64
87
  scheduler.createScheduledTaskRunner(schedule),
65
88
  gkeProviderConfig.getStringArray("parents"),
89
+ gkeProviderConfig.getOptionalString("authProvider"),
90
+ gkeProviderConfig.getOptionalString("owner"),
66
91
  clusterManagerClient
67
92
  );
68
93
  }
@@ -96,7 +121,7 @@ class GkeEntityProvider {
96
121
  annotations: {
97
122
  [pluginKubernetesCommon.ANNOTATION_KUBERNETES_API_SERVER]: `https://${cluster.endpoint}`,
98
123
  [pluginKubernetesCommon.ANNOTATION_KUBERNETES_API_SERVER_CA]: cluster.masterAuth?.clusterCaCertificate || "",
99
- [pluginKubernetesCommon.ANNOTATION_KUBERNETES_AUTH_PROVIDER]: "google",
124
+ [pluginKubernetesCommon.ANNOTATION_KUBERNETES_AUTH_PROVIDER]: this.gkeAuthProvider || "google",
100
125
  [pluginKubernetesCommon.ANNOTATION_KUBERNETES_DASHBOARD_APP]: "gke",
101
126
  [catalogModel.ANNOTATION_LOCATION]: location,
102
127
  [catalogModel.ANNOTATION_ORIGIN_LOCATION]: location,
@@ -111,7 +136,7 @@ class GkeEntityProvider {
111
136
  },
112
137
  spec: {
113
138
  type: "kubernetes-cluster",
114
- owner: "unknown"
139
+ owner: this.gkeOwner || "unknown"
115
140
  }
116
141
  }
117
142
  };
@@ -1 +1 @@
1
- {"version":3,"file":"GkeEntityProvider.cjs.js","sources":["../../src/providers/GkeEntityProvider.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 DeferredEntity,\n EntityProvider,\n EntityProviderConnection,\n} from '@backstage/plugin-catalog-node';\nimport * as container from '@google-cloud/container';\nimport {\n ANNOTATION_KUBERNETES_API_SERVER,\n ANNOTATION_KUBERNETES_API_SERVER_CA,\n ANNOTATION_KUBERNETES_AUTH_PROVIDER,\n ANNOTATION_KUBERNETES_DASHBOARD_APP,\n ANNOTATION_KUBERNETES_DASHBOARD_PARAMETERS,\n} from '@backstage/plugin-kubernetes-common';\nimport { Config } from '@backstage/config';\nimport {\n LoggerService,\n SchedulerService,\n SchedulerServiceTaskRunner,\n readSchedulerServiceTaskScheduleDefinitionFromConfig,\n} from '@backstage/backend-plugin-api';\nimport {\n ANNOTATION_LOCATION,\n ANNOTATION_ORIGIN_LOCATION,\n} from '@backstage/catalog-model';\n\n/**\n * Catalog provider to ingest GKE clusters\n *\n * @public\n */\nexport class GkeEntityProvider implements EntityProvider {\n private readonly logger: LoggerService;\n private readonly scheduleFn: () => Promise<void>;\n private readonly gkeParents: string[];\n private readonly clusterManagerClient: container.v1.ClusterManagerClient;\n private connection?: EntityProviderConnection;\n\n private constructor(\n logger: LoggerService,\n taskRunner: SchedulerServiceTaskRunner,\n gkeParents: string[],\n clusterManagerClient: container.v1.ClusterManagerClient,\n ) {\n this.logger = logger;\n this.scheduleFn = this.createScheduleFn(taskRunner);\n this.gkeParents = gkeParents;\n this.clusterManagerClient = clusterManagerClient;\n }\n\n public static fromConfig({\n logger,\n scheduler,\n config,\n }: {\n logger: LoggerService;\n scheduler: SchedulerService;\n config: Config;\n }) {\n return GkeEntityProvider.fromConfigWithClient({\n logger,\n scheduler: scheduler,\n config,\n clusterManagerClient: new container.v1.ClusterManagerClient(),\n });\n }\n\n public static fromConfigWithClient({\n logger,\n scheduler,\n config,\n clusterManagerClient,\n }: {\n logger: LoggerService;\n scheduler: SchedulerService;\n config: Config;\n clusterManagerClient: container.v1.ClusterManagerClient;\n }) {\n const gkeProviderConfig = config.getConfig('catalog.providers.gcp.gke');\n const schedule = readSchedulerServiceTaskScheduleDefinitionFromConfig(\n gkeProviderConfig.getConfig('schedule'),\n );\n return new GkeEntityProvider(\n logger,\n scheduler.createScheduledTaskRunner(schedule),\n gkeProviderConfig.getStringArray('parents'),\n clusterManagerClient,\n );\n }\n\n getProviderName(): string {\n return `gcp-gke`;\n }\n\n async connect(connection: EntityProviderConnection): Promise<void> {\n this.connection = connection;\n await this.scheduleFn();\n }\n\n private filterOutUndefinedDeferredEntity(\n e: DeferredEntity | undefined,\n ): e is DeferredEntity {\n return e !== undefined;\n }\n\n private filterOutUndefinedCluster(\n c: container.protos.google.container.v1.ICluster | null | undefined,\n ): c is container.protos.google.container.v1.ICluster {\n return c !== undefined && c !== null;\n }\n\n private clusterToResource(\n cluster: container.protos.google.container.v1.ICluster,\n project: string,\n ): DeferredEntity | undefined {\n const location = `${this.getProviderName()}:${cluster.location}`;\n\n if (\n !cluster.name ||\n !cluster.selfLink ||\n !cluster.endpoint ||\n !cluster.location\n ) {\n this.logger.warn(\n `ignoring partial cluster, one of name=${cluster.name}, endpoint=${cluster.endpoint}, selfLink=${cluster.selfLink} or location=${cluster.location} is missing`,\n );\n return undefined;\n }\n\n // TODO fix location type\n return {\n locationKey: location,\n entity: {\n apiVersion: 'backstage.io/v1alpha1',\n kind: 'Resource',\n metadata: {\n annotations: {\n [ANNOTATION_KUBERNETES_API_SERVER]: `https://${cluster.endpoint}`,\n [ANNOTATION_KUBERNETES_API_SERVER_CA]:\n cluster.masterAuth?.clusterCaCertificate || '',\n [ANNOTATION_KUBERNETES_AUTH_PROVIDER]: 'google',\n [ANNOTATION_KUBERNETES_DASHBOARD_APP]: 'gke',\n [ANNOTATION_LOCATION]: location,\n [ANNOTATION_ORIGIN_LOCATION]: location,\n [ANNOTATION_KUBERNETES_DASHBOARD_PARAMETERS]: JSON.stringify({\n projectId: project,\n region: cluster.location,\n clusterName: cluster.name,\n }),\n },\n name: cluster.name,\n namespace: 'default',\n },\n spec: {\n type: 'kubernetes-cluster',\n owner: 'unknown',\n },\n },\n };\n }\n\n private createScheduleFn(\n taskRunner: SchedulerServiceTaskRunner,\n ): () => Promise<void> {\n return async () => {\n const taskId = `${this.getProviderName()}:refresh`;\n return taskRunner.run({\n id: taskId,\n fn: async () => {\n try {\n await this.refresh();\n } catch (error) {\n this.logger.error(error);\n }\n },\n });\n };\n }\n\n private async getClusters(): Promise<DeferredEntity[]> {\n const clusters = await Promise.all(\n this.gkeParents.map(async parent => {\n const project = parent.split('/')[1];\n const request = {\n parent: parent,\n };\n const [response] = await this.clusterManagerClient.listClusters(\n request,\n );\n return (\n response.clusters\n ?.filter(this.filterOutUndefinedCluster)\n .map(c => this.clusterToResource(c, project))\n .filter(this.filterOutUndefinedDeferredEntity) ?? []\n );\n }),\n );\n return clusters.flat();\n }\n\n async refresh() {\n if (!this.connection) {\n throw new Error('Not initialized');\n }\n\n this.logger.info('Discovering GKE clusters');\n\n let resources: DeferredEntity[];\n\n try {\n resources = await this.getClusters();\n } catch (e) {\n this.logger.error('error fetching GKE clusters', e);\n return;\n }\n\n this.logger.info(\n `Ingesting GKE clusters [${resources\n .map(r => r.entity.metadata.name)\n .join(', ')}]`,\n );\n\n await this.connection.applyMutation({\n type: 'full',\n entities: resources,\n });\n }\n}\n"],"names":["container","readSchedulerServiceTaskScheduleDefinitionFromConfig","ANNOTATION_KUBERNETES_API_SERVER","ANNOTATION_KUBERNETES_API_SERVER_CA","ANNOTATION_KUBERNETES_AUTH_PROVIDER","ANNOTATION_KUBERNETES_DASHBOARD_APP","ANNOTATION_LOCATION","ANNOTATION_ORIGIN_LOCATION","ANNOTATION_KUBERNETES_DASHBOARD_PARAMETERS"],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;;AA6CO,MAAM,iBAA4C,CAAA;AAAA,EACtC,MAAA;AAAA,EACA,UAAA;AAAA,EACA,UAAA;AAAA,EACA,oBAAA;AAAA,EACT,UAAA;AAAA,EAEA,WACN,CAAA,MAAA,EACA,UACA,EAAA,UAAA,EACA,oBACA,EAAA;AACA,IAAA,IAAA,CAAK,MAAS,GAAA,MAAA;AACd,IAAK,IAAA,CAAA,UAAA,GAAa,IAAK,CAAA,gBAAA,CAAiB,UAAU,CAAA;AAClD,IAAA,IAAA,CAAK,UAAa,GAAA,UAAA;AAClB,IAAA,IAAA,CAAK,oBAAuB,GAAA,oBAAA;AAAA;AAC9B,EAEA,OAAc,UAAW,CAAA;AAAA,IACvB,MAAA;AAAA,IACA,SAAA;AAAA,IACA;AAAA,GAKC,EAAA;AACD,IAAA,OAAO,kBAAkB,oBAAqB,CAAA;AAAA,MAC5C,MAAA;AAAA,MACA,SAAA;AAAA,MACA,MAAA;AAAA,MACA,oBAAsB,EAAA,IAAIA,oBAAU,CAAA,EAAA,CAAG,oBAAqB;AAAA,KAC7D,CAAA;AAAA;AACH,EAEA,OAAc,oBAAqB,CAAA;AAAA,IACjC,MAAA;AAAA,IACA,SAAA;AAAA,IACA,MAAA;AAAA,IACA;AAAA,GAMC,EAAA;AACD,IAAM,MAAA,iBAAA,GAAoB,MAAO,CAAA,SAAA,CAAU,2BAA2B,CAAA;AACtE,IAAA,MAAM,QAAW,GAAAC,qEAAA;AAAA,MACf,iBAAA,CAAkB,UAAU,UAAU;AAAA,KACxC;AACA,IAAA,OAAO,IAAI,iBAAA;AAAA,MACT,MAAA;AAAA,MACA,SAAA,CAAU,0BAA0B,QAAQ,CAAA;AAAA,MAC5C,iBAAA,CAAkB,eAAe,SAAS,CAAA;AAAA,MAC1C;AAAA,KACF;AAAA;AACF,EAEA,eAA0B,GAAA;AACxB,IAAO,OAAA,CAAA,OAAA,CAAA;AAAA;AACT,EAEA,MAAM,QAAQ,UAAqD,EAAA;AACjE,IAAA,IAAA,CAAK,UAAa,GAAA,UAAA;AAClB,IAAA,MAAM,KAAK,UAAW,EAAA;AAAA;AACxB,EAEQ,iCACN,CACqB,EAAA;AACrB,IAAA,OAAO,CAAM,KAAA,KAAA,CAAA;AAAA;AACf,EAEQ,0BACN,CACoD,EAAA;AACpD,IAAO,OAAA,CAAA,KAAM,UAAa,CAAM,KAAA,IAAA;AAAA;AAClC,EAEQ,iBAAA,CACN,SACA,OAC4B,EAAA;AAC5B,IAAA,MAAM,WAAW,CAAG,EAAA,IAAA,CAAK,iBAAiB,CAAA,CAAA,EAAI,QAAQ,QAAQ,CAAA,CAAA;AAE9D,IACE,IAAA,CAAC,OAAQ,CAAA,IAAA,IACT,CAAC,OAAA,CAAQ,QACT,IAAA,CAAC,OAAQ,CAAA,QAAA,IACT,CAAC,OAAA,CAAQ,QACT,EAAA;AACA,MAAA,IAAA,CAAK,MAAO,CAAA,IAAA;AAAA,QACV,CAAA,sCAAA,EAAyC,OAAQ,CAAA,IAAI,CAAc,WAAA,EAAA,OAAA,CAAQ,QAAQ,CAAA,WAAA,EAAc,OAAQ,CAAA,QAAQ,CAAgB,aAAA,EAAA,OAAA,CAAQ,QAAQ,CAAA,WAAA;AAAA,OACnJ;AACA,MAAO,OAAA,KAAA,CAAA;AAAA;AAIT,IAAO,OAAA;AAAA,MACL,WAAa,EAAA,QAAA;AAAA,MACb,MAAQ,EAAA;AAAA,QACN,UAAY,EAAA,uBAAA;AAAA,QACZ,IAAM,EAAA,UAAA;AAAA,QACN,QAAU,EAAA;AAAA,UACR,WAAa,EAAA;AAAA,YACX,CAACC,uDAAgC,GAAG,CAAA,QAAA,EAAW,QAAQ,QAAQ,CAAA,CAAA;AAAA,YAC/D,CAACC,0DAAmC,GAClC,OAAA,CAAQ,YAAY,oBAAwB,IAAA,EAAA;AAAA,YAC9C,CAACC,0DAAmC,GAAG,QAAA;AAAA,YACvC,CAACC,0DAAmC,GAAG,KAAA;AAAA,YACvC,CAACC,gCAAmB,GAAG,QAAA;AAAA,YACvB,CAACC,uCAA0B,GAAG,QAAA;AAAA,YAC9B,CAACC,iEAA0C,GAAG,IAAA,CAAK,SAAU,CAAA;AAAA,cAC3D,SAAW,EAAA,OAAA;AAAA,cACX,QAAQ,OAAQ,CAAA,QAAA;AAAA,cAChB,aAAa,OAAQ,CAAA;AAAA,aACtB;AAAA,WACH;AAAA,UACA,MAAM,OAAQ,CAAA,IAAA;AAAA,UACd,SAAW,EAAA;AAAA,SACb;AAAA,QACA,IAAM,EAAA;AAAA,UACJ,IAAM,EAAA,oBAAA;AAAA,UACN,KAAO,EAAA;AAAA;AACT;AACF,KACF;AAAA;AACF,EAEQ,iBACN,UACqB,EAAA;AACrB,IAAA,OAAO,YAAY;AACjB,MAAA,MAAM,MAAS,GAAA,CAAA,EAAG,IAAK,CAAA,eAAA,EAAiB,CAAA,QAAA,CAAA;AACxC,MAAA,OAAO,WAAW,GAAI,CAAA;AAAA,QACpB,EAAI,EAAA,MAAA;AAAA,QACJ,IAAI,YAAY;AACd,UAAI,IAAA;AACF,YAAA,MAAM,KAAK,OAAQ,EAAA;AAAA,mBACZ,KAAO,EAAA;AACd,YAAK,IAAA,CAAA,MAAA,CAAO,MAAM,KAAK,CAAA;AAAA;AACzB;AACF,OACD,CAAA;AAAA,KACH;AAAA;AACF,EAEA,MAAc,WAAyC,GAAA;AACrD,IAAM,MAAA,QAAA,GAAW,MAAM,OAAQ,CAAA,GAAA;AAAA,MAC7B,IAAK,CAAA,UAAA,CAAW,GAAI,CAAA,OAAM,MAAU,KAAA;AAClC,QAAA,MAAM,OAAU,GAAA,MAAA,CAAO,KAAM,CAAA,GAAG,EAAE,CAAC,CAAA;AACnC,QAAA,MAAM,OAAU,GAAA;AAAA,UACd;AAAA,SACF;AACA,QAAA,MAAM,CAAC,QAAQ,CAAI,GAAA,MAAM,KAAK,oBAAqB,CAAA,YAAA;AAAA,UACjD;AAAA,SACF;AACA,QAAA,OACE,SAAS,QACL,EAAA,MAAA,CAAO,KAAK,yBAAyB,CAAA,CACtC,IAAI,CAAK,CAAA,KAAA,IAAA,CAAK,iBAAkB,CAAA,CAAA,EAAG,OAAO,CAAC,CAAA,CAC3C,OAAO,IAAK,CAAA,gCAAgC,KAAK,EAAC;AAAA,OAExD;AAAA,KACH;AACA,IAAA,OAAO,SAAS,IAAK,EAAA;AAAA;AACvB,EAEA,MAAM,OAAU,GAAA;AACd,IAAI,IAAA,CAAC,KAAK,UAAY,EAAA;AACpB,MAAM,MAAA,IAAI,MAAM,iBAAiB,CAAA;AAAA;AAGnC,IAAK,IAAA,CAAA,MAAA,CAAO,KAAK,0BAA0B,CAAA;AAE3C,IAAI,IAAA,SAAA;AAEJ,IAAI,IAAA;AACF,MAAY,SAAA,GAAA,MAAM,KAAK,WAAY,EAAA;AAAA,aAC5B,CAAG,EAAA;AACV,MAAK,IAAA,CAAA,MAAA,CAAO,KAAM,CAAA,6BAAA,EAA+B,CAAC,CAAA;AAClD,MAAA;AAAA;AAGF,IAAA,IAAA,CAAK,MAAO,CAAA,IAAA;AAAA,MACV,CAAA,wBAAA,EAA2B,SACxB,CAAA,GAAA,CAAI,CAAK,CAAA,KAAA,CAAA,CAAE,MAAO,CAAA,QAAA,CAAS,IAAI,CAAA,CAC/B,IAAK,CAAA,IAAI,CAAC,CAAA,CAAA;AAAA,KACf;AAEA,IAAM,MAAA,IAAA,CAAK,WAAW,aAAc,CAAA;AAAA,MAClC,IAAM,EAAA,MAAA;AAAA,MACN,QAAU,EAAA;AAAA,KACX,CAAA;AAAA;AAEL;;;;"}
1
+ {"version":3,"file":"GkeEntityProvider.cjs.js","sources":["../../src/providers/GkeEntityProvider.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 DeferredEntity,\n EntityProvider,\n EntityProviderConnection,\n} from '@backstage/plugin-catalog-node';\nimport * as container from '@google-cloud/container';\nimport {\n ANNOTATION_KUBERNETES_API_SERVER,\n ANNOTATION_KUBERNETES_API_SERVER_CA,\n ANNOTATION_KUBERNETES_AUTH_PROVIDER,\n ANNOTATION_KUBERNETES_DASHBOARD_APP,\n ANNOTATION_KUBERNETES_DASHBOARD_PARAMETERS,\n} from '@backstage/plugin-kubernetes-common';\nimport { Config } from '@backstage/config';\nimport {\n LoggerService,\n SchedulerService,\n SchedulerServiceTaskRunner,\n readSchedulerServiceTaskScheduleDefinitionFromConfig,\n} from '@backstage/backend-plugin-api';\nimport {\n ANNOTATION_LOCATION,\n ANNOTATION_ORIGIN_LOCATION,\n} from '@backstage/catalog-model';\n\n/**\n * Catalog provider to ingest GKE clusters\n *\n * @public\n */\nexport class GkeEntityProvider implements EntityProvider {\n private readonly logger: LoggerService;\n private readonly scheduleFn: () => Promise<void>;\n private readonly gkeParents: string[];\n private readonly gkeAuthProvider: string | undefined;\n private readonly gkeOwner: string | undefined;\n private readonly clusterManagerClient: container.v1.ClusterManagerClient;\n private connection?: EntityProviderConnection;\n\n private constructor(\n logger: LoggerService,\n taskRunner: SchedulerServiceTaskRunner,\n gkeParents: string[],\n gkeAuthProvider: string | undefined,\n gkeOwner: string | undefined,\n clusterManagerClient: container.v1.ClusterManagerClient,\n ) {\n this.logger = logger;\n this.scheduleFn = this.createScheduleFn(taskRunner);\n this.gkeParents = gkeParents;\n this.gkeAuthProvider = gkeAuthProvider;\n this.gkeOwner = gkeOwner;\n this.clusterManagerClient = clusterManagerClient;\n }\n\n public static fromConfig({\n logger,\n scheduler,\n config,\n }: {\n logger: LoggerService;\n scheduler: SchedulerService;\n config: Config;\n }) {\n const gkeProviderConfig = config.getConfig('catalog.providers.gcp.gke');\n const credentials = gkeProviderConfig.getOptionalString(\n 'googleServiceAccountCredentials',\n );\n\n let clusterManagerClient: container.v1.ClusterManagerClient;\n\n if (credentials && credentials.trim()) {\n // Use credentials from config\n try {\n const credentialsObject = JSON.parse(credentials);\n clusterManagerClient = new container.v1.ClusterManagerClient({\n credentials: credentialsObject,\n });\n } catch (error) {\n throw new Error(\n `Failed to parse Google Service Account credentials from config: ${\n error instanceof Error ? error.message : 'Invalid JSON'\n }`,\n );\n }\n } else {\n // Fall back to Application Default Credentials or GOOGLE_APPLICATION_CREDENTIALS\n clusterManagerClient = new container.v1.ClusterManagerClient();\n }\n\n return GkeEntityProvider.fromConfigWithClient({\n logger,\n scheduler: scheduler,\n config,\n clusterManagerClient,\n });\n }\n\n public static fromConfigWithClient({\n logger,\n scheduler,\n config,\n clusterManagerClient,\n }: {\n logger: LoggerService;\n scheduler: SchedulerService;\n config: Config;\n clusterManagerClient: container.v1.ClusterManagerClient;\n }) {\n const gkeProviderConfig = config.getConfig('catalog.providers.gcp.gke');\n const schedule = readSchedulerServiceTaskScheduleDefinitionFromConfig(\n gkeProviderConfig.getConfig('schedule'),\n );\n return new GkeEntityProvider(\n logger,\n scheduler.createScheduledTaskRunner(schedule),\n gkeProviderConfig.getStringArray('parents'),\n gkeProviderConfig.getOptionalString('authProvider'),\n gkeProviderConfig.getOptionalString('owner'),\n clusterManagerClient,\n );\n }\n\n getProviderName(): string {\n return `gcp-gke`;\n }\n\n async connect(connection: EntityProviderConnection): Promise<void> {\n this.connection = connection;\n await this.scheduleFn();\n }\n\n private filterOutUndefinedDeferredEntity(\n e: DeferredEntity | undefined,\n ): e is DeferredEntity {\n return e !== undefined;\n }\n\n private filterOutUndefinedCluster(\n c: container.protos.google.container.v1.ICluster | null | undefined,\n ): c is container.protos.google.container.v1.ICluster {\n return c !== undefined && c !== null;\n }\n\n private clusterToResource(\n cluster: container.protos.google.container.v1.ICluster,\n project: string,\n ): DeferredEntity | undefined {\n const location = `${this.getProviderName()}:${cluster.location}`;\n\n if (\n !cluster.name ||\n !cluster.selfLink ||\n !cluster.endpoint ||\n !cluster.location\n ) {\n this.logger.warn(\n `ignoring partial cluster, one of name=${cluster.name}, endpoint=${cluster.endpoint}, selfLink=${cluster.selfLink} or location=${cluster.location} is missing`,\n );\n return undefined;\n }\n\n // TODO fix location type\n return {\n locationKey: location,\n entity: {\n apiVersion: 'backstage.io/v1alpha1',\n kind: 'Resource',\n metadata: {\n annotations: {\n [ANNOTATION_KUBERNETES_API_SERVER]: `https://${cluster.endpoint}`,\n [ANNOTATION_KUBERNETES_API_SERVER_CA]:\n cluster.masterAuth?.clusterCaCertificate || '',\n [ANNOTATION_KUBERNETES_AUTH_PROVIDER]:\n this.gkeAuthProvider || 'google',\n [ANNOTATION_KUBERNETES_DASHBOARD_APP]: 'gke',\n [ANNOTATION_LOCATION]: location,\n [ANNOTATION_ORIGIN_LOCATION]: location,\n [ANNOTATION_KUBERNETES_DASHBOARD_PARAMETERS]: JSON.stringify({\n projectId: project,\n region: cluster.location,\n clusterName: cluster.name,\n }),\n },\n name: cluster.name,\n namespace: 'default',\n },\n spec: {\n type: 'kubernetes-cluster',\n owner: this.gkeOwner || 'unknown',\n },\n },\n };\n }\n\n private createScheduleFn(\n taskRunner: SchedulerServiceTaskRunner,\n ): () => Promise<void> {\n return async () => {\n const taskId = `${this.getProviderName()}:refresh`;\n return taskRunner.run({\n id: taskId,\n fn: async () => {\n try {\n await this.refresh();\n } catch (error) {\n this.logger.error(error);\n }\n },\n });\n };\n }\n\n private async getClusters(): Promise<DeferredEntity[]> {\n const clusters = await Promise.all(\n this.gkeParents.map(async parent => {\n const project = parent.split('/')[1];\n const request = {\n parent: parent,\n };\n const [response] = await this.clusterManagerClient.listClusters(\n request,\n );\n return (\n response.clusters\n ?.filter(this.filterOutUndefinedCluster)\n .map(c => this.clusterToResource(c, project))\n .filter(this.filterOutUndefinedDeferredEntity) ?? []\n );\n }),\n );\n return clusters.flat();\n }\n\n async refresh() {\n if (!this.connection) {\n throw new Error('Not initialized');\n }\n\n this.logger.info('Discovering GKE clusters');\n\n let resources: DeferredEntity[];\n\n try {\n resources = await this.getClusters();\n } catch (e) {\n this.logger.error('error fetching GKE clusters', e);\n return;\n }\n\n this.logger.info(\n `Ingesting GKE clusters [${resources\n .map(r => r.entity.metadata.name)\n .join(', ')}]`,\n );\n\n await this.connection.applyMutation({\n type: 'full',\n entities: resources,\n });\n }\n}\n"],"names":["container","readSchedulerServiceTaskScheduleDefinitionFromConfig","ANNOTATION_KUBERNETES_API_SERVER","ANNOTATION_KUBERNETES_API_SERVER_CA","ANNOTATION_KUBERNETES_AUTH_PROVIDER","ANNOTATION_KUBERNETES_DASHBOARD_APP","ANNOTATION_LOCATION","ANNOTATION_ORIGIN_LOCATION","ANNOTATION_KUBERNETES_DASHBOARD_PARAMETERS"],"mappings":"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"}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@backstage/plugin-catalog-backend-module-gcp",
3
- "version": "0.3.11-next.0",
3
+ "version": "0.3.12-next.0",
4
4
  "description": "A Backstage catalog backend module that helps integrate towards GCP",
5
5
  "backstage": {
6
6
  "role": "backend-plugin-module",
@@ -63,16 +63,16 @@
63
63
  "test": "backstage-cli package test"
64
64
  },
65
65
  "dependencies": {
66
- "@backstage/backend-plugin-api": "1.4.2-next.0",
66
+ "@backstage/backend-plugin-api": "1.4.3-next.0",
67
67
  "@backstage/catalog-model": "1.7.5",
68
68
  "@backstage/config": "1.3.3",
69
- "@backstage/plugin-catalog-node": "1.18.0-next.0",
69
+ "@backstage/plugin-catalog-node": "1.18.1-next.0",
70
70
  "@backstage/plugin-kubernetes-common": "0.9.6",
71
71
  "@google-cloud/container": "^5.0.0"
72
72
  },
73
73
  "devDependencies": {
74
- "@backstage/backend-test-utils": "1.7.1-next.0",
75
- "@backstage/cli": "0.33.2-next.0"
74
+ "@backstage/backend-test-utils": "1.9.0-next.1",
75
+ "@backstage/cli": "0.34.2-next.1"
76
76
  },
77
77
  "configSchema": "config.d.ts"
78
78
  }