@backstage/cli-node 0.0.0-nightly-20230412021859

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 ADDED
@@ -0,0 +1,27 @@
1
+ # @backstage/cli-node
2
+
3
+ ## 0.0.0-nightly-20230412021859
4
+
5
+ ### Minor Changes
6
+
7
+ - 3156b0d85dc: Introduced the new `@backstage/cli-node` package, which provides utilities for use across Backstage CLIs.
8
+
9
+ ### Patch Changes
10
+
11
+ - Updated dependencies
12
+ - @backstage/cli-common@0.1.12
13
+ - @backstage/errors@1.1.5
14
+ - @backstage/types@1.0.2
15
+
16
+ ## 0.1.0-next.0
17
+
18
+ ### Minor Changes
19
+
20
+ - 3156b0d85dc: Introduced the new `@backstage/cli-node` package, which provides utilities for use across Backstage CLIs.
21
+
22
+ ### Patch Changes
23
+
24
+ - Updated dependencies
25
+ - @backstage/cli-common@0.1.12
26
+ - @backstage/errors@1.1.5
27
+ - @backstage/types@1.0.2
package/README.md ADDED
@@ -0,0 +1,10 @@
1
+ # @backstage/cli-node
2
+
3
+ This library provides utilities for building CLI tools for Backstage.
4
+
5
+ The difference between this library and `@backstage/cli-common` is that this library is more feature rich with a larger dependency tree, with less concern for bundle size and installation speed. The `@backstage/cli-common` package on the other hand is intended to be extremely slim and only provide minimal features for use in tools like `@backstage/create-app`.
6
+
7
+ ## Documentation
8
+
9
+ - [Backstage Readme](https://github.com/backstage/backstage/blob/master/README.md)
10
+ - [Backstage Documentation](https://backstage.io/docs)
@@ -0,0 +1,572 @@
1
+ 'use strict';
2
+
3
+ Object.defineProperty(exports, '__esModule', { value: true });
4
+
5
+ var errors = require('@backstage/errors');
6
+ var cliCommon = require('@backstage/cli-common');
7
+ var child_process = require('child_process');
8
+ var util = require('util');
9
+ var fs = require('fs-extra');
10
+ var path = require('path');
11
+ var getPackages = require('@manypkg/get-packages');
12
+ var parsers = require('@yarnpkg/parsers');
13
+ var zod = require('zod');
14
+
15
+ function _interopDefaultLegacy (e) { return e && typeof e === 'object' && 'default' in e ? e : { 'default': e }; }
16
+
17
+ var fs__default = /*#__PURE__*/_interopDefaultLegacy(fs);
18
+ var path__default = /*#__PURE__*/_interopDefaultLegacy(path);
19
+
20
+ const paths = cliCommon.findPaths(__dirname);
21
+
22
+ const execFile = util.promisify(child_process.execFile);
23
+
24
+ async function runGit(...args) {
25
+ var _a, _b;
26
+ try {
27
+ const { stdout } = await execFile("git", args, {
28
+ shell: true,
29
+ cwd: paths.targetRoot
30
+ });
31
+ return stdout.trim().split(/\r\n|\r|\n/);
32
+ } catch (error) {
33
+ errors.assertError(error);
34
+ if (error.stderr || typeof error.code === "number") {
35
+ const stderr = (_a = error.stderr) == null ? void 0 : _a.toString("utf8");
36
+ const msg = (_b = stderr == null ? void 0 : stderr.trim()) != null ? _b : `with exit code ${error.code}`;
37
+ throw new Error(`git ${args[0]} failed, ${msg}`);
38
+ }
39
+ throw new errors.ForwardedError("Unknown execution error", error);
40
+ }
41
+ }
42
+ class GitUtils {
43
+ /**
44
+ * Returns a sorted list of all files that have changed since the merge base
45
+ * of the provided `ref` and HEAD, as well as all files that are not tracked by git.
46
+ */
47
+ static async listChangedFiles(ref) {
48
+ if (!ref) {
49
+ throw new Error("ref is required");
50
+ }
51
+ let diffRef = ref;
52
+ try {
53
+ const [base] = await runGit("merge-base", "HEAD", ref);
54
+ diffRef = base;
55
+ } catch {
56
+ }
57
+ const tracked = await runGit("diff", "--name-only", diffRef);
58
+ const untracked = await runGit(
59
+ "ls-files",
60
+ "--others",
61
+ "--exclude-standard"
62
+ );
63
+ return Array.from(/* @__PURE__ */ new Set([...tracked, ...untracked]));
64
+ }
65
+ /**
66
+ * Returns the contents of a file at a specific ref.
67
+ */
68
+ static async readFileAtRef(path, ref) {
69
+ let showRef = ref;
70
+ try {
71
+ const [base] = await runGit("merge-base", "HEAD", ref);
72
+ showRef = base;
73
+ } catch {
74
+ }
75
+ const { stdout } = await execFile("git", ["show", `${showRef}:${path}`], {
76
+ shell: true,
77
+ cwd: paths.targetRoot,
78
+ maxBuffer: 1024 * 1024 * 50
79
+ });
80
+ return stdout;
81
+ }
82
+ }
83
+
84
+ async function isMonoRepo() {
85
+ var _a;
86
+ const rootPackageJsonPath = paths.resolveTargetRoot("package.json");
87
+ try {
88
+ const pkg = await fs__default["default"].readJson(rootPackageJsonPath);
89
+ return Boolean((_a = pkg == null ? void 0 : pkg.workspaces) == null ? void 0 : _a.packages);
90
+ } catch (error) {
91
+ return false;
92
+ }
93
+ }
94
+
95
+ const ENTRY_PATTERN = /^((?:@[^/]+\/)?[^@/]+)@(.+)$/;
96
+ const SPECIAL_OBJECT_KEYS = [
97
+ `__metadata`,
98
+ `version`,
99
+ `resolution`,
100
+ `dependencies`,
101
+ `peerDependencies`,
102
+ `dependenciesMeta`,
103
+ `peerDependenciesMeta`,
104
+ `binaries`
105
+ ];
106
+ class Lockfile {
107
+ constructor(packages, data) {
108
+ this.packages = packages;
109
+ this.data = data;
110
+ }
111
+ /**
112
+ * Load a {@link Lockfile} from a file path.
113
+ */
114
+ static async load(path) {
115
+ const lockfileContents = await fs__default["default"].readFile(path, "utf8");
116
+ return Lockfile.parse(lockfileContents);
117
+ }
118
+ /**
119
+ * Parse lockfile contents into a {@link Lockfile}.
120
+ *
121
+ * @public
122
+ */
123
+ static parse(content) {
124
+ var _a;
125
+ let data;
126
+ try {
127
+ data = parsers.parseSyml(content);
128
+ } catch (err) {
129
+ throw new Error(`Failed yarn.lock parse, ${err}`);
130
+ }
131
+ const packages = /* @__PURE__ */ new Map();
132
+ for (const [key, value] of Object.entries(data)) {
133
+ if (SPECIAL_OBJECT_KEYS.includes(key))
134
+ continue;
135
+ const [, name, ranges] = (_a = ENTRY_PATTERN.exec(key)) != null ? _a : [];
136
+ if (!name) {
137
+ throw new Error(`Failed to parse yarn.lock entry '${key}'`);
138
+ }
139
+ let queries = packages.get(name);
140
+ if (!queries) {
141
+ queries = [];
142
+ packages.set(name, queries);
143
+ }
144
+ for (let range of ranges.split(/\s*,\s*/)) {
145
+ if (range.startsWith(`${name}@`)) {
146
+ range = range.slice(`${name}@`.length);
147
+ }
148
+ if (range.startsWith("npm:")) {
149
+ range = range.slice("npm:".length);
150
+ }
151
+ queries.push({ range, version: value.version, dataKey: key });
152
+ }
153
+ }
154
+ return new Lockfile(packages, data);
155
+ }
156
+ /**
157
+ * Creates a simplified dependency graph from the lockfile data, where each
158
+ * key is a package, and the value is a set of all packages that it depends on
159
+ * across all versions.
160
+ */
161
+ createSimplifiedDependencyGraph() {
162
+ const graph = /* @__PURE__ */ new Map();
163
+ for (const [name, entries] of this.packages) {
164
+ const dependencies = new Set(
165
+ entries.flatMap((e) => {
166
+ var _a, _b;
167
+ const data = this.data[e.dataKey];
168
+ return [
169
+ ...Object.keys((_a = data == null ? void 0 : data.dependencies) != null ? _a : {}),
170
+ ...Object.keys((_b = data == null ? void 0 : data.peerDependencies) != null ? _b : {})
171
+ ];
172
+ })
173
+ );
174
+ graph.set(name, dependencies);
175
+ }
176
+ return graph;
177
+ }
178
+ /**
179
+ * Diff with another lockfile, returning entries that have been
180
+ * added, changed, and removed compared to the other lockfile.
181
+ */
182
+ diff(otherLockfile) {
183
+ var _a;
184
+ const diff = {
185
+ added: new Array(),
186
+ changed: new Array(),
187
+ removed: new Array()
188
+ };
189
+ const remainingOldNames = new Set(this.packages.keys());
190
+ for (const [name, otherQueries] of otherLockfile.packages) {
191
+ remainingOldNames.delete(name);
192
+ const thisQueries = this.packages.get(name);
193
+ if (!thisQueries) {
194
+ diff.removed.push(...otherQueries.map((q) => ({ name, range: q.range })));
195
+ continue;
196
+ }
197
+ const remainingOldRanges = new Set(thisQueries.map((q) => q.range));
198
+ for (const otherQuery of otherQueries) {
199
+ remainingOldRanges.delete(otherQuery.range);
200
+ const thisQuery = thisQueries.find((q) => q.range === otherQuery.range);
201
+ if (!thisQuery) {
202
+ diff.removed.push({ name, range: otherQuery.range });
203
+ continue;
204
+ }
205
+ const otherPkg = otherLockfile.data[otherQuery.dataKey];
206
+ const thisPkg = this.data[thisQuery.dataKey];
207
+ if (otherPkg && thisPkg) {
208
+ const thisCheck = thisPkg.integrity || thisPkg.checksum;
209
+ const otherCheck = otherPkg.integrity || otherPkg.checksum;
210
+ if (thisCheck !== otherCheck) {
211
+ diff.changed.push({ name, range: otherQuery.range });
212
+ }
213
+ }
214
+ }
215
+ for (const thisRange of remainingOldRanges) {
216
+ diff.added.push({ name, range: thisRange });
217
+ }
218
+ }
219
+ for (const name of remainingOldNames) {
220
+ const queries = (_a = this.packages.get(name)) != null ? _a : [];
221
+ diff.added.push(...queries.map((q) => ({ name, range: q.range })));
222
+ }
223
+ return diff;
224
+ }
225
+ }
226
+
227
+ class PackageGraph extends Map {
228
+ /**
229
+ * Lists all local packages in a monorepo.
230
+ */
231
+ static async listTargetPackages() {
232
+ const { packages } = await getPackages.getPackages(paths.targetDir);
233
+ return packages;
234
+ }
235
+ /**
236
+ * Creates a package graph from a list of local packages.
237
+ */
238
+ static fromPackages(packages) {
239
+ const graph = new PackageGraph();
240
+ for (const pkg of packages) {
241
+ const name = pkg.packageJson.name;
242
+ const existingPkg = graph.get(name);
243
+ if (existingPkg) {
244
+ throw new Error(
245
+ `Duplicate package name '${name}' at ${pkg.dir} and ${existingPkg.dir}`
246
+ );
247
+ }
248
+ graph.set(name, {
249
+ name,
250
+ dir: pkg.dir,
251
+ packageJson: pkg.packageJson,
252
+ allLocalDependencies: /* @__PURE__ */ new Map(),
253
+ publishedLocalDependencies: /* @__PURE__ */ new Map(),
254
+ localDependencies: /* @__PURE__ */ new Map(),
255
+ localDevDependencies: /* @__PURE__ */ new Map(),
256
+ localOptionalDependencies: /* @__PURE__ */ new Map(),
257
+ allLocalDependents: /* @__PURE__ */ new Map(),
258
+ publishedLocalDependents: /* @__PURE__ */ new Map(),
259
+ localDependents: /* @__PURE__ */ new Map(),
260
+ localDevDependents: /* @__PURE__ */ new Map(),
261
+ localOptionalDependents: /* @__PURE__ */ new Map()
262
+ });
263
+ }
264
+ for (const node of graph.values()) {
265
+ for (const depName of Object.keys(node.packageJson.dependencies || {})) {
266
+ const depPkg = graph.get(depName);
267
+ if (depPkg) {
268
+ node.allLocalDependencies.set(depName, depPkg);
269
+ node.publishedLocalDependencies.set(depName, depPkg);
270
+ node.localDependencies.set(depName, depPkg);
271
+ depPkg.allLocalDependents.set(node.name, node);
272
+ depPkg.publishedLocalDependents.set(node.name, node);
273
+ depPkg.localDependents.set(node.name, node);
274
+ }
275
+ }
276
+ for (const depName of Object.keys(
277
+ node.packageJson.devDependencies || {}
278
+ )) {
279
+ const depPkg = graph.get(depName);
280
+ if (depPkg) {
281
+ node.allLocalDependencies.set(depName, depPkg);
282
+ node.localDevDependencies.set(depName, depPkg);
283
+ depPkg.allLocalDependents.set(node.name, node);
284
+ depPkg.localDevDependents.set(node.name, node);
285
+ }
286
+ }
287
+ for (const depName of Object.keys(
288
+ node.packageJson.optionalDependencies || {}
289
+ )) {
290
+ const depPkg = graph.get(depName);
291
+ if (depPkg) {
292
+ node.allLocalDependencies.set(depName, depPkg);
293
+ node.publishedLocalDependencies.set(depName, depPkg);
294
+ node.localOptionalDependencies.set(depName, depPkg);
295
+ depPkg.allLocalDependents.set(node.name, node);
296
+ depPkg.publishedLocalDependents.set(node.name, node);
297
+ depPkg.localOptionalDependents.set(node.name, node);
298
+ }
299
+ }
300
+ }
301
+ return graph;
302
+ }
303
+ /**
304
+ * Traverses the package graph and collects a set of package names.
305
+ *
306
+ * The traversal starts at the provided list names, and continues
307
+ * throughout all the names returned by the `collectFn`, which is
308
+ * called once for each seen package.
309
+ */
310
+ collectPackageNames(startingPackageNames, collectFn) {
311
+ const targets = /* @__PURE__ */ new Set();
312
+ const searchNames = startingPackageNames.slice();
313
+ while (searchNames.length) {
314
+ const name = searchNames.pop();
315
+ if (targets.has(name)) {
316
+ continue;
317
+ }
318
+ const node = this.get(name);
319
+ if (!node) {
320
+ throw new Error(`Package '${name}' not found`);
321
+ }
322
+ targets.add(name);
323
+ const collected = collectFn(node);
324
+ if (collected) {
325
+ searchNames.push(...collected);
326
+ }
327
+ }
328
+ return targets;
329
+ }
330
+ /**
331
+ * Lists all packages that have changed since a given git ref.
332
+ *
333
+ * @remarks
334
+ *
335
+ * If the `analyzeLockfile` option is set to true, the change detection will
336
+ * also consider changes to the dependency management lockfile.
337
+ */
338
+ async listChangedPackages(options) {
339
+ var _a, _b;
340
+ const changedFiles = await GitUtils.listChangedFiles(options.ref);
341
+ const dirMap = new Map(
342
+ Array.from(this.values()).map((pkg) => [
343
+ // relative from root, convert to posix, and add a / at the end
344
+ path__default["default"].relative(paths.targetRoot, pkg.dir).split(path__default["default"].sep).join(path__default["default"].posix.sep) + path__default["default"].posix.sep,
345
+ pkg
346
+ ])
347
+ );
348
+ const packageDirs = Array.from(dirMap.keys());
349
+ const result = new Array();
350
+ let searchIndex = 0;
351
+ changedFiles.sort();
352
+ packageDirs.sort();
353
+ for (const packageDir of packageDirs) {
354
+ while (searchIndex < changedFiles.length && changedFiles[searchIndex] < packageDir) {
355
+ searchIndex += 1;
356
+ }
357
+ if ((_a = changedFiles[searchIndex]) == null ? void 0 : _a.startsWith(packageDir)) {
358
+ searchIndex += 1;
359
+ result.push(dirMap.get(packageDir));
360
+ while ((_b = changedFiles[searchIndex]) == null ? void 0 : _b.startsWith(packageDir)) {
361
+ searchIndex += 1;
362
+ }
363
+ }
364
+ }
365
+ if (changedFiles.includes("yarn.lock") && options.analyzeLockfile) {
366
+ let thisLockfile;
367
+ let otherLockfile;
368
+ try {
369
+ thisLockfile = await Lockfile.load(
370
+ paths.resolveTargetRoot("yarn.lock")
371
+ );
372
+ otherLockfile = Lockfile.parse(
373
+ await GitUtils.readFileAtRef("yarn.lock", options.ref)
374
+ );
375
+ } catch (error) {
376
+ console.warn(
377
+ `Failed to read lockfiles, assuming all packages have changed, ${error}`
378
+ );
379
+ return Array.from(this.values());
380
+ }
381
+ const diff = thisLockfile.diff(otherLockfile);
382
+ const graph = thisLockfile.createSimplifiedDependencyGraph();
383
+ {
384
+ const otherGraph = thisLockfile.createSimplifiedDependencyGraph();
385
+ for (const [name, dependencies] of otherGraph) {
386
+ const node = graph.get(name);
387
+ if (node) {
388
+ dependencies.forEach((d) => node.add(d));
389
+ } else {
390
+ graph.set(name, dependencies);
391
+ }
392
+ }
393
+ }
394
+ const changedPackages = new Set(
395
+ [...diff.added, ...diff.changed, ...diff.removed].map((e) => e.name)
396
+ );
397
+ let changed = false;
398
+ do {
399
+ changed = false;
400
+ for (const [name, dependencies] of graph) {
401
+ if (changedPackages.has(name)) {
402
+ continue;
403
+ }
404
+ for (const dep of dependencies) {
405
+ if (changedPackages.has(dep)) {
406
+ changed = true;
407
+ changedPackages.add(name);
408
+ break;
409
+ }
410
+ }
411
+ }
412
+ } while (changed);
413
+ for (const node of this.values()) {
414
+ if (changedPackages.has(node.name) && !result.includes(node)) {
415
+ result.push(node);
416
+ }
417
+ }
418
+ }
419
+ return result;
420
+ }
421
+ }
422
+
423
+ const packageRoleInfos = [
424
+ {
425
+ role: "frontend",
426
+ platform: "web",
427
+ output: ["bundle"]
428
+ },
429
+ {
430
+ role: "backend",
431
+ platform: "node",
432
+ output: ["bundle"]
433
+ },
434
+ {
435
+ role: "cli",
436
+ platform: "node",
437
+ output: ["cjs"]
438
+ },
439
+ {
440
+ role: "web-library",
441
+ platform: "web",
442
+ output: ["types", "esm"]
443
+ },
444
+ {
445
+ role: "node-library",
446
+ platform: "node",
447
+ output: ["types", "cjs"]
448
+ },
449
+ {
450
+ role: "common-library",
451
+ platform: "common",
452
+ output: ["types", "esm", "cjs"]
453
+ },
454
+ {
455
+ role: "frontend-plugin",
456
+ platform: "web",
457
+ output: ["types", "esm"]
458
+ },
459
+ {
460
+ role: "frontend-plugin-module",
461
+ platform: "web",
462
+ output: ["types", "esm"]
463
+ },
464
+ {
465
+ role: "backend-plugin",
466
+ platform: "node",
467
+ output: ["types", "cjs"]
468
+ },
469
+ {
470
+ role: "backend-plugin-module",
471
+ platform: "node",
472
+ output: ["types", "cjs"]
473
+ }
474
+ ];
475
+ const readSchema = zod.z.object({
476
+ name: zod.z.string().optional(),
477
+ backstage: zod.z.object({
478
+ role: zod.z.string().optional()
479
+ }).optional()
480
+ });
481
+ const detectionSchema = zod.z.object({
482
+ name: zod.z.string().optional(),
483
+ scripts: zod.z.object({
484
+ start: zod.z.string().optional(),
485
+ build: zod.z.string().optional()
486
+ }).optional(),
487
+ publishConfig: zod.z.object({
488
+ main: zod.z.string().optional(),
489
+ types: zod.z.string().optional(),
490
+ module: zod.z.string().optional()
491
+ }).optional(),
492
+ main: zod.z.string().optional(),
493
+ types: zod.z.string().optional(),
494
+ module: zod.z.string().optional()
495
+ });
496
+ class PackageRoles {
497
+ /**
498
+ * Get the associated info for a package role.
499
+ */
500
+ static getRoleInfo(role) {
501
+ const roleInfo = packageRoleInfos.find((r) => r.role === role);
502
+ if (!roleInfo) {
503
+ throw new Error(`Unknown package role '${role}'`);
504
+ }
505
+ return roleInfo;
506
+ }
507
+ /**
508
+ * Given package JSON data, get the package role.
509
+ */
510
+ static getRoleFromPackage(pkgJson) {
511
+ const pkg = readSchema.parse(pkgJson);
512
+ if (pkg.backstage) {
513
+ const { role } = pkg.backstage;
514
+ if (!role) {
515
+ throw new Error(
516
+ `Package ${pkg.name} must specify a role in the "backstage" field`
517
+ );
518
+ }
519
+ return this.getRoleInfo(role).role;
520
+ }
521
+ return void 0;
522
+ }
523
+ /**
524
+ * Attempt to detect the role of a package from its package.json.
525
+ */
526
+ static detectRoleFromPackage(pkgJson) {
527
+ var _a, _b, _c, _d, _e, _f, _g, _h, _i, _j, _k, _l, _m, _n, _o;
528
+ const pkg = detectionSchema.parse(pkgJson);
529
+ if ((_b = (_a = pkg.scripts) == null ? void 0 : _a.start) == null ? void 0 : _b.includes("app:serve")) {
530
+ return "frontend";
531
+ }
532
+ if ((_d = (_c = pkg.scripts) == null ? void 0 : _c.build) == null ? void 0 : _d.includes("backend:bundle")) {
533
+ return "backend";
534
+ }
535
+ if (((_e = pkg.name) == null ? void 0 : _e.includes("plugin-")) && ((_f = pkg.name) == null ? void 0 : _f.includes("-backend-module-"))) {
536
+ return "backend-plugin-module";
537
+ }
538
+ if (((_g = pkg.name) == null ? void 0 : _g.includes("plugin-")) && ((_h = pkg.name) == null ? void 0 : _h.includes("-module-"))) {
539
+ return "frontend-plugin-module";
540
+ }
541
+ if ((_j = (_i = pkg.scripts) == null ? void 0 : _i.start) == null ? void 0 : _j.includes("plugin:serve")) {
542
+ return "frontend-plugin";
543
+ }
544
+ if ((_l = (_k = pkg.scripts) == null ? void 0 : _k.start) == null ? void 0 : _l.includes("backend:dev")) {
545
+ return "backend-plugin";
546
+ }
547
+ const mainEntry = ((_m = pkg.publishConfig) == null ? void 0 : _m.main) || pkg.main;
548
+ const moduleEntry = ((_n = pkg.publishConfig) == null ? void 0 : _n.module) || pkg.module;
549
+ const typesEntry = ((_o = pkg.publishConfig) == null ? void 0 : _o.types) || pkg.types;
550
+ if (typesEntry) {
551
+ if (mainEntry && moduleEntry) {
552
+ return "common-library";
553
+ }
554
+ if (moduleEntry || (mainEntry == null ? void 0 : mainEntry.endsWith(".esm.js"))) {
555
+ return "web-library";
556
+ }
557
+ if (mainEntry) {
558
+ return "node-library";
559
+ }
560
+ } else if (mainEntry) {
561
+ return "cli";
562
+ }
563
+ return void 0;
564
+ }
565
+ }
566
+
567
+ exports.GitUtils = GitUtils;
568
+ exports.Lockfile = Lockfile;
569
+ exports.PackageGraph = PackageGraph;
570
+ exports.PackageRoles = PackageRoles;
571
+ exports.isMonoRepo = isMonoRepo;
572
+ //# sourceMappingURL=index.cjs.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.cjs.js","sources":["../src/paths.ts","../src/run.ts","../src/git/GitUtils.ts","../src/monorepo/isMonoRepo.ts","../src/monorepo/Lockfile.ts","../src/monorepo/PackageGraph.ts","../src/roles/PackageRoles.ts"],"sourcesContent":["/*\n * Copyright 2020 The Backstage Authors\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nimport { findPaths } from '@backstage/cli-common';\n\n/* eslint-disable-next-line no-restricted-syntax */\nexport const paths = findPaths(__dirname);\n","/*\n * Copyright 2020 The Backstage Authors\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nimport { execFile as execFileCb } from 'child_process';\nimport { promisify } from 'util';\n\nexport const execFile = promisify(execFileCb);\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 { assertError, ForwardedError } from '@backstage/errors';\nimport { paths } from '../paths';\nimport { execFile } from '../run';\n\n/**\n * Run a git command, trimming the output splitting it into lines.\n */\nexport async function runGit(...args: string[]) {\n try {\n const { stdout } = await execFile('git', args, {\n shell: true,\n cwd: paths.targetRoot,\n });\n return stdout.trim().split(/\\r\\n|\\r|\\n/);\n } catch (error) {\n assertError(error);\n if (error.stderr || typeof error.code === 'number') {\n const stderr = (error.stderr as undefined | Buffer)?.toString('utf8');\n const msg = stderr?.trim() ?? `with exit code ${error.code}`;\n throw new Error(`git ${args[0]} failed, ${msg}`);\n }\n throw new ForwardedError('Unknown execution error', error);\n }\n}\n\n/**\n * Utilities for working with git.\n *\n * @public\n */\nexport class GitUtils {\n /**\n * Returns a sorted list of all files that have changed since the merge base\n * of the provided `ref` and HEAD, as well as all files that are not tracked by git.\n */\n static async listChangedFiles(ref: string) {\n if (!ref) {\n throw new Error('ref is required');\n }\n\n let diffRef = ref;\n try {\n const [base] = await runGit('merge-base', 'HEAD', ref);\n diffRef = base;\n } catch {\n // silently fall back to using the ref directly if merge base is not available\n }\n\n const tracked = await runGit('diff', '--name-only', diffRef);\n const untracked = await runGit(\n 'ls-files',\n '--others',\n '--exclude-standard',\n );\n\n return Array.from(new Set([...tracked, ...untracked]));\n }\n\n /**\n * Returns the contents of a file at a specific ref.\n */\n static async readFileAtRef(path: string, ref: string) {\n let showRef = ref;\n try {\n const [base] = await runGit('merge-base', 'HEAD', ref);\n showRef = base;\n } catch {\n // silently fall back to using the ref directly if merge base is not available\n }\n\n const { stdout } = await execFile('git', ['show', `${showRef}:${path}`], {\n shell: true,\n cwd: paths.targetRoot,\n maxBuffer: 1024 * 1024 * 50,\n });\n return stdout;\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 { paths } from '../paths';\nimport fs from 'fs-extra';\n\n/**\n * Returns try if the current project is a monorepo.\n *\n * @public\n */\nexport async function isMonoRepo(): Promise<boolean> {\n const rootPackageJsonPath = paths.resolveTargetRoot('package.json');\n try {\n const pkg = await fs.readJson(rootPackageJsonPath);\n return Boolean(pkg?.workspaces?.packages);\n } catch (error) {\n return false;\n }\n}\n","/*\n * Copyright 2020 The Backstage Authors\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nimport { parseSyml } from '@yarnpkg/parsers';\nimport fs from 'fs-extra';\n\nconst ENTRY_PATTERN = /^((?:@[^/]+\\/)?[^@/]+)@(.+)$/;\n\n/** @internal */\ntype LockfileData = {\n [entry: string]: {\n version: string;\n resolved?: string;\n integrity?: string /* old */;\n checksum?: string /* new */;\n dependencies?: { [name: string]: string };\n peerDependencies?: { [name: string]: string };\n };\n};\n\n/** @internal */\ntype LockfileQueryEntry = {\n range: string;\n version: string;\n dataKey: string;\n};\n\n/**\n * An entry for a single difference between two {@link Lockfile}s.\n *\n * @public\n */\nexport type LockfileDiffEntry = {\n name: string;\n range: string;\n};\n\n/**\n * Represents the difference between two {@link Lockfile}s.\n *\n * @public\n */\nexport type LockfileDiff = {\n added: LockfileDiffEntry[];\n changed: LockfileDiffEntry[];\n removed: LockfileDiffEntry[];\n};\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\n/**\n * Represents a package manager lockfile.\n *\n * @public\n */\nexport class Lockfile {\n /**\n * Load a {@link Lockfile} from a file path.\n */\n static async load(path: string): Promise<Lockfile> {\n const lockfileContents = await fs.readFile(path, 'utf8');\n return Lockfile.parse(lockfileContents);\n }\n\n /**\n * Parse lockfile contents into a {@link Lockfile}.\n *\n * @public\n */\n static parse(content: string): Lockfile {\n let data: LockfileData;\n try {\n data = parseSyml(content);\n } catch (err) {\n throw new Error(`Failed yarn.lock parse, ${err}`);\n }\n\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, ranges] = 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 for (let range of ranges.split(/\\s*,\\s*/)) {\n if (range.startsWith(`${name}@`)) {\n range = range.slice(`${name}@`.length);\n }\n if (range.startsWith('npm:')) {\n range = range.slice('npm:'.length);\n }\n queries.push({ range, version: value.version, dataKey: key });\n }\n }\n\n return new Lockfile(packages, data);\n }\n\n private constructor(\n private readonly packages: Map<string, LockfileQueryEntry[]>,\n private readonly data: LockfileData,\n ) {}\n\n /**\n * Creates a simplified dependency graph from the lockfile data, where each\n * key is a package, and the value is a set of all packages that it depends on\n * across all versions.\n */\n createSimplifiedDependencyGraph(): Map<string, Set<string>> {\n const graph = new Map<string, Set<string>>();\n\n for (const [name, entries] of this.packages) {\n const dependencies = new Set(\n entries.flatMap(e => {\n const data = this.data[e.dataKey];\n return [\n ...Object.keys(data?.dependencies ?? {}),\n ...Object.keys(data?.peerDependencies ?? {}),\n ];\n }),\n );\n graph.set(name, dependencies);\n }\n\n return graph;\n }\n\n /**\n * Diff with another lockfile, returning entries that have been\n * added, changed, and removed compared to the other lockfile.\n */\n diff(otherLockfile: Lockfile): LockfileDiff {\n const diff = {\n added: new Array<{ name: string; range: string }>(),\n changed: new Array<{ name: string; range: string }>(),\n removed: new Array<{ name: string; range: string }>(),\n };\n\n // Keeps track of packages that only exist in this lockfile\n const remainingOldNames = new Set(this.packages.keys());\n\n for (const [name, otherQueries] of otherLockfile.packages) {\n remainingOldNames.delete(name);\n\n const thisQueries = this.packages.get(name);\n // If the packages doesn't exist in this lockfile, add all entries\n if (!thisQueries) {\n diff.removed.push(...otherQueries.map(q => ({ name, range: q.range })));\n continue;\n }\n\n const remainingOldRanges = new Set(thisQueries.map(q => q.range));\n\n for (const otherQuery of otherQueries) {\n remainingOldRanges.delete(otherQuery.range);\n\n const thisQuery = thisQueries.find(q => q.range === otherQuery.range);\n if (!thisQuery) {\n diff.removed.push({ name, range: otherQuery.range });\n continue;\n }\n\n const otherPkg = otherLockfile.data[otherQuery.dataKey];\n const thisPkg = this.data[thisQuery.dataKey];\n if (otherPkg && thisPkg) {\n const thisCheck = thisPkg.integrity || thisPkg.checksum;\n const otherCheck = otherPkg.integrity || otherPkg.checksum;\n if (thisCheck !== otherCheck) {\n diff.changed.push({ name, range: otherQuery.range });\n }\n }\n }\n\n for (const thisRange of remainingOldRanges) {\n diff.added.push({ name, range: thisRange });\n }\n }\n\n for (const name of remainingOldNames) {\n const queries = this.packages.get(name) ?? [];\n diff.added.push(...queries.map(q => ({ name, range: q.range })));\n }\n\n return diff;\n }\n}\n","/*\n * Copyright 2020 The Backstage Authors\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nimport path from 'path';\nimport { getPackages, Package } from '@manypkg/get-packages';\nimport { paths } from '../paths';\nimport { PackageRole } from '../roles';\nimport { GitUtils } from '../git';\nimport { Lockfile } from './Lockfile';\nimport { JsonValue } from '@backstage/types';\n\n/**\n * Known fields in Backstage package.json files.\n *\n * @public\n */\nexport interface BackstagePackageJson {\n name: string;\n version: string;\n private?: boolean;\n\n main?: string;\n module?: string;\n types?: string;\n\n scripts?: {\n [key: string]: string;\n };\n // The `bundled` field is a field known within Backstage, it means\n // that the package bundles all of its dependencies in its build output.\n bundled?: boolean;\n\n backstage?: {\n role?: PackageRole;\n };\n\n exports?: JsonValue;\n typesVersions?: Record<string, Record<string, string[]>>;\n\n files?: string[];\n\n publishConfig?: {\n access?: 'public' | 'restricted';\n directory?: string;\n registry?: string;\n alphaTypes?: string;\n betaTypes?: string;\n };\n\n dependencies?: {\n [key: string]: string;\n };\n peerDependencies?: {\n [key: string]: string;\n };\n devDependencies?: {\n [key: string]: string;\n };\n optionalDependencies?: {\n [key: string]: string;\n };\n}\n\n/**\n * A local Backstage monorepo package\n *\n * @public\n */\nexport type BackstagePackage = {\n dir: string;\n packageJson: BackstagePackageJson;\n};\n\n/**\n * A local package in the monorepo package graph.\n *\n * @public\n */\nexport type PackageGraphNode = {\n /** The name of the package */\n name: string;\n /** The directory of the package */\n dir: string;\n /** The package data of the package itself */\n packageJson: BackstagePackageJson;\n\n /** All direct local dependencies of the package */\n allLocalDependencies: Map<string, PackageGraphNode>;\n /** All direct local dependencies that will be present in the published package */\n publishedLocalDependencies: Map<string, PackageGraphNode>;\n /** Local dependencies */\n localDependencies: Map<string, PackageGraphNode>;\n /** Local devDependencies */\n localDevDependencies: Map<string, PackageGraphNode>;\n /** Local optionalDependencies */\n localOptionalDependencies: Map<string, PackageGraphNode>;\n\n /** All direct incoming local dependencies of the package */\n allLocalDependents: Map<string, PackageGraphNode>;\n /** All direct incoming local dependencies that will be present in the published package */\n publishedLocalDependents: Map<string, PackageGraphNode>;\n /** Incoming local dependencies */\n localDependents: Map<string, PackageGraphNode>;\n /** Incoming local devDependencies */\n localDevDependents: Map<string, PackageGraphNode>;\n /** Incoming local optionalDependencies */\n localOptionalDependents: Map<string, PackageGraphNode>;\n};\n\n/**\n * Represents a local Backstage monorepo package graph.\n *\n * @public\n */\nexport class PackageGraph extends Map<string, PackageGraphNode> {\n /**\n * Lists all local packages in a monorepo.\n */\n static async listTargetPackages(): Promise<BackstagePackage[]> {\n const { packages } = await getPackages(paths.targetDir);\n return packages as BackstagePackage[];\n }\n\n /**\n * Creates a package graph from a list of local packages.\n */\n static fromPackages(packages: Package[]): PackageGraph {\n const graph = new PackageGraph();\n\n // Add all local packages to the graph\n for (const pkg of packages) {\n const name = pkg.packageJson.name;\n const existingPkg = graph.get(name);\n if (existingPkg) {\n throw new Error(\n `Duplicate package name '${name}' at ${pkg.dir} and ${existingPkg.dir}`,\n );\n }\n\n graph.set(name, {\n name,\n dir: pkg.dir,\n packageJson: pkg.packageJson as BackstagePackageJson,\n\n allLocalDependencies: new Map(),\n publishedLocalDependencies: new Map(),\n localDependencies: new Map(),\n localDevDependencies: new Map(),\n localOptionalDependencies: new Map(),\n\n allLocalDependents: new Map(),\n publishedLocalDependents: new Map(),\n localDependents: new Map(),\n localDevDependents: new Map(),\n localOptionalDependents: new Map(),\n });\n }\n\n // Populate the local dependency structure\n for (const node of graph.values()) {\n for (const depName of Object.keys(node.packageJson.dependencies || {})) {\n const depPkg = graph.get(depName);\n if (depPkg) {\n node.allLocalDependencies.set(depName, depPkg);\n node.publishedLocalDependencies.set(depName, depPkg);\n node.localDependencies.set(depName, depPkg);\n\n depPkg.allLocalDependents.set(node.name, node);\n depPkg.publishedLocalDependents.set(node.name, node);\n depPkg.localDependents.set(node.name, node);\n }\n }\n for (const depName of Object.keys(\n node.packageJson.devDependencies || {},\n )) {\n const depPkg = graph.get(depName);\n if (depPkg) {\n node.allLocalDependencies.set(depName, depPkg);\n node.localDevDependencies.set(depName, depPkg);\n\n depPkg.allLocalDependents.set(node.name, node);\n depPkg.localDevDependents.set(node.name, node);\n }\n }\n for (const depName of Object.keys(\n node.packageJson.optionalDependencies || {},\n )) {\n const depPkg = graph.get(depName);\n if (depPkg) {\n node.allLocalDependencies.set(depName, depPkg);\n node.publishedLocalDependencies.set(depName, depPkg);\n node.localOptionalDependencies.set(depName, depPkg);\n\n depPkg.allLocalDependents.set(node.name, node);\n depPkg.publishedLocalDependents.set(node.name, node);\n depPkg.localOptionalDependents.set(node.name, node);\n }\n }\n }\n\n return graph;\n }\n\n /**\n * Traverses the package graph and collects a set of package names.\n *\n * The traversal starts at the provided list names, and continues\n * throughout all the names returned by the `collectFn`, which is\n * called once for each seen package.\n */\n collectPackageNames(\n startingPackageNames: string[],\n collectFn: (pkg: PackageGraphNode) => Iterable<string> | undefined,\n ): Set<string> {\n const targets = new Set<string>();\n const searchNames = startingPackageNames.slice();\n\n while (searchNames.length) {\n const name = searchNames.pop()!;\n\n if (targets.has(name)) {\n continue;\n }\n\n const node = this.get(name);\n if (!node) {\n throw new Error(`Package '${name}' not found`);\n }\n\n targets.add(name);\n\n const collected = collectFn(node);\n if (collected) {\n searchNames.push(...collected);\n }\n }\n\n return targets;\n }\n\n /**\n * Lists all packages that have changed since a given git ref.\n *\n * @remarks\n *\n * If the `analyzeLockfile` option is set to true, the change detection will\n * also consider changes to the dependency management lockfile.\n */\n async listChangedPackages(options: {\n ref: string;\n analyzeLockfile?: boolean;\n }) {\n const changedFiles = await GitUtils.listChangedFiles(options.ref);\n\n const dirMap = new Map(\n Array.from(this.values()).map(pkg => [\n // relative from root, convert to posix, and add a / at the end\n path\n .relative(paths.targetRoot, pkg.dir)\n .split(path.sep)\n .join(path.posix.sep) + path.posix.sep,\n pkg,\n ]),\n );\n const packageDirs = Array.from(dirMap.keys());\n\n const result = new Array<PackageGraphNode>();\n let searchIndex = 0;\n\n changedFiles.sort();\n packageDirs.sort();\n\n for (const packageDir of packageDirs) {\n // Skip through changes that appear before our package dir\n while (\n searchIndex < changedFiles.length &&\n changedFiles[searchIndex] < packageDir\n ) {\n searchIndex += 1;\n }\n\n // Check if we arrived at a match, otherwise we move on to the next package dir\n if (changedFiles[searchIndex]?.startsWith(packageDir)) {\n searchIndex += 1;\n\n result.push(dirMap.get(packageDir)!);\n\n // Skip through the rest of the changed files for the same package\n while (changedFiles[searchIndex]?.startsWith(packageDir)) {\n searchIndex += 1;\n }\n }\n }\n\n if (changedFiles.includes('yarn.lock') && options.analyzeLockfile) {\n // Load the lockfile in the working tree and the one at the ref and diff them\n let thisLockfile: Lockfile;\n let otherLockfile: Lockfile;\n try {\n thisLockfile = await Lockfile.load(\n paths.resolveTargetRoot('yarn.lock'),\n );\n otherLockfile = Lockfile.parse(\n await GitUtils.readFileAtRef('yarn.lock', options.ref),\n );\n } catch (error) {\n console.warn(\n `Failed to read lockfiles, assuming all packages have changed, ${error}`,\n );\n return Array.from(this.values());\n }\n const diff = thisLockfile.diff(otherLockfile);\n\n // Create a simplified dependency graph only keeps track of package names\n const graph = thisLockfile.createSimplifiedDependencyGraph();\n\n // Merge the dependency graph from the other lockfile into this one in\n // order to be able to detect removals accurately.\n {\n const otherGraph = thisLockfile.createSimplifiedDependencyGraph();\n for (const [name, dependencies] of otherGraph) {\n const node = graph.get(name);\n if (node) {\n dependencies.forEach(d => node.add(d));\n } else {\n graph.set(name, dependencies);\n }\n }\n }\n\n // The check is simplified by only considering the package names rather\n // than the exact version range queries that were changed.\n // TODO(Rugvip): Use a more exact check\n const changedPackages = new Set(\n [...diff.added, ...diff.changed, ...diff.removed].map(e => e.name),\n );\n\n // Starting with our set of changed packages from the diff, we loop through\n // the full graph and add any package that has a dependency on a changed package.\n // We keep looping until all transitive dependencies have been detected.\n let changed = false;\n do {\n changed = false;\n for (const [name, dependencies] of graph) {\n if (changedPackages.has(name)) {\n continue;\n }\n for (const dep of dependencies) {\n if (changedPackages.has(dep)) {\n changed = true;\n changedPackages.add(name);\n break;\n }\n }\n }\n } while (changed);\n\n // Add all local packages that had a transitive dependency change to the result set\n for (const node of this.values()) {\n if (changedPackages.has(node.name) && !result.includes(node)) {\n result.push(node);\n }\n }\n }\n\n return result;\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 { z } from 'zod';\nimport { PackageRole, PackageRoleInfo } from './types';\n\nconst packageRoleInfos: PackageRoleInfo[] = [\n {\n role: 'frontend',\n platform: 'web',\n output: ['bundle'],\n },\n {\n role: 'backend',\n platform: 'node',\n output: ['bundle'],\n },\n {\n role: 'cli',\n platform: 'node',\n output: ['cjs'],\n },\n {\n role: 'web-library',\n platform: 'web',\n output: ['types', 'esm'],\n },\n {\n role: 'node-library',\n platform: 'node',\n output: ['types', 'cjs'],\n },\n {\n role: 'common-library',\n platform: 'common',\n output: ['types', 'esm', 'cjs'],\n },\n {\n role: 'frontend-plugin',\n platform: 'web',\n output: ['types', 'esm'],\n },\n {\n role: 'frontend-plugin-module',\n platform: 'web',\n output: ['types', 'esm'],\n },\n {\n role: 'backend-plugin',\n platform: 'node',\n output: ['types', 'cjs'],\n },\n {\n role: 'backend-plugin-module',\n platform: 'node',\n output: ['types', 'cjs'],\n },\n];\n\nconst readSchema = z.object({\n name: z.string().optional(),\n backstage: z\n .object({\n role: z.string().optional(),\n })\n .optional(),\n});\n\nconst detectionSchema = z.object({\n name: z.string().optional(),\n scripts: z\n .object({\n start: z.string().optional(),\n build: z.string().optional(),\n })\n .optional(),\n publishConfig: z\n .object({\n main: z.string().optional(),\n types: z.string().optional(),\n module: z.string().optional(),\n })\n .optional(),\n main: z.string().optional(),\n types: z.string().optional(),\n module: z.string().optional(),\n});\n\n/**\n * Utilities for working with Backstage package roles.\n *\n * @public\n */\nexport class PackageRoles {\n /**\n * Get the associated info for a package role.\n */\n static getRoleInfo(role: string): PackageRoleInfo {\n const roleInfo = packageRoleInfos.find(r => r.role === role);\n if (!roleInfo) {\n throw new Error(`Unknown package role '${role}'`);\n }\n return roleInfo;\n }\n\n /**\n * Given package JSON data, get the package role.\n */\n static getRoleFromPackage(pkgJson: unknown): PackageRole | undefined {\n const pkg = readSchema.parse(pkgJson);\n\n if (pkg.backstage) {\n const { role } = pkg.backstage;\n if (!role) {\n throw new Error(\n `Package ${pkg.name} must specify a role in the \"backstage\" field`,\n );\n }\n\n return this.getRoleInfo(role).role;\n }\n\n return undefined;\n }\n\n /**\n * Attempt to detect the role of a package from its package.json.\n */\n static detectRoleFromPackage(pkgJson: unknown): PackageRole | undefined {\n const pkg = detectionSchema.parse(pkgJson);\n\n if (pkg.scripts?.start?.includes('app:serve')) {\n return 'frontend';\n }\n if (pkg.scripts?.build?.includes('backend:bundle')) {\n return 'backend';\n }\n if (\n pkg.name?.includes('plugin-') &&\n pkg.name?.includes('-backend-module-')\n ) {\n return 'backend-plugin-module';\n }\n if (pkg.name?.includes('plugin-') && pkg.name?.includes('-module-')) {\n return 'frontend-plugin-module';\n }\n if (pkg.scripts?.start?.includes('plugin:serve')) {\n return 'frontend-plugin';\n }\n if (pkg.scripts?.start?.includes('backend:dev')) {\n return 'backend-plugin';\n }\n\n const mainEntry = pkg.publishConfig?.main || pkg.main;\n const moduleEntry = pkg.publishConfig?.module || pkg.module;\n const typesEntry = pkg.publishConfig?.types || pkg.types;\n if (typesEntry) {\n if (mainEntry && moduleEntry) {\n return 'common-library';\n }\n if (moduleEntry || mainEntry?.endsWith('.esm.js')) {\n return 'web-library';\n }\n if (mainEntry) {\n return 'node-library';\n }\n } else if (mainEntry) {\n return 'cli';\n }\n\n return undefined;\n }\n}\n"],"names":["findPaths","promisify","execFileCb","assertError","ForwardedError","fs","parseSyml","getPackages","path","z"],"mappings":"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"}
@@ -0,0 +1,248 @@
1
+ import { Package } from '@manypkg/get-packages';
2
+ import { JsonValue } from '@backstage/types';
3
+
4
+ /**
5
+ * Utilities for working with git.
6
+ *
7
+ * @public
8
+ */
9
+ declare class GitUtils {
10
+ /**
11
+ * Returns a sorted list of all files that have changed since the merge base
12
+ * of the provided `ref` and HEAD, as well as all files that are not tracked by git.
13
+ */
14
+ static listChangedFiles(ref: string): Promise<string[]>;
15
+ /**
16
+ * Returns the contents of a file at a specific ref.
17
+ */
18
+ static readFileAtRef(path: string, ref: string): Promise<string>;
19
+ }
20
+
21
+ /**
22
+ * Returns try if the current project is a monorepo.
23
+ *
24
+ * @public
25
+ */
26
+ declare function isMonoRepo(): Promise<boolean>;
27
+
28
+ /**
29
+ * Backstage package role, see {@link https://backstage.io/docs/local-dev/cli-build-system#package-roles | docs}.
30
+ *
31
+ * @public
32
+ */
33
+ type PackageRole = 'frontend' | 'backend' | 'cli' | 'web-library' | 'node-library' | 'common-library' | 'frontend-plugin' | 'frontend-plugin-module' | 'backend-plugin' | 'backend-plugin-module';
34
+ /**
35
+ * A type of platform that a package can be built for.
36
+ *
37
+ * @public
38
+ */
39
+ type PackagePlatform = 'node' | 'web' | 'common';
40
+ /**
41
+ * The type of output that a package can produce.
42
+ *
43
+ * @public
44
+ */
45
+ type PackageOutputType = 'bundle' | 'types' | 'esm' | 'cjs';
46
+ /**
47
+ * Information about a package role.
48
+ *
49
+ * @public
50
+ */
51
+ interface PackageRoleInfo {
52
+ role: PackageRole;
53
+ platform: PackagePlatform;
54
+ output: PackageOutputType[];
55
+ }
56
+
57
+ /**
58
+ * Utilities for working with Backstage package roles.
59
+ *
60
+ * @public
61
+ */
62
+ declare class PackageRoles {
63
+ /**
64
+ * Get the associated info for a package role.
65
+ */
66
+ static getRoleInfo(role: string): PackageRoleInfo;
67
+ /**
68
+ * Given package JSON data, get the package role.
69
+ */
70
+ static getRoleFromPackage(pkgJson: unknown): PackageRole | undefined;
71
+ /**
72
+ * Attempt to detect the role of a package from its package.json.
73
+ */
74
+ static detectRoleFromPackage(pkgJson: unknown): PackageRole | undefined;
75
+ }
76
+
77
+ /**
78
+ * Known fields in Backstage package.json files.
79
+ *
80
+ * @public
81
+ */
82
+ interface BackstagePackageJson {
83
+ name: string;
84
+ version: string;
85
+ private?: boolean;
86
+ main?: string;
87
+ module?: string;
88
+ types?: string;
89
+ scripts?: {
90
+ [key: string]: string;
91
+ };
92
+ bundled?: boolean;
93
+ backstage?: {
94
+ role?: PackageRole;
95
+ };
96
+ exports?: JsonValue;
97
+ typesVersions?: Record<string, Record<string, string[]>>;
98
+ files?: string[];
99
+ publishConfig?: {
100
+ access?: 'public' | 'restricted';
101
+ directory?: string;
102
+ registry?: string;
103
+ alphaTypes?: string;
104
+ betaTypes?: string;
105
+ };
106
+ dependencies?: {
107
+ [key: string]: string;
108
+ };
109
+ peerDependencies?: {
110
+ [key: string]: string;
111
+ };
112
+ devDependencies?: {
113
+ [key: string]: string;
114
+ };
115
+ optionalDependencies?: {
116
+ [key: string]: string;
117
+ };
118
+ }
119
+ /**
120
+ * A local Backstage monorepo package
121
+ *
122
+ * @public
123
+ */
124
+ type BackstagePackage = {
125
+ dir: string;
126
+ packageJson: BackstagePackageJson;
127
+ };
128
+ /**
129
+ * A local package in the monorepo package graph.
130
+ *
131
+ * @public
132
+ */
133
+ type PackageGraphNode = {
134
+ /** The name of the package */
135
+ name: string;
136
+ /** The directory of the package */
137
+ dir: string;
138
+ /** The package data of the package itself */
139
+ packageJson: BackstagePackageJson;
140
+ /** All direct local dependencies of the package */
141
+ allLocalDependencies: Map<string, PackageGraphNode>;
142
+ /** All direct local dependencies that will be present in the published package */
143
+ publishedLocalDependencies: Map<string, PackageGraphNode>;
144
+ /** Local dependencies */
145
+ localDependencies: Map<string, PackageGraphNode>;
146
+ /** Local devDependencies */
147
+ localDevDependencies: Map<string, PackageGraphNode>;
148
+ /** Local optionalDependencies */
149
+ localOptionalDependencies: Map<string, PackageGraphNode>;
150
+ /** All direct incoming local dependencies of the package */
151
+ allLocalDependents: Map<string, PackageGraphNode>;
152
+ /** All direct incoming local dependencies that will be present in the published package */
153
+ publishedLocalDependents: Map<string, PackageGraphNode>;
154
+ /** Incoming local dependencies */
155
+ localDependents: Map<string, PackageGraphNode>;
156
+ /** Incoming local devDependencies */
157
+ localDevDependents: Map<string, PackageGraphNode>;
158
+ /** Incoming local optionalDependencies */
159
+ localOptionalDependents: Map<string, PackageGraphNode>;
160
+ };
161
+ /**
162
+ * Represents a local Backstage monorepo package graph.
163
+ *
164
+ * @public
165
+ */
166
+ declare class PackageGraph extends Map<string, PackageGraphNode> {
167
+ /**
168
+ * Lists all local packages in a monorepo.
169
+ */
170
+ static listTargetPackages(): Promise<BackstagePackage[]>;
171
+ /**
172
+ * Creates a package graph from a list of local packages.
173
+ */
174
+ static fromPackages(packages: Package[]): PackageGraph;
175
+ /**
176
+ * Traverses the package graph and collects a set of package names.
177
+ *
178
+ * The traversal starts at the provided list names, and continues
179
+ * throughout all the names returned by the `collectFn`, which is
180
+ * called once for each seen package.
181
+ */
182
+ collectPackageNames(startingPackageNames: string[], collectFn: (pkg: PackageGraphNode) => Iterable<string> | undefined): Set<string>;
183
+ /**
184
+ * Lists all packages that have changed since a given git ref.
185
+ *
186
+ * @remarks
187
+ *
188
+ * If the `analyzeLockfile` option is set to true, the change detection will
189
+ * also consider changes to the dependency management lockfile.
190
+ */
191
+ listChangedPackages(options: {
192
+ ref: string;
193
+ analyzeLockfile?: boolean;
194
+ }): Promise<PackageGraphNode[]>;
195
+ }
196
+
197
+ /**
198
+ * An entry for a single difference between two {@link Lockfile}s.
199
+ *
200
+ * @public
201
+ */
202
+ type LockfileDiffEntry = {
203
+ name: string;
204
+ range: string;
205
+ };
206
+ /**
207
+ * Represents the difference between two {@link Lockfile}s.
208
+ *
209
+ * @public
210
+ */
211
+ type LockfileDiff = {
212
+ added: LockfileDiffEntry[];
213
+ changed: LockfileDiffEntry[];
214
+ removed: LockfileDiffEntry[];
215
+ };
216
+ /**
217
+ * Represents a package manager lockfile.
218
+ *
219
+ * @public
220
+ */
221
+ declare class Lockfile {
222
+ private readonly packages;
223
+ private readonly data;
224
+ /**
225
+ * Load a {@link Lockfile} from a file path.
226
+ */
227
+ static load(path: string): Promise<Lockfile>;
228
+ /**
229
+ * Parse lockfile contents into a {@link Lockfile}.
230
+ *
231
+ * @public
232
+ */
233
+ static parse(content: string): Lockfile;
234
+ private constructor();
235
+ /**
236
+ * Creates a simplified dependency graph from the lockfile data, where each
237
+ * key is a package, and the value is a set of all packages that it depends on
238
+ * across all versions.
239
+ */
240
+ createSimplifiedDependencyGraph(): Map<string, Set<string>>;
241
+ /**
242
+ * Diff with another lockfile, returning entries that have been
243
+ * added, changed, and removed compared to the other lockfile.
244
+ */
245
+ diff(otherLockfile: Lockfile): LockfileDiff;
246
+ }
247
+
248
+ export { BackstagePackage, BackstagePackageJson, GitUtils, Lockfile, LockfileDiff, LockfileDiffEntry, PackageGraph, PackageGraphNode, PackageOutputType, PackagePlatform, PackageRole, PackageRoleInfo, PackageRoles, isMonoRepo };
package/package.json ADDED
@@ -0,0 +1,41 @@
1
+ {
2
+ "name": "@backstage/cli-node",
3
+ "description": "Node.js library for Backstage CLIs",
4
+ "version": "0.0.0-nightly-20230412021859",
5
+ "main": "dist/index.cjs.js",
6
+ "types": "dist/index.d.ts",
7
+ "license": "Apache-2.0",
8
+ "publishConfig": {
9
+ "access": "public",
10
+ "main": "dist/index.cjs.js",
11
+ "types": "dist/index.d.ts"
12
+ },
13
+ "backstage": {
14
+ "role": "node-library"
15
+ },
16
+ "scripts": {
17
+ "build": "backstage-cli package build",
18
+ "lint": "backstage-cli package lint",
19
+ "test": "backstage-cli package test",
20
+ "clean": "backstage-cli package clean",
21
+ "prepack": "backstage-cli package prepack",
22
+ "postpack": "backstage-cli package postpack"
23
+ },
24
+ "dependencies": {
25
+ "@backstage/cli-common": "^0.1.12",
26
+ "@backstage/errors": "^1.1.5",
27
+ "@backstage/types": "^1.0.2",
28
+ "@manypkg/get-packages": "^1.1.3",
29
+ "@yarnpkg/parsers": "^3.0.0-rc.4",
30
+ "fs-extra": "10.1.0",
31
+ "semver": "^7.3.2",
32
+ "zod": "^3.21.4"
33
+ },
34
+ "devDependencies": {
35
+ "@backstage/cli": "^0.0.0-nightly-20230412021859",
36
+ "mock-fs": "^5.2.0"
37
+ },
38
+ "files": [
39
+ "dist"
40
+ ]
41
+ }