Skip to content

Commit 70f5a9f

Browse files
authored
feat: create markdown snippets linter (nodejs#7431)
* feat: create markdown snippets linter * chore: review * chore: rollback test
1 parent 7efa770 commit 70f5a9f

File tree

8 files changed

+152
-32
lines changed

8 files changed

+152
-32
lines changed

apps/site/package.json

+2-1
Original file line numberDiff line numberDiff line change
@@ -10,9 +10,10 @@
1010
"deploy": "cross-env NEXT_PUBLIC_STATIC_EXPORT=true npm run build",
1111
"check-types": "tsc --noEmit",
1212
"lint:js": "eslint \"**/*.{js,mjs,ts,tsx}\"",
13+
"lint:snippets": "node ./scripts/lint-snippets/index.mjs",
1314
"lint:md": "eslint \"**/*.md?(x)\" --cache --cache-strategy=content --cache-location=.eslintmdcache",
1415
"lint:css": "stylelint \"**/*.css\" --allow-empty-input --cache --cache-strategy=content --cache-location=.stylelintcache",
15-
"lint": "turbo run lint:md lint:js lint:css",
16+
"lint": "turbo run lint:md lint:snippets lint:js lint:css",
1617
"lint:fix": "turbo run lint:md lint:js lint:css --no-cache -- --fix",
1718
"sync-orama": "node ./scripts/orama-search/sync-orama-cloud.mjs",
1819
"storybook": "cross-env NODE_NO_WARNINGS=1 storybook dev -p 6006 --no-open",

apps/site/pages/en/learn/modules/how-to-use-streams.md

+41-27
Original file line numberDiff line numberDiff line change
@@ -248,18 +248,24 @@ class MyStream extends Writable {
248248
process.stdout.write(data.toString().toUpperCase() + '\n', cb);
249249
}
250250
}
251-
const stream = new MyStream();
252251

