piral-cli 0.15.0-alpha.3564 → 0.15.0-alpha.3589

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.
@@ -10,6 +10,7 @@ export * from './emoji';
10
10
  export * from './envs';
11
11
  export * from './hash';
12
12
  export * from './http';
13
+ export * from './importmap';
13
14
  export * from './info';
14
15
  export * from './injectors';
15
16
  export * from './io';
@@ -6,13 +6,13 @@ import { getDependencies, getDevDependencies } from './language';
6
6
  import { SourceLanguage, ForceOverwrite } from './enums';
7
7
  import { checkAppShellCompatibility } from './compatibility';
8
8
  import { deepMerge } from './merge';
9
- import { getHashFromUrl } from './http';
10
9
  import { applyTemplate } from './template';
10
+ import { readImportmap } from './importmap';
11
11
  import { isGitPackage, isLocalPackage, makeGitUrl, makeFilePath, makePiletExternals, makeExternals } from './npm';
12
12
  import { filesTar, filesOnceTar, declarationEntryExtensions } from './constants';
13
13
  import { getHash, checkIsDirectory, matchFiles } from './io';
14
14
  import { readJson, copy, updateExistingJson, findFile, checkExists } from './io';
15
- import { Framework, FileInfo, PiletsInfo, TemplateFileLocation, SharedDependency } from '../types';
15
+ import { Framework, FileInfo, PiletsInfo, TemplateFileLocation } from '../types';
16
16
 
