node-backpack 0.0.4
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/.gitattributes +23 -0
- package/CODE_OF_CONDUCT.md +4 -0
- package/CONTRIBUTING.md +59 -0
- package/LICENSE +202 -0
- package/NOTICE +1 -0
- package/README.md +160 -0
- package/bin/node-backpack +2 -0
- package/lib/api/_attributions.d.ts +64 -0
- package/lib/api/_attributions.js +161 -0
- package/lib/api/_shell.d.ts +5 -0
- package/lib/api/_shell.js +37 -0
- package/lib/api/bundle.d.ts +163 -0
- package/lib/api/bundle.js +352 -0
- package/lib/api/index.d.ts +2 -0
- package/lib/api/index.js +19 -0
- package/lib/api/violation.d.ts +70 -0
- package/lib/api/violation.js +71 -0
- package/lib/cli.d.ts +1 -0
- package/lib/cli.js +119 -0
- package/lib/index.d.ts +1 -0
- package/lib/index.js +18 -0
- package/package.json +101 -0
|
@@ -0,0 +1,161 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
3
|
+
if (k2 === undefined) k2 = k;
|
|
4
|
+
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
5
|
+
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
|
6
|
+
desc = { enumerable: true, get: function() { return m[k]; } };
|
|
7
|
+
}
|
|
8
|
+
Object.defineProperty(o, k2, desc);
|
|
9
|
+
}) : (function(o, m, k, k2) {
|
|
10
|
+
if (k2 === undefined) k2 = k;
|
|
11
|
+
o[k2] = m[k];
|
|
12
|
+
}));
|
|
13
|
+
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
|
|
14
|
+
Object.defineProperty(o, "default", { enumerable: true, value: v });
|
|
15
|
+
}) : function(o, v) {
|
|
16
|
+
o["default"] = v;
|
|
17
|
+
});
|
|
18
|
+
var __importStar = (this && this.__importStar) || function (mod) {
|
|
19
|
+
if (mod && mod.__esModule) return mod;
|
|
20
|
+
var result = {};
|
|
21
|
+
if (mod != null) for (var k in mod) if (k !== "default" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k);
|
|
22
|
+
__setModuleDefault(result, mod);
|
|
23
|
+
return result;
|
|
24
|
+
};
|
|
25
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
26
|
+
exports.Attributions = void 0;
|
|
27
|
+
const path = __importStar(require("path"));
|
|
28
|
+
const fs = __importStar(require("fs-extra"));
|
|
29
|
+
const _shell_1 = require("./_shell");
|
|
30
|
+
const violation_1 = require("./violation");
|
|
31
|
+
const ATTRIBUTION_SEPARATOR = '\n----------------\n';
|
|
32
|
+
/**
|
|
33
|
+
* `Attributions` represents an attributions file containing third-party license information.
|
|
34
|
+
*/
|
|
35
|
+
class Attributions {
|
|
36
|
+
constructor(props) {
|
|
37
|
+
this.packageDir = props.packageDir;
|
|
38
|
+
this.packageName = props.packageName;
|
|
39
|
+
this.filePath = path.join(this.packageDir, props.filePath);
|
|
40
|
+
this.dependencies = props.dependencies.filter(d => !props.exclude || !new RegExp(props.exclude).test(d.name));
|
|
41
|
+
this.allowedLicenses = props.allowedLicenses.map(l => l.toLowerCase());
|
|
42
|
+
this.dependenciesRoot = props.dependenciesRoot;
|
|
43
|
+
// without the generated notice content, this object is pretty much
|
|
44
|
+
// useless, so lets generate those of the bat.
|
|
45
|
+
this.attributions = this.generateAttributions();
|
|
46
|
+
this.content = this.render(this.attributions);
|
|
47
|
+
}
|
|
48
|
+
/**
|
|
49
|
+
* Validate the current notice file.
|
|
50
|
+
*
|
|
51
|
+
* This method never throws. The Caller is responsible for inspecting the report returned and act accordinagly.
|
|
52
|
+
*/
|
|
53
|
+
validate() {
|
|
54
|
+
const violations = [];
|
|
55
|
+
const relNoticePath = path.relative(this.packageDir, this.filePath);
|
|
56
|
+
const fix = () => this.flush();
|
|
57
|
+
const missing = !fs.existsSync(this.filePath);
|
|
58
|
+
const attributions = missing ? undefined : fs.readFileSync(this.filePath, { encoding: 'utf-8' });
|
|
59
|
+
const outdated = attributions !== undefined && attributions !== this.content;
|
|
60
|
+
if (missing) {
|
|
61
|
+
violations.push({ type: violation_1.ViolationType.MISSING_NOTICE, message: `${relNoticePath} is missing`, fix });
|
|
62
|
+
}
|
|
63
|
+
if (outdated) {
|
|
64
|
+
violations.push({ type: violation_1.ViolationType.OUTDATED_ATTRIBUTIONS, message: `${relNoticePath} is outdated`, fix });
|
|
65
|
+
}
|
|
66
|
+
const invalidLicense = Array.from(this.attributions.values())
|
|
67
|
+
.filter(a => a.licenses.length === 1 && !this.allowedLicenses.includes(a.licenses[0].toLowerCase()))
|
|
68
|
+
.map(a => ({ type: violation_1.ViolationType.INVALID_LICENSE, message: `Dependency ${a.package} has an invalid license: ${a.licenses[0]}` }));
|
|
69
|
+
const noLicense = Array.from(this.attributions.values())
|
|
70
|
+
.filter(a => a.licenses.length === 0)
|
|
71
|
+
.map(a => ({ type: violation_1.ViolationType.NO_LICENSE, message: `Dependency ${a.package} has no license` }));
|
|
72
|
+
const multiLicense = Array.from(this.attributions.values())
|
|
73
|
+
.filter(a => a.licenses.length > 1)
|
|
74
|
+
.map(a => ({ type: violation_1.ViolationType.MULTIPLE_LICENSE, message: `Dependency ${a.package} has multiple licenses: ${a.licenses}` }));
|
|
75
|
+
violations.push(...invalidLicense);
|
|
76
|
+
violations.push(...noLicense);
|
|
77
|
+
violations.push(...multiLicense);
|
|
78
|
+
return new violation_1.ViolationsReport(violations);
|
|
79
|
+
}
|
|
80
|
+
/**
|
|
81
|
+
* Flush the generated notice file to disk.
|
|
82
|
+
*/
|
|
83
|
+
flush() {
|
|
84
|
+
fs.writeFileSync(this.filePath, this.content);
|
|
85
|
+
}
|
|
86
|
+
render(attributions) {
|
|
87
|
+
const content = [];
|
|
88
|
+
if (attributions.size > 0) {
|
|
89
|
+
content.push(`The ${this.packageName} package includes the following third-party software/licensing:`);
|
|
90
|
+
content.push('');
|
|
91
|
+
}
|
|
92
|
+
// sort the attributions so the file doesn't change due to ordering issues
|
|
93
|
+
const ordered = Array.from(attributions.values()).sort((a1, a2) => a1.package.localeCompare(a2.package));
|
|
94
|
+
for (const attr of ordered) {
|
|
95
|
+
content.push(`** ${attr.package} - ${attr.url} | ${attr.licenses[0]}`);
|
|
96
|
+
// prefer notice over license
|
|
97
|
+
if (attr.noticeText) {
|
|
98
|
+
content.push(attr.noticeText);
|
|
99
|
+
}
|
|
100
|
+
else if (attr.licenseText) {
|
|
101
|
+
content.push(attr.licenseText);
|
|
102
|
+
}
|
|
103
|
+
content.push(ATTRIBUTION_SEPARATOR);
|
|
104
|
+
}
|
|
105
|
+
return content
|
|
106
|
+
// since we are embedding external files, those can different line
|
|
107
|
+
// endings, so we standardize to LF.
|
|
108
|
+
.map(l => l.replace(/\r\n/g, '\n'))
|
|
109
|
+
.join('\n');
|
|
110
|
+
}
|
|
111
|
+
generateAttributions() {
|
|
112
|
+
var _a, _b;
|
|
113
|
+
if (this.dependencies.length === 0) {
|
|
114
|
+
return new Map();
|
|
115
|
+
}
|
|
116
|
+
const attributions = new Map();
|
|
117
|
+
const pkg = (d) => `${d.name}@${d.version}`;
|
|
118
|
+
const packages = this.dependencies.map(d => pkg(d));
|
|
119
|
+
function fetchInfos(_cwd, _packages) {
|
|
120
|
+
// we don't use the programmatic API since it only offers an async API.
|
|
121
|
+
// prefer to stay sync for now since its easier to integrate with other tooling.
|
|
122
|
+
// will offer an async API further down the road.
|
|
123
|
+
const command = `${require.resolve('license-checker/bin/license-checker')} --json --packages "${_packages.join(';')}"`;
|
|
124
|
+
const output = (0, _shell_1.shell)(command, { cwd: _cwd, quiet: true });
|
|
125
|
+
return JSON.parse(output);
|
|
126
|
+
}
|
|
127
|
+
// first run a global command to fetch as much information in one shot
|
|
128
|
+
const infos = fetchInfos(this.dependenciesRoot, packages);
|
|
129
|
+
for (const dep of this.dependencies) {
|
|
130
|
+
const key = pkg(dep);
|
|
131
|
+
// sometimes the dependency might not exist from fetching information globally,
|
|
132
|
+
// so we try fetching a concrete package. this can happen for example when
|
|
133
|
+
// two different major versions exist of the same dependency.
|
|
134
|
+
const info = (_a = infos[key]) !== null && _a !== void 0 ? _a : fetchInfos(dep.path, [pkg(dep)])[key];
|
|
135
|
+
if (!info) {
|
|
136
|
+
// make sure all dependencies are accounted for.
|
|
137
|
+
throw new Error(`Unable to locate license information for ${key} (${dep.path})`);
|
|
138
|
+
}
|
|
139
|
+
const noticeText = info.noticeFile ? fs.readFileSync(info.noticeFile, { encoding: 'utf-8' }) : undefined;
|
|
140
|
+
// for some reason, the license-checker package falls back to the README.md file of the package for license
|
|
141
|
+
// text. this seems strange, disabling that for now.
|
|
142
|
+
// see https://github.com/davglass/license-checker/blob/master/lib/license-files.js#L9
|
|
143
|
+
// note that a non existing license file is ok as long as the license type could be extracted.
|
|
144
|
+
const licenseFile = ((_b = info.licenseFile) === null || _b === void 0 ? void 0 : _b.toLowerCase().endsWith('.md')) ? undefined : info.licenseFile;
|
|
145
|
+
const licenseText = licenseFile ? fs.readFileSync(licenseFile, { encoding: 'utf-8' }) : undefined;
|
|
146
|
+
// the licenses key comes in different types but we convert it here
|
|
147
|
+
// to always be an array.
|
|
148
|
+
const licenses = !info.licenses ? undefined : (Array.isArray(info.licenses) ? info.licenses : [info.licenses]);
|
|
149
|
+
attributions.set(key, {
|
|
150
|
+
package: key,
|
|
151
|
+
url: `https://www.npmjs.com/package/${dep.name}/v/${dep.version}`,
|
|
152
|
+
licenses: licenses !== null && licenses !== void 0 ? licenses : [],
|
|
153
|
+
licenseText,
|
|
154
|
+
noticeText,
|
|
155
|
+
});
|
|
156
|
+
}
|
|
157
|
+
return attributions;
|
|
158
|
+
}
|
|
159
|
+
}
|
|
160
|
+
exports.Attributions = Attributions;
|
|
161
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"_attributions.js","sourceRoot":"","sources":["../../src/api/_attributions.ts"],"names":[],"mappings":"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","sourcesContent":["import * as path from 'path';\nimport * as fs from 'fs-extra';\nimport type { ModuleInfo } from 'license-checker';\nimport { shell } from './_shell';\nimport type { Package } from './bundle';\nimport { Violation, ViolationType, ViolationsReport } from './violation';\n\n\nconst ATTRIBUTION_SEPARATOR = '\\n----------------\\n';\n\n/**\n * Properties for `Attributions`.\n */\nexport interface AttributionsProps {\n  /**\n   * The package root directory.\n   */\n  readonly packageDir: string;\n  /**\n   * The name of the package.\n   */\n  readonly packageName: string;\n  /**\n   * Package dependencies.\n   */\n  readonly dependencies: Package[];\n  /**\n   * The parent directory underwhich all dependencies live.\n   */\n  readonly dependenciesRoot: string;\n  /**\n   * Path to the notice file to created / validated.\n   */\n  readonly filePath: string;\n  /**\n   * List of allowed licenses.\n   *\n   */\n  readonly allowedLicenses: string[];\n  /**\n   * Dependencies matching this pattern will be excluded from attribution.\n   *\n   * @default - no exclusions.\n   */\n  readonly exclude?: string;\n}\n\n/**\n * `Attributions` represents an attributions file containing third-party license information.\n */\nexport class Attributions {\n\n  private readonly packageDir: string;\n  private readonly packageName: string;\n  private readonly dependencies: Package[];\n  private readonly allowedLicenses: string[];\n  private readonly dependenciesRoot: string;\n  private readonly filePath: string;\n\n  private readonly attributions: Map<string, Attribution>;\n  private readonly content: string;\n\n  constructor(props: AttributionsProps) {\n    this.packageDir = props.packageDir;\n    this.packageName = props.packageName;\n    this.filePath = path.join(this.packageDir, props.filePath);\n    this.dependencies = props.dependencies.filter(d => !props.exclude || !new RegExp(props.exclude).test(d.name));\n    this.allowedLicenses = props.allowedLicenses.map(l => l.toLowerCase());\n    this.dependenciesRoot = props.dependenciesRoot;\n\n    // without the generated notice content, this object is pretty much\n    // useless, so lets generate those of the bat.\n    this.attributions = this.generateAttributions();\n    this.content = this.render(this.attributions);\n  }\n\n  /**\n   * Validate the current notice file.\n   *\n   * This method never throws. The Caller is responsible for inspecting the report returned and act accordinagly.\n   */\n  public validate(): ViolationsReport {\n\n    const violations: Violation[] = [];\n    const relNoticePath = path.relative(this.packageDir, this.filePath);\n\n    const fix = () => this.flush();\n\n    const missing = !fs.existsSync(this.filePath);\n    const attributions = missing ? undefined : fs.readFileSync(this.filePath, { encoding: 'utf-8' });\n    const outdated = attributions !== undefined && attributions !== this.content;\n\n    if (missing) {\n      violations.push({ type: ViolationType.MISSING_NOTICE, message: `${relNoticePath} is missing`, fix });\n    }\n\n    if (outdated) {\n      violations.push({ type: ViolationType.OUTDATED_ATTRIBUTIONS, message: `${relNoticePath} is outdated`, fix });\n    }\n\n    const invalidLicense: Violation[] = Array.from(this.attributions.values())\n      .filter(a => a.licenses.length === 1 && !this.allowedLicenses.includes(a.licenses[0].toLowerCase()))\n      .map(a => ({ type: ViolationType.INVALID_LICENSE, message: `Dependency ${a.package} has an invalid license: ${a.licenses[0]}` }));\n\n    const noLicense: Violation[] = Array.from(this.attributions.values())\n      .filter(a => a.licenses.length === 0)\n      .map(a => ({ type: ViolationType.NO_LICENSE, message: `Dependency ${a.package} has no license` }));\n\n    const multiLicense: Violation[] = Array.from(this.attributions.values())\n      .filter(a => a.licenses.length > 1)\n      .map(a => ({ type: ViolationType.MULTIPLE_LICENSE, message: `Dependency ${a.package} has multiple licenses: ${a.licenses}` }));\n\n    violations.push(...invalidLicense);\n    violations.push(...noLicense);\n    violations.push(...multiLicense);\n\n    return new ViolationsReport(violations);\n  }\n\n  /**\n   * Flush the generated notice file to disk.\n   */\n  public flush() {\n    fs.writeFileSync(this.filePath, this.content);\n  }\n\n  private render(attributions: Map<string, Attribution>): string {\n\n    const content = [];\n\n    if (attributions.size > 0) {\n      content.push(`The ${this.packageName} package includes the following third-party software/licensing:`);\n      content.push('');\n    }\n\n    // sort the attributions so the file doesn't change due to ordering issues\n    const ordered = Array.from(attributions.values()).sort((a1, a2) => a1.package.localeCompare(a2.package));\n\n    for (const attr of ordered) {\n      content.push(`** ${attr.package} - ${attr.url} | ${attr.licenses[0]}`);\n\n      // prefer notice over license\n      if (attr.noticeText) {\n        content.push(attr.noticeText);\n      } else if (attr.licenseText) {\n        content.push(attr.licenseText);\n      }\n      content.push(ATTRIBUTION_SEPARATOR);\n    }\n\n    return content\n      // since we are embedding external files, those can different line\n      // endings, so we standardize to LF.\n      .map(l => l.replace(/\\r\\n/g, '\\n'))\n      .join('\\n');\n\n  }\n\n  private generateAttributions(): Map<string, Attribution> {\n\n    if (this.dependencies.length === 0) {\n      return new Map();\n    }\n\n    const attributions: Map<string, Attribution> = new Map();\n\n    const pkg = (d: Package) => `${d.name}@${d.version}`;\n\n    const packages = this.dependencies.map(d => pkg(d));\n\n    function fetchInfos(_cwd: string, _packages: string[]) {\n      // we don't use the programmatic API since it only offers an async API.\n      // prefer to stay sync for now since its easier to integrate with other tooling.\n      // will offer an async API further down the road.\n      const command = `${require.resolve('license-checker/bin/license-checker')} --json --packages \"${_packages.join(';')}\"`;\n      const output = shell(command, { cwd: _cwd, quiet: true });\n      return JSON.parse(output);\n    }\n\n    // first run a global command to fetch as much information in one shot\n    const infos = fetchInfos(this.dependenciesRoot, packages);\n\n    for (const dep of this.dependencies) {\n      const key = pkg(dep);\n\n      // sometimes the dependency might not exist from fetching information globally,\n      // so we try fetching a concrete package. this can happen for example when\n      // two different major versions exist of the same dependency.\n      const info: ModuleInfo = infos[key] ?? fetchInfos(dep.path, [pkg(dep)])[key];\n\n      if (!info) {\n        // make sure all dependencies are accounted for.\n        throw new Error(`Unable to locate license information for ${key} (${dep.path})`);\n      }\n\n      const noticeText = info.noticeFile ? fs.readFileSync(info.noticeFile, { encoding: 'utf-8' }) : undefined;\n\n      // for some reason, the license-checker package falls back to the README.md file of the package for license\n      // text. this seems strange, disabling that for now.\n      // see https://github.com/davglass/license-checker/blob/master/lib/license-files.js#L9\n      // note that a non existing license file is ok as long as the license type could be extracted.\n      const licenseFile = info.licenseFile?.toLowerCase().endsWith('.md') ? undefined : info.licenseFile;\n      const licenseText = licenseFile ? fs.readFileSync(licenseFile, { encoding: 'utf-8' }) : undefined;\n\n      // the licenses key comes in different types but we convert it here\n      // to always be an array.\n      const licenses = !info.licenses ? undefined : (Array.isArray(info.licenses) ? info.licenses : [info.licenses]);\n\n      attributions.set(key, {\n        package: key,\n        url: `https://www.npmjs.com/package/${dep.name}/v/${dep.version}`,\n        licenses: licenses ?? [],\n        licenseText,\n        noticeText,\n      });\n    }\n\n    return attributions;\n  }\n\n}\n\n/**\n * Attribution of a specific dependency.\n */\ninterface Attribution {\n  /**\n   * Attributed package (name + version)\n   */\n  readonly package: string;\n  /**\n   * URL to the package.\n   */\n  readonly url: string;\n  /**\n   * Package licenses.\n   *\n   * Note that some packages will may have multiple licenses,\n   * which is why this is an array. In such cases, the license\n   * validation will fail since we currently disallow this.\n   */\n  readonly licenses: string[];\n  /**\n   * Package license content.\n   *\n   * In case a package has multiple licenses, this will\n   * contain...one of them. It currently doesn't matter which\n   * one since it will not pass validation anyway.\n   */\n  readonly licenseText?: string;\n  /**\n   * Package notice.\n   */\n  readonly noticeText?: string;\n}\n"]}
|
|
@@ -0,0 +1,37 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
3
|
+
if (k2 === undefined) k2 = k;
|
|
4
|
+
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
5
|
+
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
|
6
|
+
desc = { enumerable: true, get: function() { return m[k]; } };
|
|
7
|
+
}
|
|
8
|
+
Object.defineProperty(o, k2, desc);
|
|
9
|
+
}) : (function(o, m, k, k2) {
|
|
10
|
+
if (k2 === undefined) k2 = k;
|
|
11
|
+
o[k2] = m[k];
|
|
12
|
+
}));
|
|
13
|
+
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
|
|
14
|
+
Object.defineProperty(o, "default", { enumerable: true, value: v });
|
|
15
|
+
}) : function(o, v) {
|
|
16
|
+
o["default"] = v;
|
|
17
|
+
});
|
|
18
|
+
var __importStar = (this && this.__importStar) || function (mod) {
|
|
19
|
+
if (mod && mod.__esModule) return mod;
|
|
20
|
+
var result = {};
|
|
21
|
+
if (mod != null) for (var k in mod) if (k !== "default" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k);
|
|
22
|
+
__setModuleDefault(result, mod);
|
|
23
|
+
return result;
|
|
24
|
+
};
|
|
25
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
26
|
+
exports.shell = void 0;
|
|
27
|
+
const child_process = __importStar(require("child_process"));
|
|
28
|
+
function shell(command, options = {}) {
|
|
29
|
+
const stdio = options.quiet ? ['ignore', 'pipe', 'pipe'] : ['ignore', 'inherit', 'inherit'];
|
|
30
|
+
const buffer = child_process.execSync(command, {
|
|
31
|
+
cwd: options.cwd,
|
|
32
|
+
stdio: stdio,
|
|
33
|
+
});
|
|
34
|
+
return buffer ? buffer.toString('utf-8').trim() : '';
|
|
35
|
+
}
|
|
36
|
+
exports.shell = shell;
|
|
37
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiX3NoZWxsLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL2FwaS9fc2hlbGwudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7QUFBQSw2REFBK0M7QUFPL0MsU0FBZ0IsS0FBSyxDQUFDLE9BQWUsRUFBRSxVQUF3QixFQUFFO0lBQy9ELE1BQU0sS0FBSyxHQUErQixPQUFPLENBQUMsS0FBSyxDQUFDLENBQUMsQ0FBQyxDQUFDLFFBQVEsRUFBRSxNQUFNLEVBQUUsTUFBTSxDQUFDLENBQUMsQ0FBQyxDQUFDLENBQUMsUUFBUSxFQUFFLFNBQVMsRUFBRSxTQUFTLENBQUMsQ0FBQztJQUN4SCxNQUFNLE1BQU0sR0FBRyxhQUFhLENBQUMsUUFBUSxDQUFDLE9BQU8sRUFBRTtRQUM3QyxHQUFHLEVBQUUsT0FBTyxDQUFDLEdBQUc7UUFDaEIsS0FBSyxFQUFFLEtBQUs7S0FDYixDQUFDLENBQUM7SUFDSCxPQUFPLE1BQU0sQ0FBQyxDQUFDLENBQUMsTUFBTSxDQUFDLFFBQVEsQ0FBQyxPQUFPLENBQUMsQ0FBQyxJQUFJLEVBQUUsQ0FBQyxDQUFDLENBQUMsRUFBRSxDQUFDO0FBQ3ZELENBQUM7QUFQRCxzQkFPQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCAqIGFzIGNoaWxkX3Byb2Nlc3MgZnJvbSAnY2hpbGRfcHJvY2Vzcyc7XG5cbmV4cG9ydCBpbnRlcmZhY2UgU2hlbGxPcHRpb25zIHtcbiAgcmVhZG9ubHkgY3dkPzogc3RyaW5nO1xuICByZWFkb25seSBxdWlldD86IGJvb2xlYW47XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBzaGVsbChjb21tYW5kOiBzdHJpbmcsIG9wdGlvbnM6IFNoZWxsT3B0aW9ucyA9IHt9KTogc3RyaW5nIHtcbiAgY29uc3Qgc3RkaW86IGNoaWxkX3Byb2Nlc3MuU3RkaW9PcHRpb25zID0gb3B0aW9ucy5xdWlldCA/IFsnaWdub3JlJywgJ3BpcGUnLCAncGlwZSddIDogWydpZ25vcmUnLCAnaW5oZXJpdCcsICdpbmhlcml0J107XG4gIGNvbnN0IGJ1ZmZlciA9IGNoaWxkX3Byb2Nlc3MuZXhlY1N5bmMoY29tbWFuZCwge1xuICAgIGN3ZDogb3B0aW9ucy5jd2QsXG4gICAgc3RkaW86IHN0ZGlvLFxuICB9KTtcbiAgcmV0dXJuIGJ1ZmZlciA/IGJ1ZmZlci50b1N0cmluZygndXRmLTgnKS50cmltKCkgOiAnJztcbn1cbiJdfQ==
|
|
@@ -0,0 +1,163 @@
|
|
|
1
|
+
import { ViolationsReport } from './violation';
|
|
2
|
+
/**
|
|
3
|
+
* Bundling properties.
|
|
4
|
+
*/
|
|
5
|
+
export interface BundleProps {
|
|
6
|
+
/**
|
|
7
|
+
* Directory where the package to bundle is located at.
|
|
8
|
+
*/
|
|
9
|
+
readonly packageDir: string;
|
|
10
|
+
/**
|
|
11
|
+
* List of entry-points to bundle.
|
|
12
|
+
*
|
|
13
|
+
* @default - the 'main' file as specified in package.json.
|
|
14
|
+
*/
|
|
15
|
+
readonly entryPoints?: string[];
|
|
16
|
+
/**
|
|
17
|
+
* Path to attributions file that will be created / validated.
|
|
18
|
+
* This path is relative to the package directory.
|
|
19
|
+
*
|
|
20
|
+
* @default 'THIRD_PARTY_LICENSES'
|
|
21
|
+
*/
|
|
22
|
+
readonly attributionsFile?: string;
|
|
23
|
+
/**
|
|
24
|
+
* External packages that cannot be bundled.
|
|
25
|
+
*
|
|
26
|
+
* @default - no external references.
|
|
27
|
+
*/
|
|
28
|
+
readonly externals?: Externals;
|
|
29
|
+
/**
|
|
30
|
+
* External resources that need to be embedded in the bundle.
|
|
31
|
+
*
|
|
32
|
+
* These will be copied over to the appropriate paths before packaging.
|
|
33
|
+
*/
|
|
34
|
+
readonly resources?: {
|
|
35
|
+
[src: string]: string;
|
|
36
|
+
};
|
|
37
|
+
/**
|
|
38
|
+
* A list of licenses that are allowed for bundling.
|
|
39
|
+
* If any dependency contains a license not in this list, bundling will fail.
|
|
40
|
+
*
|
|
41
|
+
* @default - Default list
|
|
42
|
+
*/
|
|
43
|
+
readonly allowedLicenses?: string[];
|
|
44
|
+
/**
|
|
45
|
+
* Packages matching this regular expression will be excluded from attribution.
|
|
46
|
+
*/
|
|
47
|
+
readonly dontAttribute?: string;
|
|
48
|
+
/**
|
|
49
|
+
* Basic sanity check to run against the created bundle.
|
|
50
|
+
*
|
|
51
|
+
* @default - no check.
|
|
52
|
+
*/
|
|
53
|
+
readonly test?: string;
|
|
54
|
+
}
|
|
55
|
+
/**
|
|
56
|
+
* Options for `Bundle.pack`.
|
|
57
|
+
*/
|
|
58
|
+
export interface BundlePackOptions {
|
|
59
|
+
/**
|
|
60
|
+
* The target directory to create the package in.
|
|
61
|
+
*
|
|
62
|
+
* @default - the package directory.
|
|
63
|
+
*/
|
|
64
|
+
readonly target?: string;
|
|
65
|
+
}
|
|
66
|
+
export interface BundleValidateOptions {
|
|
67
|
+
/**
|
|
68
|
+
* Automatically fix any (fixable) violations.
|
|
69
|
+
*
|
|
70
|
+
* @default false
|
|
71
|
+
*/
|
|
72
|
+
readonly fix?: boolean;
|
|
73
|
+
}
|
|
74
|
+
/**
|
|
75
|
+
* Package on the local file system.
|
|
76
|
+
*/
|
|
77
|
+
export interface Package {
|
|
78
|
+
/**
|
|
79
|
+
* Path of the dependency on the local file system.
|
|
80
|
+
*/
|
|
81
|
+
readonly path: string;
|
|
82
|
+
/**
|
|
83
|
+
* Dependency name.
|
|
84
|
+
*/
|
|
85
|
+
readonly name: string;
|
|
86
|
+
/**
|
|
87
|
+
* Dependency version.
|
|
88
|
+
*/
|
|
89
|
+
readonly version: string;
|
|
90
|
+
}
|
|
91
|
+
/**
|
|
92
|
+
* External packages that cannot be bundled.
|
|
93
|
+
*/
|
|
94
|
+
export interface Externals {
|
|
95
|
+
/**
|
|
96
|
+
* External packages that should be listed in the `dependencies` section
|
|
97
|
+
* of the manifest.
|
|
98
|
+
*/
|
|
99
|
+
readonly dependencies?: readonly string[];
|
|
100
|
+
/**
|
|
101
|
+
* External packages that should be listed in the `optionalDependencies` section
|
|
102
|
+
* of the manifest.
|
|
103
|
+
*/
|
|
104
|
+
readonly optionalDependencies?: readonly string[];
|
|
105
|
+
}
|
|
106
|
+
/**
|
|
107
|
+
* Bundle class to validate and pack nodejs bundles.
|
|
108
|
+
*/
|
|
109
|
+
export declare class Bundle {
|
|
110
|
+
private readonly manifest;
|
|
111
|
+
private readonly noticePath;
|
|
112
|
+
private readonly packageDir;
|
|
113
|
+
private readonly entryPoints;
|
|
114
|
+
private readonly externals;
|
|
115
|
+
private readonly resources;
|
|
116
|
+
private readonly allowedLicenses;
|
|
117
|
+
private readonly dontAttribute?;
|
|
118
|
+
private readonly test?;
|
|
119
|
+
private _bundle?;
|
|
120
|
+
private _dependencies?;
|
|
121
|
+
private _dependenciesRoot?;
|
|
122
|
+
private _attributions?;
|
|
123
|
+
constructor(props: BundleProps);
|
|
124
|
+
/**
|
|
125
|
+
* Validate the bundle for violations.
|
|
126
|
+
*
|
|
127
|
+
* If `fix` is set to true, this method will return the remaining
|
|
128
|
+
* violations after the fixes were applied.
|
|
129
|
+
*
|
|
130
|
+
* This method never throws. The Caller is responsible for inspecting the
|
|
131
|
+
* returned report and act accordingly.
|
|
132
|
+
*/
|
|
133
|
+
validate(options?: BundleValidateOptions): ViolationsReport;
|
|
134
|
+
/**
|
|
135
|
+
* Write the bundle version of the project to a temp directory.
|
|
136
|
+
* This directory is what the tool will end up packing.
|
|
137
|
+
*
|
|
138
|
+
* Returns the temp directory location.
|
|
139
|
+
*/
|
|
140
|
+
write(): string;
|
|
141
|
+
/**
|
|
142
|
+
* Write the bundle and create the tarball.
|
|
143
|
+
*
|
|
144
|
+
* Returns the location of the tarball.
|
|
145
|
+
*/
|
|
146
|
+
pack(options?: BundlePackOptions): string;
|
|
147
|
+
private get bundle();
|
|
148
|
+
private get dependencies();
|
|
149
|
+
private get dependenciesRoot();
|
|
150
|
+
private get attributions();
|
|
151
|
+
private findExternalDependencyVersion;
|
|
152
|
+
private closestPackagePath;
|
|
153
|
+
private createPackage;
|
|
154
|
+
private esbuild;
|
|
155
|
+
private validateCircularImports;
|
|
156
|
+
private validateResources;
|
|
157
|
+
private validateAttributions;
|
|
158
|
+
private addExternals;
|
|
159
|
+
private removeDependencies;
|
|
160
|
+
private writeOutputs;
|
|
161
|
+
private writeResources;
|
|
162
|
+
private writeManifest;
|
|
163
|
+
}
|