253-
for (let i = 0; i < 10; i++) {
254-
const waitDrain = !stream.write('hello');
252+
async function main() {
253+
const stream = new MyStream();
255254

256-
if (waitDrain) {
257-
console.log('>> wait drain');
258-
await once(stream, 'drain');
255+
for (let i = 0; i < 10; i++) {
256+
const waitDrain = !stream.write('hello');
257+
258+
if (waitDrain) {
259+
console.log('>> wait drain');
260+
await once(stream, 'drain');
261+
}
259262
}
263+
264+
stream.end('world');
260265
}
261266

262-
stream.end('world');
267+
// Call the async function
268+
main().catch(console.error);
263269
```
264270

265271
```mjs
@@ -663,15 +669,19 @@ Here's an example demonstrating the use of async iterators with a readable strea
663669
const fs = require('node:fs');
664670
const { pipeline } = require('node:stream/promises');
665671

666-
await pipeline(
667-
fs.createReadStream(import.meta.filename),
668-
async function* (source) {
669-
for await (let chunk of source) {
670-
yield chunk.toString().toUpperCase();
671-
}
672-
},
673-
process.stdout
674-
);
672+
async function main() {
673+
await pipeline(
674+
fs.createReadStream(__filename),
675+
async function* (source) {
676+
for await (let chunk of source) {
677+
yield chunk.toString().toUpperCase();
678+
}
679+
},
680+
process.stdout
681+
);
682+
}
683+
684+
main().catch(console.error);
675685
```
676686
677687
```mjs
@@ -798,18 +808,22 @@ The helper functions are useful if you need to return a Web Stream from a Node.j
798808
```cjs
799809
const { pipeline } = require('node:stream/promises');
800810

801-
const { body } = await fetch('https://nodejs.org/api/stream.html');
811+
async function main() {
812+
const { body } = await fetch('https://nodejs.org/api/stream.html');
802813

803-
await pipeline(
804-
body,
805-
new TextDecoderStream(),
806-
async function* (source) {
807-
for await (const chunk of source) {
808-
yield chunk.toString().toUpperCase();
809-
}
810-
},
811-
process.stdout
812-
);
814+
await pipeline(
815+
body,
816+
new TextDecoderStream(),
817+
async function* (source) {
818+
for await (const chunk of source) {
819+
yield chunk.toString().toUpperCase();
820+
}
821+
},
822+
process.stdout
823+
);
824+
}
825+
826+
main().catch(console.error);
813827
```
814828
815829
```mjs

apps/site/pages/en/learn/test-runner/mocking.md

+5-4
Original file line numberDiff line numberDiff line change
@@ -149,20 +149,19 @@ This leverages [`mock`](https://nodejs.org/api/test.html#class-mocktracker) from
149149
import assert from 'node:assert/strict';
150150
import { before, describe, it, mock } from 'node:test';
151151

152-
153152
describe('foo', { concurrency: true }, () => {
154153
let barMock = mock.fn();
155154
let foo;
156155

157156
before(async () => {
158157
const barNamedExports = await import('./bar.mjs')
159158
// discard the original default export
160-
.then(({ default, ...rest }) => rest);
159+
.then(({ default: _, ...rest }) => rest);
161160

162161
// It's usually not necessary to manually call restore() after each
163162
// nor reset() after all (node does this automatically).
164163
mock.module('./bar.mjs', {
165-
defaultExport: barMock
164+
defaultExport: barMock,
166165
// Keep the other exports that you don't want to mock.
167166
namedExports: barNamedExports,
168167
});
@@ -173,7 +172,9 @@ describe('foo', { concurrency: true }, () => {
173172
});
174173

175174
it('should do the thing', () => {
176-
barMock.mockImplementationOnce(function bar_mock() {/**/});
175+
barMock.mockImplementationOnce(function bar_mock() {
176+
/**/
177+
});
177178

178179
assert.equal(foo(), 42);
179180
});
+96
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,96 @@
1+
import { readFile } from 'node:fs/promises';
2+
3+
import { parse } from 'acorn';
4+
import { glob } from 'glob';
5+
import remarkParse from 'remark-parse';
6+
import { unified } from 'unified';
7+
import { visit } from 'unist-util-visit';
8+
9+
const SUPPORTED_LANGUAGES = ['js', 'mjs', 'cjs'];
10+
11+
// Initialize the markdown parser
12+
const markdownParser = unified().use(remarkParse);
13+
14+
/**
15+
* Parse JavaScript code using Acorn
16+
*
17+
* @param {string} code - The code to parse
18+
* @param {string} language - The language identifier
19+
* @returns {void}
20+
* @throws {Error} If parsing fails
21+
*/
22+
function parseJavaScript(code, language) {
23+
parse(code, {
24+
ecmaVersion: 'latest',
25+
sourceType: language === 'cjs' ? 'script' : 'module',
26+
allowReturnOutsideFunction: true,
27+
});
28+
}
29+
30+
/**
31+
* Validate code blocks in a markdown file
32+
*
33+
* @param {string} filePath - Path to the markdown file
34+
* @returns {Array<{path: string, position: number, message: string}>} Array of errors
35+
*/
36+
async function validateFile(filePath) {
37+
const errors = [];
38+
39+
const content = await readFile(filePath, 'utf-8');
40+
const tree = markdownParser.parse(content);
41+
42+
visit(tree, 'code', node => {
43+
// TODO: Add TypeScript support
44+
if (!SUPPORTED_LANGUAGES.includes(node.lang)) {
45+
return;
46+
}
47+
48+
try {
49+
parseJavaScript(node.value, node.lang);
50+
} catch (err) {
51+
errors.push({
52+
path: filePath,
53+
position: node.position.start.line,
54+
message: err.message,
55+
});
56+
}
57+
});
58+
59+
return errors;
60+
}
61+
62+
/**
63+
* Print validation errors to console
64+
*
65+
* @param {Array<{path: string, position: number, message: string}>} errors
66+
* @returns {void}
67+
*/
68+
function reportErrors(errors) {
69+
if (errors.length === 0) {
70+
return;
71+
}
72+
73+
console.error('Errors found in the following files:');
74+
errors.forEach(({ path, position, message }) => {
75+
console.error(`- ${path}:${position}: ${message}`);
76+
});
77+
}
78+
79+
// Get all markdown files
80+
const filePaths = await glob('**/*.md', {
81+
root: process.cwd(),
82+
cwd: 'apps/site/pages/en/learn/',
83+
absolute: true,
84+
});
85+
86+
// Validate all files and collect errors
87+
const allErrors = await Promise.all(filePaths.map(validateFile));
88+
89+
// Flatten the array of errors
90+
const flattenedErrors = allErrors.flat();
91+
92+
// Report errors if any
93+
reportErrors(flattenedErrors);
94+
95+
// Exit with appropriate code
96+
process.exit(flattenedErrors.length > 0 ? 1 : 0);

apps/site/turbo.json

+4
Original file line numberDiff line numberDiff line change
@@ -108,6 +108,10 @@
108108
"inputs": ["{app,pages}/**/*.{md,mdx}", "*.{md,mdx}"],
109109
"outputs": [".eslintmdcache"]
110110
},
111+
"lint:snippets": {
112+
"inputs": ["{app,pages}/**/*.{md,mdx}", "*.{md,mdx}"],
113+
"outputs": []
114+
},
111115
"lint:css": {
112116
"inputs": ["{app,components,layouts,pages,styles}/**/*.css"],
113117
"outputs": [".stylelintcache"]

package-lock.json

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

package.json

+1
Original file line numberDiff line numberDiff line change
@@ -35,6 +35,7 @@
3535
"prepare": "husky"
3636
},
3737
"dependencies": {
38+
"acorn": "^8.14.0",
3839
"husky": "9.1.7",
3940
"lint-staged": "15.3.0",
4041
"turbo": "2.3.3"

turbo.json

+2
Original file line numberDiff line numberDiff line change
@@ -11,13 +11,15 @@
1111
"lint": {
1212
"dependsOn": [
1313
"@node-core/website#lint:md",
14+
"@node-core/website#lint:snippets",
1415
"@node-core/website#lint:css",
1516
"lint:js"
1617
]
1718
},
1819
"lint:lint-staged": {
1920
"dependsOn": [
2021
"@node-core/website#lint:md",
22+
"@node-core/website#lint:snippets",
2123
"@node-core/website#lint:css",
2224
"@node-core/website#lint:js"
2325
]

0 commit comments

Comments
 (0)