@backstage/plugin-catalog-backend-module-gcp 0.1.19-next.2 → 0.1.20-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,30 @@
1
1
  # @backstage/plugin-catalog-backend-module-gcp
2
2
 
3
+ ## 0.1.20-next.0
4
+
5
+ ### Patch Changes
6
+
7
+ - Updated dependencies
8
+ - @backstage/backend-tasks@0.5.24-next.0
9
+ - @backstage/backend-common@0.22.1-next.0
10
+ - @backstage/backend-plugin-api@0.6.19-next.0
11
+ - @backstage/plugin-catalog-node@1.12.1-next.0
12
+ - @backstage/catalog-model@1.5.0
13
+ - @backstage/config@1.2.0
14
+ - @backstage/plugin-kubernetes-common@0.7.6
15
+
16
+ ## 0.1.19
17
+
18
+ ### Patch Changes
19
+
20
+ - Updated dependencies
21
+ - @backstage/plugin-catalog-node@1.12.0
22
+ - @backstage/catalog-model@1.5.0
23
+ - @backstage/backend-common@0.22.0
24
+ - @backstage/backend-plugin-api@0.6.18
25
+ - @backstage/backend-tasks@0.5.23
26
+ - @backstage/plugin-kubernetes-common@0.7.6
27
+
3
28
  ## 0.1.19-next.2
4
29
 
5
30
  ### Patch Changes
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@backstage/plugin-catalog-backend-module-gcp",
3
- "version": "0.1.19-next.2",
3
+ "version": "0.1.20-next.0",
4
4
  "main": "../dist/alpha.cjs.js",
5
5
  "types": "../dist/alpha.d.ts"
6
6
  }
