@backstage/plugin-devtools-backend 0.4.1-next.0 → 0.4.1

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,36 @@
1
1
  # @backstage/plugin-devtools-backend
2
2
 
3
+ ## 0.4.1
4
+
5
+ ### Patch Changes
6
+
7
+ - 094eaa3: Remove references to in-repo backend-common
8
+ - Updated dependencies
9
+ - @backstage/plugin-permission-node@0.8.4
10
+ - @backstage/backend-plugin-api@1.0.1
11
+ - @backstage/cli-common@0.1.14
12
+ - @backstage/config@1.2.0
13
+ - @backstage/config-loader@1.9.1
14
+ - @backstage/errors@1.2.4
15
+ - @backstage/types@1.1.1
16
+ - @backstage/plugin-devtools-common@0.1.12
17
+ - @backstage/plugin-permission-common@0.8.1
18
+
19
+ ## 0.4.1-next.1
20
+
21
+ ### Patch Changes
22
+
23
+ - Updated dependencies
24
+ - @backstage/backend-plugin-api@1.0.1-next.1
25
+ - @backstage/cli-common@0.1.14
26
+ - @backstage/config@1.2.0
27
+ - @backstage/config-loader@1.9.1
28
+ - @backstage/errors@1.2.4
29
+ - @backstage/types@1.1.1
30
+ - @backstage/plugin-devtools-common@0.1.12
31
+ - @backstage/plugin-permission-common@0.8.1
32
+ - @backstage/plugin-permission-node@0.8.4-next.1
33
+
3
34
  ## 0.4.1-next.0
4
35
 
5
36
  ### Patch Changes
