@docusaurus/utils 2.0.0-beta.8e9b829d9 → 2.0.0-beta.9

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.
@@ -129,6 +129,19 @@ describe('createExcerpt', () => {
129
129
  `),
130
130
  ).toEqual('Markdown title');
131
131
  });
132
+
133
+ test('should create excerpt for content with various code blocks', () => {
134
+ expect(
135
+ createExcerpt(dedent`
136
+ \`\`\`jsx
137
+ import React from 'react';
138
+ import Layout from '@theme/Layout';
139
+ \`\`\`
140
+
141
+ Lorem \`ipsum\` dolor sit amet, consectetur \`adipiscing elit\`.
142
+ `),
143
+ ).toEqual('Lorem ipsum dolor sit amet, consectetur adipiscing elit.');
144
+ });
132
145
  });
133
146
 
134
147
  describe('parseMarkdownContentTitle', () => {
@@ -0,0 +1,133 @@
1
+ /**
2
+ * Copyright (c) Facebook, Inc. and its affiliates.
3
+ *
4
+ * This source code is licensed under the MIT license found in the
5
+ * LICENSE file in the root directory of this source tree.
6
+ */
7
+
8
+ import {mdxToHtml} from '../mdxUtils';
9
+
10
+ describe('mdxToHtml', () => {
11
+ test('work with simple markdown', () => {
12
+ const mdxString = `
13
+ # title
14
+
15
+ title text **bold**
16
+
17
+ ## subtitle
18
+
19
+ subtitle text *italic*
20
+
21
+ > Quote
22
+
23
+ `;
24
+
25
+ expect(mdxToHtml(mdxString)).toMatchInlineSnapshot(
26
+ `"<h1>title</h1><p>title text <strong>bold</strong></p><h2>subtitle</h2><p>subtitle text <em>italic</em></p><blockquote><p>Quote</p></blockquote>"`,
27
+ );
28
+ });
29
+
30
+ test('work with MDX imports', () => {
31
+ const mdxString = `
32
+ # title
33
+
34
+ import Tabs from '@theme/Tabs';
35
+ import TabItem from '@theme/TabItem';
36
+
37
+ text
38
+
39
+ `;
40
+
41
+ expect(mdxToHtml(mdxString)).toMatchInlineSnapshot(
42
+ `"<h1>title</h1><p>text</p>"`,
43
+ );
44
+ });
45
+
46
+ test('work with MDX exports', () => {
47
+ const mdxString = `
48
+ # title
49
+
50
+ export const someExport = 42
51
+
52
+ export const MyLocalComponent = () => "result"
53
+
54
+ export const toc = [
55
+ {id: "title",label: "title"}
56
+ ]
57
+
58
+ text
59
+
60
+
61
+ `;
62
+
63
+ expect(mdxToHtml(mdxString)).toMatchInlineSnapshot(
64
+ `"<h1>title</h1><p>text</p>"`,
65
+ );
66
+ });
67
+
68
+ test('work with MDX Tabs', () => {
69
+ const mdxString = `
70
+ # title
71
+
72
+ import Tabs from '@theme/Tabs';
73
+ import TabItem from '@theme/TabItem';
74
+
75
+ <Tabs>
76
+ <TabItem value="apple" label="Apple">
77
+ This is an apple 🍎
78
+ </TabItem>
79
+ <TabItem value="orange" label="Orange">
80
+ This is an orange 🍊
81
+ </TabItem>
82
+ </Tabs>
83
+
84
+ text
85
+
86
+
87
+ `;
88
+
89
+ // TODO this is not an ideal behavior!
90
+ // There is a warning "Component TabItem was not imported, exported, or provided by MDXProvider as global scope"
91
+ // Theme + MDX config should provide a list of React components to put in MDX scope
92
+ expect(mdxToHtml(mdxString)).toMatchInlineSnapshot(
93
+ `"<h1>title</h1><div><div value=\\"apple\\" label=\\"Apple\\">This is an apple 🍎</div><div value=\\"orange\\" label=\\"Orange\\">This is an orange 🍊</div></div><p>text</p>"`,
94
+ );
95
+ });
96
+
97
+ test('work with MDX Tabs with ```mdx-code-block', () => {
98
+ const mdxString = `
99
+ # title
100
+
101
+ import Tabs from '@theme/Tabs';
102
+ import TabItem from '@theme/TabItem';
103
+
104
+ \`\`\`mdx-code-block
105
+ <Tabs>
106
+ <TabItem value="apple" label="Apple">
107
+ This is an apple 🍎
108
+ </TabItem>
109
+ <TabItem value="orange" label="Orange">
110
+ This is an orange 🍊
111
+ </TabItem>
112
+ </Tabs>
113
+ \`\`\`
114
+
115
+ text
116
+
117
+ `;
118
+
119
+ // TODO bad behavior!
120
+ // ```mdx-code-block should be unwrapped and inner MDX content should be evaluated
121
+ expect(mdxToHtml(mdxString)).toMatchInlineSnapshot(`
122
+ "<h1>title</h1><pre><code class=\\"language-mdx-code-block\\">&lt;Tabs&gt;
123
+ &lt;TabItem value=&quot;apple&quot; label=&quot;Apple&quot;&gt;
124
+ This is an apple 🍎
125
+ &lt;/TabItem&gt;
126
+ &lt;TabItem value=&quot;orange&quot; label=&quot;Orange&quot;&gt;
127
+ This is an orange 🍊
128
+ &lt;/TabItem&gt;
129
+ &lt;/Tabs&gt;
130
+ </code></pre><p>text</p>"
131
+ `);
132
+ });
133
+ });
@@ -0,0 +1,117 @@
1
+ /**
2
+ * Copyright (c) Facebook, Inc. and its affiliates.
3
+ *
4
+ * This source code is licensed under the MIT license found in the
5
+ * LICENSE file in the root directory of this source tree.
6
+ */
7
+
8
+ import {normalizeUrl} from '../normalizeUrl';
9
+
10
+ describe('normalizeUrl', () => {
11
+ test('should normalize urls correctly', () => {
12
+ const asserts = [
13
+ {
14
+ input: ['/', ''],
15
+ output: '/',
16
+ },
17
+ {
18
+ input: ['', '/'],
19
+ output: '/',
20
+ },
21
+ {
22
+ input: ['/'],
23
+ output: '/',
24
+ },
25
+ {
26
+ input: [''],
27
+ output: '',
28
+ },
29
+ {
30
+ input: ['/', '/'],
31
+ output: '/',
32
+ },
33
+ {
34
+ input: ['/', 'docs'],
35
+ output: '/docs',
36
+ },
37
+ {
38
+ input: ['/', 'docs', 'en', 'next', 'blog'],
39
+ output: '/docs/en/next/blog',
40
+ },
41
+ {
42
+ input: ['/test/', '/docs', 'ro', 'doc1'],
43
+ output: '/test/docs/ro/doc1',
44
+ },
45
+ {
46
+ input: ['/test/', '/', 'ro', 'doc1'],
47
+ output: '/test/ro/doc1',
48
+ },
49
+ {
50
+ input: ['/', '/', '2020/02/29/leap-day'],
51
+ output: '/2020/02/29/leap-day',
52
+ },
53
+ {
54
+ input: ['', '/', 'ko', 'hello'],
55
+ output: '/ko/hello',
56
+ },
57
+ {
58
+ input: ['hello', 'world'],
59
+ output: 'hello/world',
60
+ },
61
+ {
62
+ input: ['http://www.google.com/', 'foo/bar', '?test=123'],
63
+ output: 'http://www.google.com/foo/bar?test=123',
64
+ },
65
+ {
66
+ input: ['http:', 'www.google.com///', 'foo/bar', '?test=123'],
67
+ output: 'http://www.google.com/foo/bar?test=123',
68
+ },
69
+ {
70
+ input: ['http://foobar.com', '', 'test'],
71
+ output: 'http://foobar.com/test',
72
+ },
73
+ {
74
+ input: ['http://foobar.com', '', 'test', '/'],
75
+ output: 'http://foobar.com/test/',
76
+ },
77
+ {
78
+ input: ['/', '', 'hello', '', '/', '/', '', '/', '/world'],
79
+ output: '/hello/world',
80
+ },
81
+ {
82
+ input: ['', '', '/tt', 'ko', 'hello'],
83
+ output: '/tt/ko/hello',
84
+ },
85
+ {
86
+ input: ['', '///hello///', '', '///world'],
87
+ output: '/hello/world',
88
+ },
89
+ {
90
+ input: ['', '/hello/', ''],
91
+ output: '/hello/',
92
+ },
93
+ {
94
+ input: ['', '/', ''],
95
+ output: '/',
96
+ },
97
+ {
98
+ input: ['///', '///'],
99
+ output: '/',
100
+ },
101
+ {
102
+ input: ['/', '/hello/world/', '///'],
103
+ output: '/hello/world/',
104
+ },
105
+ ];
106
+ asserts.forEach((testCase) => {
107
+ expect(normalizeUrl(testCase.input)).toBe(testCase.output);
108
+ });
109
+
110
+ expect(() =>
111
+ // @ts-expect-error undefined for test
112
+ normalizeUrl(['http:example.com', undefined]),
113
+ ).toThrowErrorMatchingInlineSnapshot(
114
+ `"Url must be a string. Received undefined"`,
115
+ );
116
+ });
117
+ });
@@ -17,8 +17,10 @@ describe('pathUtils', () => {
17
17
  'endi-lie-fd3': false,
18
18
  'yangshun-tay-48d': false,
19
19
  'yangshun-tay-f3b': false,
20
- 'foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-d46': true,
21
- 'foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-test-1-test-2-787': true,
20
+ 'foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-d46':
21
+ true,
22
+ 'foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-foo-bar-test-1-test-2-787':
23
+ true,
22
24
  };
23
25
  Object.keys(asserts).forEach((path) => {
24
26
  expect(isNameTooLong(path)).toBe(asserts[path]);
@@ -0,0 +1,183 @@
1
+ /**
2
+ * Copyright (c) Facebook, Inc. and its affiliates.
3
+ *
4
+ * This source code is licensed under the MIT license found in the
5
+ * LICENSE file in the root directory of this source tree.
6
+ */
7
+
8
+ import {
9
+ normalizeFrontMatterTag,
10
+ normalizeFrontMatterTags,
11
+ groupTaggedItems,
12
+ Tag,
13
+ } from '../tags';
14
+
15
+ describe('normalizeFrontMatterTag', () => {
16
+ type Input = Parameters<typeof normalizeFrontMatterTag>[1];
17
+ type Output = ReturnType<typeof normalizeFrontMatterTag>;
18
+
19
+ test('should normalize simple string tag', () => {
20
+ const tagsPath = '/all/tags';
21
+ const input: Input = 'tag';
22
+ const expectedOutput: Output = {
23
+ label: 'tag',
24
+ permalink: `${tagsPath}/tag`,
25
+ };
26
+ expect(normalizeFrontMatterTag(tagsPath, input)).toEqual(expectedOutput);
27
+ });
28
+
29
+ test('should normalize complex string tag', () => {
30
+ const tagsPath = '/all/tags';
31
+ const input: Input = 'some more Complex_tag';
32
+ const expectedOutput: Output = {
33
+ label: 'some more Complex_tag',
34
+ permalink: `${tagsPath}/some-more-complex-tag`,
35
+ };
36
+ expect(normalizeFrontMatterTag(tagsPath, input)).toEqual(expectedOutput);
37
+ });
38
+
39
+ test('should normalize simple object tag', () => {
40
+ const tagsPath = '/all/tags';
41
+ const input: Input = {label: 'tag', permalink: 'tagPermalink'};
42
+ const expectedOutput: Output = {
43
+ label: 'tag',
44
+ permalink: `${tagsPath}/tagPermalink`,
45
+ };
46
+ expect(normalizeFrontMatterTag(tagsPath, input)).toEqual(expectedOutput);
47
+ });
48
+
49
+ test('should normalize complex string tag', () => {
50
+ const tagsPath = '/all/tags';
51
+ const input: Input = {
52
+ label: 'tag complex Label',
53
+ permalink: '/MoreComplex/Permalink',
54
+ };
55
+ const expectedOutput: Output = {
56
+ label: 'tag complex Label',
57
+ permalink: `${tagsPath}/MoreComplex/Permalink`,
58
+ };
59
+ expect(normalizeFrontMatterTag(tagsPath, input)).toEqual(expectedOutput);
60
+ });
61
+ });
62
+
63
+ describe('normalizeFrontMatterTags', () => {
64
+ type Input = Parameters<typeof normalizeFrontMatterTags>[1];
65
+ type Output = ReturnType<typeof normalizeFrontMatterTags>;
66
+
67
+ test('should normalize string list', () => {
68
+ const tagsPath = '/all/tags';
69
+ const input: Input = ['tag 1', 'tag-1', 'tag 3', 'tag1', 'tag-2'];
70
+ // Keep user input order but remove tags that lead to same permalink
71
+ const expectedOutput: Output = [
72
+ {
73
+ label: 'tag 1',
74
+ permalink: `${tagsPath}/tag-1`,
75
+ },
76
+ {
77
+ label: 'tag 3',
78
+ permalink: `${tagsPath}/tag-3`,
79
+ },
80
+ {
81
+ label: 'tag-2',
82
+ permalink: `${tagsPath}/tag-2`,
83
+ },
84
+ ];
85
+ expect(normalizeFrontMatterTags(tagsPath, input)).toEqual(expectedOutput);
86
+ });
87
+
88
+ test('should normalize complex mixed list', () => {
89
+ const tagsPath = '/all/tags';
90
+ const input: Input = [
91
+ 'tag 1',
92
+ {label: 'tag-1', permalink: '/tag-1'},
93
+ 'tag 3',
94
+ 'tag1',
95
+ {label: 'tag 4', permalink: '/tag4Permalink'},
96
+ ];
97
+ // Keep user input order but remove tags that lead to same permalink
98
+ const expectedOutput: Output = [
99
+ {
100
+ label: 'tag 1',
101
+ permalink: `${tagsPath}/tag-1`,
102
+ },
103
+ {
104
+ label: 'tag 3',
105
+ permalink: `${tagsPath}/tag-3`,
106
+ },
107
+ {
108
+ label: 'tag 4',
109
+ permalink: `${tagsPath}/tag4Permalink`,
110
+ },
111
+ ];
112
+ expect(normalizeFrontMatterTags(tagsPath, input)).toEqual(expectedOutput);
113
+ });
114
+ });
115
+
116
+ describe('groupTaggedItems', () => {
117
+ type SomeTaggedItem = {
118
+ id: string;
119
+ nested: {
120
+ tags: Tag[];
121
+ };
122
+ };
123
+ function groupItems(items: SomeTaggedItem[]) {
124
+ return groupTaggedItems(items, (item) => item.nested.tags);
125
+ }
126
+
127
+ type Input = Parameters<typeof groupItems>[0];
128
+ type Output = ReturnType<typeof groupItems>;
129
+
130
+ test('should group items by tag permalink', () => {
131
+ const tagGuide = {label: 'Guide', permalink: '/guide'};
132
+ const tagTutorial = {label: 'Tutorial', permalink: '/tutorial'};
133
+ const tagAPI = {label: 'API', permalink: '/api'};
134
+
135
+ // This one will be grouped under same permalink and label is ignored
136
+ const tagTutorialOtherLabel = {
137
+ label: 'TutorialOtherLabel',
138
+ permalink: '/tutorial',
139
+ };
140
+
141
+ const item1: SomeTaggedItem = {
142
+ id: '1',
143
+ nested: {
144
+ tags: [
145
+ tagGuide,
146
+ tagTutorial,
147
+ tagAPI,
148
+ // Add some duplicates on purpose: they should be filtered
149
+ tagGuide,
150
+ tagTutorialOtherLabel,
151
+ ],
152
+ },
153
+ };
154
+ const item2: SomeTaggedItem = {
155
+ id: '2',
156
+ nested: {
157
+ tags: [tagAPI],
158
+ },
159
+ };
160
+ const item3: SomeTaggedItem = {
161
+ id: '3',
162
+ nested: {
163
+ tags: [tagTutorial],
164
+ },
165
+ };
166
+ const item4: SomeTaggedItem = {
167
+ id: '4',
168
+ nested: {
169
+ tags: [tagTutorialOtherLabel],
170
+ },
171
+ };
172
+
173
+ const input: Input = [item1, item2, item3, item4];
174
+
175
+ const expectedOutput: Output = {
176
+ '/guide': {tag: tagGuide, items: [item1]},
177
+ '/tutorial': {tag: tagTutorial, items: [item1, item3, item4]},
178
+ '/api': {tag: tagAPI, items: [item1, item2]},
179
+ };
180
+
181
+ expect(groupItems(input)).toEqual(expectedOutput);
182
+ });
183
+ });
@@ -0,0 +1,12 @@
1
+ /**
2
+ * Copyright (c) Facebook, Inc. and its affiliates.
3
+ *
4
+ * This source code is licensed under the MIT license found in the
5
+ * LICENSE file in the root directory of this source tree.
6
+ */
7
+
8
+ // Dependencies with missing typedefs
9
+
10
+ declare module '@mdx-js/runtime';
11
+ declare module 'remark-mdx-remove-imports';
12
+ declare module 'remark-mdx-remove-exports';
package/src/index.ts CHANGED
@@ -18,11 +18,15 @@ import {
18
18
  TranslationFile,
19
19
  } from '@docusaurus/types';
20
20
 
21
- // @ts-expect-error: no typedefs :s
22
21
  import resolvePathnameUnsafe from 'resolve-pathname';
23
22
 
24
23
  import {posixPath as posixPathImport} from './posixPath';
25
24
  import {simpleHash, docuHash} from './hashUtils';
25
+ import {normalizeUrl} from './normalizeUrl';
26
+
27
+ export * from './mdxUtils';
28
+ export * from './normalizeUrl';
29
+ export * from './tags';
26
30
 
27
31
  export const posixPath = posixPathImport;
28
32
 
@@ -190,80 +194,6 @@ export function getSubFolder(file: string, refDir: string): string | null {
190
194
  return match && match[1];
191
195
  }
192
196
 
193
- export function normalizeUrl(rawUrls: string[]): string {
194
- const urls = [...rawUrls];
195
- const resultArray = [];
196
-
197
- let hasStartingSlash = false;
198
- let hasEndingSlash = false;
199
-
200
- // If the first part is a plain protocol, we combine it with the next part.
201
- if (urls[0].match(/^[^/:]+:\/*$/) && urls.length > 1) {
202
- const first = urls.shift();
203
- urls[0] = first + urls[0];
204
- }
205
-
206
- // There must be two or three slashes in the file protocol,
207
- // two slashes in anything else.
208
- const replacement = urls[0].match(/^file:\/\/\//) ? '$1:///' : '$1://';
209
- urls[0] = urls[0].replace(/^([^/:]+):\/*/, replacement);
210
-
211
- // eslint-disable-next-line
212
- for (let i = 0; i < urls.length; i++) {
213
- let component = urls[i];
214
-
215
- if (typeof component !== 'string') {
216
- throw new TypeError(`Url must be a string. Received ${typeof component}`);
217
- }
218
-
219
- if (component === '') {
220
- if (i === urls.length - 1 && hasEndingSlash) {
221
- resultArray.push('/');
222
- }
223
- // eslint-disable-next-line
224
- continue;
225
- }
226
-
227
- if (component !== '/') {
228
- if (i > 0) {
229
- // Removing the starting slashes for each component but the first.
230
- component = component.replace(
231
- /^[/]+/,
232
- // Special case where the first element of rawUrls is empty ["", "/hello"] => /hello
233
- component[0] === '/' && !hasStartingSlash ? '/' : '',
234
- );
235
- }
236
-
237
- hasEndingSlash = component[component.length - 1] === '/';
238
- // Removing the ending slashes for each component but the last.
239
- // For the last component we will combine multiple slashes to a single one.
240
- component = component.replace(/[/]+$/, i < urls.length - 1 ? '' : '/');
241
- }
242
-
243
- hasStartingSlash = true;
244
- resultArray.push(component);
245
- }
246
-
247
- let str = resultArray.join('/');
248
- // Each input component is now separated by a single slash
249
- // except the possible first plain protocol part.
250
-
251
- // Remove trailing slash before parameters or hash.
252
- str = str.replace(/\/(\?|&|#[^!])/g, '$1');
253
-
254
- // Replace ? in parameters with &.
255
- const parts = str.split('?');
256
- str = parts.shift() + (parts.length > 0 ? '?' : '') + parts.join('&');
257
-
258
- // Dedupe forward slashes in the entire path, avoiding protocol slashes.
259
- str = str.replace(/([^:]\/)\/+/g, '$1');
260
-
261
- // Dedupe forward slashes at the beginning of the path.
262
- str = str.replace(/^\/+/g, '/');
263
-
264
- return str;
265
- }
266
-
267
197
  /**
268
198
  * Alias filepath relative to site directory, very useful so that we
269
199
  * don't expose user's site structure.
@@ -497,9 +427,7 @@ export function updateTranslationFileMessages(
497
427
 
498
428
  // Input: ## Some heading {#some-heading}
499
429
  // Output: {text: "## Some heading", id: "some-heading"}
500
- export function parseMarkdownHeadingId(
501
- heading: string,
502
- ): {
430
+ export function parseMarkdownHeadingId(heading: string): {
503
431
  text: string;
504
432
  id?: string;
505
433
  } {
@@ -18,6 +18,7 @@ export function createExcerpt(fileString: string): string | undefined {
18
18
  // Remove Markdown alternate title
19
19
  .replace(/^[^\n]*\n[=]+/g, '')
20
20
  .split('\n');
21
+ let inCode = false;
21
22
 
22
23
  /* eslint-disable no-continue */
23
24
  // eslint-disable-next-line no-restricted-syntax
@@ -32,6 +33,14 @@ export function createExcerpt(fileString: string): string | undefined {
32
33
  continue;
33
34
  }
34
35
 
36
+ // Skip code block line.
37
+ if (fileLine.trim().startsWith('```')) {
38
+ inCode = !inCode;
39
+ continue;
40
+ } else if (inCode) {
41
+ continue;
42
+ }
43
+
35
44
  const cleanedLine = fileLine
36
45
  // Remove HTML tags.
37
46
  .replace(/<[^>]*>/g, '')
@@ -67,9 +76,7 @@ export function createExcerpt(fileString: string): string | undefined {
67
76
  return undefined;
68
77
  }
69
78
 
70
- export function parseFrontMatter(
71
- markdownFileContent: string,
72
- ): {
79
+ export function parseFrontMatter(markdownFileContent: string): {
73
80
  frontMatter: Record<string, unknown>;
74
81
  content: string;
75
82
  } {
@@ -98,10 +105,11 @@ export function parseMarkdownContentTitle(
98
105
 
99
106
  const content = contentUntrimmed.trim();
100
107
 
101
- const IMPORT_STATEMENT = /import\s+(([\w*{}\s\n,]+)from\s+)?["'\s]([@\w/_.-]+)["'\s];?|\n/
102
- .source;
103
- const REGULAR_TITLE = /(?<pattern>#\s*(?<title>[^#\n{]*)+[ \t]*(?<suffix>({#*[\w-]+})|#)?\n*?)/
104
- .source;
108
+ const IMPORT_STATEMENT =
109
+ /import\s+(([\w*{}\s\n,]+)from\s+)?["'\s]([@\w/_.-]+)["'\s];?|\n/.source;
110
+ const REGULAR_TITLE =
111
+ /(?<pattern>#\s*(?<title>[^#\n{]*)+[ \t]*(?<suffix>({#*[\w-]+})|#)?\n*?)/
112
+ .source;
105
113
  const ALTERNATE_TITLE = /(?<pattern>\s*(?<title>[^\n]*)\s*\n[=]+)/.source;
106
114
 
107
115
  const regularTitleMatch = new RegExp(
@@ -141,9 +149,8 @@ export function parseMarkdownString(
141
149
  options?: {removeContentTitle?: boolean},
142
150
  ): ParsedMarkdown {
143
151
  try {
144
- const {frontMatter, content: contentWithoutFrontMatter} = parseFrontMatter(
145
- markdownFileContent,
146
- );
152
+ const {frontMatter, content: contentWithoutFrontMatter} =
153
+ parseFrontMatter(markdownFileContent);
147
154
 
148
155
  const {content, contentTitle} = parseMarkdownContentTitle(
149
156
  contentWithoutFrontMatter,
@@ -176,7 +183,7 @@ export async function parseMarkdownFile(
176
183
  return parseMarkdownString(markdownString, options);
177
184
  } catch (e) {
178
185
  throw new Error(
179
- `Error while parsing Markdown file ${source}: "${e.message}".`,
186
+ `Error while parsing Markdown file ${source}: "${(e as Error).message}".`,
180
187
  );
181
188
  }
182
189
  }
@@ -0,0 +1,32 @@
1
+ /**
2
+ * Copyright (c) Facebook, Inc. and its affiliates.
3
+ *
4
+ * This source code is licensed under the MIT license found in the
5
+ * LICENSE file in the root directory of this source tree.
6
+ */
7
+
8
+ import React from 'react';
9
+ import ReactDOMServer from 'react-dom/server';
10
+ import MDX from '@mdx-js/runtime';
11
+ import removeImports from 'remark-mdx-remove-imports';
12
+ import removeExports from 'remark-mdx-remove-exports';
13
+
14
+ /**
15
+ * Transform mdx text to plain html text
16
+ * Initially created to convert MDX blog posts to HTML for the RSS feed
17
+ * without import/export nodes
18
+ *
19
+ * TODO not ideal implementation, won't work well with MDX elements!
20
+ * TODO theme+global site config should be able to declare MDX comps in scope for rendering the RSS feeds
21
+ * see also https://github.com/facebook/docusaurus/issues/4625
22
+ */
23
+ export function mdxToHtml(
24
+ mdxStr: string,
25
+ // TODO allow providing components/scope here, see https://github.com/mdx-js/mdx/tree/v1.6.13/packages/runtime
26
+ ): string {
27
+ return ReactDOMServer.renderToString(
28
+ React.createElement(MDX, {remarkPlugins: [removeImports, removeExports]}, [
29
+ mdxStr,
30
+ ]),
31
+ );
32
+ }