package/dist/index.cjs.js CHANGED
@@ -29,19 +29,13 @@ function _interopNamespaceCompat(e) {
29
29
 
30
30
  var container__namespace = /*#__PURE__*/_interopNamespaceCompat(container);
31
31
 
32
- var __defProp = Object.defineProperty;
33
- var __defNormalProp = (obj, key, value) => key in obj ? __defProp(obj, key, { enumerable: true, configurable: true, writable: true, value }) : obj[key] = value;
34
- var __publicField = (obj, key, value) => {
35
- __defNormalProp(obj, typeof key !== "symbol" ? key + "" : key, value);
36
- return value;
37
- };
38
32
  class GkeEntityProvider {
33
+ logger;
34
+ scheduleFn;
35
+ gkeParents;
36
+ clusterManagerClient;
37
+ connection;
39
38
  constructor(logger, taskRunner, gkeParents, clusterManagerClient) {
40
- __publicField(this, "logger");
41
- __publicField(this, "scheduleFn");
42
- __publicField(this, "gkeParents");
43
- __publicField(this, "clusterManagerClient");
44
- __publicField(this, "connection");
45
39
  this.logger = logger;
46
40
  this.scheduleFn = this.createScheduleFn(taskRunner);
47
41
  this.gkeParents = gkeParents;
@@ -90,7 +84,6 @@ class GkeEntityProvider {
90
84
  return c !== void 0 && c !== null;
91
85
  }
92
86
  clusterToResource(cluster, project) {
93
- var _a;
94
87
  const location = `${this.getProviderName()}:${cluster.location}`;
95
88
  if (!cluster.name || !cluster.selfLink || !cluster.endpoint || !cluster.location) {
96
89
  this.logger.warn(
@@ -106,7 +99,7 @@ class GkeEntityProvider {
106
99
  metadata: {
107
100
  annotations: {
108
101
  [pluginKubernetesCommon.ANNOTATION_KUBERNETES_API_SERVER]: `https://${cluster.endpoint}`,
109
- [pluginKubernetesCommon.ANNOTATION_KUBERNETES_API_SERVER_CA]: ((_a = cluster.masterAuth) == null ? void 0 : _a.clusterCaCertificate) || "",
102
+ [pluginKubernetesCommon.ANNOTATION_KUBERNETES_API_SERVER_CA]: cluster.masterAuth?.clusterCaCertificate || "",
110
103
  [pluginKubernetesCommon.ANNOTATION_KUBERNETES_AUTH_PROVIDER]: "google",
111
104
  [pluginKubernetesCommon.ANNOTATION_KUBERNETES_DASHBOARD_APP]: "gke",
112
105
  [catalogModel.ANNOTATION_LOCATION]: location,
@@ -145,7 +138,6 @@ class GkeEntityProvider {
145
138
  async getClusters() {
146
139
  const clusters = await Promise.all(
147
140
  this.gkeParents.map(async (parent) => {
148
- var _a, _b;
149
141
  const project = parent.split("/")[1];
150
142
  const request = {
151
143
  parent
@@ -153,7 +145,7 @@ class GkeEntityProvider {
153
145
  const [response] = await this.clusterManagerClient.listClusters(
154
146
  request
155
147
  );
156
- return (_b = (_a = response.clusters) == null ? void 0 : _a.filter(this.filterOutUndefinedCluster).map((c) => this.clusterToResource(c, project)).filter(this.filterOutUndefinedDeferredEntity)) != null ? _b : [];
148
+ return response.clusters?.filter(this.filterOutUndefinedCluster).map((c) => this.clusterToResource(c, project)).filter(this.filterOutUndefinedDeferredEntity) ?? [];
157
149
  })
158
150
  );
159
151
  return clusters.flat();
@@ -1 +1 @@
1
- {"version":3,"file":"index.cjs.js","sources":["../src/providers/GkeEntityProvider.ts","../src/module/catalogModuleGcpGkeEntityProvider.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 readTaskScheduleDefinitionFromConfig,\n TaskRunner,\n} from '@backstage/backend-tasks';\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 { LoggerService, SchedulerService } 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: TaskRunner,\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 = readTaskScheduleDefinitionFromConfig(\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(taskRunner: TaskRunner): () => 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","/*\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":["container","readTaskScheduleDefinitionFromConfig","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","createBackendModule","coreServices","catalogProcessingExtensionPoint"],"mappings":"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"}
1
+ {"version":3,"file":"index.cjs.js","sources":["../src/providers/GkeEntityProvider.ts","../src/module/catalogModuleGcpGkeEntityProvider.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 readTaskScheduleDefinitionFromConfig,\n TaskRunner,\n} from '@backstage/backend-tasks';\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 { LoggerService, SchedulerService } 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: TaskRunner,\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 = readTaskScheduleDefinitionFromConfig(\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(taskRunner: TaskRunner): () => 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","/*\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":["container","readTaskScheduleDefinitionFromConfig","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","createBackendModule","coreServices","catalogProcessingExtensionPoint"],"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.1.19-next.2",
3
+ "version": "0.1.20-next.0",
4
4
  "description": "A Backstage catalog backend module that helps integrate towards GCP",
5
5
  "backstage": {
6
6
  "role": "backend-plugin-module"
@@ -48,18 +48,18 @@
48
48
  "test": "backstage-cli package test"
49
49
  },
50
50
  "dependencies": {
51
- "@backstage/backend-common": "^0.22.0-next.2",
52
- "@backstage/backend-plugin-api": "^0.6.18-next.1",
53
- "@backstage/backend-tasks": "^0.5.23-next.1",
54
- "@backstage/catalog-model": "^1.5.0-next.0",
51
+ "@backstage/backend-common": "^0.22.1-next.0",
52
+ "@backstage/backend-plugin-api": "^0.6.19-next.0",
53
+ "@backstage/backend-tasks": "^0.5.24-next.0",
54
+ "@backstage/catalog-model": "^1.5.0",
55
55
  "@backstage/config": "^1.2.0",
56
- "@backstage/plugin-catalog-node": "^1.12.0-next.2",
57
- "@backstage/plugin-kubernetes-common": "^0.7.6-next.0",
56
+ "@backstage/plugin-catalog-node": "^1.12.1-next.0",
57
+ "@backstage/plugin-kubernetes-common": "^0.7.6",
58
58
  "@google-cloud/container": "^5.0.0"
59
59
  },
60
60
  "devDependencies": {
61
- "@backstage/backend-test-utils": "^0.3.8-next.2",
62
- "@backstage/cli": "^0.26.5-next.1"
61
+ "@backstage/backend-test-utils": "^0.3.9-next.0",
62
+ "@backstage/cli": "^0.26.6-next.0"
63
63
  },
64
64
  "configSchema": "config.d.ts"
65
65
  }