@@ -0,0 +1,188 @@
1
+ 'use strict';
2
+
3
+ var config = require('@backstage/config');
4
+ var configLoader = require('@backstage/config-loader');
5
+ var pluginDevtoolsCommon = require('@backstage/plugin-devtools-common');
6
+ var fetch = require('node-fetch');
7
+ var cliCommon = require('@backstage/cli-common');
8
+ var getPackages = require('@manypkg/get-packages');
9
+ var ping = require('ping');
10
+ var os = require('os');
11
+ var fs = require('fs-extra');
12
+ var Lockfile = require('../util/Lockfile.cjs.js');
13
+ var lodash = require('lodash');
14
+ var errors = require('@backstage/errors');
15
+
16
+ function _interopDefaultCompat (e) { return e && typeof e === 'object' && 'default' in e ? e : { default: e }; }
17
+
18
+ var fetch__default = /*#__PURE__*/_interopDefaultCompat(fetch);
19
+ var ping__default = /*#__PURE__*/_interopDefaultCompat(ping);
20
+ var os__default = /*#__PURE__*/_interopDefaultCompat(os);
21
+ var fs__default = /*#__PURE__*/_interopDefaultCompat(fs);
22
+
23
+ class DevToolsBackendApi {
24
+ constructor(logger, config) {
25
+ this.logger = logger;
26
+ this.config = config;
27
+ }
28
+ async listExternalDependencyDetails() {
29
+ const result = [];
30
+ const endpoints = this.config.getOptional(
31
+ "devTools.externalDependencies.endpoints"
32
+ );
33
+ if (!endpoints) {
34
+ return result;
35
+ }
36
+ for (const endpoint of endpoints) {
37
+ this.logger?.info(
38
+ `Checking external dependency "${endpoint.name}" at "${endpoint.target}"`
39
+ );
40
+ switch (endpoint.type) {
41
+ case "ping": {
42
+ const pingResult = await this.pingExternalDependency(endpoint);
43
+ result.push(pingResult);
44
+ break;
45
+ }
46
+ case "fetch": {
47
+ const fetchResult = await this.fetchExternalDependency(endpoint);
48
+ result.push(fetchResult);
49
+ break;
50
+ }
51
+ default:
52
+ return result;
53
+ }
54
+ }
55
+ return result;
56
+ }
57
+ async fetchExternalDependency(endpoint) {
58
+ let status;
59
+ let error;
60
+ await fetch__default.default(endpoint.target).then((res) => {
61
+ status = res.status === 200 ? pluginDevtoolsCommon.ExternalDependencyStatus.healthy : pluginDevtoolsCommon.ExternalDependencyStatus.unhealthy;
62
+ this.logger.debug(
63
+ `Fetch for ${endpoint.name} resulted in status code "${res.status}"`
64
+ );
65
+ }).catch((err) => {
66
+ this.logger.error(`Fetch failed for ${endpoint.name} - ${err.message}`);
67
+ error = err.message;
68
+ });
69
+ const result = {
70
+ name: endpoint.name,
71
+ type: endpoint.type,
72
+ target: endpoint.target,
73
+ status: status ?? pluginDevtoolsCommon.ExternalDependencyStatus.unhealthy,
74
+ error: error ?? void 0
75
+ };
76
+ return result;
77
+ }
78
+ async pingExternalDependency(endpoint) {
79
+ const pingResult = await ping__default.default.promise.probe(endpoint.target);
80
+ let error;
81
+ if (pingResult.packetLoss === "100.000" || pingResult.packetLoss === "unknown") {
82
+ this.logger.error(
83
+ `Ping failed for ${endpoint.name} - ${pingResult.output}`
84
+ );
85
+ error = pingResult.output === "" ? `${endpoint.target} - Unknown` : pingResult.output;
86
+ }
87
+ this.logger.debug(
88
+ `Ping results for ${endpoint.name}: ${pingResult.output}`
89
+ );
90
+ const result = {
91
+ name: endpoint.name,
92
+ type: endpoint.type,
93
+ target: endpoint.target,
94
+ status: pingResult.alive ? pluginDevtoolsCommon.ExternalDependencyStatus.healthy : pluginDevtoolsCommon.ExternalDependencyStatus.unhealthy,
95
+ error: error ?? void 0
96
+ };
97
+ return result;
98
+ }
99
+ async listConfig() {
100
+ const paths = cliCommon.findPaths(__dirname);
101
+ const { packages } = await getPackages.getPackages(paths.targetDir);
102
+ const schemaFunc = async () => {
103
+ return await configLoader.loadConfigSchema({
104
+ dependencies: packages.map((p) => p.packageJson.name)
105
+ });
106
+ };
107
+ const schemaMemo = lodash.memoize(schemaFunc);
108
+ const schema = await schemaMemo();
109
+ const configInfo = {
110
+ config: void 0,
111
+ error: void 0
112
+ };
113
+ try {
114
+ const config$1 = {
115
+ data: this.config.get(),
116
+ context: "inline"
117
+ };
118
+ const sanitizedConfigs = schema.process([config$1], {
119
+ ignoreSchemaErrors: false,
120
+ valueTransform: (value, context) => context.visibility === "secret" ? "<secret>" : value
121
+ });
122
+ const data = config.ConfigReader.fromConfigs(sanitizedConfigs).get();
123
+ configInfo.config = data;
124
+ } catch (error) {
125
+ errors.assertError(error);
126
+ const config$1 = {
127
+ data: this.config.get(),
128
+ context: "inline"
129
+ };
130
+ const sanitizedConfigs = schema.process([config$1], {
131
+ ignoreSchemaErrors: true,
132
+ valueTransform: (value, context) => context.visibility === "secret" ? "<secret>" : value
133
+ });
134
+ const data = config.ConfigReader.fromConfigs(sanitizedConfigs).get();
135
+ configInfo.config = data;
136
+ configInfo.error = {
137
+ name: error.name,
138
+ message: error.message,
139
+ messages: error.messages,
140
+ stack: error.stack
141
+ };
142
+ }
143
+ return configInfo;
144
+ }
145
+ async listInfo() {
146
+ const operatingSystem = `${os__default.default.hostname()}: ${os__default.default.type} ${os__default.default.release} - ${os__default.default.platform}/${os__default.default.arch}`;
147
+ const usedMem = Math.floor((os__default.default.totalmem() - os__default.default.freemem()) / (1024 * 1024));
148
+ const resources = `Memory: ${usedMem}/${Math.floor(
149
+ os__default.default.totalmem() / (1024 * 1024)
150
+ )}MB - Load: ${os__default.default.loadavg().map((v) => v.toFixed(2)).join("/")}`;
151
+ const nodeJsVersion = process.version;
152
+ const paths = cliCommon.findPaths(__dirname);
153
+ const backstageFile = paths.resolveTargetRoot("backstage.json");
154
+ let backstageJson = void 0;
155
+ if (fs__default.default.existsSync(backstageFile)) {
156
+ const buffer = await fs__default.default.readFile(backstageFile);
157
+ backstageJson = JSON.parse(buffer.toString());
158
+ }
159
+ const lockfilePath = paths.resolveTargetRoot("yarn.lock");
160
+ const lockfile = await Lockfile.Lockfile.load(lockfilePath);
161
+ const prefixes = ["@backstage", "@internal"].concat(
162
+ this.config.getOptionalStringArray("devTools.info.packagePrefixes") ?? []
163
+ );
164
+ const deps = [...lockfile.keys()].filter(
165
+ (n) => prefixes.some((prefix) => n.startsWith(prefix))
166
+ );
167
+ const infoDependencies = [];
168
+ for (const dep of deps) {
169
+ const versions = new Set(lockfile.get(dep).map((i) => i.version));
170
+ const infoDependency = {
171
+ name: dep,
172
+ versions: [...versions].join(", ")
173
+ };
174
+ infoDependencies.push(infoDependency);
175
+ }
176
+ const info = {
177
+ operatingSystem: operatingSystem ?? "N/A",
178
+ resourceUtilization: resources ?? "N/A",
179
+ nodeJsVersion: nodeJsVersion ?? "N/A",
180
+ backstageVersion: backstageJson && backstageJson.version ? backstageJson.version : "N/A",
181
+ dependencies: infoDependencies
182
+ };
183
+ return info;
184
+ }
185
+ }
186
+
187
+ exports.DevToolsBackendApi = DevToolsBackendApi;
188
+ //# sourceMappingURL=DevToolsBackendApi.cjs.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"DevToolsBackendApi.cjs.js","sources":["../../src/api/DevToolsBackendApi.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 { Config, ConfigReader } from '@backstage/config';\nimport { loadConfigSchema } from '@backstage/config-loader';\nimport {\n ConfigInfo,\n DevToolsInfo,\n Endpoint,\n ExternalDependency,\n ExternalDependencyStatus,\n PackageDependency,\n} from '@backstage/plugin-devtools-common';\n\nimport { JsonObject } from '@backstage/types';\nimport fetch from 'node-fetch';\nimport { findPaths } from '@backstage/cli-common';\nimport { getPackages } from '@manypkg/get-packages';\nimport ping from 'ping';\nimport os from 'os';\nimport fs from 'fs-extra';\nimport { Lockfile } from '../util/Lockfile';\nimport { memoize } from 'lodash';\nimport { assertError } from '@backstage/errors';\nimport { LoggerService } from '@backstage/backend-plugin-api';\n\n/** @public */\nexport class DevToolsBackendApi {\n public constructor(\n private readonly logger: LoggerService,\n private readonly config: Config,\n ) {}\n\n public async listExternalDependencyDetails(): Promise<ExternalDependency[]> {\n const result: ExternalDependency[] = [];\n\n const endpoints = this.config.getOptional<Endpoint[]>(\n 'devTools.externalDependencies.endpoints',\n );\n if (!endpoints) {\n // No external dependency endpoints configured\n return result;\n }\n for (const endpoint of endpoints) {\n this.logger?.info(\n `Checking external dependency \"${endpoint.name}\" at \"${endpoint.target}\"`,\n );\n\n switch (endpoint.type) {\n case 'ping': {\n const pingResult = await this.pingExternalDependency(endpoint);\n result.push(pingResult);\n break;\n }\n case 'fetch': {\n const fetchResult = await this.fetchExternalDependency(endpoint);\n result.push(fetchResult);\n break;\n }\n default:\n return result;\n }\n }\n\n return result;\n }\n\n private async fetchExternalDependency(\n endpoint: Endpoint,\n ): Promise<ExternalDependency> {\n let status;\n let error;\n\n await fetch(endpoint.target)\n .then(res => {\n status =\n res.status === 200\n ? ExternalDependencyStatus.healthy\n : ExternalDependencyStatus.unhealthy;\n this.logger.debug(\n `Fetch for ${endpoint.name} resulted in status code \"${res.status}\"`,\n );\n })\n .catch((err: Error) => {\n this.logger.error(`Fetch failed for ${endpoint.name} - ${err.message}`);\n error = err.message;\n });\n\n const result: ExternalDependency = {\n name: endpoint.name,\n type: endpoint.type,\n target: endpoint.target,\n status: status ?? ExternalDependencyStatus.unhealthy,\n error: error ?? undefined,\n };\n\n return result;\n }\n\n private async pingExternalDependency(\n endpoint: Endpoint,\n ): Promise<ExternalDependency> {\n const pingResult = await ping.promise.probe(endpoint.target);\n\n let error;\n if (\n pingResult.packetLoss === '100.000' ||\n pingResult.packetLoss === 'unknown'\n ) {\n this.logger.error(\n `Ping failed for ${endpoint.name} - ${pingResult.output}`,\n );\n error =\n pingResult.output === ''\n ? `${endpoint.target} - Unknown`\n : pingResult.output;\n }\n\n this.logger.debug(\n `Ping results for ${endpoint.name}: ${pingResult.output}`,\n );\n\n const result: ExternalDependency = {\n name: endpoint.name,\n type: endpoint.type,\n target: endpoint.target,\n status: pingResult.alive\n ? ExternalDependencyStatus.healthy\n : ExternalDependencyStatus.unhealthy,\n error: error ?? undefined,\n };\n\n return result;\n }\n\n public async listConfig(): Promise<ConfigInfo> {\n /* eslint-disable-next-line no-restricted-syntax */\n const paths = findPaths(__dirname);\n\n const { packages } = await getPackages(paths.targetDir);\n const schemaFunc = async () => {\n return await loadConfigSchema({\n dependencies: packages.map(p => p.packageJson.name),\n });\n };\n\n const schemaMemo = memoize(schemaFunc);\n const schema = await schemaMemo();\n\n const configInfo: ConfigInfo = {\n config: undefined,\n error: undefined,\n };\n try {\n const config = {\n data: this.config.get() as JsonObject,\n context: 'inline',\n };\n const sanitizedConfigs = schema.process([config], {\n ignoreSchemaErrors: false,\n valueTransform: (value, context) =>\n context.visibility === 'secret' ? '<secret>' : value,\n });\n\n const data = ConfigReader.fromConfigs(sanitizedConfigs).get();\n configInfo.config = data;\n } catch (error) {\n assertError(error);\n // The config is not valid for some reason but we want to be able to see it still\n const config = {\n data: this.config.get() as JsonObject,\n context: 'inline',\n };\n const sanitizedConfigs = schema.process([config], {\n ignoreSchemaErrors: true,\n valueTransform: (value, context) =>\n context.visibility === 'secret' ? '<secret>' : value,\n });\n\n const data = ConfigReader.fromConfigs(sanitizedConfigs).get();\n configInfo.config = data;\n configInfo.error = {\n name: error.name,\n message: error.message,\n messages: error.messages as string[] | undefined,\n stack: error.stack,\n };\n }\n\n return configInfo;\n }\n\n public async listInfo(): Promise<DevToolsInfo> {\n const operatingSystem = `${os.hostname()}: ${os.type} ${os.release} - ${\n os.platform\n }/${os.arch}`;\n const usedMem = Math.floor((os.totalmem() - os.freemem()) / (1024 * 1024));\n const resources = `Memory: ${usedMem}/${Math.floor(\n os.totalmem() / (1024 * 1024),\n )}MB - Load: ${os\n .loadavg()\n .map(v => v.toFixed(2))\n .join('/')}`;\n const nodeJsVersion = process.version;\n\n /* eslint-disable-next-line no-restricted-syntax */\n const paths = findPaths(__dirname);\n const backstageFile = paths.resolveTargetRoot('backstage.json');\n let backstageJson = undefined;\n if (fs.existsSync(backstageFile)) {\n const buffer = await fs.readFile(backstageFile);\n backstageJson = JSON.parse(buffer.toString());\n }\n\n const lockfilePath = paths.resolveTargetRoot('yarn.lock');\n const lockfile = await Lockfile.load(lockfilePath);\n\n const prefixes = ['@backstage', '@internal'].concat(\n this.config.getOptionalStringArray('devTools.info.packagePrefixes') ?? [],\n );\n const deps = [...lockfile.keys()].filter(n =>\n prefixes.some(prefix => n.startsWith(prefix)),\n );\n\n const infoDependencies: PackageDependency[] = [];\n for (const dep of deps) {\n const versions = new Set(lockfile.get(dep)!.map(i => i.version));\n const infoDependency: PackageDependency = {\n name: dep,\n versions: [...versions].join(', '),\n };\n infoDependencies.push(infoDependency);\n }\n\n const info: DevToolsInfo = {\n operatingSystem: operatingSystem ?? 'N/A',\n resourceUtilization: resources ?? 'N/A',\n nodeJsVersion: nodeJsVersion ?? 'N/A',\n backstageVersion:\n backstageJson && backstageJson.version ? backstageJson.version : 'N/A',\n dependencies: infoDependencies,\n };\n\n return info;\n }\n}\n\nexport function isValidUrl(url: string): boolean {\n try {\n // eslint-disable-next-line no-new\n new URL(url);\n return true;\n } catch {\n return false;\n }\n}\n"],"names":["fetch","ExternalDependencyStatus","ping","findPaths","getPackages","loadConfigSchema","memoize","config","ConfigReader","assertError","os","fs","Lockfile"],"mappings":"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"}
package/dist/index.cjs.js CHANGED
@@ -2,367 +2,13 @@
2
2
 
3
3
  Object.defineProperty(exports, '__esModule', { value: true });
4
4
 
5
- var config = require('@backstage/config');
6
- var configLoader = require('@backstage/config-loader');
7
- var pluginDevtoolsCommon = require('@backstage/plugin-devtools-common');
8
- var fetch = require('node-fetch');
9
- var cliCommon = require('@backstage/cli-common');
10
- var getPackages = require('@manypkg/get-packages');
11
- var ping = require('ping');
12
- var os = require('os');
13
- var fs = require('fs-extra');
14
- var parsers = require('@yarnpkg/parsers');
15
- var lockfile = require('@yarnpkg/lockfile');
16
- var lodash = require('lodash');
17
- var errors = require('@backstage/errors');
18
- var pluginPermissionCommon = require('@backstage/plugin-permission-common');
19
- var Router = require('express-promise-router');
20
- var backendCommon = require('@backstage/backend-common');
21
- var express = require('express');
22
- var pluginPermissionNode = require('@backstage/plugin-permission-node');
23
- var backendPluginApi = require('@backstage/backend-plugin-api');
5
+ var DevToolsBackendApi = require('./api/DevToolsBackendApi.cjs.js');
6
+ var router = require('./service/router.cjs.js');
7
+ var plugin = require('./plugin.cjs.js');
24
8
 
25
- function _interopDefaultCompat (e) { return e && typeof e === 'object' && 'default' in e ? e : { default: e }; }
26
9
 
