From 79afb61e4e48e365cee558b5f3f3c6210d18cfd3 Mon Sep 17 00:00:00 2001
From: ozaki <29860391+OzakIOne@users.noreply.github.com>
Date: Tue, 24 Oct 2023 16:15:49 +0200
Subject: [PATCH] feat(mdx-loader): Remark plugin to report unused MDX /
Markdown directives (#9394)
Co-authored-by: sebastienlorber
---
argos/playwright.config.ts | 2 +
argos/tests/screenshot.spec.ts | 8 +-
packages/docusaurus-mdx-loader/src/loader.ts | 4 +
.../docusaurus-mdx-loader/src/processor.ts | 8 +-
.../src/remark/contentTitle/index.ts | 1 +
.../src/remark/mermaid/index.ts | 6 +-
.../src/remark/transformImage/index.ts | 16 +-
.../src/remark/transformLinks/index.ts | 16 +-
.../__fixtures__/containerDirectives.md | 19 ++
.../__tests__/__fixtures__/leafDirectives.md | 5 +
.../__tests__/__fixtures__/textDirectives.md | 17 ++
.../__snapshots__/index.test.ts.snap | 86 ++++++++++
.../unusedDirectives/__tests__/index.test.ts | 111 ++++++++++++
.../src/remark/unusedDirectives/index.ts | 162 ++++++++++++++++++
.../src/remark/utils/index.ts | 23 ++-
.../src/vfile-datamap.d.mts | 21 +++
packages/docusaurus-utils/src/index.ts | 6 +-
packages/docusaurus-utils/src/webpackUtils.ts | 20 ++-
project-words.txt | 1 +
19 files changed, 506 insertions(+), 26 deletions(-)
create mode 100644 packages/docusaurus-mdx-loader/src/remark/unusedDirectives/__tests__/__fixtures__/containerDirectives.md
create mode 100644 packages/docusaurus-mdx-loader/src/remark/unusedDirectives/__tests__/__fixtures__/leafDirectives.md
create mode 100644 packages/docusaurus-mdx-loader/src/remark/unusedDirectives/__tests__/__fixtures__/textDirectives.md
create mode 100644 packages/docusaurus-mdx-loader/src/remark/unusedDirectives/__tests__/__snapshots__/index.test.ts.snap
create mode 100644 packages/docusaurus-mdx-loader/src/remark/unusedDirectives/__tests__/index.test.ts
create mode 100644 packages/docusaurus-mdx-loader/src/remark/unusedDirectives/index.ts
create mode 100644 packages/docusaurus-mdx-loader/src/vfile-datamap.d.mts
diff --git a/argos/playwright.config.ts b/argos/playwright.config.ts
index 31409769110c2..519227c75121b 100644
--- a/argos/playwright.config.ts
+++ b/argos/playwright.config.ts
@@ -14,6 +14,8 @@ import type {PlaywrightTestConfig} from '@playwright/test';
const config: PlaywrightTestConfig = {
testDir: './tests',
+ timeout: 60000,
+
reporter: [['list'], ['@argos-ci/playwright/reporter']],
// Run website production built
diff --git a/argos/tests/screenshot.spec.ts b/argos/tests/screenshot.spec.ts
index 70bee2af15c35..f6829a13a9d25 100644
--- a/argos/tests/screenshot.spec.ts
+++ b/argos/tests/screenshot.spec.ts
@@ -36,8 +36,12 @@ function isBlacklisted(pathname: string) {
}
// Some paths explicitly blacklisted
const BlacklistedPathnames: string[] = [
- '/feature-requests', // Flaky because of Canny widget
- '/community/canary', // Flaky because of dynamic canary version fetched from npm
+ // Flaky because of Canny widget
+ '/feature-requests',
+ // Flaky because of dynamic canary version fetched from npm
+ '/community/canary',
+ // Long blog post with many image carousels, often timeouts
+ '/blog/2022/08/01/announcing-docusaurus-2.0',
];
return (
diff --git a/packages/docusaurus-mdx-loader/src/loader.ts b/packages/docusaurus-mdx-loader/src/loader.ts
index 519b5da3b9cdc..a475220cd5c63 100644
--- a/packages/docusaurus-mdx-loader/src/loader.ts
+++ b/packages/docusaurus-mdx-loader/src/loader.ts
@@ -11,6 +11,7 @@ import {
parseFrontMatter,
escapePath,
getFileLoaderUtils,
+ getWebpackLoaderCompilerName,
} from '@docusaurus/utils';
import stringifyObject from 'stringify-object';
import preprocessor from './preprocessor';
@@ -134,10 +135,12 @@ export async function mdxLoader(
this: LoaderContext,
fileString: string,
): Promise {
+ const compilerName = getWebpackLoaderCompilerName(this);
const callback = this.async();
const filePath = this.resourcePath;
const reqOptions: Options = this.getOptions();
const {query} = this;
+
ensureMarkdownConfig(reqOptions);
const {frontMatter} = parseFrontMatter(fileString);
@@ -165,6 +168,7 @@ export async function mdxLoader(
content: preprocessedContent,
filePath,
frontMatter,
+ compilerName,
});
} catch (errorUnknown) {
const error = errorUnknown as Error;
diff --git a/packages/docusaurus-mdx-loader/src/processor.ts b/packages/docusaurus-mdx-loader/src/processor.ts
index b7d500c64a0d0..032cdf4e2ff5f 100644
--- a/packages/docusaurus-mdx-loader/src/processor.ts
+++ b/packages/docusaurus-mdx-loader/src/processor.ts
@@ -15,8 +15,10 @@ import details from './remark/details';
import head from './remark/head';
import mermaid from './remark/mermaid';
import transformAdmonitions from './remark/admonitions';
+import unusedDirectivesWarning from './remark/unusedDirectives';
import codeCompatPlugin from './remark/mdx1Compat/codeCompatPlugin';
import {getFormat} from './format';
+import type {WebpackCompilerName} from '@docusaurus/utils';
import type {MDXFrontMatter} from './frontMatter';
import type {Options} from './loader';
import type {AdmonitionOptions} from './remark/admonitions';
@@ -37,10 +39,12 @@ type SimpleProcessor = {
content,
filePath,
frontMatter,
+ compilerName,
}: {
content: string;
filePath: string;
frontMatter: {[key: string]: unknown};
+ compilerName: WebpackCompilerName;
}) => Promise;
};
@@ -123,6 +127,7 @@ async function createProcessorFactory() {
gfm,
options.markdownConfig.mdx1Compat.comments ? comment : null,
...(options.remarkPlugins ?? []),
+ unusedDirectivesWarning,
].filter((plugin): plugin is MDXPlugin => Boolean(plugin));
// codeCompatPlugin needs to be applied last after user-provided plugins
@@ -167,12 +172,13 @@ async function createProcessorFactory() {
});
return {
- process: async ({content, filePath, frontMatter}) => {
+ process: async ({content, filePath, frontMatter, compilerName}) => {
const vfile = new VFile({
value: content,
path: filePath,
data: {
frontMatter,
+ compilerName,
},
});
return mdxProcessor.process(vfile).then((result) => ({
diff --git a/packages/docusaurus-mdx-loader/src/remark/contentTitle/index.ts b/packages/docusaurus-mdx-loader/src/remark/contentTitle/index.ts
index e2e9977dad923..c26bd1f0cce66 100644
--- a/packages/docusaurus-mdx-loader/src/remark/contentTitle/index.ts
+++ b/packages/docusaurus-mdx-loader/src/remark/contentTitle/index.ts
@@ -35,6 +35,7 @@ const plugin: Plugin = function plugin(
const {toString} = await import('mdast-util-to-string');
visit(root, 'heading', (headingNode: Heading, index, parent) => {
if (headingNode.depth === 1) {
+ vfile.data.compilerName;
vfile.data.contentTitle = toString(headingNode);
if (removeContentTitle) {
parent!.children.splice(index, 1);
diff --git a/packages/docusaurus-mdx-loader/src/remark/mermaid/index.ts b/packages/docusaurus-mdx-loader/src/remark/mermaid/index.ts
index 7947d3387e1f9..7a5eac045601c 100644
--- a/packages/docusaurus-mdx-loader/src/remark/mermaid/index.ts
+++ b/packages/docusaurus-mdx-loader/src/remark/mermaid/index.ts
@@ -6,6 +6,8 @@
*/
import visit from 'unist-util-visit';
+import {transformNode} from '../utils';
+
// @ts-expect-error: TODO see https://github.com/microsoft/TypeScript/issues/49721
import type {Transformer} from 'unified';
import type {Code} from 'mdast';
@@ -16,10 +18,10 @@ import type {Code} from 'mdast';
// by theme-mermaid itself
export default function plugin(): Transformer {
return (root) => {
- visit(root, 'code', (node: Code, index, parent) => {
+ visit(root, 'code', (node: Code) => {
if (node.lang === 'mermaid') {
// TODO migrate to mdxJsxFlowElement? cf admonitions
- parent!.children.splice(index, 1, {
+ transformNode(node, {
type: 'mermaidCodeBlock',
data: {
hName: 'mermaid',
diff --git a/packages/docusaurus-mdx-loader/src/remark/transformImage/index.ts b/packages/docusaurus-mdx-loader/src/remark/transformImage/index.ts
index 5bd9c1d9a9915..956de8a8df477 100644
--- a/packages/docusaurus-mdx-loader/src/remark/transformImage/index.ts
+++ b/packages/docusaurus-mdx-loader/src/remark/transformImage/index.ts
@@ -20,7 +20,7 @@ import visit from 'unist-util-visit';
import escapeHtml from 'escape-html';
import sizeOf from 'image-size';
import logger from '@docusaurus/logger';
-import {assetRequireAttributeValue} from '../utils';
+import {assetRequireAttributeValue, transformNode} from '../utils';
// @ts-expect-error: TODO see https://github.com/microsoft/TypeScript/issues/49721
import type {Transformer} from 'unified';
// @ts-expect-error: TODO see https://github.com/microsoft/TypeScript/issues/49721
@@ -110,14 +110,12 @@ ${(err as Error).message}`;
}
}
- Object.keys(jsxNode).forEach(
- (key) => delete jsxNode[key as keyof typeof jsxNode],
- );
-
- jsxNode.type = 'mdxJsxTextElement';
- jsxNode.name = 'img';
- jsxNode.attributes = attributes;
- jsxNode.children = [];
+ transformNode(jsxNode, {
+ type: 'mdxJsxTextElement',
+ name: 'img',
+ attributes,
+ children: [],
+ });
}
async function ensureImageFileExist(imagePath: string, sourceFilePath: string) {
diff --git a/packages/docusaurus-mdx-loader/src/remark/transformLinks/index.ts b/packages/docusaurus-mdx-loader/src/remark/transformLinks/index.ts
index 71086023345ce..bddf42613f4f3 100644
--- a/packages/docusaurus-mdx-loader/src/remark/transformLinks/index.ts
+++ b/packages/docusaurus-mdx-loader/src/remark/transformLinks/index.ts
@@ -17,7 +17,7 @@ import {
} from '@docusaurus/utils';
import visit from 'unist-util-visit';
import escapeHtml from 'escape-html';
-import {assetRequireAttributeValue} from '../utils';
+import {assetRequireAttributeValue, transformNode} from '../utils';
// @ts-expect-error: TODO see https://github.com/microsoft/TypeScript/issues/49721
import type {Transformer} from 'unified';
// @ts-expect-error: TODO see https://github.com/microsoft/TypeScript/issues/49721
@@ -90,14 +90,12 @@ async function toAssetRequireNode(
const {children} = node;
- Object.keys(jsxNode).forEach(
- (key) => delete jsxNode[key as keyof typeof jsxNode],
- );
-
- jsxNode.type = 'mdxJsxTextElement';
- jsxNode.name = 'a';
- jsxNode.attributes = attributes;
- jsxNode.children = children;
+ transformNode(jsxNode, {
+ type: 'mdxJsxTextElement',
+ name: 'a',
+ attributes,
+ children,
+ });
}
async function ensureAssetFileExist(assetPath: string, sourceFilePath: string) {
diff --git a/packages/docusaurus-mdx-loader/src/remark/unusedDirectives/__tests__/__fixtures__/containerDirectives.md b/packages/docusaurus-mdx-loader/src/remark/unusedDirectives/__tests__/__fixtures__/containerDirectives.md
new file mode 100644
index 0000000000000..2a416f0ca72bc
--- /dev/null
+++ b/packages/docusaurus-mdx-loader/src/remark/unusedDirectives/__tests__/__fixtures__/containerDirectives.md
@@ -0,0 +1,19 @@
+:::danger
+
+Take care of snowstorms...
+
+:::
+
+:::unusedDirective
+
+unused directive content
+
+:::
+
+:::NotAContainerDirective with a phrase after
+
+:::
+
+Phrase before :::NotAContainerDirective
+
+:::
diff --git a/packages/docusaurus-mdx-loader/src/remark/unusedDirectives/__tests__/__fixtures__/leafDirectives.md b/packages/docusaurus-mdx-loader/src/remark/unusedDirectives/__tests__/__fixtures__/leafDirectives.md
new file mode 100644
index 0000000000000..73fd60c77b676
--- /dev/null
+++ b/packages/docusaurus-mdx-loader/src/remark/unusedDirectives/__tests__/__fixtures__/leafDirectives.md
@@ -0,0 +1,5 @@
+::unusedLeafDirective
+
+Leaf directive in a phrase ::NotALeafDirective
+
+::NotALeafDirective with a phrase after
diff --git a/packages/docusaurus-mdx-loader/src/remark/unusedDirectives/__tests__/__fixtures__/textDirectives.md b/packages/docusaurus-mdx-loader/src/remark/unusedDirectives/__tests__/__fixtures__/textDirectives.md
new file mode 100644
index 0000000000000..9410feb923bec
--- /dev/null
+++ b/packages/docusaurus-mdx-loader/src/remark/unusedDirectives/__tests__/__fixtures__/textDirectives.md
@@ -0,0 +1,17 @@
+Simple: textDirective1
+
+```sh
+Simple: textDirectiveCode
+```
+
+Simple:textDirective2
+
+Simple:textDirective3[label]
+
+Simple:textDirective4{age=42}
+
+Simple:textDirective5
+
+```sh
+Simple:textDirectiveCode
+```
diff --git a/packages/docusaurus-mdx-loader/src/remark/unusedDirectives/__tests__/__snapshots__/index.test.ts.snap b/packages/docusaurus-mdx-loader/src/remark/unusedDirectives/__tests__/__snapshots__/index.test.ts.snap
new file mode 100644
index 0000000000000..46285b8894e14
--- /dev/null
+++ b/packages/docusaurus-mdx-loader/src/remark/unusedDirectives/__tests__/__snapshots__/index.test.ts.snap
@@ -0,0 +1,86 @@
+// Jest Snapshot v1, https://goo.gl/fbAQLP
+
+exports[`directives remark plugin - client compiler default behavior for container directives: console 1`] = `
+[
+ [
+ "[WARNING] Docusaurus found 1 unused Markdown directives in file "packages/docusaurus-mdx-loader/src/remark/unusedDirectives/__tests__/__fixtures__/containerDirectives.md"
+- :::unusedDirective (7:1)
+Your content might render in an unexpected way. Visit https://github.com/facebook/docusaurus/pull/9394 to find out why and how to fix it.",
+ ],
+]
+`;
+
+exports[`directives remark plugin - client compiler default behavior for container directives: result 1`] = `
+"Take care of snowstorms...
+
+:::NotAContainerDirective with a phrase after
+:::
+Phrase before :::NotAContainerDirective
+:::
"
+`;
+
+exports[`directives remark plugin - client compiler default behavior for leaf directives: console 1`] = `
+[
+ [
+ "[WARNING] Docusaurus found 1 unused Markdown directives in file "packages/docusaurus-mdx-loader/src/remark/unusedDirectives/__tests__/__fixtures__/leafDirectives.md"
+- ::unusedLeafDirective (1:1)
+Your content might render in an unexpected way. Visit https://github.com/facebook/docusaurus/pull/9394 to find out why and how to fix it.",
+ ],
+]
+`;
+
+exports[`directives remark plugin - client compiler default behavior for leaf directives: result 1`] = `
+"
+Leaf directive in a phrase ::NotALeafDirective
+::NotALeafDirective with a phrase after
"
+`;
+
+exports[`directives remark plugin - client compiler default behavior for text directives: console 1`] = `
+[
+ [
+ "[WARNING] Docusaurus found 2 unused Markdown directives in file "packages/docusaurus-mdx-loader/src/remark/unusedDirectives/__tests__/__fixtures__/textDirectives.md"
+- :textDirective3 (9:7)
+- :textDirective4 (11:7)
+Your content might render in an unexpected way. Visit https://github.com/facebook/docusaurus/pull/9394 to find out why and how to fix it.",
+ ],
+]
+`;
+
+exports[`directives remark plugin - client compiler default behavior for text directives: result 1`] = `
+"Simple: textDirective1
+Simple: textDirectiveCode
+
+Simple:textDirective2
+Simple
label
+Simple
+Simple:textDirective5
+Simple:textDirectiveCode
+
"
+`;
+
+exports[`directives remark plugin - server compiler default behavior for container directives: result 1`] = `
+"Take care of snowstorms...
+
+:::NotAContainerDirective with a phrase after
+:::
+Phrase before :::NotAContainerDirective
+:::
"
+`;
+
+exports[`directives remark plugin - server compiler default behavior for leaf directives: result 1`] = `
+"
+Leaf directive in a phrase ::NotALeafDirective
+::NotALeafDirective with a phrase after
"
+`;
+
+exports[`directives remark plugin - server compiler default behavior for text directives: result 1`] = `
+"Simple: textDirective1
+Simple: textDirectiveCode
+
+Simple:textDirective2
+Simple
label
+Simple
+Simple:textDirective5
+Simple:textDirectiveCode
+
"
+`;
diff --git a/packages/docusaurus-mdx-loader/src/remark/unusedDirectives/__tests__/index.test.ts b/packages/docusaurus-mdx-loader/src/remark/unusedDirectives/__tests__/index.test.ts
new file mode 100644
index 0000000000000..77910e1e442c3
--- /dev/null
+++ b/packages/docusaurus-mdx-loader/src/remark/unusedDirectives/__tests__/index.test.ts
@@ -0,0 +1,111 @@
+/**
+ * Copyright (c) Facebook, Inc. and its affiliates.
+ *
+ * This source code is licensed under the MIT license found in the
+ * LICENSE file in the root directory of this source tree.
+ */
+
+import path from 'path';
+import remark2rehype from 'remark-rehype';
+import stringify from 'rehype-stringify';
+import vfile from 'to-vfile';
+import plugin from '../index';
+import admonition from '../../admonitions';
+import type {WebpackCompilerName} from '@docusaurus/utils';
+
+const processFixture = async (
+ name: string,
+ {compilerName}: {compilerName: WebpackCompilerName},
+) => {
+ const {remark} = await import('remark');
+ const {default: directives} = await import('remark-directive');
+
+ const filePath = path.join(__dirname, '__fixtures__', `${name}.md`);
+ const file = await vfile.read(filePath);
+ file.data.compilerName = compilerName;
+
+ const result = await remark()
+ .use(directives)
+ .use(admonition)
+ .use(plugin)
+ .use(remark2rehype)
+ .use(stringify)
+ .process(file);
+
+ return result.value;
+};
+
+describe('directives remark plugin - client compiler', () => {
+ const consoleMock = jest.spyOn(console, 'warn').mockImplementation(() => {});
+ beforeEach(() => jest.clearAllMocks());
+
+ const options = {compilerName: 'client'} as const;
+
+ it('default behavior for container directives', async () => {
+ const result = await processFixture('containerDirectives', options);
+ expect(result).toMatchSnapshot('result');
+ expect(consoleMock).toHaveBeenCalledTimes(1);
+ expect(consoleMock.mock.calls).toMatchSnapshot('console');
+ });
+
+ it('default behavior for leaf directives', async () => {
+ const result = await processFixture('leafDirectives', options);
+ expect(result).toMatchSnapshot('result');
+ expect(consoleMock).toHaveBeenCalledTimes(1);
+ expect(consoleMock.mock.calls).toMatchSnapshot('console');
+ });
+
+ it('default behavior for text directives', async () => {
+ const result = await processFixture('textDirectives', options);
+ expect(result).toMatchSnapshot('result');
+ expect(consoleMock).toHaveBeenCalledTimes(1);
+ expect(consoleMock.mock.calls).toMatchSnapshot('console');
+ });
+});
+
+describe('directives remark plugin - server compiler', () => {
+ const consoleMock = jest.spyOn(console, 'warn').mockImplementation(() => {});
+ beforeEach(() => jest.clearAllMocks());
+
+ const options = {compilerName: 'server'} as const;
+
+ it('default behavior for container directives', async () => {
+ const result = await processFixture('containerDirectives', options);
+ expect(result).toMatchSnapshot('result');
+ expect(consoleMock).toHaveBeenCalledTimes(0);
+ });
+
+ it('default behavior for leaf directives', async () => {
+ const result = await processFixture('leafDirectives', options);
+ expect(result).toMatchSnapshot('result');
+ expect(consoleMock).toHaveBeenCalledTimes(0);
+ });
+
+ it('default behavior for text directives', async () => {
+ const result = await processFixture('textDirectives', options);
+ expect(result).toMatchSnapshot('result');
+ expect(consoleMock).toHaveBeenCalledTimes(0);
+ });
+});
+
+describe('directives remark plugin - client result === server result', () => {
+ // It is important that client/server outputs are exactly the same
+ // otherwise React hydration mismatches can occur
+ async function testSameResult(name: string) {
+ const resultClient = await processFixture(name, {compilerName: 'client'});
+ const resultServer = await processFixture(name, {compilerName: 'server'});
+ expect(resultClient).toEqual(resultServer);
+ }
+
+ it('for containerDirectives', async () => {
+ await testSameResult('containerDirectives');
+ });
+
+ it('for leafDirectives', async () => {
+ await testSameResult('leafDirectives');
+ });
+
+ it('for textDirectives', async () => {
+ await testSameResult('textDirectives');
+ });
+});
diff --git a/packages/docusaurus-mdx-loader/src/remark/unusedDirectives/index.ts b/packages/docusaurus-mdx-loader/src/remark/unusedDirectives/index.ts
new file mode 100644
index 0000000000000..99d0a1dfa6e70
--- /dev/null
+++ b/packages/docusaurus-mdx-loader/src/remark/unusedDirectives/index.ts
@@ -0,0 +1,162 @@
+/**
+ * Copyright (c) Facebook, Inc. and its affiliates.
+ *
+ * This source code is licensed under the MIT license found in the
+ * LICENSE file in the root directory of this source tree.
+ */
+import path from 'path';
+import process from 'process';
+import visit from 'unist-util-visit';
+import logger from '@docusaurus/logger';
+import {posixPath} from '@docusaurus/utils';
+import {transformNode} from '../utils';
+
+// @ts-expect-error: TODO see https://github.com/microsoft/TypeScript/issues/49721
+import type {Transformer, Processor, Parent} from 'unified';
+import type {
+ Directive,
+ TextDirective,
+ // @ts-expect-error: TODO see https://github.com/microsoft/TypeScript/issues/49721
+} from 'mdast-util-directive';
+
+// TODO as of April 2023, no way to import/re-export this ESM type easily :/
+// This might change soon, likely after TS 5.2
+// See https://github.com/microsoft/TypeScript/issues/49721#issuecomment-1517839391
+// import type {Plugin} from 'unified';
+type Plugin = any; // TODO fix this asap
+
+type DirectiveType = Directive['type'];
+
+const directiveTypes: DirectiveType[] = [
+ 'containerDirective',
+ 'leafDirective',
+ 'textDirective',
+];
+
+const directivePrefixMap: {[key in DirectiveType]: string} = {
+ textDirective: ':',
+ leafDirective: '::',
+ containerDirective: ':::',
+};
+
+function formatDirectiveName(directive: Directive) {
+ const prefix = directivePrefixMap[directive.type];
+ if (!prefix) {
+ throw new Error(
+ `unexpected, no prefix found for directive of type ${directive.type}`,
+ );
+ }
+ // To simplify we don't display the eventual label/props of directives
+ return `${prefix}${directive.name}`;
+}
+
+function formatDirectivePosition(directive: Directive): string | undefined {
+ return directive.position?.start
+ ? logger.interpolate`number=${directive.position.start.line}:number=${directive.position.start.column}`
+ : undefined;
+}
+
+function formatUnusedDirectiveMessage(directive: Directive) {
+ const name = formatDirectiveName(directive);
+ const position = formatDirectivePosition(directive);
+
+ return `- ${name} ${position ? `(${position})` : ''}`;
+}
+
+function formatUnusedDirectivesMessage({
+ directives,
+ filePath,
+}: {
+ directives: Directive[];
+ filePath: string;
+}): string {
+ const supportUrl = 'https://github.com/facebook/docusaurus/pull/9394';
+ const customPath = posixPath(path.relative(process.cwd(), filePath));
+ const warningTitle = logger.interpolate`Docusaurus found ${directives.length} unused Markdown directives in file path=${customPath}`;
+ const customSupportUrl = logger.interpolate`url=${supportUrl}`;
+ const warningMessages = directives
+ .map(formatUnusedDirectiveMessage)
+ .join('\n');
+
+ return `${warningTitle}
+${warningMessages}
+Your content might render in an unexpected way. Visit ${customSupportUrl} to find out why and how to fix it.`;
+}
+
+function logUnusedDirectivesWarning({
+ directives,
+ filePath,
+}: {
+ directives: Directive[];
+ filePath: string;
+}) {
+ if (directives.length > 0) {
+ const message = formatUnusedDirectivesMessage({
+ directives,
+ filePath,
+ });
+ logger.warn(message);
+ }
+}
+
+function isTextDirective(directive: Directive): directive is TextDirective {
+ return directive.type === 'textDirective';
+}
+
+// A simple text directive is one without any label/props
+function isSimpleTextDirective(
+ directive: Directive,
+): directive is TextDirective {
+ if (isTextDirective(directive)) {
+ // Attributes in MDAST = Directive props
+ const hasAttributes =
+ directive.attributes && Object.keys(directive.attributes).length > 0;
+ // Children in MDAST = Directive label
+ const hasChildren = directive.children.length > 0;
+ return !hasAttributes && !hasChildren;
+ }
+ return false;
+}
+
+function transformSimpleTextDirectiveToString(textDirective: Directive) {
+ transformNode(textDirective, {
+ type: 'text',
+ value: `:${textDirective.name}`, // We ignore label/props on purpose here
+ });
+}
+
+function isUnusedDirective(directive: Directive) {
+ // If directive data is set (notably hName/hProperties set by admonitions)
+ // this usually means the directive has been handled by another plugin
+ return !directive.data;
+}
+
+const plugin: Plugin = function plugin(this: Processor): Transformer {
+ return (tree, file) => {
+ const unusedDirectives: Directive[] = [];
+
+ visit(tree, directiveTypes, (directive: Directive) => {
+ // If directive data is set (notably hName/hProperties set by admonitions)
+ // this usually means the directive has been handled by another plugin
+ if (isUnusedDirective(directive)) {
+ if (isSimpleTextDirective(directive)) {
+ transformSimpleTextDirectiveToString(directive);
+ } else {
+ unusedDirectives.push(directive);
+ }
+ }
+ });
+
+ // We only enable these warnings for the client compiler
+ // This avoids emitting duplicate warnings in prod mode
+ // Note: the client compiler is used in both dev/prod modes
+ if (file.data.compilerName === 'client') {
+ logUnusedDirectivesWarning({
+ directives: unusedDirectives,
+ filePath: file.path,
+ });
+ }
+ };
+};
+
+export default plugin;
diff --git a/packages/docusaurus-mdx-loader/src/remark/utils/index.ts b/packages/docusaurus-mdx-loader/src/remark/utils/index.ts
index 46e9591762397..3b29be807e2db 100644
--- a/packages/docusaurus-mdx-loader/src/remark/utils/index.ts
+++ b/packages/docusaurus-mdx-loader/src/remark/utils/index.ts
@@ -6,7 +6,7 @@
*/
import escapeHtml from 'escape-html';
-import type {Parent} from 'unist';
+import type {Parent, Node} from 'unist';
import type {PhrasingContent, Heading} from 'mdast';
import type {
MdxJsxAttribute,
@@ -15,6 +15,27 @@ import type {
// @ts-expect-error: TODO see https://github.com/microsoft/TypeScript/issues/49721
} from 'mdast-util-mdx';
+/**
+ * Util to transform one node type to another node type
+ * The input node is mutated in place
+ * @param node the node to mutate
+ * @param newNode what the original node should become become
+ */
+export function transformNode(
+ node: Node,
+ newNode: NewNode,
+): NewNode {
+ Object.keys(node).forEach((key) => {
+ // @ts-expect-error: unsafe but ok
+ delete node[key];
+ });
+ Object.keys(newNode).forEach((key) => {
+ // @ts-expect-error: unsafe but ok
+ node[key] = newNode[key];
+ });
+ return node as NewNode;
+}
+
export function stringifyContent(
node: Parent,
toString: (param: unknown) => string, // TODO weird but works
diff --git a/packages/docusaurus-mdx-loader/src/vfile-datamap.d.mts b/packages/docusaurus-mdx-loader/src/vfile-datamap.d.mts
new file mode 100644
index 0000000000000..2538305411c82
--- /dev/null
+++ b/packages/docusaurus-mdx-loader/src/vfile-datamap.d.mts
@@ -0,0 +1,21 @@
+/**
+ * Copyright (c) Facebook, Inc. and its affiliates.
+ *
+ * This source code is licensed under the MIT license found in the
+ * LICENSE file in the root directory of this source tree.
+ */
+
+import type {WebpackCompilerName} from '@docusaurus/utils';
+
+declare module 'vfile' {
+ /*
+ This map registers the type of the data key of a VFile (TypeScript type).
+ This type can be augmented to register custom data types.
+ See https://github.com/vfile/vfile#datamap
+ */
+ interface DataMap {
+ frontMatter: {[key: string]: unknown};
+ compilerName: WebpackCompilerName;
+ contentTitle?: string;
+ }
+}
diff --git a/packages/docusaurus-utils/src/index.ts b/packages/docusaurus-utils/src/index.ts
index d5cabc65301aa..5bb77a0c054f0 100644
--- a/packages/docusaurus-utils/src/index.ts
+++ b/packages/docusaurus-utils/src/index.ts
@@ -98,7 +98,11 @@ export {
createMatcher,
createAbsoluteFilePathMatcher,
} from './globUtils';
-export {getFileLoaderUtils} from './webpackUtils';
+export {
+ getFileLoaderUtils,
+ getWebpackLoaderCompilerName,
+ type WebpackCompilerName,
+} from './webpackUtils';
export {escapeShellArg} from './shellUtils';
export {loadFreshModule} from './moduleUtils';
export {
diff --git a/packages/docusaurus-utils/src/webpackUtils.ts b/packages/docusaurus-utils/src/webpackUtils.ts
index c1195d5764b37..411b5a0fee22b 100644
--- a/packages/docusaurus-utils/src/webpackUtils.ts
+++ b/packages/docusaurus-utils/src/webpackUtils.ts
@@ -11,7 +11,25 @@ import {
WEBPACK_URL_LOADER_LIMIT,
OUTPUT_STATIC_ASSETS_DIR_NAME,
} from './constants';
-import type {RuleSetRule} from 'webpack';
+import type {RuleSetRule, LoaderContext} from 'webpack';
+
+export type WebpackCompilerName = 'server' | 'client';
+
+export function getWebpackLoaderCompilerName(
+ context: LoaderContext,
+): WebpackCompilerName {
+ // eslint-disable-next-line no-underscore-dangle
+ const compilerName = context._compiler?.name;
+ switch (compilerName) {
+ case 'server':
+ case 'client':
+ return compilerName;
+ default:
+ throw new Error(
+ `Cannot get valid Docusaurus webpack compiler name. Found compilerName=${compilerName}`,
+ );
+ }
+}
type AssetFolder = 'images' | 'files' | 'fonts' | 'medias';
diff --git a/project-words.txt b/project-words.txt
index 1f9e8bada8efd..fdc8d43a57fc6 100644
--- a/project-words.txt
+++ b/project-words.txt
@@ -62,6 +62,7 @@ customizability
dabit
daishi
datagit
+datamap
datas
dbaeumer
décembre