Skip to content

Commit 5ff30dd

Browse files
feat: support parsing existing package.json descriptions as markdown (#2132)
## PR Checklist - [x] Addresses an existing open issue: fixes #1934 - [x] That issue was marked as [`status: accepting prs`](https://github.com/JoshuaKGoldberg/create-typescript-app/issues?q=is%3Aopen+is%3Aissue+label%3A%22status%3A+accepting+prs%22) - [x] Steps in [CONTRIBUTING.md](https://github.com/JoshuaKGoldberg/create-typescript-app/blob/main/.github/CONTRIBUTING.md) were taken ## Overview Makes parsing around HTML (`README.md`) vs. markdown (`package.json`) descriptions smarter. [`marked`](https://marked.js.org) in inline mode is used to parse existing `package.json` descriptions, before they're checked against the README.md one. Also normalizes the output `package.json`'s `description` to be plain text, since neither GitHub nor npm support rich text (#1934 (comment)). Example improvement: JoshuaKGoldberg/eslint-plugin-erasable-syntax-only@c43f07b 🎁
1 parent 4679042 commit 5ff30dd

10 files changed

+119
-40
lines changed

package.json

+1
Original file line numberDiff line numberDiff line change
@@ -54,6 +54,7 @@
5454
"json5": "^2.2.3",
5555
"lazy-value": "^3.0.0",
5656
"lodash": "^4.17.21",
57+
"marked": "^15.0.7",
5758
"npm-user": "^6.1.1",
5859
"object-strings-deep": "^0.1.1",
5960
"parse-author": "^2.0.0",

pnpm-lock.yaml

+10
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

src/base.ts

+2-1
Original file line numberDiff line numberDiff line change
@@ -185,7 +185,8 @@ export const base = createBase({
185185
const getEmoji = lazyValue(async () => await readEmoji(getDescription));
186186

187187
const getDescription = lazyValue(
188-
async () => await readDescription(getPackageData, getReadme),
188+
async () =>
189+
await readDescription(getPackageData, getReadme, getRepository),
189190
);
190191

191192
const getDevelopmentDocumentation = lazyValue(

src/blocks/blockPackageJson.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -5,8 +5,8 @@ import { z } from "zod";
55
import { PackageJson } from "zod-package-json";
66

77
import { base } from "../base.js";
8+
import { htmlToTextSafe } from "../utils/htmlToTextSafe.js";
89
import { blockRemoveFiles } from "./blockRemoveFiles.js";
9-
import { htmlToTextSafe } from "./html/htmlToTextSafe.js";
1010
import { CommandPhase } from "./phases.js";
1111

1212
const PackageJsonWithNullableScripts = PackageJson.partial().extend({

src/blocks/blockRepositorySettings.test.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -43,7 +43,7 @@ describe("blockRepositorySettings", () => {
4343
`);
4444
});
4545

46-
test("with a long description", () => {
46+
test("with a long HTML description", () => {
4747
const creation = testBlock(blockRepositorySettings, {
4848
options: {
4949
...optionsBase,

src/blocks/blockRepositorySettings.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
import { base } from "../base.js";
2-
import { htmlToTextSafe } from "./html/htmlToTextSafe.js";
2+
import { htmlToTextSafe } from "../utils/htmlToTextSafe.js";
33

44
export const blockRepositorySettings = base.createBlock({
55
about: {

src/blocks/html/htmlToTextSafe.ts

-5
This file was deleted.

src/options/readDescription.test.ts

+58-25
Original file line numberDiff line numberDiff line change
@@ -1,79 +1,112 @@
1-
import { describe, expect, it, vi } from "vitest";
1+
import { describe, expect, it } from "vitest";
22

3+
import { packageData } from "../data/packageData.js";
34
import { readDescription } from "./readDescription.js";
45

5-
const mockPackageDataDescription = vi.fn<() => string>();
6-
7-
vi.mock("../data/packageData.js", () => ({
8-
packageData: {
9-
get description() {
10-
return mockPackageDataDescription();
11-
},
12-
},
13-
}));
14-
156
describe(readDescription, () => {
167
it("returns undefined when the there is no package.json description", async () => {
178
const description = await readDescription(
189
() => Promise.resolve({}),
1910
() => Promise.resolve(""),
11+
() => Promise.resolve("other-repository"),
2012
);
2113

2214
expect(description).toBeUndefined();
23-
expect(mockPackageDataDescription).not.toHaveBeenCalled();
2415
});
2516

26-
it("returns undefined when the description matches the current package.json description", async () => {
17+
it("returns the README.md description when it matches the current package.json description", async () => {
2718
const existing = "Same description.";
2819

29-
mockPackageDataDescription.mockReturnValueOnce(existing);
30-
3120
const description = await readDescription(
3221
() => Promise.resolve({ description: existing }),
3322
() => Promise.resolve(""),
23+
() => Promise.resolve("other-repository"),
3424
);
3525

36-
expect(description).toBeUndefined();
26+
expect(description).toBe(existing);
3727
});
3828

39-
it("returns the updated description when neither description nor name match the current package.json", async () => {
29+
it("returns the updated package.json description when neither description nor name match the current package.json", async () => {
4030
const updated = "Updated description.";
4131

42-
mockPackageDataDescription.mockReturnValueOnce("Existing description");
43-
4432
const description = await readDescription(
4533
() => Promise.resolve({ description: updated }),
4634
() => Promise.resolve(""),
35+
() => Promise.resolve("other-repository"),
4736
);
4837

4938
expect(description).toBe(updated);
5039
});
5140

52-
it("uses the README.md HTML description when it matches what's inferred from package.json plus HTML tags", async () => {
41+
it("returns the existing description when they are equal to crate-typescript-app's and the repository is create-typescript-app", async () => {
42+
const description = await readDescription(
43+
() => Promise.resolve({ description: packageData.description }),
44+
() => Promise.resolve(`<p align="center">${packageData.description}</p>`),
45+
() => Promise.resolve("create-typescript-app"),
46+
);
47+
48+
expect(description).toBe(packageData.description);
49+
});
50+
51+
it("returns undefined when the descriptions are create-typescript-app's and the repository is not", async () => {
52+
const description = await readDescription(
53+
() => Promise.resolve({ description: packageData.description }),
54+
() => Promise.resolve(`<p align="center">${packageData.description}</p>`),
55+
() => Promise.resolve("other-repository"),
56+
);
57+
58+
expect(description).toBeUndefined();
59+
});
60+
61+
it("uses a README.md HTML description when it matches plain text from package.json plus HTML tags", async () => {
5362
const plaintext = "Updated description.";
5463
const encoded = "Updated <code>description</code>.";
5564

56-
mockPackageDataDescription.mockReturnValueOnce("Existing description");
57-
5865
const description = await readDescription(
5966
() => Promise.resolve({ description: plaintext }),
6067
() => Promise.resolve(`<p align="center">${encoded}</p>`),
68+
() => Promise.resolve("other-repository"),
6169
);
6270

6371
expect(description).toBe(encoded);
6472
});
6573

66-
it("uses the package.json description when the README.md HTML description doesn't match what's inferred from package.json plus HTML tags", async () => {
74+
it("uses a README.md HTML description when it matches markdown from package.json plus HTML tags", async () => {
75+
const markdown = "Before `inner` after.";
76+
const html = `Before <a href="https://create.bingo"><code>inner</code></a> after.`;
77+
78+
const description = await readDescription(
79+
() => Promise.resolve({ description: markdown }),
80+
() => Promise.resolve(`<p align="center">${html}</p>`),
81+
() => Promise.resolve("other-repository"),
82+
);
83+
84+
expect(description).toBe(html);
85+
});
86+
87+
it("uses a plain text package.json description when the README.md HTML description doesn't match what's inferred from package.json plus HTML tags", async () => {
6788
const plaintext = "Updated description.";
6889
const encoded = "Incorrect <code>description</code>.";
6990

70-
mockPackageDataDescription.mockReturnValueOnce("Existing description");
71-
7291
const description = await readDescription(
7392
() => Promise.resolve({ description: plaintext }),
7493
() => Promise.resolve(`<p align="center">${encoded}</p>`),
94+
() => Promise.resolve("other-repository"),
7595
);
7696

7797
expect(description).toBe(plaintext);
7898
});
99+
100+
it("uses a markdown package.json description parsed to HTML when the README.md does not have a description and the package.json description is markdown", async () => {
101+
const inPackageJson = "Updated _description_.";
102+
const inReadme = "Incorrect <code>description</code>.";
103+
104+
const description = await readDescription(
105+
() => Promise.resolve({ description: inPackageJson }),
106+
() => Promise.resolve(`<p align="center">${inReadme}</p>`),
107+
() => Promise.resolve("other-repository"),
108+
);
109+
110+
expect(description).toBe("Updated <em>description</em>.");
111+
});
79112
});

src/options/readDescription.ts

+32-6
Original file line numberDiff line numberDiff line change
@@ -1,22 +1,48 @@
1+
import { marked } from "marked";
2+
13
import { packageData } from "../data/packageData.js";
24
import { PartialPackageData } from "../types.js";
5+
import { htmlToTextSafe } from "../utils/htmlToTextSafe.js";
36
import { readDescriptionFromReadme } from "./readDescriptionFromReadme.js";
47

58
export async function readDescription(
69
getPackageData: () => Promise<PartialPackageData>,
710
getReadme: () => Promise<string>,
11+
getRepository: () => Promise<string | undefined>,
812
) {
9-
const { description: inferred } = await getPackageData();
10-
if (!inferred) {
13+
// If we there is no package.json yet, this is probably setup mode.
14+
const { description: fromPackageJson } = await getPackageData();
15+
if (!fromPackageJson) {
1116
return undefined;
1217
}
1318

14-
const { description: existing } = packageData;
19+
// If only a a package.json exists, this is probably transition mode.
20+
// We can use the package.json's description as the only source of truth.
1521
const fromReadme = await readDescriptionFromReadme(getReadme);
22+
if (!fromReadme) {
23+
return marked.parseInline(fromPackageJson);
24+
}
25+
26+
// If the package.json is create-typescript-app's but the repository isn't,
27+
// we're almost certainly in transition mode after cloning the template.
28+
if (
29+
(await getRepository()) !== "create-typescript-app" &&
30+
fromPackageJson === packageData.description
31+
) {
32+
return undefined;
33+
}
34+
35+
const fromPackageJsonNormalized = htmlToTextSafe(
36+
await marked.parseInline(fromPackageJson),
37+
);
38+
const fromReadmeNormalized = htmlToTextSafe(fromReadme);
1639

17-
if (fromReadme?.replaceAll(/<\s*(?:\/\s*)?\w+\s*>/gu, "") === inferred) {
18-
return fromReadme;
40+
// If the package.json and README.md don't match, we prefer the package.json,
41+
// as it's what is used in publishing to npm.
42+
if (fromReadmeNormalized !== fromPackageJsonNormalized) {
43+
return await marked.parseInline(fromPackageJson);
1944
}
2045

21-
return existing === inferred ? undefined : inferred;
46+
// Otherwise, if they do match, the README.md may have more rich HTML text.
47+
return fromReadme;
2248
}

src/utils/htmlToTextSafe.ts

+13
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,13 @@
1+
import { convert } from "html-to-text";
2+
3+
export function htmlToTextSafe(raw: string) {
4+
return convert(raw, {
5+
selectors: [
6+
{
7+
options: { ignoreHref: true },
8+
selector: "a",
9+
},
10+
],
11+
wordwrap: false,
12+
});
13+
}

0 commit comments

Comments
 (0)