Skip to content

Commit 9e3b3e5

Browse files
authored
Merge pull request #440 from TobyAndToby/ts/http-replacements
Http replacement source
2 parents 4eb9258 + 9ef7e5f commit 9e3b3e5

File tree

17 files changed

+399
-63
lines changed

17 files changed

+399
-63
lines changed

e2e/config-file/test/cli/replacement/__snapshots__/replacement.e2e.spec.ts.snap

+15-12
Original file line numberDiff line numberDiff line change
@@ -4,30 +4,33 @@ exports[`cli should match snapshot when a package is replaced 1`] = `
44
"This file was generated with the generate-license-file npm package!
55
https://www.npmjs.com/package/generate-license-file
66
7-
The following npm package may be included in this product:
7+
The following npm packages may be included in this product:
88
9-
9+
10+
1011
11-
This package contains the following license and notice below:
12+
These packages each contain the following license and notice below:
1213
13-
# Dep Four
14+
# Dep Two
1415
15-
This license file is spelt \`LICENCE\`.
16+
This license file is spelt \`LICENCE.md\`.
1617
This license should be found.
1718
1819
-----------
1920
20-
The following npm packages may be included in this product:
21+
The following npm package may be included in this product:
2122
22-
23-
23+
2424
25-
These packages each contain the following license and notice below:
25+
This package contains the following license and notice below:
2626
27-
# Dep Two
27+
# Remote license
2828
29-
This license file is spelt \`LICENCE.md\`.
30-
This license should be found.
29+
This file is NOT an actual license for this package.
30+
31+
It is used as a resource for our e2e's, as a file we can fetch over the network, and have control over.
32+
33+
This content should appear in the generated license file output!
3134
3235
-----------
3336

e2e/config-file/test/cli/replacement/config.js

+2
Original file line numberDiff line numberDiff line change
@@ -8,5 +8,7 @@ module.exports = {
88

99
"dep-three": "./some-path-that-we-dont-want-to-use.txt",
1010
"[email protected]": "./name-and-version-replacement-content.txt",
11+
"dep-four":
12+
"https://raw.githubusercontent.com/TobyAndToby/generate-license-file/main/e2e/.remote-licenses/license.md",
1113
},
1214
};

src/packages/generate-license-file/src/lib/cli/config/index.ts

+15-8
Original file line numberDiff line numberDiff line change
@@ -17,17 +17,20 @@ export const loadConfigFile = async (path?: string): Promise<ConfigSchema> => {
1717
const findAndParseConfig = async (directory: string): Promise<ConfigSchema> => {
1818
const configFile = await findConfig(directory);
1919

20-
return parseConfig(configFile, directory);
20+
return await parseConfig(configFile, directory);
2121
};
2222

2323
const loadAndParseConfig = async (filePath: string) => {
2424
const configFile = await loadConfig(filePath);
2525
const directory = dirname(filePath);
2626

27-
return parseConfig(configFile, directory);
27+
return await parseConfig(configFile, directory);
2828
};
2929

30-
const parseConfig = (configFile: ConfigFile | undefined, directory: string): ConfigSchema => {
30+
const parseConfig = async (
31+
configFile: ConfigFile | undefined,
32+
directory: string,
33+
): Promise<ConfigSchema> => {
3134
const config = parseSchema(configFile?.config);
3235

3336
if (config === undefined) {
@@ -37,12 +40,16 @@ const parseConfig = (configFile: ConfigFile | undefined, directory: string): Con
3740
for (const replacement in config?.replace) {
3841
const replacementPath = config.replace[replacement];
3942

40-
if (isAbsolute(replacementPath)) {
41-
continue;
42-
}
43-
43+
// The replacement value could be multiple things (e.g. file path, or a
44+
// URL). If it's a file path, then at this stage the CLI is aware of the
45+
// execution directory and needs to make the path absolute before handing
46+
// it off to the library implementation. Otherwise, pass the raw replacement
47+
// value into the library so it can handle it however it wants.
4448
const absolutePath = join(directory, replacementPath);
45-
config.replace[replacement] = absolutePath;
49+
50+
if (await doesFileExist(absolutePath)) {
51+
config.replace[replacement] = absolutePath;
52+
}
4653
}
4754

4855
if (config.append) {

src/packages/generate-license-file/src/lib/internal/resolveDependencies/resolveNpmDependencies.ts

+11-2
Original file line numberDiff line numberDiff line change
@@ -3,6 +3,7 @@ import { resolveLicenseContent } from "../resolveLicenseContent";
33
import { dirname, isAbsolute, join } from "path";
44
import { Dependency, LicenseContent } from "../resolveLicenses";
55
import { readPackageJson } from "../../utils/packageJson.utils";
6+
import logger from "../../utils/console.utils";
67

78
type ResolveLicensesOptions = {
89
replace?: Record<string, string>;
@@ -36,9 +37,9 @@ export const resolveDependenciesForNpmProject = async (
3637
return;
3738
}
3839

39-
const licenseContent = await resolveLicenseContent(node.realpath, packageJson, replacements);
40+
try {
41+
const licenseContent = await resolveLicenseContent(node.realpath, packageJson, replacements);
4042

41-
if (licenseContent) {
4243
const dependencies = licensesMap.get(licenseContent) ?? [];
4344

4445
const alreadyExists = dependencies.find(
@@ -50,6 +51,14 @@ export const resolveDependenciesForNpmProject = async (
5051
}
5152

5253
licensesMap.set(licenseContent, dependencies);
54+
} catch (error) {
55+
const warningLines = [
56+
`Unable to determine license content for ${packageJson.name}@${packageJson.version} with error:`,
57+
error instanceof Error ? error.message : error?.toString(),
58+
"", // Empty line for spacing
59+
];
60+
61+
logger.warn(warningLines.join("\n"));
5362
}
5463

5564
for (const child of node.children.values()) {

src/packages/generate-license-file/src/lib/internal/resolveDependencies/resolvePnpmDependencies.ts

+15-2
Original file line numberDiff line numberDiff line change
@@ -3,6 +3,7 @@ import { dirname, join } from "path";
33
import { getPnpmProjectDependencies, getPnpmVersion } from "../../utils/pnpmCli.utils";
44
import { Dependency, LicenseContent } from "../resolveLicenses";
55
import { readPackageJson } from "../../utils/packageJson.utils";
6+
import logger from "../../utils/console.utils";
67

78
type ResolveLicensesOptions = {
89
replace?: Record<string, string>;
@@ -34,9 +35,13 @@ export const resolveDependenciesForPnpmProject = async (
3435
continue;
3536
}
3637

37-
const licenseContent = await resolveLicenseContent(dependencyPath, packageJson, replacements);
38+
try {
39+
const licenseContent = await resolveLicenseContent(
40+
dependencyPath,
41+
packageJson,
42+
replacements,
43+
);
3844

39-
if (licenseContent) {
4045
const dependencies = licensesMap.get(licenseContent) ?? [];
4146

4247
const alreadyExists = dependencies.find(
@@ -48,6 +53,14 @@ export const resolveDependenciesForPnpmProject = async (
4853
}
4954

5055
licensesMap.set(licenseContent, dependencies);
56+
} catch (error) {
57+
const warningLines = [
58+
`Unable to determine license content for ${packageJson.name}@${packageJson.version} with error:`,
59+
error instanceof Error ? error.message : error?.toString(),
60+
"", // Empty line for spacing
61+
];
62+
63+
logger.warn(warningLines.join("\n"));
5164
}
5265
}
5366
}

src/packages/generate-license-file/src/lib/internal/resolveLicenseContent/index.ts

+28-6
Original file line numberDiff line numberDiff line change
@@ -2,37 +2,59 @@ import { PackageJson } from "../../utils/packageJson.utils";
22
import { packageJsonLicense } from "./packageJsonLicense";
33
import { licenseFile } from "./licenseFile";
44
import { spdxExpression } from "./spdxExpression";
5-
import { readFile } from "../../utils/file.utils";
5+
import { replacementFile } from "./replacementFile";
6+
import { replacementHttp } from "./replacementHttp";
67

78
export interface ResolutionInputs {
89
directory: string;
910
packageJson: PackageJson;
1011
}
1112

1213
export type Resolution = (inputs: ResolutionInputs) => Promise<string | null>;
13-
1414
const resolutions: Resolution[] = [packageJsonLicense, licenseFile, spdxExpression];
1515

16+
export type ReplacementResolution = (location: string) => Promise<string | null>;
17+
const replacementResolutions: ReplacementResolution[] = [replacementHttp, replacementFile];
18+
1619
export const resolveLicenseContent = async (
1720
directory: string,
1821
packageJson: PackageJson,
1922
replacements: Record<string, string>,
20-
): Promise<string | null> => {
23+
): Promise<string> => {
2124
const replacementPath =
2225
replacements[`${packageJson.name}@${packageJson.version}`] ||
2326
replacements[`${packageJson.name}`];
2427

2528
if (replacementPath) {
26-
return await readFile(replacementPath, { encoding: "utf-8" });
29+
return runReplacementResolutions(replacementPath, packageJson);
30+
}
31+
32+
const resolutionInputs: ResolutionInputs = { directory, packageJson };
33+
return runResolutions(resolutionInputs, packageJson);
34+
};
35+
36+
const runReplacementResolutions = async (replacementPath: string, packageJson: PackageJson) => {
37+
for (const resolution of replacementResolutions) {
38+
const result = await resolution(replacementPath);
39+
40+
if (result) {
41+
return result;
42+
}
2743
}
2844

45+
throw new Error(
46+
`Could not find replacement content at ${replacementPath} for ${packageJson.name}@${packageJson.version}`,
47+
);
48+
};
49+
50+
const runResolutions = async (inputs: ResolutionInputs, packageJson: PackageJson) => {
2951
for (const resolution of resolutions) {
30-
const result = await resolution({ directory, packageJson });
52+
const result = await resolution(inputs);
3153

3254
if (result) {
3355
return result;
3456
}
3557
}
3658

37-
return null;
59+
throw new Error(`Could not find license content for ${packageJson.name}@${packageJson.version}`);
3860
};

src/packages/generate-license-file/src/lib/internal/resolveLicenseContent/packageJsonLicense.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -86,7 +86,7 @@ const parseArrayLicense = (license: PackageJsonLicense[], packageJson: PackageJs
8686
}
8787

8888
const warningLines = [
89-
`The license key for ${packageJson.name}@${packageJson.version} contains multiple licenses"`,
89+
`The license key for ${packageJson.name}@${packageJson.version} contains multiple licenses`,
9090
"We suggest you determine which license applies to your project and replace the license content",
9191
`for ${packageJson.name}@${packageJson.version} using a generate-license-file config file.`,
9292
"See: https://generate-license-file.js.org/docs/cli/config-file for more information.",
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,10 @@
1+
import { ReplacementResolution } from ".";
2+
import { readFile, doesFileExist } from "../../utils/file.utils";
3+
4+
export const replacementFile: ReplacementResolution = async location => {
5+
if (!(await doesFileExist(location))) {
6+
return null;
7+
}
8+
9+
return readFile(location, { encoding: "utf-8" });
10+
};
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,10 @@
1+
import { ReplacementResolution } from ".";
2+
import { fetchString } from "../../utils/http.utils";
3+
4+
export const replacementHttp: ReplacementResolution = async location => {
5+
if (!location.startsWith("http") && !location.startsWith("www")) {
6+
return null;
7+
}
8+
9+
return fetchString(location);
10+
};
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,6 @@
1+
// istanbul ignore file
2+
3+
export const fetchString = async (url: string): Promise<string> => {
4+
const response = await fetch(url);
5+
return response.text();
6+
};