27
- var fetch__default = /*#__PURE__*/_interopDefaultCompat(fetch);
28
- var ping__default = /*#__PURE__*/_interopDefaultCompat(ping);
29
- var os__default = /*#__PURE__*/_interopDefaultCompat(os);
30
- var fs__default = /*#__PURE__*/_interopDefaultCompat(fs);
31
- var Router__default = /*#__PURE__*/_interopDefaultCompat(Router);
32
- var express__default = /*#__PURE__*/_interopDefaultCompat(express);
33
10
 
34
- const ENTRY_PATTERN = /^((?:@[^/]+\/)?[^@/]+)@(.+)$/;
35
- function parseLockfile(lockfileContents) {
36
- try {
37
- return {
38
- object: parsers.parseSyml(lockfileContents),
39
- type: "success"
40
- };
41
- } catch (err) {
42
- return {
43
- object: null,
44
- type: err
45
- };
46
- }
47
- }
48
- const NEW_HEADER = `${[
49
- `# This file is generated by running "yarn install" inside your project.
50
- `,
51
- `# Manual changes might be lost - proceed with caution!
52
- `
53
- ].join(``)}
54
- `;
55
- function stringifyLockfile(data, legacy) {
56
- return legacy ? lockfile.stringify(data) : NEW_HEADER + parsers.stringifySyml(data);
57
- }
58
- const LEGACY_REGEX = /^(#.*(\r?\n))*?#\s+yarn\s+lockfile\s+v1\r?\n/i;
59
- const SPECIAL_OBJECT_KEYS = [
60
- `__metadata`,
61
- `version`,
62
- `resolution`,
63
- `dependencies`,
64
- `peerDependencies`,
65
- `dependenciesMeta`,
66
- `peerDependenciesMeta`,
67
- `binaries`
68
- ];
69
- class Lockfile {
70
- constructor(packages, data, legacy = false) {
71
- this.packages = packages;
72
- this.data = data;
73
- this.legacy = legacy;
74
- }
75
- static async load(path) {
76
- const lockfileContents = await fs__default.default.readFile(path, "utf8");
77
- const legacy = LEGACY_REGEX.test(lockfileContents);
78
- const lockfile = parseLockfile(lockfileContents);
79
- if (lockfile.type !== "success") {
80
- throw new Error(`Failed yarn.lock parse with ${lockfile.type}`);
81
- }
82
- const data = lockfile.object;
83
- const packages = /* @__PURE__ */ new Map();
84
- for (const [key, value] of Object.entries(data)) {
85
- if (SPECIAL_OBJECT_KEYS.includes(key)) continue;
86
- const [, name, range] = ENTRY_PATTERN.exec(key) ?? [];
87
- if (!name) {
88
- throw new Error(`Failed to parse yarn.lock entry '${key}'`);
89
- }
90
- let queries = packages.get(name);
91
- if (!queries) {
92
- queries = [];
93
- packages.set(name, queries);
94
- }
95
- queries.push({ range, version: value.version });
96
- }
97
- return new Lockfile(packages, data, legacy);
98
- }
99
- /** Get the entries for a single package in the lockfile */
100
- get(name) {
101
- return this.packages.get(name);
102
- }
103
- /** Returns the name of all packages available in the lockfile */
104
- keys() {
105
- return this.packages.keys();
106
- }
107
- toString() {
108
- return stringifyLockfile(this.data, this.legacy);
109
- }
110
- }
111
-
112
- class DevToolsBackendApi {
113
- constructor(logger, config) {
114
- this.logger = logger;
115
- this.config = config;
116
- }
117
- async listExternalDependencyDetails() {
118
- const result = [];
119
- const endpoints = this.config.getOptional(
120
- "devTools.externalDependencies.endpoints"
121
- );
122
- if (!endpoints) {
123
- return result;
124
- }
125
- for (const endpoint of endpoints) {
126
- this.logger?.info(
127
- `Checking external dependency "${endpoint.name}" at "${endpoint.target}"`
128
- );
129
- switch (endpoint.type) {
130
- case "ping": {
131
- const pingResult = await this.pingExternalDependency(endpoint);
132
- result.push(pingResult);
133
- break;
134
- }
135
- case "fetch": {
136
- const fetchResult = await this.fetchExternalDependency(endpoint);
137
- result.push(fetchResult);
138
- break;
139
- }
140
- default:
141
- return result;
142
- }
143
- }
144
- return result;
145
- }
146
- async fetchExternalDependency(endpoint) {
147
- let status;
148
- let error;
149
- await fetch__default.default(endpoint.target).then((res) => {
150
- status = res.status === 200 ? pluginDevtoolsCommon.ExternalDependencyStatus.healthy : pluginDevtoolsCommon.ExternalDependencyStatus.unhealthy;
151
- this.logger.debug(
152
- `Fetch for ${endpoint.name} resulted in status code "${res.status}"`
153
- );
154
- }).catch((err) => {
155
- this.logger.error(`Fetch failed for ${endpoint.name} - ${err.message}`);
156
- error = err.message;
157
- });
158
- const result = {
159
- name: endpoint.name,
160
- type: endpoint.type,
161
- target: endpoint.target,
162
- status: status ?? pluginDevtoolsCommon.ExternalDependencyStatus.unhealthy,
163
- error: error ?? void 0
164
- };
165
- return result;
166
- }
167
- async pingExternalDependency(endpoint) {
168
- const pingResult = await ping__default.default.promise.probe(endpoint.target);
169
- let error;
170
- if (pingResult.packetLoss === "100.000" || pingResult.packetLoss === "unknown") {
171
- this.logger.error(
172
- `Ping failed for ${endpoint.name} - ${pingResult.output}`
173
- );
174
- error = pingResult.output === "" ? `${endpoint.target} - Unknown` : pingResult.output;
175
- }
176
- this.logger.debug(
177
- `Ping results for ${endpoint.name}: ${pingResult.output}`
178
- );
179
- const result = {
180
- name: endpoint.name,
181
- type: endpoint.type,
182
- target: endpoint.target,
183
- status: pingResult.alive ? pluginDevtoolsCommon.ExternalDependencyStatus.healthy : pluginDevtoolsCommon.ExternalDependencyStatus.unhealthy,
184
- error: error ?? void 0
185
- };
186
- return result;
187
- }
188
- async listConfig() {
189
- const paths = cliCommon.findPaths(__dirname);
190
- const { packages } = await getPackages.getPackages(paths.targetDir);
191
- const schemaFunc = async () => {
192
- return await configLoader.loadConfigSchema({
193
- dependencies: packages.map((p) => p.packageJson.name)
194
- });
195
- };
196
- const schemaMemo = lodash.memoize(schemaFunc);
197
- const schema = await schemaMemo();
198
- const configInfo = {
199
- config: void 0,
200
- error: void 0
201
- };
202
- try {
203
- const config$1 = {
204
- data: this.config.get(),
205
- context: "inline"
206
- };
207
- const sanitizedConfigs = schema.process([config$1], {
208
- ignoreSchemaErrors: false,
209
- valueTransform: (value, context) => context.visibility === "secret" ? "<secret>" : value
210
- });
211
- const data = config.ConfigReader.fromConfigs(sanitizedConfigs).get();
212
- configInfo.config = data;
213
- } catch (error) {
214
- errors.assertError(error);
215
- const config$1 = {
216
- data: this.config.get(),
217
- context: "inline"
218
- };
219
- const sanitizedConfigs = schema.process([config$1], {
220
- ignoreSchemaErrors: true,
221
- valueTransform: (value, context) => context.visibility === "secret" ? "<secret>" : value
222
- });
223
- const data = config.ConfigReader.fromConfigs(sanitizedConfigs).get();
224
- configInfo.config = data;
225
- configInfo.error = {
226
- name: error.name,
227
- message: error.message,
228
- messages: error.messages,
229
- stack: error.stack
230
- };
231
- }
232
- return configInfo;
233
- }
234
- async listInfo() {
235
- const operatingSystem = `${os__default.default.hostname()}: ${os__default.default.type} ${os__default.default.release} - ${os__default.default.platform}/${os__default.default.arch}`;
236
- const usedMem = Math.floor((os__default.default.totalmem() - os__default.default.freemem()) / (1024 * 1024));
237
- const resources = `Memory: ${usedMem}/${Math.floor(
238
- os__default.default.totalmem() / (1024 * 1024)
239
- )}MB - Load: ${os__default.default.loadavg().map((v) => v.toFixed(2)).join("/")}`;
240
- const nodeJsVersion = process.version;
241
- const paths = cliCommon.findPaths(__dirname);
242
- const backstageFile = paths.resolveTargetRoot("backstage.json");
243
- let backstageJson = void 0;
244
- if (fs__default.default.existsSync(backstageFile)) {
245
- const buffer = await fs__default.default.readFile(backstageFile);
246
- backstageJson = JSON.parse(buffer.toString());
247
- }
248
- const lockfilePath = paths.resolveTargetRoot("yarn.lock");
249
- const lockfile = await Lockfile.load(lockfilePath);
250
- const prefixes = ["@backstage", "@internal"].concat(
251
- this.config.getOptionalStringArray("devTools.info.packagePrefixes") ?? []
252
- );
253
- const deps = [...lockfile.keys()].filter(
254
- (n) => prefixes.some((prefix) => n.startsWith(prefix))
255
- );
256
- const infoDependencies = [];
257
- for (const dep of deps) {
258
- const versions = new Set(lockfile.get(dep).map((i) => i.version));
259
- const infoDependency = {
260
- name: dep,
261
- versions: [...versions].join(", ")
262
- };
263
- infoDependencies.push(infoDependency);
264
- }
265
- const info = {
266
- operatingSystem: operatingSystem ?? "N/A",
267
- resourceUtilization: resources ?? "N/A",
268
- nodeJsVersion: nodeJsVersion ?? "N/A",
269
- backstageVersion: backstageJson && backstageJson.version ? backstageJson.version : "N/A",
270
- dependencies: infoDependencies
271
- };
272
- return info;
273
- }
274
- }
275
-
276
- async function createRouter(options) {
277
- const { logger, config, permissions } = options;
278
- const { httpAuth } = backendCommon.createLegacyAuthAdapters(options);
279
- const devToolsBackendApi = options.devToolsBackendApi || new DevToolsBackendApi(logger, config);
280
- const router = Router__default.default();
281
- router.use(express__default.default.json());
282
- router.use(
283
- pluginPermissionNode.createPermissionIntegrationRouter({
284
- permissions: pluginDevtoolsCommon.devToolsPermissions
285
- })
286
- );
287
- router.get("/health", (_req, res) => {
288
- res.status(200).json({ status: "ok" });
289
- });
290
- router.get("/info", async (req, response) => {
291
- const decision = (await permissions.authorize(
292
- [{ permission: pluginDevtoolsCommon.devToolsInfoReadPermission }],
293
- { credentials: await httpAuth.credentials(req) }
294
- ))[0];
295
- if (decision.result === pluginPermissionCommon.AuthorizeResult.DENY) {
296
- throw new errors.NotAllowedError("Unauthorized");
297
- }
298
- const info = await devToolsBackendApi.listInfo();
299
- response.status(200).json(info);
300
- });
301
- router.get("/config", async (req, response) => {
302
- const decision = (await permissions.authorize(
303
- [{ permission: pluginDevtoolsCommon.devToolsConfigReadPermission }],
304
- { credentials: await httpAuth.credentials(req) }
305
- ))[0];
306
- if (decision.result === pluginPermissionCommon.AuthorizeResult.DENY) {
307
- throw new errors.NotAllowedError("Unauthorized");
308
- }
309
- const configList = await devToolsBackendApi.listConfig();
310
- response.status(200).json(configList);
311
- });
312
- router.get("/external-dependencies", async (req, response) => {
313
- const decision = (await permissions.authorize(
314
- [{ permission: pluginDevtoolsCommon.devToolsExternalDependenciesReadPermission }],
315
- { credentials: await httpAuth.credentials(req) }
316
- ))[0];
317
- if (decision.result === pluginPermissionCommon.AuthorizeResult.DENY) {
318
- throw new errors.NotAllowedError("Unauthorized");
319
- }
320
- const health = await devToolsBackendApi.listExternalDependencyDetails();
321
- response.status(200).json(health);
322
- });
323
- router.use(backendCommon.errorHandler());
324
- return router;
325
- }
326
-
327
- const devtoolsPlugin = backendPluginApi.createBackendPlugin({
328
- pluginId: "devtools",
329
- register(env) {
330
- env.registerInit({
331
- deps: {
332
- config: backendPluginApi.coreServices.rootConfig,
333
- logger: backendPluginApi.coreServices.logger,
334
- permissions: backendPluginApi.coreServices.permissions,
335
- httpRouter: backendPluginApi.coreServices.httpRouter,
336
- discovery: backendPluginApi.coreServices.discovery,
337
- httpAuth: backendPluginApi.coreServices.httpAuth
338
- },
339
- async init({
340
- config,
341
- logger,
342
- permissions,
343
- httpRouter,
344
- discovery,
345
- httpAuth
346
- }) {
347
- httpRouter.use(
348
- await createRouter({
349
- config,
350
- logger,
351
- permissions,
352
- discovery,
353
- httpAuth
354
- })
355
- );
356
- httpRouter.addAuthPolicy({
357
- path: "/health",
358
- allow: "unauthenticated"
359
- });
360
- }
361
- });
362
- }
363
- });
364
-
365
- exports.DevToolsBackendApi = DevToolsBackendApi;
366
- exports.createRouter = createRouter;
367
- exports.default = devtoolsPlugin;
11
+ exports.DevToolsBackendApi = DevToolsBackendApi.DevToolsBackendApi;
12
+ exports.createRouter = router.createRouter;
13
+ exports.default = plugin.devtoolsPlugin;
368
14
  //# sourceMappingURL=index.cjs.js.map
@@ -1 +1 @@
1
- {"version":3,"file":"index.cjs.js","sources":["../src/util/Lockfile.ts","../src/api/DevToolsBackendApi.ts","../src/service/router.ts","../src/plugin.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 fs from 'fs-extra';\nimport { parseSyml, stringifySyml } from '@yarnpkg/parsers';\nimport { stringify as legacyStringifyLockfile } from '@yarnpkg/lockfile';\n\nconst ENTRY_PATTERN = /^((?:@[^/]+\\/)?[^@/]+)@(.+)$/;\n\ntype LockfileData = {\n [entry: string]: {\n version: string;\n resolved?: string;\n integrity?: string;\n dependencies?: { [name: string]: string };\n };\n};\n\ntype LockfileQueryEntry = {\n range: string;\n version: string;\n};\n\nfunction parseLockfile(lockfileContents: string) {\n try {\n return {\n object: parseSyml(lockfileContents),\n type: 'success',\n };\n } catch (err) {\n return {\n object: null,\n type: err,\n };\n }\n}\n\n// the new yarn header is handled out of band of the parsing\n// https://github.com/yarnpkg/berry/blob/0c5974f193a9397630e9aee2b3876cca62611149/packages/yarnpkg-core/sources/Project.ts#L1741-L1746\nconst NEW_HEADER = `${[\n `# This file is generated by running \"yarn install\" inside your project.\\n`,\n `# Manual changes might be lost - proceed with caution!\\n`,\n].join(``)}\\n`;\n\nfunction stringifyLockfile(data: LockfileData, legacy: boolean) {\n return legacy\n ? legacyStringifyLockfile(data)\n : NEW_HEADER + stringifySyml(data);\n}\n// taken from yarn parser package\n// https://github.com/yarnpkg/berry/blob/0c5974f193a9397630e9aee2b3876cca62611149/packages/yarnpkg-parsers/sources/syml.ts#L136\nconst LEGACY_REGEX = /^(#.*(\\r?\\n))*?#\\s+yarn\\s+lockfile\\s+v1\\r?\\n/i;\n\n// these are special top level yarn keys.\n// https://github.com/yarnpkg/berry/blob/9bd61fbffb83d0b8166a9cc26bec3a58743aa453/packages/yarnpkg-parsers/sources/syml.ts#L9\nconst SPECIAL_OBJECT_KEYS = [\n `__metadata`,\n `version`,\n `resolution`,\n `dependencies`,\n `peerDependencies`,\n `dependenciesMeta`,\n `peerDependenciesMeta`,\n `binaries`,\n];\n\nexport class Lockfile {\n static async load(path: string) {\n const lockfileContents = await fs.readFile(path, 'utf8');\n const legacy = LEGACY_REGEX.test(lockfileContents);\n const lockfile = parseLockfile(lockfileContents);\n if (lockfile.type !== 'success') {\n throw new Error(`Failed yarn.lock parse with ${lockfile.type}`);\n }\n\n const data = lockfile.object as LockfileData;\n const packages = new Map<string, LockfileQueryEntry[]>();\n\n for (const [key, value] of Object.entries(data)) {\n if (SPECIAL_OBJECT_KEYS.includes(key)) continue;\n\n const [, name, range] = ENTRY_PATTERN.exec(key) ?? [];\n if (!name) {\n throw new Error(`Failed to parse yarn.lock entry '${key}'`);\n }\n\n let queries = packages.get(name);\n if (!queries) {\n queries = [];\n packages.set(name, queries);\n }\n queries.push({ range, version: value.version });\n }\n\n return new Lockfile(packages, data, legacy);\n }\n\n private constructor(\n private readonly packages: Map<string, LockfileQueryEntry[]>,\n private readonly data: LockfileData,\n private readonly legacy: boolean = false,\n ) {}\n\n /** Get the entries for a single package in the lockfile */\n get(name: string): LockfileQueryEntry[] | undefined {\n return this.packages.get(name);\n }\n\n /** Returns the name of all packages available in the lockfile */\n keys(): IterableIterator<string> {\n return this.packages.keys();\n }\n\n toString() {\n return stringifyLockfile(this.data, this.legacy);\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 { Config, ConfigReader } from '@backstage/config';\nimport { loadConfigSchema } from '@backstage/config-loader';\nimport {\n ConfigInfo,\n DevToolsInfo,\n Endpoint,\n ExternalDependency,\n ExternalDependencyStatus,\n PackageDependency,\n} from '@backstage/plugin-devtools-common';\n\nimport { JsonObject } from '@backstage/types';\nimport fetch from 'node-fetch';\nimport { findPaths } from '@backstage/cli-common';\nimport { getPackages } from '@manypkg/get-packages';\nimport ping from 'ping';\nimport os from 'os';\nimport fs from 'fs-extra';\nimport { Lockfile } from '../util/Lockfile';\nimport { memoize } from 'lodash';\nimport { assertError } from '@backstage/errors';\nimport { LoggerService } from '@backstage/backend-plugin-api';\n\n/** @public */\nexport class DevToolsBackendApi {\n public constructor(\n private readonly logger: LoggerService,\n private readonly config: Config,\n ) {}\n\n public async listExternalDependencyDetails(): Promise<ExternalDependency[]> {\n const result: ExternalDependency[] = [];\n\n const endpoints = this.config.getOptional<Endpoint[]>(\n 'devTools.externalDependencies.endpoints',\n );\n if (!endpoints) {\n // No external dependency endpoints configured\n return result;\n }\n for (const endpoint of endpoints) {\n this.logger?.info(\n `Checking external dependency \"${endpoint.name}\" at \"${endpoint.target}\"`,\n );\n\n switch (endpoint.type) {\n case 'ping': {\n const pingResult = await this.pingExternalDependency(endpoint);\n result.push(pingResult);\n break;\n }\n case 'fetch': {\n const fetchResult = await this.fetchExternalDependency(endpoint);\n result.push(fetchResult);\n break;\n }\n default:\n return result;\n }\n }\n\n return result;\n }\n\n private async fetchExternalDependency(\n endpoint: Endpoint,\n ): Promise<ExternalDependency> {\n let status;\n let error;\n\n await fetch(endpoint.target)\n .then(res => {\n status =\n res.status === 200\n ? ExternalDependencyStatus.healthy\n : ExternalDependencyStatus.unhealthy;\n this.logger.debug(\n `Fetch for ${endpoint.name} resulted in status code \"${res.status}\"`,\n );\n })\n .catch((err: Error) => {\n this.logger.error(`Fetch failed for ${endpoint.name} - ${err.message}`);\n error = err.message;\n });\n\n const result: ExternalDependency = {\n name: endpoint.name,\n type: endpoint.type,\n target: endpoint.target,\n status: status ?? ExternalDependencyStatus.unhealthy,\n error: error ?? undefined,\n };\n\n return result;\n }\n\n private async pingExternalDependency(\n endpoint: Endpoint,\n ): Promise<ExternalDependency> {\n const pingResult = await ping.promise.probe(endpoint.target);\n\n let error;\n if (\n pingResult.packetLoss === '100.000' ||\n pingResult.packetLoss === 'unknown'\n ) {\n this.logger.error(\n `Ping failed for ${endpoint.name} - ${pingResult.output}`,\n );\n error =\n pingResult.output === ''\n ? `${endpoint.target} - Unknown`\n : pingResult.output;\n }\n\n this.logger.debug(\n `Ping results for ${endpoint.name}: ${pingResult.output}`,\n );\n\n const result: ExternalDependency = {\n name: endpoint.name,\n type: endpoint.type,\n target: endpoint.target,\n status: pingResult.alive\n ? ExternalDependencyStatus.healthy\n : ExternalDependencyStatus.unhealthy,\n error: error ?? undefined,\n };\n\n return result;\n }\n\n public async listConfig(): Promise<ConfigInfo> {\n /* eslint-disable-next-line no-restricted-syntax */\n const paths = findPaths(__dirname);\n\n const { packages } = await getPackages(paths.targetDir);\n const schemaFunc = async () => {\n return await loadConfigSchema({\n dependencies: packages.map(p => p.packageJson.name),\n });\n };\n\n const schemaMemo = memoize(schemaFunc);\n const schema = await schemaMemo();\n\n const configInfo: ConfigInfo = {\n config: undefined,\n error: undefined,\n };\n try {\n const config = {\n data: this.config.get() as JsonObject,\n context: 'inline',\n };\n const sanitizedConfigs = schema.process([config], {\n ignoreSchemaErrors: false,\n valueTransform: (value, context) =>\n context.visibility === 'secret' ? '<secret>' : value,\n });\n\n const data = ConfigReader.fromConfigs(sanitizedConfigs).get();\n configInfo.config = data;\n } catch (error) {\n assertError(error);\n // The config is not valid for some reason but we want to be able to see it still\n const config = {\n data: this.config.get() as JsonObject,\n context: 'inline',\n };\n const sanitizedConfigs = schema.process([config], {\n ignoreSchemaErrors: true,\n valueTransform: (value, context) =>\n context.visibility === 'secret' ? '<secret>' : value,\n });\n\n const data = ConfigReader.fromConfigs(sanitizedConfigs).get();\n configInfo.config = data;\n configInfo.error = {\n name: error.name,\n message: error.message,\n messages: error.messages as string[] | undefined,\n stack: error.stack,\n };\n }\n\n return configInfo;\n }\n\n public async listInfo(): Promise<DevToolsInfo> {\n const operatingSystem = `${os.hostname()}: ${os.type} ${os.release} - ${\n os.platform\n }/${os.arch}`;\n const usedMem = Math.floor((os.totalmem() - os.freemem()) / (1024 * 1024));\n const resources = `Memory: ${usedMem}/${Math.floor(\n os.totalmem() / (1024 * 1024),\n )}MB - Load: ${os\n .loadavg()\n .map(v => v.toFixed(2))\n .join('/')}`;\n const nodeJsVersion = process.version;\n\n /* eslint-disable-next-line no-restricted-syntax */\n const paths = findPaths(__dirname);\n const backstageFile = paths.resolveTargetRoot('backstage.json');\n let backstageJson = undefined;\n if (fs.existsSync(backstageFile)) {\n const buffer = await fs.readFile(backstageFile);\n backstageJson = JSON.parse(buffer.toString());\n }\n\n const lockfilePath = paths.resolveTargetRoot('yarn.lock');\n const lockfile = await Lockfile.load(lockfilePath);\n\n const prefixes = ['@backstage', '@internal'].concat(\n this.config.getOptionalStringArray('devTools.info.packagePrefixes') ?? [],\n );\n const deps = [...lockfile.keys()].filter(n =>\n prefixes.some(prefix => n.startsWith(prefix)),\n );\n\n const infoDependencies: PackageDependency[] = [];\n for (const dep of deps) {\n const versions = new Set(lockfile.get(dep)!.map(i => i.version));\n const infoDependency: PackageDependency = {\n name: dep,\n versions: [...versions].join(', '),\n };\n infoDependencies.push(infoDependency);\n }\n\n const info: DevToolsInfo = {\n operatingSystem: operatingSystem ?? 'N/A',\n resourceUtilization: resources ?? 'N/A',\n nodeJsVersion: nodeJsVersion ?? 'N/A',\n backstageVersion:\n backstageJson && backstageJson.version ? backstageJson.version : 'N/A',\n dependencies: infoDependencies,\n };\n\n return info;\n }\n}\n\nexport function isValidUrl(url: string): boolean {\n try {\n // eslint-disable-next-line no-new\n new URL(url);\n return true;\n } catch {\n return false;\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 */\nimport { AuthorizeResult } from '@backstage/plugin-permission-common';\nimport {\n devToolsConfigReadPermission,\n devToolsExternalDependenciesReadPermission,\n devToolsInfoReadPermission,\n devToolsPermissions,\n} from '@backstage/plugin-devtools-common';\n\nimport { DevToolsBackendApi } from '../api';\nimport { NotAllowedError } from '@backstage/errors';\nimport Router from 'express-promise-router';\nimport {\n createLegacyAuthAdapters,\n errorHandler,\n} from '@backstage/backend-common';\nimport express from 'express';\nimport { createPermissionIntegrationRouter } from '@backstage/plugin-permission-node';\nimport {\n DiscoveryService,\n HttpAuthService,\n LoggerService,\n PermissionsService,\n RootConfigService,\n} from '@backstage/backend-plugin-api';\n\n/**\n * @public\n * @deprecated Please migrate to the new backend system as this will be removed in the future.\n */\nexport interface RouterOptions {\n devToolsBackendApi?: DevToolsBackendApi;\n logger: LoggerService;\n config: RootConfigService;\n permissions: PermissionsService;\n discovery: DiscoveryService;\n httpAuth?: HttpAuthService;\n}\n\n/**\n * @deprecated Please migrate to the new backend system as this will be removed in the future.\n * @public\n * */\nexport async function createRouter(\n options: RouterOptions,\n): Promise<express.Router> {\n const { logger, config, permissions } = options;\n\n const { httpAuth } = createLegacyAuthAdapters(options);\n\n const devToolsBackendApi =\n options.devToolsBackendApi || new DevToolsBackendApi(logger, config);\n\n const router = Router();\n router.use(express.json());\n router.use(\n createPermissionIntegrationRouter({\n permissions: devToolsPermissions,\n }),\n );\n\n router.get('/health', (_req, res) => {\n res.status(200).json({ status: 'ok' });\n });\n\n router.get('/info', async (req, response) => {\n const decision = (\n await permissions.authorize(\n [{ permission: devToolsInfoReadPermission }],\n { credentials: await httpAuth.credentials(req) },\n )\n )[0];\n\n if (decision.result === AuthorizeResult.DENY) {\n throw new NotAllowedError('Unauthorized');\n }\n\n const info = await devToolsBackendApi.listInfo();\n\n response.status(200).json(info);\n });\n\n router.get('/config', async (req, response) => {\n const decision = (\n await permissions.authorize(\n [{ permission: devToolsConfigReadPermission }],\n { credentials: await httpAuth.credentials(req) },\n )\n )[0];\n\n if (decision.result === AuthorizeResult.DENY) {\n throw new NotAllowedError('Unauthorized');\n }\n\n const configList = await devToolsBackendApi.listConfig();\n\n response.status(200).json(configList);\n });\n\n router.get('/external-dependencies', async (req, response) => {\n const decision = (\n await permissions.authorize(\n [{ permission: devToolsExternalDependenciesReadPermission }],\n { credentials: await httpAuth.credentials(req) },\n )\n )[0];\n\n if (decision.result === AuthorizeResult.DENY) {\n throw new NotAllowedError('Unauthorized');\n }\n\n const health = await devToolsBackendApi.listExternalDependencyDetails();\n\n response.status(200).json(health);\n });\n\n router.use(errorHandler());\n return router;\n}\n","/*\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 coreServices,\n createBackendPlugin,\n} from '@backstage/backend-plugin-api';\nimport { createRouter } from './service/router';\n\n/**\n * DevTools backend plugin\n *\n * @public\n */\nexport const devtoolsPlugin = createBackendPlugin({\n pluginId: 'devtools',\n register(env) {\n env.registerInit({\n deps: {\n config: coreServices.rootConfig,\n logger: coreServices.logger,\n permissions: coreServices.permissions,\n httpRouter: coreServices.httpRouter,\n discovery: coreServices.discovery,\n httpAuth: coreServices.httpAuth,\n },\n async init({\n config,\n logger,\n permissions,\n httpRouter,\n discovery,\n httpAuth,\n }) {\n httpRouter.use(\n await createRouter({\n config,\n logger,\n permissions,\n discovery,\n httpAuth,\n }),\n );\n httpRouter.addAuthPolicy({\n path: '/health',\n allow: 'unauthenticated',\n });\n },\n });\n },\n});\n"],"names":["parseSyml","legacyStringifyLockfile","stringifySyml","fs","fetch","ExternalDependencyStatus","ping","findPaths","getPackages","loadConfigSchema","memoize","config","ConfigReader","assertError","os","createLegacyAuthAdapters","Router","express","createPermissionIntegrationRouter","devToolsPermissions","devToolsInfoReadPermission","AuthorizeResult","NotAllowedError","devToolsConfigReadPermission","devToolsExternalDependenciesReadPermission","errorHandler","createBackendPlugin","coreServices"],"mappings":"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"}
1
+ {"version":3,"file":"index.cjs.js","sources":[],"sourcesContent":[],"names":[],"mappings":";;;;;;;;;;;;"}
@@ -0,0 +1,45 @@
1
+ 'use strict';
2
+
3
+ var backendPluginApi = require('@backstage/backend-plugin-api');
4
+ var router = require('./service/router.cjs.js');
5
+
6
+ const devtoolsPlugin = backendPluginApi.createBackendPlugin({
7
+ pluginId: "devtools",
8
+ register(env) {
9
+ env.registerInit({
10
+ deps: {
11
+ config: backendPluginApi.coreServices.rootConfig,
12
+ logger: backendPluginApi.coreServices.logger,
13
+ permissions: backendPluginApi.coreServices.permissions,
14
+ httpRouter: backendPluginApi.coreServices.httpRouter,
15
+ discovery: backendPluginApi.coreServices.discovery,
16
+ httpAuth: backendPluginApi.coreServices.httpAuth
17
+ },
18
+ async init({
19
+ config,
20
+ logger,
21
+ permissions,
22
+ httpRouter,
23
+ discovery,
24
+ httpAuth
25
+ }) {
26
+ httpRouter.use(
27
+ await router.createRouter({
28
+ config,
29
+ logger,
30
+ permissions,
31
+ discovery,
32
+ httpAuth
33
+ })
34
+ );
35
+ httpRouter.addAuthPolicy({
36
+ path: "/health",
37
+ allow: "unauthenticated"
38
+ });
39
+ }
40
+ });
41
+ }
42
+ });
43
+
44
+ exports.devtoolsPlugin = devtoolsPlugin;
45
+ //# sourceMappingURL=plugin.cjs.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"plugin.cjs.js","sources":["../src/plugin.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 coreServices,\n createBackendPlugin,\n} from '@backstage/backend-plugin-api';\nimport { createRouter } from './service/router';\n\n/**\n * DevTools backend plugin\n *\n * @public\n */\nexport const devtoolsPlugin = createBackendPlugin({\n pluginId: 'devtools',\n register(env) {\n env.registerInit({\n deps: {\n config: coreServices.rootConfig,\n logger: coreServices.logger,\n permissions: coreServices.permissions,\n httpRouter: coreServices.httpRouter,\n discovery: coreServices.discovery,\n httpAuth: coreServices.httpAuth,\n },\n async init({\n config,\n logger,\n permissions,\n httpRouter,\n discovery,\n httpAuth,\n }) {\n httpRouter.use(\n await createRouter({\n config,\n logger,\n permissions,\n discovery,\n httpAuth,\n }),\n );\n httpRouter.addAuthPolicy({\n path: '/health',\n allow: 'unauthenticated',\n });\n },\n });\n },\n});\n"],"names":["createBackendPlugin","coreServices","createRouter"],"mappings":";;;;;AA2BO,MAAM,iBAAiBA,oCAAoB,CAAA;AAAA,EAChD,QAAU,EAAA,UAAA;AAAA,EACV,SAAS,GAAK,EAAA;AACZ,IAAA,GAAA,CAAI,YAAa,CAAA;AAAA,MACf,IAAM,EAAA;AAAA,QACJ,QAAQC,6BAAa,CAAA,UAAA;AAAA,QACrB,QAAQA,6BAAa,CAAA,MAAA;AAAA,QACrB,aAAaA,6BAAa,CAAA,WAAA;AAAA,QAC1B,YAAYA,6BAAa,CAAA,UAAA;AAAA,QACzB,WAAWA,6BAAa,CAAA,SAAA;AAAA,QACxB,UAAUA,6BAAa,CAAA,QAAA;AAAA,OACzB;AAAA,MACA,MAAM,IAAK,CAAA;AAAA,QACT,MAAA;AAAA,QACA,MAAA;AAAA,QACA,WAAA;AAAA,QACA,UAAA;AAAA,QACA,SAAA;AAAA,QACA,QAAA;AAAA,OACC,EAAA;AACD,QAAW,UAAA,CAAA,GAAA;AAAA,UACT,MAAMC,mBAAa,CAAA;AAAA,YACjB,MAAA;AAAA,YACA,MAAA;AAAA,YACA,WAAA;AAAA,YACA,SAAA;AAAA,YACA,QAAA;AAAA,WACD,CAAA;AAAA,SACH,CAAA;AACA,QAAA,UAAA,CAAW,aAAc,CAAA;AAAA,UACvB,IAAM,EAAA,SAAA;AAAA,UACN,KAAO,EAAA,iBAAA;AAAA,SACR,CAAA,CAAA;AAAA,OACH;AAAA,KACD,CAAA,CAAA;AAAA,GACH;AACF,CAAC;;;;"}
@@ -0,0 +1,69 @@
1
+ 'use strict';
2
+
3
+ var pluginPermissionCommon = require('@backstage/plugin-permission-common');
4
+ var pluginDevtoolsCommon = require('@backstage/plugin-devtools-common');
5
+ var DevToolsBackendApi = require('../api/DevToolsBackendApi.cjs.js');
6
+ var errors = require('@backstage/errors');
7
+ var Router = require('express-promise-router');
8
+ var backendCommon = require('@backstage/backend-common');
9
+ var express = require('express');
10
+ var pluginPermissionNode = require('@backstage/plugin-permission-node');
11
+
12
+ function _interopDefaultCompat (e) { return e && typeof e === 'object' && 'default' in e ? e : { default: e }; }
13
+
14
+ var Router__default = /*#__PURE__*/_interopDefaultCompat(Router);
15
+ var express__default = /*#__PURE__*/_interopDefaultCompat(express);
16
+
17
+ async function createRouter(options) {
18
+ const { logger, config, permissions } = options;
19
+ const { httpAuth } = backendCommon.createLegacyAuthAdapters(options);
20
+ const devToolsBackendApi = options.devToolsBackendApi || new DevToolsBackendApi.DevToolsBackendApi(logger, config);
21
+ const router = Router__default.default();
22
+ router.use(express__default.default.json());
23
+ router.use(
24
+ pluginPermissionNode.createPermissionIntegrationRouter({
25
+ permissions: pluginDevtoolsCommon.devToolsPermissions
26
+ })
27
+ );
28
+ router.get("/health", (_req, res) => {
29
+ res.status(200).json({ status: "ok" });
30
+ });
31
+ router.get("/info", async (req, response) => {
32
+ const decision = (await permissions.authorize(
33
+ [{ permission: pluginDevtoolsCommon.devToolsInfoReadPermission }],
34
+ { credentials: await httpAuth.credentials(req) }
35
+ ))[0];
36
+ if (decision.result === pluginPermissionCommon.AuthorizeResult.DENY) {
37
+ throw new errors.NotAllowedError("Unauthorized");
38
+ }
39
+ const info = await devToolsBackendApi.listInfo();
40
+ response.status(200).json(info);
41
+ });
42
+ router.get("/config", async (req, response) => {
43
+ const decision = (await permissions.authorize(
44
+ [{ permission: pluginDevtoolsCommon.devToolsConfigReadPermission }],
45
+ { credentials: await httpAuth.credentials(req) }
46
+ ))[0];
47
+ if (decision.result === pluginPermissionCommon.AuthorizeResult.DENY) {
48
+ throw new errors.NotAllowedError("Unauthorized");
49
+ }
50
+ const configList = await devToolsBackendApi.listConfig();
51
+ response.status(200).json(configList);
52
+ });
53
+ router.get("/external-dependencies", async (req, response) => {
54
+ const decision = (await permissions.authorize(
55
+ [{ permission: pluginDevtoolsCommon.devToolsExternalDependenciesReadPermission }],
56
+ { credentials: await httpAuth.credentials(req) }
57
+ ))[0];
58
+ if (decision.result === pluginPermissionCommon.AuthorizeResult.DENY) {
59
+ throw new errors.NotAllowedError("Unauthorized");
60
+ }
61
+ const health = await devToolsBackendApi.listExternalDependencyDetails();
62
+ response.status(200).json(health);
63
+ });
64
+ router.use(backendCommon.errorHandler());
65
+ return router;
66
+ }
67
+
68
+ exports.createRouter = createRouter;
69
+ //# sourceMappingURL=router.cjs.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"router.cjs.js","sources":["../../src/service/router.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 */\nimport { AuthorizeResult } from '@backstage/plugin-permission-common';\nimport {\n devToolsConfigReadPermission,\n devToolsExternalDependenciesReadPermission,\n devToolsInfoReadPermission,\n devToolsPermissions,\n} from '@backstage/plugin-devtools-common';\n\nimport { DevToolsBackendApi } from '../api';\nimport { NotAllowedError } from '@backstage/errors';\nimport Router from 'express-promise-router';\nimport {\n createLegacyAuthAdapters,\n errorHandler,\n} from '@backstage/backend-common';\nimport express from 'express';\nimport { createPermissionIntegrationRouter } from '@backstage/plugin-permission-node';\nimport {\n DiscoveryService,\n HttpAuthService,\n LoggerService,\n PermissionsService,\n RootConfigService,\n} from '@backstage/backend-plugin-api';\n\n/**\n * @public\n * @deprecated Please migrate to the new backend system as this will be removed in the future.\n */\nexport interface RouterOptions {\n devToolsBackendApi?: DevToolsBackendApi;\n logger: LoggerService;\n config: RootConfigService;\n permissions: PermissionsService;\n discovery: DiscoveryService;\n httpAuth?: HttpAuthService;\n}\n\n/**\n * @deprecated Please migrate to the new backend system as this will be removed in the future.\n * @public\n * */\nexport async function createRouter(\n options: RouterOptions,\n): Promise<express.Router> {\n const { logger, config, permissions } = options;\n\n const { httpAuth } = createLegacyAuthAdapters(options);\n\n const devToolsBackendApi =\n options.devToolsBackendApi || new DevToolsBackendApi(logger, config);\n\n const router = Router();\n router.use(express.json());\n router.use(\n createPermissionIntegrationRouter({\n permissions: devToolsPermissions,\n }),\n );\n\n router.get('/health', (_req, res) => {\n res.status(200).json({ status: 'ok' });\n });\n\n router.get('/info', async (req, response) => {\n const decision = (\n await permissions.authorize(\n [{ permission: devToolsInfoReadPermission }],\n { credentials: await httpAuth.credentials(req) },\n )\n )[0];\n\n if (decision.result === AuthorizeResult.DENY) {\n throw new NotAllowedError('Unauthorized');\n }\n\n const info = await devToolsBackendApi.listInfo();\n\n response.status(200).json(info);\n });\n\n router.get('/config', async (req, response) => {\n const decision = (\n await permissions.authorize(\n [{ permission: devToolsConfigReadPermission }],\n { credentials: await httpAuth.credentials(req) },\n )\n )[0];\n\n if (decision.result === AuthorizeResult.DENY) {\n throw new NotAllowedError('Unauthorized');\n }\n\n const configList = await devToolsBackendApi.listConfig();\n\n response.status(200).json(configList);\n });\n\n router.get('/external-dependencies', async (req, response) => {\n const decision = (\n await permissions.authorize(\n [{ permission: devToolsExternalDependenciesReadPermission }],\n { credentials: await httpAuth.credentials(req) },\n )\n )[0];\n\n if (decision.result === AuthorizeResult.DENY) {\n throw new NotAllowedError('Unauthorized');\n }\n\n const health = await devToolsBackendApi.listExternalDependencyDetails();\n\n response.status(200).json(health);\n });\n\n router.use(errorHandler());\n return router;\n}\n"],"names":["createLegacyAuthAdapters","DevToolsBackendApi","Router","express","createPermissionIntegrationRouter","devToolsPermissions","devToolsInfoReadPermission","AuthorizeResult","NotAllowedError","devToolsConfigReadPermission","devToolsExternalDependenciesReadPermission","errorHandler"],"mappings":"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"}
@@ -0,0 +1,90 @@
1
+ 'use strict';
2
+
3
+ var fs = require('fs-extra');
4
+ var parsers = require('@yarnpkg/parsers');
5
+ var lockfile = require('@yarnpkg/lockfile');
6
+
7
+ function _interopDefaultCompat (e) { return e && typeof e === 'object' && 'default' in e ? e : { default: e }; }
8
+
9
+ var fs__default = /*#__PURE__*/_interopDefaultCompat(fs);
10
+
11
+ const ENTRY_PATTERN = /^((?:@[^/]+\/)?[^@/]+)@(.+)$/;
12
+ function parseLockfile(lockfileContents) {
13
+ try {
14
+ return {
15
+ object: parsers.parseSyml(lockfileContents),
16
+ type: "success"
17
+ };
18
+ } catch (err) {
19
+ return {
20
+ object: null,
21
+ type: err
22
+ };
23
+ }
24
+ }
25
+ const NEW_HEADER = `${[
26
+ `# This file is generated by running "yarn install" inside your project.
27
+ `,
28
+ `# Manual changes might be lost - proceed with caution!
29
+ `
30
+ ].join(``)}
31
+ `;
32
+ function stringifyLockfile(data, legacy) {
33
+ return legacy ? lockfile.stringify(data) : NEW_HEADER + parsers.stringifySyml(data);
34
+ }
35
+ const LEGACY_REGEX = /^(#.*(\r?\n))*?#\s+yarn\s+lockfile\s+v1\r?\n/i;
36
+ const SPECIAL_OBJECT_KEYS = [
37
+ `__metadata`,
38
+ `version`,
39
+ `resolution`,
40
+ `dependencies`,
41
+ `peerDependencies`,
42
+ `dependenciesMeta`,
43
+ `peerDependenciesMeta`,
44
+ `binaries`
45
+ ];
46
+ class Lockfile {
47
+ constructor(packages, data, legacy = false) {
48
+ this.packages = packages;
49
+ this.data = data;
50
+ this.legacy = legacy;
51
+ }
52
+ static async load(path) {
53
+ const lockfileContents = await fs__default.default.readFile(path, "utf8");
54
+ const legacy = LEGACY_REGEX.test(lockfileContents);
55
+ const lockfile = parseLockfile(lockfileContents);
56
+ if (lockfile.type !== "success") {
57
+ throw new Error(`Failed yarn.lock parse with ${lockfile.type}`);
58
+ }
59
+ const data = lockfile.object;
60
+ const packages = /* @__PURE__ */ new Map();
61
+ for (const [key, value] of Object.entries(data)) {
62
+ if (SPECIAL_OBJECT_KEYS.includes(key)) continue;
63
+ const [, name, range] = ENTRY_PATTERN.exec(key) ?? [];
64
+ if (!name) {
65
+ throw new Error(`Failed to parse yarn.lock entry '${key}'`);
66
+ }
67
+ let queries = packages.get(name);
68
+ if (!queries) {
69
+ queries = [];
70
+ packages.set(name, queries);
71
+ }
72
+ queries.push({ range, version: value.version });
73
+ }
74
+ return new Lockfile(packages, data, legacy);
75
+ }
76
+ /** Get the entries for a single package in the lockfile */
77
+ get(name) {
78
+ return this.packages.get(name);
79
+ }
80
+ /** Returns the name of all packages available in the lockfile */
81
+ keys() {
82
+ return this.packages.keys();
83
+ }
84
+ toString() {
85
+ return stringifyLockfile(this.data, this.legacy);
86
+ }
87
+ }
88
+
89
+ exports.Lockfile = Lockfile;
90
+ //# sourceMappingURL=Lockfile.cjs.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"Lockfile.cjs.js","sources":["../../src/util/Lockfile.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 fs from 'fs-extra';\nimport { parseSyml, stringifySyml } from '@yarnpkg/parsers';\nimport { stringify as legacyStringifyLockfile } from '@yarnpkg/lockfile';\n\nconst ENTRY_PATTERN = /^((?:@[^/]+\\/)?[^@/]+)@(.+)$/;\n\ntype LockfileData = {\n [entry: string]: {\n version: string;\n resolved?: string;\n integrity?: string;\n dependencies?: { [name: string]: string };\n };\n};\n\ntype LockfileQueryEntry = {\n range: string;\n version: string;\n};\n\nfunction parseLockfile(lockfileContents: string) {\n try {\n return {\n object: parseSyml(lockfileContents),\n type: 'success',\n };\n } catch (err) {\n return {\n object: null,\n type: err,\n };\n }\n}\n\n// the new yarn header is handled out of band of the parsing\n// https://github.com/yarnpkg/berry/blob/0c5974f193a9397630e9aee2b3876cca62611149/packages/yarnpkg-core/sources/Project.ts#L1741-L1746\nconst NEW_HEADER = `${[\n `# This file is generated by running \"yarn install\" inside your project.\\n`,\n `# Manual changes might be lost - proceed with caution!\\n`,\n].join(``)}\\n`;\n\nfunction stringifyLockfile(data: LockfileData, legacy: boolean) {\n return legacy\n ? legacyStringifyLockfile(data)\n : NEW_HEADER + stringifySyml(data);\n}\n// taken from yarn parser package\n// https://github.com/yarnpkg/berry/blob/0c5974f193a9397630e9aee2b3876cca62611149/packages/yarnpkg-parsers/sources/syml.ts#L136\nconst LEGACY_REGEX = /^(#.*(\\r?\\n))*?#\\s+yarn\\s+lockfile\\s+v1\\r?\\n/i;\n\n// these are special top level yarn keys.\n// https://github.com/yarnpkg/berry/blob/9bd61fbffb83d0b8166a9cc26bec3a58743aa453/packages/yarnpkg-parsers/sources/syml.ts#L9\nconst SPECIAL_OBJECT_KEYS = [\n `__metadata`,\n `version`,\n `resolution`,\n `dependencies`,\n `peerDependencies`,\n `dependenciesMeta`,\n `peerDependenciesMeta`,\n `binaries`,\n];\n\nexport class Lockfile {\n static async load(path: string) {\n const lockfileContents = await fs.readFile(path, 'utf8');\n const legacy = LEGACY_REGEX.test(lockfileContents);\n const lockfile = parseLockfile(lockfileContents);\n if (lockfile.type !== 'success') {\n throw new Error(`Failed yarn.lock parse with ${lockfile.type}`);\n }\n\n const data = lockfile.object as LockfileData;\n const packages = new Map<string, LockfileQueryEntry[]>();\n\n for (const [key, value] of Object.entries(data)) {\n if (SPECIAL_OBJECT_KEYS.includes(key)) continue;\n\n const [, name, range] = ENTRY_PATTERN.exec(key) ?? [];\n if (!name) {\n throw new Error(`Failed to parse yarn.lock entry '${key}'`);\n }\n\n let queries = packages.get(name);\n if (!queries) {\n queries = [];\n packages.set(name, queries);\n }\n queries.push({ range, version: value.version });\n }\n\n return new Lockfile(packages, data, legacy);\n }\n\n private constructor(\n private readonly packages: Map<string, LockfileQueryEntry[]>,\n private readonly data: LockfileData,\n private readonly legacy: boolean = false,\n ) {}\n\n /** Get the entries for a single package in the lockfile */\n get(name: string): LockfileQueryEntry[] | undefined {\n return this.packages.get(name);\n }\n\n /** Returns the name of all packages available in the lockfile */\n keys(): IterableIterator<string> {\n return this.packages.keys();\n }\n\n toString() {\n return stringifyLockfile(this.data, this.legacy);\n }\n}\n"],"names":["parseSyml","legacyStringifyLockfile","stringifySyml","fs"],"mappings":";;;;;;;;;;AAoBA,MAAM,aAAgB,GAAA,8BAAA,CAAA;AAgBtB,SAAS,cAAc,gBAA0B,EAAA;AAC/C,EAAI,IAAA;AACF,IAAO,OAAA;AAAA,MACL,MAAA,EAAQA,kBAAU,gBAAgB,CAAA;AAAA,MAClC,IAAM,EAAA,SAAA;AAAA,KACR,CAAA;AAAA,WACO,GAAK,EAAA;AACZ,IAAO,OAAA;AAAA,MACL,MAAQ,EAAA,IAAA;AAAA,MACR,IAAM,EAAA,GAAA;AAAA,KACR,CAAA;AAAA,GACF;AACF,CAAA;AAIA,MAAM,aAAa,CAAG,EAAA;AAAA,EACpB,CAAA;AAAA,CAAA;AAAA,EACA,CAAA;AAAA,CAAA;AACF,CAAE,CAAA,IAAA,CAAK,EAAE,CAAC,CAAA;AAAA,CAAA,CAAA;AAEV,SAAS,iBAAA,CAAkB,MAAoB,MAAiB,EAAA;AAC9D,EAAA,OAAO,SACHC,kBAAwB,CAAA,IAAI,CAC5B,GAAA,UAAA,GAAaC,sBAAc,IAAI,CAAA,CAAA;AACrC,CAAA;AAGA,MAAM,YAAe,GAAA,+CAAA,CAAA;AAIrB,MAAM,mBAAsB,GAAA;AAAA,EAC1B,CAAA,UAAA,CAAA;AAAA,EACA,CAAA,OAAA,CAAA;AAAA,EACA,CAAA,UAAA,CAAA;AAAA,EACA,CAAA,YAAA,CAAA;AAAA,EACA,CAAA,gBAAA,CAAA;AAAA,EACA,CAAA,gBAAA,CAAA;AAAA,EACA,CAAA,oBAAA,CAAA;AAAA,EACA,CAAA,QAAA,CAAA;AACF,CAAA,CAAA;AAEO,MAAM,QAAS,CAAA;AAAA,EA+BZ,WACW,CAAA,QAAA,EACA,IACA,EAAA,MAAA,GAAkB,KACnC,EAAA;AAHiB,IAAA,IAAA,CAAA,QAAA,GAAA,QAAA,CAAA;AACA,IAAA,IAAA,CAAA,IAAA,GAAA,IAAA,CAAA;AACA,IAAA,IAAA,CAAA,MAAA,GAAA,MAAA,CAAA;AAAA,GAChB;AAAA,EAlCH,aAAa,KAAK,IAAc,EAAA;AAC9B,IAAA,MAAM,gBAAmB,GAAA,MAAMC,mBAAG,CAAA,QAAA,CAAS,MAAM,MAAM,CAAA,CAAA;AACvD,IAAM,MAAA,MAAA,GAAS,YAAa,CAAA,IAAA,CAAK,gBAAgB,CAAA,CAAA;AACjD,IAAM,MAAA,QAAA,GAAW,cAAc,gBAAgB,CAAA,CAAA;AAC/C,IAAI,IAAA,QAAA,CAAS,SAAS,SAAW,EAAA;AAC/B,MAAA,MAAM,IAAI,KAAA,CAAM,CAA+B,4BAAA,EAAA,QAAA,CAAS,IAAI,CAAE,CAAA,CAAA,CAAA;AAAA,KAChE;AAEA,IAAA,MAAM,OAAO,QAAS,CAAA,MAAA,CAAA;AACtB,IAAM,MAAA,QAAA,uBAAe,GAAkC,EAAA,CAAA;AAEvD,IAAA,KAAA,MAAW,CAAC,GAAK,EAAA,KAAK,KAAK,MAAO,CAAA,OAAA,CAAQ,IAAI,CAAG,EAAA;AAC/C,MAAI,IAAA,mBAAA,CAAoB,QAAS,CAAA,GAAG,CAAG,EAAA,SAAA;AAEvC,MAAM,MAAA,GAAG,IAAM,EAAA,KAAK,IAAI,aAAc,CAAA,IAAA,CAAK,GAAG,CAAA,IAAK,EAAC,CAAA;AACpD,MAAA,IAAI,CAAC,IAAM,EAAA;AACT,QAAA,MAAM,IAAI,KAAA,CAAM,CAAoC,iCAAA,EAAA,GAAG,CAAG,CAAA,CAAA,CAAA,CAAA;AAAA,OAC5D;AAEA,MAAI,IAAA,OAAA,GAAU,QAAS,CAAA,GAAA,CAAI,IAAI,CAAA,CAAA;AAC/B,MAAA,IAAI,CAAC,OAAS,EAAA;AACZ,QAAA,OAAA,GAAU,EAAC,CAAA;AACX,QAAS,QAAA,CAAA,GAAA,CAAI,MAAM,OAAO,CAAA,CAAA;AAAA,OAC5B;AACA,MAAA,OAAA,CAAQ,KAAK,EAAE,KAAA,EAAO,OAAS,EAAA,KAAA,CAAM,SAAS,CAAA,CAAA;AAAA,KAChD;AAEA,IAAA,OAAO,IAAI,QAAA,CAAS,QAAU,EAAA,IAAA,EAAM,MAAM,CAAA,CAAA;AAAA,GAC5C;AAAA;AAAA,EASA,IAAI,IAAgD,EAAA;AAClD,IAAO,OAAA,IAAA,CAAK,QAAS,CAAA,GAAA,CAAI,IAAI,CAAA,CAAA;AAAA,GAC/B;AAAA;AAAA,EAGA,IAAiC,GAAA;AAC/B,IAAO,OAAA,IAAA,CAAK,SAAS,IAAK,EAAA,CAAA;AAAA,GAC5B;AAAA,EAEA,QAAW,GAAA;AACT,IAAA,OAAO,iBAAkB,CAAA,IAAA,CAAK,IAAM,EAAA,IAAA,CAAK,MAAM,CAAA,CAAA;AAAA,GACjD;AACF;;;;"}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@backstage/plugin-devtools-backend",
3
- "version": "0.4.1-next.0",
3
+ "version": "0.4.1",
4
4
  "backstage": {
5
5
  "role": "backend-plugin",
6
6
  "pluginId": "devtools",
@@ -39,14 +39,14 @@
39
39
  },
40
40
  "dependencies": {
41
41
  "@backstage/backend-common": "^0.25.0",
42
- "@backstage/backend-plugin-api": "^1.0.1-next.0",
42
+ "@backstage/backend-plugin-api": "^1.0.1",
43
43
  "@backstage/cli-common": "^0.1.14",
44
44
  "@backstage/config": "^1.2.0",
45
45
  "@backstage/config-loader": "^1.9.1",
46
46
  "@backstage/errors": "^1.2.4",
47
47
  "@backstage/plugin-devtools-common": "^0.1.12",
48
48
  "@backstage/plugin-permission-common": "^0.8.1",
49
- "@backstage/plugin-permission-node": "^0.8.4-next.0",
49
+ "@backstage/plugin-permission-node": "^0.8.4",
50
50
  "@backstage/types": "^1.1.1",
51
51
  "@manypkg/get-packages": "^1.1.3",
52
52
  "@types/express": "*",
@@ -62,9 +62,9 @@
62
62
  "yn": "^4.0.0"
63
63
  },
64
64
  "devDependencies": {
65
- "@backstage/backend-defaults": "^0.5.1-next.0",
66
- "@backstage/backend-test-utils": "^1.0.1-next.0",
67
- "@backstage/cli": "^0.28.0-next.0",
65
+ "@backstage/backend-defaults": "^0.5.1",
66
+ "@backstage/backend-test-utils": "^1.0.1",
67
+ "@backstage/cli": "^0.28.0",
68
68
  "@types/ping": "^0.4.1",
69
69
  "@types/supertest": "^2.0.8",
70
70
  "@types/yarnpkg__lockfile": "^1.1.4",