17
17
  function getDependencyVersion(
18
18
  name: string,
@@ -574,108 +574,6 @@ export function checkAppShellPackage(appPackage: any) {
574
574
  return false;
575
575
  }
576
576
 
577
- function tryResolve(baseDir: string, name: string) {
578
- try {
579
- return require.resolve(name, {
580
- paths: [baseDir],
581
- });
582
- } catch (ex) {
583
- log('generalDebug_0003', `Could not resolve the package "${name}" in "${baseDir}": ${ex}`);
584
- return undefined;
585
- }
586
- }
587
-
588
- interface Importmap {
589
- imports: Record<string, string>;
590
- }
591
-
592
- function normalizeDepName(s: string) {
593
- return (s.startsWith('@') ? s.substring(1) : s).replace(/[\/\.]/g, '-').replace(/(\-)+/, '-');
594
- }
595
-
596
- async function resolveImportmap(dir: string, importmap: Importmap) {
597
- const dependencies: Array<SharedDependency> = [];
598
- const sharedImports = importmap?.imports;
599
-
600
- if (typeof sharedImports === 'object' && sharedImports) {
601
- for (const depName of Object.keys(sharedImports)) {
602
- const url = sharedImports[depName];
603
- const assetName = normalizeDepName(depName);
604
-
605
- if (typeof url !== 'string') {
606
- } else if (/^https?:\/\//.test(url)) {
607
- const hash = await getHashFromUrl(url);
608
-
609
- dependencies.push({
610
- id: `${depName}@${hash}`,
611
- entry: url,
612
- name: depName,
613
- ref: url,
614
- type: 'remote',
615
- });
616
- } else if (url === depName) {
617
- const entry = tryResolve(dir, depName);
618
-
619
- if (entry) {
620
- const packageJson = await findFile(dirname(entry), 'package.json');
621
- const details = require(packageJson);
622
-
623
- dependencies.push({
624
- id: `${depName}@${details.version}`,
625
- entry,
626
- ref: `${assetName}.js`,
627
- name: depName,
628
- type: 'local',
629
- });
630
- }
631
- } else {
632
- const entry = resolve(dir, url);
633
- const exists = await checkExists(entry);
634
-
635
- if (exists) {
636
- const packageJson = await findFile(dirname(entry), 'package.json');
637
-
638
- if (packageJson) {
639
- const details = require(packageJson);
640
-
641
- dependencies.push({
642
- id: `${depName}@${details.version}`,
643
- entry,
644
- name: depName,
645
- ref: `${assetName}.js`,
646
- type: 'local',
647
- });
648
- } else {
649
- const hash = await getHash(entry);
650
-
651
- dependencies.push({
652
- id: `${depName}@${hash}`,
653
- entry,
654
- name: depName,
655
- ref: `${assetName}.js`,
656
- type: 'local',
657
- });
658
- }
659
- }
660
- }
661
- }
662
- }
663
-
664
- return dependencies;
665
- }
666
-
667
- export async function readImportmap(dir: string, packageDetails: any) {
668
- const importmap = packageDetails.importmap;
669
-
670
- if (typeof importmap === 'string') {
671
- const content = await readJson(dir, importmap);
672
- const baseDir = dirname(resolve(dir, importmap));
673
- return resolveImportmap(baseDir, content);
674
- }
675
-
676
- return resolveImportmap(dir, importmap);
677
- }
678
-
679
577
  export async function retrievePiletData(target: string, app?: string) {
680
578
  const packageJson = await findFile(target, 'package.json');
681
579
 
@@ -0,0 +1,233 @@
1
+ import { satisfies, validate } from './version';
2
+
3
+ describe('semver check module', () => {
4
+ it('specific publish version is valid', () => {
5
+ const result = validate('1.2.3');
6
+ expect(result).toBeTruthy();
7
+ });
8
+
9
+ it('shortened publish version is valid', () => {
10
+ const result = validate('1.2');
11
+ expect(result).toBeTruthy();
12
+ });
13
+
14
+ it('some hash is invalid', () => {
15
+ const result = validate('fabcde');
16
+ expect(result).toBeFalsy();
17
+ });
18
+
19
+ it('some longer hash is invalid', () => {
20
+ const result = validate('123fabdef0012');
21
+ expect(result).toBeFalsy();
22
+ });
23
+
24
+ it('major publish version is valid', () => {
25
+ const result = validate('1');
26
+ expect(result).toBeTruthy();
27
+ });
28
+
29
+ it('any publish version is valid', () => {
30
+ const result = validate('1.2.x');
31
+ expect(result).toBeTruthy();
32
+ });
33
+
34
+ it('specific preview version is valid', () => {
35
+ const result = validate('1.2.3-pre.123');
36
+ expect(result).toBeTruthy();
37
+ });
38
+
39
+ it('caret version is specifier valid', () => {
40
+ const result = validate('^1.2.3');
41
+ expect(result).toBeTruthy();
42
+ });
43
+
44
+ it('tilde version is specifier valid', () => {
45
+ const result = validate('~1.2.3');
46
+ expect(result).toBeTruthy();
47
+ });
48
+
49
+ it('greater version is specifier valid', () => {
50
+ const result = validate('>1.2.3');
51
+ expect(result).toBeTruthy();
52
+ });
53
+
54
+ it('greater equals version is specifier valid', () => {
55
+ const result = validate('>=1.2.3');
56
+ expect(result).toBeTruthy();
57
+ });
58
+
59
+ it('greater equals with x version is specifier valid', () => {
60
+ const result = validate('>=1.x');
61
+ expect(result).toBeTruthy();
62
+ });
63
+
64
+ it('satisfies exact version match', () => {
65
+ const result = satisfies('1.2.3', '1.2.3');
66
+ expect(result).toBeTruthy();
67
+ });
68
+
69
+ it('does not satisfy exact version mismatch greater', () => {
70
+ const result = satisfies('1.2.3', '1.2.4');
71
+ expect(result).toBeFalsy();
72
+ });
73
+
74
+ it('does not satisfy exact version mismatch smaller', () => {
75
+ const result = satisfies('1.2.3', '1.2.2');
76
+ expect(result).toBeFalsy();
77
+ });
78
+
79
+ it('satisfies constraint with caret patch', () => {
80
+ const result = satisfies('1.2.3', '^1.2.0');
81
+ expect(result).toBeTruthy();
82
+ });
83
+
84
+ it('satisfies constraint with caret minor', () => {
85
+ const result = satisfies('1.3.0', '^1.2.0');
86
+ expect(result).toBeTruthy();
87
+ });
88
+
89
+ it('satisfies constraint with caret exact', () => {
90
+ const result = satisfies('1.3.0', '^1.3.0');
91
+ expect(result).toBeTruthy();
92
+ });
93
+
94
+ it('satisfies constraint with caret exact with dropped number', () => {
95
+ const result = satisfies('1.3.0', '^1.3');
96
+ expect(result).toBeTruthy();
97
+ });
98
+
99
+ it('satisfies constraint with caret exact with dropped numbers', () => {
100
+ const result = satisfies('1', '^1');
101
+ expect(result).toBeTruthy();
102
+ });
103
+
104
+ it('does not satisfy constraint with caret major', () => {
105
+ const result = satisfies('2.0.0', '^1.2.0');
106
+ expect(result).toBeFalsy();
107
+ });
108
+
109
+ it('satisfies constraint with tilde patch', () => {
110
+ const result = satisfies('1.2.3', '~1.2.0');
111
+ expect(result).toBeTruthy();
112
+ });
113
+
114
+ it('does not satisfy constraint with tilde minor', () => {
115
+ const result = satisfies('1.3.0', '~1.2.0');
116
+ expect(result).toBeFalsy();
117
+ });
118
+
119
+ it('does not satisfy constraint with tilde patch greater', () => {
120
+ const result = satisfies('1.3.0', '~1.3.2');
121
+ expect(result).toBeFalsy();
122
+ });
123
+
124
+ it('satisfies constraint with tilde patch lighter', () => {
125
+ const result = satisfies('1.3.3', '~1.3.2');
126
+ expect(result).toBeTruthy();
127
+ });
128
+
129
+ it('satisfies constraint with x patch', () => {
130
+ const result = satisfies('1.2.3', '1.x.3');
131
+ expect(result).toBeTruthy();
132
+ });
133
+
134
+ it('does not satisfy constraint x minor', () => {
135
+ const result = satisfies('1.3.0', '1.2.x');
136
+ expect(result).toBeFalsy();
137
+ });
138
+
139
+ it('satsfies greater than matching', () => {
140
+ const result = satisfies('1.3.0', '>=1.2.0');
141
+ expect(result).toBeTruthy();
142
+ });
143
+
144
+ it('satsfies less than matching', () => {
145
+ const result = satisfies('1.3.0', '<=1.3.0');
146
+ expect(result).toBeTruthy();
147
+ });
148
+
149
+ it('does not satisfy only greater than non-matching', () => {
150
+ const result = satisfies('1.3.0', '>1.3.0');
151
+ expect(result).toBeFalsy();
152
+ });
153
+
154
+ it('does not satisfy greater than non-matching', () => {
155
+ const result = satisfies('1.3.0', '>=1.3.1');
156
+ expect(result).toBeFalsy();
157
+ });
158
+
159
+ it('does not satisfy less than non-matching', () => {
160
+ const result = satisfies('1.3.0', '<=1.2.0');
161
+ expect(result).toBeFalsy();
162
+ });
163
+
164
+ it('does not satisfy less than non-matching', () => {
165
+ const result = satisfies('1.3.0', '<1.3.0');
166
+ expect(result).toBeFalsy();
167
+ });
168
+
169
+ it('does not satisfy less than non-matching', () => {
170
+ const result = satisfies('1.3.0', '<=1.2.0');
171
+ expect(result).toBeFalsy();
172
+ });
173
+
174
+ it('satisfies exact version match with previews', () => {
175
+ const result = satisfies('1.2.3-pre.5', '1.2.3-pre.5');
176
+ expect(result).toBeTruthy();
177
+ });
178
+
179
+ it('satisfies exact version match with dropped number', () => {
180
+ const result = satisfies('1.0', '1');
181
+ expect(result).toBeTruthy();
182
+ });
183
+
184
+ it('satisfies exact version match with dropped numbers', () => {
185
+ const result = satisfies('1', '1.0.0');
186
+ expect(result).toBeTruthy();
187
+ });
188
+
189
+ it('does not satisfy exact version mismatch preview', () => {
190
+ const result = satisfies('1.2.4-pre.5', '1.2.4-pre.6');
191
+ expect(result).toBeFalsy();
192
+ });
193
+
194
+ it('does not satisfy exact version mismatch preview version offered', () => {
195
+ const result = satisfies('1.2.4-beta', '1.2.4-beta.5');
196
+ expect(result).toBeFalsy();
197
+ });
198
+
199
+ it('does not satisfy exact version mismatch preview version demanded', () => {
200
+ const result = satisfies('1.2.4-beta.5', '1.2.4-beta');
201
+ expect(result).toBeFalsy();
202
+ });
203
+
204
+ it('does not satisfy exact version mismatch preview higher', () => {
205
+ const result = satisfies('1.2.4-gamma', '1.2.4-beta');
206
+ expect(result).toBeFalsy();
207
+ });
208
+
209
+ it('does not satisfy exact version mismatch preview smaller', () => {
210
+ const result = satisfies('1.2.4-alpha', '1.2.4-beta');
211
+ expect(result).toBeFalsy();
212
+ });
213
+
214
+ it('does not satisfy exact and preview mismatch', () => {
215
+ const result = satisfies('1.2.4-pre.5', '1.2.4');
216
+ expect(result).toBeFalsy();
217
+ });
218
+
219
+ it('satisfies just an x', () => {
220
+ const result = satisfies('1.2.3', 'x');
221
+ expect(result).toBeTruthy();
222
+ });
223
+
224
+ it('satisfies just a *', () => {
225
+ const result = satisfies('1.2.3', '*');
226
+ expect(result).toBeTruthy();
227
+ });
228
+
229
+ it('satisfies just a >=0', () => {
230
+ const result = satisfies('1.2.3', '>=0');
231
+ expect(result).toBeTruthy();
232
+ });
233
+ });
@@ -0,0 +1,142 @@
1
+ const semver =
2
+ /^[v^~<>=]*?(\d+)(?:\.([x*]|\d+)(?:\.([x*]|\d+)(?:\.([x*]|\d+))?(?:-([\da-z\-]+(?:\.[\da-z\-]+)*))?(?:\+[\da-z\-]+(?:\.[\da-z\-]+)*)?)?)?$/i;
3
+
4
+ const acceptsAll = ['*', 'x', '>=0'];
5
+
6
+ const operatorResMap: Record<string, [number, number?]> = {
7
+ '>': [1],
8
+ '>=': [0, 1],
9
+ '=': [0],
10
+ '<=': [-1, 0],
11
+ '<': [-1],
12
+ };
13
+
14
+ function indexOrEnd(str: string, q: string) {
15
+ return str.indexOf(q) === -1 ? str.length : str.indexOf(q);
16
+ }
17
+
18
+ function splitVersion(v: string) {
19
+ var c = v.replace(/^v/, '').replace(/\+.*$/, '');
20
+ var patchIndex = indexOrEnd(c, '-');
21
+ var arr = c.substring(0, patchIndex).split('.');
22
+ arr.push(c.substring(patchIndex + 1));
23
+ return arr;
24
+ }
25
+
26
+ function parseSegment(v: string) {
27
+ var n = parseInt(v, 10);
28
+ return isNaN(n) ? v : n;
29
+ }
30
+
31
+ function validateAndParse(v: string) {
32
+ const match = v.match(semver);
33
+ match.shift();
34
+ return match;
35
+ }
36
+
37
+ function compareStrings(a: string, b: string) {
38
+ const ap = parseSegment(a);
39
+ const bp = parseSegment(b);
40
+
41
+ if (ap > bp) {
42
+ return 1;
43
+ } else if (ap < bp) {
44
+ return -1;
45
+ } else {
46
+ return 0;
47
+ }
48
+ }
49
+
50
+ function compareSegments(a: [string, string], b: [string, string]) {
51
+ for (let i = 0; i < 2; i++) {
52
+ const r = compareStrings(a[i] || '0', b[i] || '0');
53
+
54
+ if (r !== 0) {
55
+ return r;
56
+ }
57
+ }
58
+
59
+ return 0;
60
+ }
61
+
62
+ function compareVersions(v1: string, v2: string) {
63
+ const s1 = splitVersion(v1);
64
+ const s2 = splitVersion(v2);
65
+ const len = Math.max(s1.length - 1, s2.length - 1);
66
+
67
+ for (let i = 0; i < len; i++) {
68
+ const n1 = parseInt(s1[i] || '0', 10);
69
+ const n2 = parseInt(s2[i] || '0', 10);
70
+
71
+ if (n1 > n2) {
72
+ return 1;
73
+ } else if (n2 > n1) {
74
+ return -1;
75
+ }
76
+ }
77
+
78
+ const sp1 = s1[s1.length - 1];
79
+ const sp2 = s2[s2.length - 1];
80
+
81
+ if (sp1 && sp2) {
82
+ const p1 = sp1.split('.').map(parseSegment);
83
+ const p2 = sp2.split('.').map(parseSegment);
84
+ const len = Math.max(p1.length, p2.length);
85
+
86
+ for (let i = 0; i < len; i++) {
87
+ if (p1[i] === undefined || (typeof p2[i] === 'string' && typeof p1[i] === 'number')) {
88
+ return -1;
89
+ } else if (p2[i] === undefined || (typeof p1[i] === 'string' && typeof p2[i] === 'number')) {
90
+ return 1;
91
+ } else if (p1[i] > p2[i]) {
92
+ return 1;
93
+ } else if (p2[i] > p1[i]) {
94
+ return -1;
95
+ }
96
+ }
97
+ } else if (sp1 || sp2) {
98
+ return sp1 ? -1 : 1;
99
+ }
100
+
101
+ return 0;
102
+ }
103
+
104
+ function compare(v1: string, v2: string, operator: string) {
105
+ // since result of compareVersions can only be -1 or 0 or 1
106
+ // a simple map can be used to replace switch
107
+ const res = compareVersions(v1, v2);
108
+ return operatorResMap[operator].indexOf(res) > -1;
109
+ }
110
+
111
+ export function validate(version: string) {
112
+ return acceptsAll.includes(version) || semver.test(version);
113
+ }
114
+
115
+ export function satisfies(v: string, r: string) {
116
+ if (!acceptsAll.includes(r)) {
117
+ // if no range operator then "="
118
+ const match = r.match(/^([<>=~^]+)/);
119
+ const op = match ? match[1] : '=';
120
+
121
+ // if gt/lt/eq then operator compare
122
+ if (op !== '^' && op !== '~') {
123
+ return compare(v, r, op);
124
+ }
125
+
126
+ // else range of either "~" or "^" is assumed
127
+ const [v1, v2, v3] = validateAndParse(v);
128
+ const [m1, m2, m3] = validateAndParse(r);
129
+
130
+ if (compareStrings(v1, m1) !== 0) {
131
+ return false;
132
+ } else if (op === '^') {
133
+ return compareSegments([v2, v3], [m2, m3]) >= 0;
134
+ } else if (compareStrings(v2, m2) !== 0) {
135
+ return false;
136
+ }
137
+
138
+ return compareStrings(v3, m3) >= 0;
139
+ }
140
+
141
+ return true;
142
+ }
package/src/messages.ts CHANGED
@@ -392,6 +392,205 @@ export function packageVersionInvalid_0024(version: string): QuickMessage {
392
392
  return [LogLevels.error, '0024', `The given package version "${version}" is invalid.`];
393
393
  }
394
394
 
395
+ /**
396
+ * @kind Error
397
+ *
398
+ * @summary
399
+ * Cannot add the shared dependency since its version constraint is not satisfied.
400
+ *
401
+ * @abstract
402
+ * The importmap definition allows you to define a version specifier separated with
403
+ * the '@' character. If you write down a version specifier then it has to be
404
+ * fulfilled already from the local version, otherwise the packaged version can
405
+ * potentially not be resolved at runtime. This would resolve in a pilet that fails
406
+ * when running in isolation and most likely fails when running with other pilets.
407
+ *
408
+ * @see
409
+ * - [import-maps specification](https://github.com/WICG/import-maps)
410
+ *
411
+ * @example
412
+ * Check the contents of the available package.json:
413
+ *
414
+ * ```sh
415
+ * cat package.json
416
+ * ```
417
+ *
418
+ * The displayed content should look similar to (i.e., contain an importmap such as):
419
+ *
420
+ * ```json
421
+ * {
422
+ * "importmap": {
423
+ * "imports": {
424
+ * "foo@^3.2.1": "foo"
425
+ * }
426
+ * }
427
+ * }
428
+ * ```
429
+ *
430
+ * For the error to occur the specifier (^3.2.1) potentially does not match the version (e.g., if
431
+ * the version of the dependency is 3.1.2).
432
+ *
433
+ * One strategy is to remove the specifier, which will automatically use the exact current version
434
+ * as specifier:
435
+ *
436
+ * ```json
437
+ * {
438
+ * "importmap": {
439
+ * "imports": {
440
+ * "foo": "foo"
441
+ * }
442
+ * }
443
+ * }
444
+ * ```
445
+ *
446
+ * The best way, however, is to look at the used version and adjust the specifier to be correct again.
447
+ * Alternatively, change the used version to satisfy the constraint again.
448
+ */
449
+ export function importMapVersionSpecNotSatisfied_0025(depName: string, version: string): QuickMessage {
450
+ return [LogLevels.error, '0025', `The dependency "${depName}" in only available in version "${version}".`];
451
+ }
452
+
453
+ /**
454
+ * @kind Error
455
+ *
456
+ * @summary
457
+ * The version spec part of the importmap identifer is invalid.
458
+ *
459
+ * @abstract
460
+ * The importmap definition allows you to define a version specifier separated with
461
+ * the '@' character. This part has to follow the semver convention and rules.
462
+ *
463
+ * Check your specifier via online tools such as "Semver check" to verify it is
464
+ * valid and follows the semver specification.
465
+ *
466
+ * @see
467
+ * - [Online Checker](https://jubianchi.github.io/semver-check/)
468
+ *
469
+ * @example
470
+ * Check the contents of the available package.json:
471
+ *
472
+ * ```sh
473
+ * cat package.json
474
+ * ```
475
+ *
476
+ * The displayed content should look similar to (i.e., contain an importmap such as):
477
+ *
478
+ * ```json
479
+ * {
480
+ * "importmap": {
481
+ * "imports": {
482
+ * "foo@bar": "foo"
483
+ * }
484
+ * }
485
+ * }
486
+ * ```
487
+ *
488
+ * For the error to occur the specifier (bar) is not following the semver specification.
489
+ *
490
+ * One way is to remove the version spec, which will resolve to an exact version specifier
491
+ * and therefore always works:
492
+ *
493
+ * ```json
494
+ * {
495
+ * "importmap": {
496
+ * "imports": {
497
+ * "foo": "foo"
498
+ * }
499
+ * }
500
+ * }
501
+ * ```
502
+ *
503
+ * The best way, however, is to look at the used version and adjust the specifier to be correct again,
504
+ * such as "^1.2.3" or "1.x" or "3" etc.
505
+ */
506
+ export function importMapVersionSpecInvalid_0026(depName: string): QuickMessage {
507
+ return [LogLevels.error, '0026', `The dependency "${depName}" has an invalid version spec.`];
508
+ }
509
+
510
+ /**
511
+ * @kind Error
512
+ *
513
+ * @summary
514
+ * The provided importmap reference could not be resolved.
515
+ *
516
+ * @abstract
517
+ * The importmap consists of keys and values. The keys represent the packages names and optional
518
+ * version specifiers to demand at runtime. The values represent the entry point or URL to use
519
+ * when the dependency is not yet loaded.
520
+ *
521
+ * In case of a non-URL value the reference has either to be a valid package name or a file path
522
+ * that leads to either a package or valid JS module. Either way, it needs to exist. If the path
523
+ * is invalid an error will be emitted.
524
+ *
525
+ * @see
526
+ * - [npm Install](https://docs.npmjs.com/cli/install)
527
+ *
528
+ * @example
529
+ * Check the contents of the available package.json:
530
+ *
531
+ * ```sh
532
+ * cat package.json
533
+ * ```
534
+ *
535
+ * The displayed content should look similar to (i.e., contain an importmap such as):
536
+ *
537
+ * ```json
538
+ * {
539
+ * "importmap": {
540
+ * "imports": {
541
+ * "foo@bar": "./node_modules/prect"
542
+ * }
543
+ * }
544
+ * }
545
+ * ```
546
+ *
547
+ * Note the potential misspelling. It maybe should have been "./node_modules/preact". In such
548
+ * cases the reference may not be resolved locally. If everything was written correctly the
549
+ * node modules are most likely not installed (correctly).
550
+ */
551
+ export function importMapReferenceNotFound_0027(dir: string, reference: string): QuickMessage {
552
+ return [LogLevels.error, '0027', `The reference to "${reference}" could not be resolved from "${dir}". Are you sure the file or package exists?`];
553
+ }
554
+
555
+ /**
556
+ * @kind Error
557
+ *
558
+ * @summary
559
+ * The provided importmap file could not be found.
560
+ *
561
+ * @abstract
562
+ * The importmap can be referenced in a file from the package.json. If the named
563
+ * file cannot be found the build process has to be stopped. Make sure that the
564
+ * file has been specified relative to the package.json where it was referenced
565
+ * from.
566
+ *
567
+ * @see
568
+ * - [import-maps specification](https://github.com/WICG/import-maps)
569
+ *
570
+ * @example
571
+ * Check the contents of the available package.json:
572
+ *
573
+ * ```sh
574
+ * cat package.json
575
+ * ```
576
+ *
577
+ * The displayed content should look similar to (i.e., contain an importmap such as):
578
+ *
579
+ * ```json
580
+ * {
581
+ * "importmap": "./import-map.json"
582
+ * }
583
+ * ```
584
+ *
585
+ * If the importmap has instead been (re)named "importmap.json" then this will not work.
586
+ * Likewise, with the reference above the file is expected to be in the same directory
587
+ * as the package.json. If it is, e.g., in the "src" subfolder you'd should reference it
588
+ * as "./src/import-map.json" instead.
589
+ */
590
+ export function importMapFileNotFound_0028(dir: string, file: string): QuickMessage {
591
+ return [LogLevels.error, '0028', `The importmap "${file}" could not be found at "${dir}".`];
592
+ }
593
+
395
594
  /**
396
595
  * @kind Error
397
596
  *