src/packages/generate-license-file/test/internal/resolveDependencies/resolveNpmDependencies.spec.ts

+19
Original file line numberDiff line numberDiff line change
@@ -6,19 +6,22 @@ import { Dependency, LicenseContent } from "../../../src/lib/internal/resolveLic
66
import { PackageJson } from "../../../src/lib/utils/packageJson.utils";
77
import { join } from "path";
88
import { doesFileExist, readFile } from "../../../src/lib/utils/file.utils";
9+
import logger from "../../../src/lib/utils/console.utils";
910

1011
jest.mock("@npmcli/arborist", () => ({
1112
__esModule: true,
1213
default: jest.fn(),
1314
}));
1415

1516
jest.mock("../../../src/lib/utils/file.utils");
17+
jest.mock("../../../src/lib/utils/console.utils");
1618

1719
jest.mock("../../../src/lib/internal/resolveLicenseContent", () => ({
1820
resolveLicenseContent: jest.fn(),
1921
}));
2022

2123
describe("resolveNpmDependencies", () => {
24+
const mockedLogger = jest.mocked(logger);
2225
const mockedReadFile = jest.mocked(readFile);
2326
const mockedDoesFileExist = jest.mocked(doesFileExist);
2427

@@ -258,6 +261,22 @@ describe("resolveNpmDependencies", () => {
258261
expect(replacements3).toBe(replacements);
259262
});
260263

264+
it.each([new Error("Something went wrong"), "Something went wrong"])(
265+
"should warning log if resolveLicenseContent throws an error",
266+
async error => {
267+
when(mockedResolveLicenseContent)
268+
.calledWith(child1Realpath, expect.anything(), expect.anything())
269+
.mockRejectedValue(error);
270+
271+
await resolveDependenciesForNpmProject("/some/path/package.json", new Map());
272+
273+
expect(mockedLogger.warn).toHaveBeenCalledTimes(1);
274+
expect(mockedLogger.warn).toHaveBeenCalledWith(
275+
`Unable to determine license content for ${child1Name}@${child1Version} with error:\nSomething went wrong\n`,
276+
);
277+
},
278+
);
279+
261280
describe("when no options are provided", () => {
262281
it("should include non-dev dependencies in the result", async () => {
263282
const licensesMap = new Map<LicenseContent, Dependency[]>();

src/packages/generate-license-file/test/internal/resolveDependencies/resolvePnpmDependencies.spec.ts

+26-3
Original file line numberDiff line numberDiff line change
@@ -8,6 +8,7 @@ import { resolveLicenseContent } from "../../../src/lib/internal/resolveLicenseC
88
import { when } from "jest-when";
99
import { doesFileExist, readFile } from "../../../src/lib/utils/file.utils";
1010
import { PackageJson } from "../../../src/lib/utils/packageJson.utils";
11+
import logger from "../../../src/lib/utils/console.utils";
1112
import { join } from "path";
1213
import { Dependency, LicenseContent } from "../../../src/lib/internal/resolveLicenses";
1314

@@ -17,6 +18,7 @@ jest.mock("../../../src/lib/utils/pnpmCli.utils", () => ({
1718
}));
1819

1920
jest.mock("../../../src/lib/utils/file.utils");
21+
jest.mock("../../../src/lib/utils/console.utils");
2022

2123
jest.mock("../../../src/lib/internal/resolveLicenseContent", () => ({
2224
resolveLicenseContent: jest.fn(),
@@ -39,8 +41,8 @@ describe("resolveDependenciesForPnpmProject", () => {
3941
name: "dependency3",
4042
paths: ["/some/path/dependency3"],
4143
};
42-
const dependency3LicenseContent = null as unknown as string;
4344

45+
const mockedLogger = jest.mocked(logger);
4446
const mockedReadFile = jest.mocked(readFile);
4547
const mockedDoesFileExist = jest.mocked(doesFileExist);
4648
const mockedGetPnpmVersion = jest.mocked(getPnpmVersion);
@@ -67,7 +69,9 @@ describe("resolveDependenciesForPnpmProject", () => {
6769

6870
when(mockedResolveLicenseContent)
6971
.calledWith(dependency3.paths[0], expect.anything(), expect.anything())
70-
.mockResolvedValue(dependency3LicenseContent);
72+
.mockImplementation(() => {
73+
throw new Error("Cannot find license content");
74+
});
7175
setUpPackageJson(dependency3.paths[0], { name: dependency3.name, version: "1.0.0" });
7276
});
7377

@@ -198,9 +202,28 @@ describe("resolveDependenciesForPnpmProject", () => {
198202
.get(dependency2LicenseContent)
199203
?.find(d => d.name === "dependency2" && d.version === "2.0.0"),
200204
).toBeDefined();
201-
expect(licensesMap.get(dependency3LicenseContent)).toBeUndefined();
202205
});
203206

207+
it.each([new Error("Something went wrong"), "Something went wrong"])(
208+
"should warning log if resolveLicenseContent throws an error",
209+
async error => {
210+
mockedGetPnpmVersion.mockResolvedValue(pnpmVersion);
211+
mockedGetPnpmProjectDependencies.mockResolvedValue([dependency1, dependency2, dependency3]);
212+
213+
when(mockedResolveLicenseContent)
214+
.calledWith(dependency1.paths[0], expect.anything(), expect.anything())
215+
.mockRejectedValue(error);
216+
217+
const licensesMap = new Map<LicenseContent, Dependency[]>();
218+
219+
await resolveDependenciesForPnpmProject("/some/path/package.json", licensesMap);
220+
221+
expect(mockedLogger.warn).toHaveBeenCalledWith(
222+
`Unable to determine license content for ${dependency1.name}@1.0.0 with error:\nSomething went wrong\n`,
223+
);
224+
},
225+
);
226+
204227
describe("when the dependency is in the exclude list", () => {
205228
it("should not call resolveLicenseContent", async () => {
206229
mockedGetPnpmVersion.mockResolvedValue(pnpmVersion);

0 commit comments

Comments
 (0)