@docusaurus/plugin-content-docs 2.0.0-beta.1ec2c95e3 → 2.0.0-beta.21
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.
- package/lib/categoryGeneratedIndex.d.ts +12 -0
- package/lib/categoryGeneratedIndex.js +37 -0
- package/lib/cli.d.ts +3 -2
- package/lib/cli.js +58 -72
- package/lib/client/docsClientUtils.d.ts +9 -28
- package/lib/client/docsClientUtils.js +34 -43
- package/lib/client/index.d.ts +22 -0
- package/lib/client/index.js +59 -0
- package/lib/constants.d.ts +4 -0
- package/lib/constants.js +4 -1
- package/lib/docs.d.ts +32 -3
- package/lib/docs.js +164 -63
- package/{src/__tests__/__fixtures__/site-with-autogenerated-sidebar/docusaurus.config.js → lib/frontMatter.d.ts} +4 -8
- package/lib/{docFrontMatter.js → frontMatter.js} +12 -3
- package/lib/globalData.d.ts +3 -3
- package/lib/globalData.js +35 -6
- package/lib/index.d.ts +3 -3
- package/lib/index.js +123 -149
- package/lib/lastUpdate.d.ts +4 -6
- package/lib/lastUpdate.js +22 -26
- package/lib/markdown/index.d.ts +3 -6
- package/lib/markdown/index.js +3 -3
- package/lib/markdown/linkify.d.ts +1 -1
- package/lib/markdown/linkify.js +6 -3
- package/lib/numberPrefix.d.ts +1 -1
- package/lib/numberPrefix.js +16 -21
- package/lib/options.d.ts +3 -5
- package/lib/options.js +54 -13
- package/lib/props.d.ts +7 -2
- package/lib/props.js +70 -14
- package/lib/routes.d.ts +29 -0
- package/lib/routes.js +96 -0
- package/lib/server-export.d.ts +9 -0
- package/lib/server-export.js +22 -0
- package/lib/{sidebarItemsGenerator.d.ts → sidebars/generator.d.ts} +1 -6
- package/lib/sidebars/generator.js +209 -0
- package/lib/sidebars/index.d.ts +13 -0
- package/lib/sidebars/index.js +92 -0
- package/lib/sidebars/normalization.d.ts +13 -0
- package/lib/sidebars/normalization.js +55 -0
- package/lib/sidebars/postProcessor.d.ts +11 -0
- package/lib/sidebars/postProcessor.js +81 -0
- package/lib/sidebars/processor.d.ts +10 -0
- package/lib/sidebars/processor.js +79 -0
- package/lib/sidebars/types.d.ts +178 -0
- package/lib/{docFrontMatter.d.ts → sidebars/types.js} +2 -2
- package/lib/sidebars/utils.d.ts +54 -0
- package/lib/sidebars/utils.js +255 -0
- package/lib/sidebars/validation.d.ts +11 -0
- package/lib/sidebars/validation.js +142 -0
- package/lib/slug.d.ts +5 -4
- package/lib/slug.js +28 -18
- package/{src/__tests__/__fixtures__/sidebars/sidebars-first-level-not-category.js → lib/tags.d.ts} +3 -14
- package/lib/tags.js +21 -0
- package/lib/translations.d.ts +3 -3
- package/lib/translations.js +83 -93
- package/lib/types.d.ts +13 -184
- package/lib/versions/files.d.ts +51 -0
- package/lib/versions/files.js +143 -0
- package/lib/versions/index.d.ts +36 -0
- package/lib/versions/index.js +155 -0
- package/lib/versions/validation.d.ts +17 -0
- package/lib/versions/validation.js +71 -0
- package/package.json +34 -29
- package/src/categoryGeneratedIndex.ts +60 -0
- package/src/cli.ts +85 -114
- package/src/client/docsClientUtils.ts +44 -71
- package/src/client/index.ts +106 -0
- package/src/constants.ts +4 -2
- package/{types.d.ts → src/deps.d.ts} +1 -1
- package/src/docs.ts +228 -65
- package/src/{docFrontMatter.ts → frontMatter.ts} +20 -10
- package/src/globalData.ts +57 -7
- package/src/index.ts +179 -216
- package/src/lastUpdate.ts +26 -37
- package/src/markdown/index.ts +10 -16
- package/src/markdown/linkify.ts +7 -4
- package/src/numberPrefix.ts +19 -26
- package/src/options.ts +59 -18
- package/src/plugin-content-docs.d.ts +642 -89
- package/src/props.ts +103 -21
- package/src/routes.ts +159 -0
- package/src/server-export.ts +22 -0
- package/src/sidebars/README.md +10 -0
- package/src/sidebars/generator.ts +292 -0
- package/src/sidebars/index.ts +118 -0
- package/src/sidebars/normalization.ts +85 -0
- package/src/sidebars/postProcessor.ts +112 -0
- package/src/sidebars/processor.ts +123 -0
- package/src/sidebars/types.ts +275 -0
- package/src/sidebars/utils.ts +390 -0
- package/src/sidebars/validation.ts +178 -0
- package/src/slug.ts +41 -22
- package/src/tags.ts +20 -0
- package/src/translations.ts +129 -123
- package/src/types.ts +18 -249
- package/src/versions/files.ts +220 -0
- package/src/versions/index.ts +247 -0
- package/src/versions/validation.ts +115 -0
- package/lib/.tsbuildinfo +0 -1
- package/lib/sidebarItemsGenerator.js +0 -211
- package/lib/sidebars.d.ts +0 -43
- package/lib/sidebars.js +0 -320
- package/lib/theme/hooks/useDocs.d.ts +0 -20
- package/lib/theme/hooks/useDocs.js +0 -75
- package/lib/versions.d.ts +0 -16
- package/lib/versions.js +0 -319
- package/src/__tests__/__fixtures__/bad-id-site/docs/invalid-id.md +0 -5
- package/src/__tests__/__fixtures__/bad-slug-on-doc-home-site/docs/docWithSlug.md +0 -5
- package/src/__tests__/__fixtures__/empty-site/docusaurus.config.js +0 -16
- package/src/__tests__/__fixtures__/empty-site/sidebars.json +0 -1
- package/src/__tests__/__fixtures__/sidebars/sidebars-category-shorthand.js +0 -34
- package/src/__tests__/__fixtures__/sidebars/sidebars-category-wrong-items.json +0 -11
- package/src/__tests__/__fixtures__/sidebars/sidebars-category-wrong-label.json +0 -11
- package/src/__tests__/__fixtures__/sidebars/sidebars-category.js +0 -44
- package/src/__tests__/__fixtures__/sidebars/sidebars-collapsed-first-level.json +0 -20
- package/src/__tests__/__fixtures__/sidebars/sidebars-collapsed.json +0 -21
- package/src/__tests__/__fixtures__/sidebars/sidebars-doc-id-not-string.json +0 -10
- package/src/__tests__/__fixtures__/sidebars/sidebars-link-wrong-href.json +0 -11
- package/src/__tests__/__fixtures__/sidebars/sidebars-link-wrong-label.json +0 -11
- package/src/__tests__/__fixtures__/sidebars/sidebars-link.json +0 -11
- package/src/__tests__/__fixtures__/sidebars/sidebars-unknown-type.json +0 -14
- package/src/__tests__/__fixtures__/sidebars/sidebars-wrong-field.json +0 -20
- package/src/__tests__/__fixtures__/sidebars/sidebars.json +0 -20
- package/src/__tests__/__fixtures__/simple-site/docs/foo/bar.md +0 -69
- package/src/__tests__/__fixtures__/simple-site/docs/foo/baz.md +0 -70
- package/src/__tests__/__fixtures__/simple-site/docs/headingAsTitle.md +0 -1
- package/src/__tests__/__fixtures__/simple-site/docs/hello.md +0 -53
- package/src/__tests__/__fixtures__/simple-site/docs/ipsum.md +0 -5
- package/src/__tests__/__fixtures__/simple-site/docs/lorem.md +0 -6
- package/src/__tests__/__fixtures__/simple-site/docs/rootAbsoluteSlug.md +0 -5
- package/src/__tests__/__fixtures__/simple-site/docs/rootRelativeSlug.md +0 -5
- package/src/__tests__/__fixtures__/simple-site/docs/rootResolvedSlug.md +0 -5
- package/src/__tests__/__fixtures__/simple-site/docs/rootTryToEscapeSlug.md +0 -5
- package/src/__tests__/__fixtures__/simple-site/docs/slugs/absoluteSlug.md +0 -5
- package/src/__tests__/__fixtures__/simple-site/docs/slugs/relativeSlug.md +0 -5
- package/src/__tests__/__fixtures__/simple-site/docs/slugs/resolvedSlug.md +0 -5
- package/src/__tests__/__fixtures__/simple-site/docs/slugs/tryToEscapeSlug.md +0 -5
- package/src/__tests__/__fixtures__/simple-site/docusaurus.config.js +0 -14
- package/src/__tests__/__fixtures__/simple-site/sidebars.json +0 -23
- package/src/__tests__/__fixtures__/simple-site/wrong-sidebars.json +0 -7
- package/src/__tests__/__fixtures__/site-with-autogenerated-sidebar/docs/0-getting-started.md +0 -3
- package/src/__tests__/__fixtures__/site-with-autogenerated-sidebar/docs/1-installation.md +0 -3
- package/src/__tests__/__fixtures__/site-with-autogenerated-sidebar/docs/3-API/00_api-overview.md +0 -3
- package/src/__tests__/__fixtures__/site-with-autogenerated-sidebar/docs/3-API/01_Core APIs/0 --- Client API.md +0 -1
- package/src/__tests__/__fixtures__/site-with-autogenerated-sidebar/docs/3-API/01_Core APIs/1 --- Server API.md +0 -1
- package/src/__tests__/__fixtures__/site-with-autogenerated-sidebar/docs/3-API/02_Extension APIs/0. Plugin API.md +0 -1
- package/src/__tests__/__fixtures__/site-with-autogenerated-sidebar/docs/3-API/02_Extension APIs/1. Theme API.md +0 -1
- package/src/__tests__/__fixtures__/site-with-autogenerated-sidebar/docs/3-API/02_Extension APIs/_category_.yml +0 -1
- package/src/__tests__/__fixtures__/site-with-autogenerated-sidebar/docs/3-API/03_api-end.md +0 -3
- package/src/__tests__/__fixtures__/site-with-autogenerated-sidebar/docs/3-API/_category_.json +0 -3
- package/src/__tests__/__fixtures__/site-with-autogenerated-sidebar/docs/Guides/0-guide2.5.md +0 -8
- package/src/__tests__/__fixtures__/site-with-autogenerated-sidebar/docs/Guides/02-guide2.md +0 -7
- package/src/__tests__/__fixtures__/site-with-autogenerated-sidebar/docs/Guides/_category_.json +0 -3
- package/src/__tests__/__fixtures__/site-with-autogenerated-sidebar/docs/Guides/a-guide4.md +0 -7
- package/src/__tests__/__fixtures__/site-with-autogenerated-sidebar/docs/Guides/b-guide5.md +0 -7
- package/src/__tests__/__fixtures__/site-with-autogenerated-sidebar/docs/Guides/guide3.md +0 -8
- package/src/__tests__/__fixtures__/site-with-autogenerated-sidebar/docs/Guides/z-guide1.md +0 -8
- package/src/__tests__/__fixtures__/site-with-autogenerated-sidebar/partialAutogeneratedSidebars.js +0 -23
- package/src/__tests__/__fixtures__/site-with-autogenerated-sidebar/partialAutogeneratedSidebars2.js +0 -16
- package/src/__tests__/__fixtures__/site-with-doc-label/docs/hello-1.md +0 -7
- package/src/__tests__/__fixtures__/site-with-doc-label/docs/hello-2.md +0 -8
- package/src/__tests__/__fixtures__/site-with-doc-label/docusaurus.config.js +0 -14
- package/src/__tests__/__fixtures__/site-with-doc-label/sidebars.json +0 -14
- package/src/__tests__/__fixtures__/versioned-site/community/team.md +0 -1
- package/src/__tests__/__fixtures__/versioned-site/community_sidebars.json +0 -3
- package/src/__tests__/__fixtures__/versioned-site/community_versioned_docs/version-1.0.0/team.md +0 -1
- package/src/__tests__/__fixtures__/versioned-site/community_versioned_sidebars/version-1.0.0-sidebars.json +0 -3
- package/src/__tests__/__fixtures__/versioned-site/community_versions.json +0 -1
- package/src/__tests__/__fixtures__/versioned-site/docs/foo/bar.md +0 -4
- package/src/__tests__/__fixtures__/versioned-site/docs/hello.md +0 -1
- package/src/__tests__/__fixtures__/versioned-site/docs/slugs/absoluteSlug.md +0 -5
- package/src/__tests__/__fixtures__/versioned-site/docs/slugs/relativeSlug.md +0 -5
- package/src/__tests__/__fixtures__/versioned-site/docs/slugs/resolvedSlug.md +0 -5
- package/src/__tests__/__fixtures__/versioned-site/docs/slugs/tryToEscapeSlug.md +0 -5
- package/src/__tests__/__fixtures__/versioned-site/docusaurus.config.js +0 -18
- package/src/__tests__/__fixtures__/versioned-site/i18n/en/docusaurus-plugin-content-docs/version-1.0.0/hello.md +0 -1
- package/src/__tests__/__fixtures__/versioned-site/i18n/en/docusaurus-plugin-content-docs-community/current/team.md +0 -5
- package/src/__tests__/__fixtures__/versioned-site/i18n/fr/docusaurus-plugin-content-docs/version-1.0.0/hello.md +0 -1
- package/src/__tests__/__fixtures__/versioned-site/sidebars.json +0 -10
- package/src/__tests__/__fixtures__/versioned-site/versioned_docs/version-1.0.0/foo/bar.md +0 -4
- package/src/__tests__/__fixtures__/versioned-site/versioned_docs/version-1.0.0/foo/baz.md +0 -1
- package/src/__tests__/__fixtures__/versioned-site/versioned_docs/version-1.0.0/hello.md +0 -1
- package/src/__tests__/__fixtures__/versioned-site/versioned_docs/version-1.0.1/foo/bar.md +0 -1
- package/src/__tests__/__fixtures__/versioned-site/versioned_docs/version-1.0.1/hello.md +0 -1
- package/src/__tests__/__fixtures__/versioned-site/versioned_docs/version-withSlugs/rootAbsoluteSlug.md +0 -5
- package/src/__tests__/__fixtures__/versioned-site/versioned_docs/version-withSlugs/rootRelativeSlug.md +0 -5
- package/src/__tests__/__fixtures__/versioned-site/versioned_docs/version-withSlugs/rootResolvedSlug.md +0 -5
- package/src/__tests__/__fixtures__/versioned-site/versioned_docs/version-withSlugs/rootTryToEscapeSlug.md +0 -5
- package/src/__tests__/__fixtures__/versioned-site/versioned_docs/version-withSlugs/slugs/absoluteSlug.md +0 -5
- package/src/__tests__/__fixtures__/versioned-site/versioned_docs/version-withSlugs/slugs/relativeSlug.md +0 -5
- package/src/__tests__/__fixtures__/versioned-site/versioned_docs/version-withSlugs/slugs/resolvedSlug.md +0 -5
- package/src/__tests__/__fixtures__/versioned-site/versioned_docs/version-withSlugs/slugs/tryToEscapeSlug.md +0 -5
- package/src/__tests__/__fixtures__/versioned-site/versioned_sidebars/version-1.0.0-sidebars.json +0 -11
- package/src/__tests__/__fixtures__/versioned-site/versioned_sidebars/version-1.0.1-sidebars.json +0 -10
- package/src/__tests__/__fixtures__/versioned-site/versioned_sidebars/version-withSlugs-sidebars.json +0 -5
- package/src/__tests__/__fixtures__/versioned-site/versions.json +0 -5
- package/src/__tests__/__snapshots__/cli.test.ts.snap +0 -90
- package/src/__tests__/__snapshots__/index.test.ts.snap +0 -1916
- package/src/__tests__/__snapshots__/sidebars.test.ts.snap +0 -218
- package/src/__tests__/__snapshots__/translations.test.ts.snap +0 -487
- package/src/__tests__/cli.test.ts +0 -333
- package/src/__tests__/docFrontMatter.test.ts +0 -244
- package/src/__tests__/docs.test.ts +0 -878
- package/src/__tests__/index.test.ts +0 -1871
- package/src/__tests__/lastUpdate.test.ts +0 -69
- package/src/__tests__/numberPrefix.test.ts +0 -199
- package/src/__tests__/options.test.ts +0 -231
- package/src/__tests__/sidebarItemsGenerator.test.ts +0 -336
- package/src/__tests__/sidebars.test.ts +0 -639
- package/src/__tests__/slug.test.ts +0 -109
- package/src/__tests__/translations.test.ts +0 -159
- package/src/__tests__/versions.test.ts +0 -741
- package/src/client/__tests__/docsClientUtils.test.ts +0 -371
- package/src/markdown/__tests__/__fixtures__/docs/doc-localized.md +0 -1
- package/src/markdown/__tests__/__fixtures__/docs/doc1.md +0 -13
- package/src/markdown/__tests__/__fixtures__/docs/doc2.md +0 -12
- package/src/markdown/__tests__/__fixtures__/docs/doc4.md +0 -19
- package/src/markdown/__tests__/__fixtures__/docs/doc5.md +0 -6
- package/src/markdown/__tests__/__fixtures__/docs/subdir/doc3.md +0 -3
- package/src/markdown/__tests__/__fixtures__/versioned_docs/version-1.0.0/doc2.md +0 -7
- package/src/markdown/__tests__/__fixtures__/versioned_docs/version-1.0.0/subdir/doc1.md +0 -3
- package/src/markdown/__tests__/__snapshots__/linkify.test.ts.snap +0 -82
- package/src/markdown/__tests__/linkify.test.ts +0 -190
- package/src/sidebarItemsGenerator.ts +0 -307
- package/src/sidebars.ts +0 -522
- package/src/theme/hooks/useDocs.ts +0 -103
- package/src/versions.ts +0 -572
- package/tsconfig.json +0 -9
|
@@ -0,0 +1,390 @@
|
|
|
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 _ from 'lodash';
|
|
9
|
+
import {toMessageRelativeFilePath} from '@docusaurus/utils';
|
|
10
|
+
import type {
|
|
11
|
+
Sidebars,
|
|
12
|
+
Sidebar,
|
|
13
|
+
SidebarItem,
|
|
14
|
+
SidebarItemCategory,
|
|
15
|
+
SidebarItemLink,
|
|
16
|
+
SidebarItemDoc,
|
|
17
|
+
SidebarItemType,
|
|
18
|
+
SidebarCategoriesShorthand,
|
|
19
|
+
SidebarItemConfig,
|
|
20
|
+
SidebarItemCategoryWithGeneratedIndex,
|
|
21
|
+
SidebarNavigationItem,
|
|
22
|
+
} from './types';
|
|
23
|
+
import type {
|
|
24
|
+
DocMetadataBase,
|
|
25
|
+
PropNavigationLink,
|
|
26
|
+
} from '@docusaurus/plugin-content-docs';
|
|
27
|
+
|
|
28
|
+
export function isCategoriesShorthand(
|
|
29
|
+
item: SidebarItemConfig,
|
|
30
|
+
): item is SidebarCategoriesShorthand {
|
|
31
|
+
return typeof item === 'object' && !item.type;
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
export function transformSidebarItems(
|
|
35
|
+
sidebar: Sidebar,
|
|
36
|
+
updateFn: (item: SidebarItem) => SidebarItem,
|
|
37
|
+
): Sidebar {
|
|
38
|
+
function transformRecursive(item: SidebarItem): SidebarItem {
|
|
39
|
+
if (item.type === 'category') {
|
|
40
|
+
return updateFn({
|
|
41
|
+
...item,
|
|
42
|
+
items: item.items.map(transformRecursive),
|
|
43
|
+
});
|
|
44
|
+
}
|
|
45
|
+
return updateFn(item);
|
|
46
|
+
}
|
|
47
|
+
return sidebar.map(transformRecursive);
|
|
48
|
+
}
|
|
49
|
+
|
|
50
|
+
/**
|
|
51
|
+
* Flatten sidebar items into a single flat array (containing categories/docs on
|
|
52
|
+
* the same level). Order matters (useful for next/prev nav), top categories
|
|
53
|
+
* appear before their child elements
|
|
54
|
+
*/
|
|
55
|
+
function flattenSidebarItems(items: SidebarItem[]): SidebarItem[] {
|
|
56
|
+
function flattenRecursive(item: SidebarItem): SidebarItem[] {
|
|
57
|
+
return item.type === 'category'
|
|
58
|
+
? [item, ...item.items.flatMap(flattenRecursive)]
|
|
59
|
+
: [item];
|
|
60
|
+
}
|
|
61
|
+
return items.flatMap(flattenRecursive);
|
|
62
|
+
}
|
|
63
|
+
|
|
64
|
+
function collectSidebarItemsOfType<
|
|
65
|
+
Type extends SidebarItemType,
|
|
66
|
+
Item extends SidebarItem & {type: SidebarItemType},
|
|
67
|
+
>(type: Type, sidebar: Sidebar): Item[] {
|
|
68
|
+
return flattenSidebarItems(sidebar).filter(
|
|
69
|
+
(item) => item.type === type,
|
|
70
|
+
) as Item[];
|
|
71
|
+
}
|
|
72
|
+
|
|
73
|
+
export function collectSidebarDocItems(sidebar: Sidebar): SidebarItemDoc[] {
|
|
74
|
+
return collectSidebarItemsOfType('doc', sidebar);
|
|
75
|
+
}
|
|
76
|
+
export function collectSidebarCategories(
|
|
77
|
+
sidebar: Sidebar,
|
|
78
|
+
): SidebarItemCategory[] {
|
|
79
|
+
return collectSidebarItemsOfType('category', sidebar);
|
|
80
|
+
}
|
|
81
|
+
export function collectSidebarLinks(sidebar: Sidebar): SidebarItemLink[] {
|
|
82
|
+
return collectSidebarItemsOfType('link', sidebar);
|
|
83
|
+
}
|
|
84
|
+
|
|
85
|
+
// /!\ docId order matters for navigation!
|
|
86
|
+
export function collectSidebarDocIds(sidebar: Sidebar): string[] {
|
|
87
|
+
return flattenSidebarItems(sidebar).flatMap((item) => {
|
|
88
|
+
if (item.type === 'category') {
|
|
89
|
+
return item.link?.type === 'doc' ? [item.link.id] : [];
|
|
90
|
+
}
|
|
91
|
+
if (item.type === 'doc') {
|
|
92
|
+
return [item.id];
|
|
93
|
+
}
|
|
94
|
+
return [];
|
|
95
|
+
});
|
|
96
|
+
}
|
|
97
|
+
|
|
98
|
+
export function collectSidebarNavigation(
|
|
99
|
+
sidebar: Sidebar,
|
|
100
|
+
): SidebarNavigationItem[] {
|
|
101
|
+
return flattenSidebarItems(sidebar).flatMap((item) => {
|
|
102
|
+
if (item.type === 'category' && item.link) {
|
|
103
|
+
return [item as SidebarNavigationItem];
|
|
104
|
+
}
|
|
105
|
+
if (item.type === 'doc') {
|
|
106
|
+
return [item];
|
|
107
|
+
}
|
|
108
|
+
return [];
|
|
109
|
+
});
|
|
110
|
+
}
|
|
111
|
+
|
|
112
|
+
export function collectSidebarsDocIds(sidebars: Sidebars): {
|
|
113
|
+
[sidebarId: string]: string[];
|
|
114
|
+
} {
|
|
115
|
+
return _.mapValues(sidebars, collectSidebarDocIds);
|
|
116
|
+
}
|
|
117
|
+
|
|
118
|
+
export function collectSidebarsNavigations(sidebars: Sidebars): {
|
|
119
|
+
[sidebarId: string]: SidebarNavigationItem[];
|
|
120
|
+
} {
|
|
121
|
+
return _.mapValues(sidebars, collectSidebarNavigation);
|
|
122
|
+
}
|
|
123
|
+
|
|
124
|
+
export type SidebarNavigation = {
|
|
125
|
+
sidebarName: string | undefined;
|
|
126
|
+
previous: SidebarNavigationItem | undefined;
|
|
127
|
+
next: SidebarNavigationItem | undefined;
|
|
128
|
+
};
|
|
129
|
+
|
|
130
|
+
// A convenient and performant way to query the sidebars content
|
|
131
|
+
export type SidebarsUtils = {
|
|
132
|
+
sidebars: Sidebars;
|
|
133
|
+
getFirstDocIdOfFirstSidebar: () => string | undefined;
|
|
134
|
+
getSidebarNameByDocId: (docId: string) => string | undefined;
|
|
135
|
+
getDocNavigation: (
|
|
136
|
+
unversionedId: string,
|
|
137
|
+
versionedId: string,
|
|
138
|
+
displayedSidebar: string | null | undefined,
|
|
139
|
+
) => SidebarNavigation;
|
|
140
|
+
getCategoryGeneratedIndexList: () => SidebarItemCategoryWithGeneratedIndex[];
|
|
141
|
+
getCategoryGeneratedIndexNavigation: (
|
|
142
|
+
categoryGeneratedIndexPermalink: string,
|
|
143
|
+
) => SidebarNavigation;
|
|
144
|
+
/**
|
|
145
|
+
* This function may return undefined. This is usually a user mistake, because
|
|
146
|
+
* it means this sidebar will never be displayed; however, we can still use
|
|
147
|
+
* `displayed_sidebar` to make it displayed. Pretty weird but valid use-case
|
|
148
|
+
*/
|
|
149
|
+
getFirstLink: (sidebarId: string) =>
|
|
150
|
+
| {
|
|
151
|
+
type: 'doc';
|
|
152
|
+
id: string;
|
|
153
|
+
label: string;
|
|
154
|
+
}
|
|
155
|
+
| {
|
|
156
|
+
type: 'generated-index';
|
|
157
|
+
permalink: string;
|
|
158
|
+
label: string;
|
|
159
|
+
}
|
|
160
|
+
| undefined;
|
|
161
|
+
|
|
162
|
+
checkSidebarsDocIds: (validDocIds: string[], sidebarFilePath: string) => void;
|
|
163
|
+
};
|
|
164
|
+
|
|
165
|
+
export function createSidebarsUtils(sidebars: Sidebars): SidebarsUtils {
|
|
166
|
+
const sidebarNameToDocIds = collectSidebarsDocIds(sidebars);
|
|
167
|
+
const sidebarNameToNavigationItems = collectSidebarsNavigations(sidebars);
|
|
168
|
+
|
|
169
|
+
// Reverse mapping
|
|
170
|
+
const docIdToSidebarName = Object.fromEntries(
|
|
171
|
+
Object.entries(sidebarNameToDocIds).flatMap(([sidebarName, docIds]) =>
|
|
172
|
+
docIds.map((docId) => [docId, sidebarName]),
|
|
173
|
+
),
|
|
174
|
+
);
|
|
175
|
+
|
|
176
|
+
function getFirstDocIdOfFirstSidebar(): string | undefined {
|
|
177
|
+
return Object.values(sidebarNameToDocIds)[0]?.[0];
|
|
178
|
+
}
|
|
179
|
+
|
|
180
|
+
function getSidebarNameByDocId(docId: string): string | undefined {
|
|
181
|
+
return docIdToSidebarName[docId];
|
|
182
|
+
}
|
|
183
|
+
|
|
184
|
+
function emptySidebarNavigation(): SidebarNavigation {
|
|
185
|
+
return {
|
|
186
|
+
sidebarName: undefined,
|
|
187
|
+
previous: undefined,
|
|
188
|
+
next: undefined,
|
|
189
|
+
};
|
|
190
|
+
}
|
|
191
|
+
|
|
192
|
+
function getDocNavigation(
|
|
193
|
+
unversionedId: string,
|
|
194
|
+
versionedId: string,
|
|
195
|
+
displayedSidebar: string | null | undefined,
|
|
196
|
+
): SidebarNavigation {
|
|
197
|
+
// TODO legacy id retro-compatibility!
|
|
198
|
+
let docId = unversionedId;
|
|
199
|
+
let sidebarName =
|
|
200
|
+
displayedSidebar === undefined
|
|
201
|
+
? getSidebarNameByDocId(docId)
|
|
202
|
+
: displayedSidebar;
|
|
203
|
+
if (sidebarName === undefined) {
|
|
204
|
+
docId = versionedId;
|
|
205
|
+
sidebarName = getSidebarNameByDocId(docId);
|
|
206
|
+
}
|
|
207
|
+
|
|
208
|
+
if (!sidebarName) {
|
|
209
|
+
return emptySidebarNavigation();
|
|
210
|
+
}
|
|
211
|
+
const navigationItems = sidebarNameToNavigationItems[sidebarName];
|
|
212
|
+
if (!navigationItems) {
|
|
213
|
+
throw new Error(
|
|
214
|
+
`Doc with ID ${docId} wants to display sidebar ${sidebarName} but a sidebar with this name doesn't exist`,
|
|
215
|
+
);
|
|
216
|
+
}
|
|
217
|
+
const currentItemIndex = navigationItems.findIndex((item) => {
|
|
218
|
+
if (item.type === 'doc') {
|
|
219
|
+
return item.id === docId;
|
|
220
|
+
}
|
|
221
|
+
if (item.type === 'category' && item.link.type === 'doc') {
|
|
222
|
+
return item.link.id === docId;
|
|
223
|
+
}
|
|
224
|
+
return false;
|
|
225
|
+
});
|
|
226
|
+
if (currentItemIndex === -1) {
|
|
227
|
+
return {sidebarName, next: undefined, previous: undefined};
|
|
228
|
+
}
|
|
229
|
+
|
|
230
|
+
return {
|
|
231
|
+
sidebarName,
|
|
232
|
+
previous: navigationItems[currentItemIndex - 1],
|
|
233
|
+
next: navigationItems[currentItemIndex + 1],
|
|
234
|
+
};
|
|
235
|
+
}
|
|
236
|
+
|
|
237
|
+
function getCategoryGeneratedIndexList(): SidebarItemCategoryWithGeneratedIndex[] {
|
|
238
|
+
return Object.values(sidebarNameToNavigationItems)
|
|
239
|
+
.flat()
|
|
240
|
+
.flatMap((item) => {
|
|
241
|
+
if (item.type === 'category' && item.link.type === 'generated-index') {
|
|
242
|
+
return [item as SidebarItemCategoryWithGeneratedIndex];
|
|
243
|
+
}
|
|
244
|
+
return [];
|
|
245
|
+
});
|
|
246
|
+
}
|
|
247
|
+
|
|
248
|
+
/**
|
|
249
|
+
* We identity the category generated index by its permalink (should be
|
|
250
|
+
* unique). More reliable than using object identity
|
|
251
|
+
*/
|
|
252
|
+
function getCategoryGeneratedIndexNavigation(
|
|
253
|
+
categoryGeneratedIndexPermalink: string,
|
|
254
|
+
): SidebarNavigation {
|
|
255
|
+
function isCurrentCategoryGeneratedIndexItem(
|
|
256
|
+
item: SidebarNavigationItem,
|
|
257
|
+
): boolean {
|
|
258
|
+
return (
|
|
259
|
+
item.type === 'category' &&
|
|
260
|
+
item.link.type === 'generated-index' &&
|
|
261
|
+
item.link.permalink === categoryGeneratedIndexPermalink
|
|
262
|
+
);
|
|
263
|
+
}
|
|
264
|
+
|
|
265
|
+
const sidebarName = Object.entries(sidebarNameToNavigationItems).find(
|
|
266
|
+
([, navigationItems]) =>
|
|
267
|
+
navigationItems.find(isCurrentCategoryGeneratedIndexItem),
|
|
268
|
+
)![0];
|
|
269
|
+
const navigationItems = sidebarNameToNavigationItems[sidebarName]!;
|
|
270
|
+
const currentItemIndex = navigationItems.findIndex(
|
|
271
|
+
isCurrentCategoryGeneratedIndexItem,
|
|
272
|
+
);
|
|
273
|
+
return {
|
|
274
|
+
sidebarName,
|
|
275
|
+
previous: navigationItems[currentItemIndex - 1],
|
|
276
|
+
next: navigationItems[currentItemIndex + 1],
|
|
277
|
+
};
|
|
278
|
+
}
|
|
279
|
+
|
|
280
|
+
function checkSidebarsDocIds(validDocIds: string[], sidebarFilePath: string) {
|
|
281
|
+
const allSidebarDocIds = Object.values(sidebarNameToDocIds).flat();
|
|
282
|
+
const invalidSidebarDocIds = _.difference(allSidebarDocIds, validDocIds);
|
|
283
|
+
if (invalidSidebarDocIds.length > 0) {
|
|
284
|
+
throw new Error(
|
|
285
|
+
`Invalid sidebar file at "${toMessageRelativeFilePath(
|
|
286
|
+
sidebarFilePath,
|
|
287
|
+
)}".
|
|
288
|
+
These sidebar document ids do not exist:
|
|
289
|
+
- ${invalidSidebarDocIds.sort().join('\n- ')}
|
|
290
|
+
|
|
291
|
+
Available document ids are:
|
|
292
|
+
- ${_.uniq(validDocIds).sort().join('\n- ')}`,
|
|
293
|
+
);
|
|
294
|
+
}
|
|
295
|
+
}
|
|
296
|
+
|
|
297
|
+
function getFirstLink(sidebar: Sidebar):
|
|
298
|
+
| {
|
|
299
|
+
type: 'doc';
|
|
300
|
+
id: string;
|
|
301
|
+
label: string;
|
|
302
|
+
}
|
|
303
|
+
| {
|
|
304
|
+
type: 'generated-index';
|
|
305
|
+
permalink: string;
|
|
306
|
+
label: string;
|
|
307
|
+
}
|
|
308
|
+
| undefined {
|
|
309
|
+
for (const item of sidebar) {
|
|
310
|
+
if (item.type === 'doc') {
|
|
311
|
+
return {
|
|
312
|
+
type: 'doc',
|
|
313
|
+
id: item.id,
|
|
314
|
+
label: item.label ?? item.id,
|
|
315
|
+
};
|
|
316
|
+
} else if (item.type === 'category') {
|
|
317
|
+
if (item.link?.type === 'doc') {
|
|
318
|
+
return {
|
|
319
|
+
type: 'doc',
|
|
320
|
+
id: item.link.id,
|
|
321
|
+
label: item.label,
|
|
322
|
+
};
|
|
323
|
+
} else if (item.link?.type === 'generated-index') {
|
|
324
|
+
return {
|
|
325
|
+
type: 'generated-index',
|
|
326
|
+
permalink: item.link.permalink,
|
|
327
|
+
label: item.label,
|
|
328
|
+
};
|
|
329
|
+
}
|
|
330
|
+
const firstSubItem = getFirstLink(item.items);
|
|
331
|
+
if (firstSubItem) {
|
|
332
|
+
return firstSubItem;
|
|
333
|
+
}
|
|
334
|
+
}
|
|
335
|
+
}
|
|
336
|
+
return undefined;
|
|
337
|
+
}
|
|
338
|
+
|
|
339
|
+
return {
|
|
340
|
+
sidebars,
|
|
341
|
+
getFirstDocIdOfFirstSidebar,
|
|
342
|
+
getSidebarNameByDocId,
|
|
343
|
+
getDocNavigation,
|
|
344
|
+
getCategoryGeneratedIndexList,
|
|
345
|
+
getCategoryGeneratedIndexNavigation,
|
|
346
|
+
checkSidebarsDocIds,
|
|
347
|
+
getFirstLink: (id) => getFirstLink(sidebars[id]!),
|
|
348
|
+
};
|
|
349
|
+
}
|
|
350
|
+
|
|
351
|
+
export function toDocNavigationLink(doc: DocMetadataBase): PropNavigationLink {
|
|
352
|
+
const {
|
|
353
|
+
title,
|
|
354
|
+
permalink,
|
|
355
|
+
frontMatter: {
|
|
356
|
+
pagination_label: paginationLabel,
|
|
357
|
+
sidebar_label: sidebarLabel,
|
|
358
|
+
},
|
|
359
|
+
} = doc;
|
|
360
|
+
return {title: paginationLabel ?? sidebarLabel ?? title, permalink};
|
|
361
|
+
}
|
|
362
|
+
|
|
363
|
+
export function toNavigationLink(
|
|
364
|
+
navigationItem: SidebarNavigationItem | undefined,
|
|
365
|
+
docsById: {[docId: string]: DocMetadataBase},
|
|
366
|
+
): PropNavigationLink | undefined {
|
|
367
|
+
function getDocById(docId: string) {
|
|
368
|
+
const doc = docsById[docId];
|
|
369
|
+
if (!doc) {
|
|
370
|
+
throw new Error(
|
|
371
|
+
`Can't create navigation link: no doc found with id=${docId}`,
|
|
372
|
+
);
|
|
373
|
+
}
|
|
374
|
+
return doc;
|
|
375
|
+
}
|
|
376
|
+
|
|
377
|
+
if (!navigationItem) {
|
|
378
|
+
return undefined;
|
|
379
|
+
}
|
|
380
|
+
|
|
381
|
+
if (navigationItem.type === 'category') {
|
|
382
|
+
return navigationItem.link.type === 'doc'
|
|
383
|
+
? toDocNavigationLink(getDocById(navigationItem.link.id))
|
|
384
|
+
: {
|
|
385
|
+
title: navigationItem.label,
|
|
386
|
+
permalink: navigationItem.link.permalink,
|
|
387
|
+
};
|
|
388
|
+
}
|
|
389
|
+
return toDocNavigationLink(getDocById(navigationItem.id));
|
|
390
|
+
}
|
|
@@ -0,0 +1,178 @@
|
|
|
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 {Joi, URISchema} from '@docusaurus/utils-validation';
|
|
9
|
+
import type {
|
|
10
|
+
SidebarItemConfig,
|
|
11
|
+
SidebarItemBase,
|
|
12
|
+
SidebarItemAutogenerated,
|
|
13
|
+
SidebarItemDoc,
|
|
14
|
+
SidebarItemHtml,
|
|
15
|
+
SidebarItemLink,
|
|
16
|
+
SidebarItemCategoryConfig,
|
|
17
|
+
SidebarItemCategoryLink,
|
|
18
|
+
SidebarItemCategoryLinkDoc,
|
|
19
|
+
SidebarItemCategoryLinkGeneratedIndex,
|
|
20
|
+
NormalizedSidebars,
|
|
21
|
+
NormalizedSidebarItem,
|
|
22
|
+
NormalizedSidebarItemCategory,
|
|
23
|
+
CategoryMetadataFile,
|
|
24
|
+
} from './types';
|
|
25
|
+
|
|
26
|
+
// NOTE: we don't add any default values during validation on purpose!
|
|
27
|
+
// Config types are exposed to users for typechecking and we use the same type
|
|
28
|
+
// in normalization
|
|
29
|
+
|
|
30
|
+
const sidebarItemBaseSchema = Joi.object<SidebarItemBase>({
|
|
31
|
+
className: Joi.string(),
|
|
32
|
+
customProps: Joi.object().unknown(),
|
|
33
|
+
});
|
|
34
|
+
|
|
35
|
+
const sidebarItemAutogeneratedSchema =
|
|
36
|
+
sidebarItemBaseSchema.append<SidebarItemAutogenerated>({
|
|
37
|
+
type: 'autogenerated',
|
|
38
|
+
dirName: Joi.string()
|
|
39
|
+
.required()
|
|
40
|
+
.pattern(/^[^/](?:.*[^/])?$/)
|
|
41
|
+
.message(
|
|
42
|
+
'"dirName" must be a dir path relative to the docs folder root, and should not start or end with slash',
|
|
43
|
+
),
|
|
44
|
+
});
|
|
45
|
+
|
|
46
|
+
const sidebarItemDocSchema = sidebarItemBaseSchema.append<SidebarItemDoc>({
|
|
47
|
+
type: Joi.string().valid('doc', 'ref').required(),
|
|
48
|
+
id: Joi.string().required(),
|
|
49
|
+
label: Joi.string(),
|
|
50
|
+
});
|
|
51
|
+
|
|
52
|
+
const sidebarItemHtmlSchema = sidebarItemBaseSchema.append<SidebarItemHtml>({
|
|
53
|
+
type: 'html',
|
|
54
|
+
value: Joi.string().required(),
|
|
55
|
+
defaultStyle: Joi.boolean(),
|
|
56
|
+
});
|
|
57
|
+
|
|
58
|
+
const sidebarItemLinkSchema = sidebarItemBaseSchema.append<SidebarItemLink>({
|
|
59
|
+
type: 'link',
|
|
60
|
+
href: URISchema.required(),
|
|
61
|
+
label: Joi.string()
|
|
62
|
+
.required()
|
|
63
|
+
.messages({'any.unknown': '"label" must be a string'}),
|
|
64
|
+
});
|
|
65
|
+
|
|
66
|
+
const sidebarItemCategoryLinkSchema = Joi.object<SidebarItemCategoryLink>()
|
|
67
|
+
.allow(null)
|
|
68
|
+
.when('.type', {
|
|
69
|
+
switch: [
|
|
70
|
+
{
|
|
71
|
+
is: 'doc',
|
|
72
|
+
then: Joi.object<SidebarItemCategoryLinkDoc>({
|
|
73
|
+
type: 'doc',
|
|
74
|
+
id: Joi.string().required(),
|
|
75
|
+
}),
|
|
76
|
+
},
|
|
77
|
+
{
|
|
78
|
+
is: 'generated-index',
|
|
79
|
+
then: Joi.object<SidebarItemCategoryLinkGeneratedIndex>({
|
|
80
|
+
type: 'generated-index',
|
|
81
|
+
slug: Joi.string().optional(),
|
|
82
|
+
// This one is not in the user config, only in the normalized version
|
|
83
|
+
// permalink: Joi.string().optional(),
|
|
84
|
+
title: Joi.string().optional(),
|
|
85
|
+
description: Joi.string().optional(),
|
|
86
|
+
image: Joi.string().optional(),
|
|
87
|
+
keywords: [Joi.string(), Joi.array().items(Joi.string())],
|
|
88
|
+
}),
|
|
89
|
+
},
|
|
90
|
+
{
|
|
91
|
+
is: Joi.required(),
|
|
92
|
+
then: Joi.forbidden().messages({
|
|
93
|
+
'any.unknown': 'Unknown sidebar category link type "{.type}".',
|
|
94
|
+
}),
|
|
95
|
+
},
|
|
96
|
+
],
|
|
97
|
+
});
|
|
98
|
+
|
|
99
|
+
const sidebarItemCategorySchema =
|
|
100
|
+
sidebarItemBaseSchema.append<SidebarItemCategoryConfig>({
|
|
101
|
+
type: 'category',
|
|
102
|
+
label: Joi.string()
|
|
103
|
+
.required()
|
|
104
|
+
.messages({'any.unknown': '"label" must be a string'}),
|
|
105
|
+
items: Joi.array()
|
|
106
|
+
.required()
|
|
107
|
+
.messages({'any.unknown': '"items" must be an array'}),
|
|
108
|
+
// TODO: Joi doesn't allow mutual recursion. See https://github.com/sideway/joi/issues/2611
|
|
109
|
+
// .items(Joi.link('#sidebarItemSchema')),
|
|
110
|
+
link: sidebarItemCategoryLinkSchema,
|
|
111
|
+
collapsed: Joi.boolean().messages({
|
|
112
|
+
'any.unknown': '"collapsed" must be a boolean',
|
|
113
|
+
}),
|
|
114
|
+
collapsible: Joi.boolean().messages({
|
|
115
|
+
'any.unknown': '"collapsible" must be a boolean',
|
|
116
|
+
}),
|
|
117
|
+
});
|
|
118
|
+
|
|
119
|
+
const sidebarItemSchema = Joi.object<SidebarItemConfig>().when('.type', {
|
|
120
|
+
switch: [
|
|
121
|
+
{is: 'link', then: sidebarItemLinkSchema},
|
|
122
|
+
{
|
|
123
|
+
is: Joi.string().valid('doc', 'ref').required(),
|
|
124
|
+
then: sidebarItemDocSchema,
|
|
125
|
+
},
|
|
126
|
+
{is: 'html', then: sidebarItemHtmlSchema},
|
|
127
|
+
{is: 'autogenerated', then: sidebarItemAutogeneratedSchema},
|
|
128
|
+
{is: 'category', then: sidebarItemCategorySchema},
|
|
129
|
+
{
|
|
130
|
+
is: Joi.any().required(),
|
|
131
|
+
then: Joi.forbidden().messages({
|
|
132
|
+
'any.unknown': 'Unknown sidebar item type "{.type}".',
|
|
133
|
+
}),
|
|
134
|
+
},
|
|
135
|
+
],
|
|
136
|
+
});
|
|
137
|
+
// .id('sidebarItemSchema');
|
|
138
|
+
|
|
139
|
+
function validateSidebarItem(
|
|
140
|
+
item: unknown,
|
|
141
|
+
): asserts item is NormalizedSidebarItem {
|
|
142
|
+
// TODO: remove once with proper Joi support
|
|
143
|
+
// Because we can't use Joi to validate nested items (see above), we do it
|
|
144
|
+
// manually
|
|
145
|
+
Joi.assert(item, sidebarItemSchema);
|
|
146
|
+
|
|
147
|
+
if ((item as NormalizedSidebarItem).type === 'category') {
|
|
148
|
+
(item as NormalizedSidebarItemCategory).items.forEach(validateSidebarItem);
|
|
149
|
+
}
|
|
150
|
+
}
|
|
151
|
+
|
|
152
|
+
export function validateSidebars(sidebars: {
|
|
153
|
+
[sidebarId: string]: unknown;
|
|
154
|
+
}): asserts sidebars is NormalizedSidebars {
|
|
155
|
+
Object.values(sidebars as NormalizedSidebars).forEach((sidebar) => {
|
|
156
|
+
sidebar.forEach(validateSidebarItem);
|
|
157
|
+
});
|
|
158
|
+
}
|
|
159
|
+
|
|
160
|
+
const categoryMetadataFileSchema = Joi.object<CategoryMetadataFile>({
|
|
161
|
+
label: Joi.string(),
|
|
162
|
+
position: Joi.number(),
|
|
163
|
+
collapsed: Joi.boolean(),
|
|
164
|
+
collapsible: Joi.boolean(),
|
|
165
|
+
className: Joi.string(),
|
|
166
|
+
link: sidebarItemCategoryLinkSchema,
|
|
167
|
+
customProps: Joi.object().unknown(),
|
|
168
|
+
});
|
|
169
|
+
|
|
170
|
+
export function validateCategoryMetadataFile(
|
|
171
|
+
unsafeContent: unknown,
|
|
172
|
+
): CategoryMetadataFile {
|
|
173
|
+
const {error, value} = categoryMetadataFileSchema.validate(unsafeContent);
|
|
174
|
+
if (error) {
|
|
175
|
+
throw error;
|
|
176
|
+
}
|
|
177
|
+
return value;
|
|
178
|
+
}
|
package/src/slug.ts
CHANGED
|
@@ -15,50 +15,69 @@ import {
|
|
|
15
15
|
DefaultNumberPrefixParser,
|
|
16
16
|
stripPathNumberPrefixes,
|
|
17
17
|
} from './numberPrefix';
|
|
18
|
-
import {
|
|
18
|
+
import {isCategoryIndex, toCategoryIndexMatcherParam} from './docs';
|
|
19
|
+
import type {
|
|
20
|
+
NumberPrefixParser,
|
|
21
|
+
DocMetadataBase,
|
|
22
|
+
} from '@docusaurus/plugin-content-docs';
|
|
19
23
|
|
|
20
24
|
export default function getSlug({
|
|
21
25
|
baseID,
|
|
22
|
-
|
|
23
|
-
|
|
26
|
+
frontMatterSlug,
|
|
27
|
+
source,
|
|
28
|
+
sourceDirName,
|
|
24
29
|
stripDirNumberPrefixes = true,
|
|
25
30
|
numberPrefixParser = DefaultNumberPrefixParser,
|
|
26
31
|
}: {
|
|
27
32
|
baseID: string;
|
|
28
|
-
|
|
29
|
-
|
|
33
|
+
frontMatterSlug?: string;
|
|
34
|
+
source: DocMetadataBase['source'];
|
|
35
|
+
sourceDirName: DocMetadataBase['sourceDirName'];
|
|
30
36
|
stripDirNumberPrefixes?: boolean;
|
|
31
37
|
numberPrefixParser?: NumberPrefixParser;
|
|
32
38
|
}): string {
|
|
33
|
-
|
|
34
|
-
let slug: string;
|
|
35
|
-
if (baseSlug.startsWith('/')) {
|
|
36
|
-
slug = baseSlug;
|
|
37
|
-
} else {
|
|
39
|
+
function getDirNameSlug(): string {
|
|
38
40
|
const dirNameStripped = stripDirNumberPrefixes
|
|
39
|
-
? stripPathNumberPrefixes(
|
|
40
|
-
:
|
|
41
|
+
? stripPathNumberPrefixes(sourceDirName, numberPrefixParser)
|
|
42
|
+
: sourceDirName;
|
|
41
43
|
const resolveDirname =
|
|
42
|
-
|
|
44
|
+
sourceDirName === '.'
|
|
43
45
|
? '/'
|
|
44
46
|
: addLeadingSlash(addTrailingSlash(dirNameStripped));
|
|
45
|
-
|
|
47
|
+
return resolveDirname;
|
|
46
48
|
}
|
|
47
49
|
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
50
|
+
function computeSlug(): string {
|
|
51
|
+
if (frontMatterSlug?.startsWith('/')) {
|
|
52
|
+
return frontMatterSlug;
|
|
53
|
+
}
|
|
54
|
+
const dirNameSlug = getDirNameSlug();
|
|
55
|
+
if (
|
|
56
|
+
!frontMatterSlug &&
|
|
57
|
+
isCategoryIndex(toCategoryIndexMatcherParam({source, sourceDirName}))
|
|
58
|
+
) {
|
|
59
|
+
return dirNameSlug;
|
|
60
|
+
}
|
|
61
|
+
const baseSlug = frontMatterSlug ?? baseID;
|
|
62
|
+
return resolvePathname(baseSlug, getDirNameSlug());
|
|
63
|
+
}
|
|
64
|
+
|
|
65
|
+
function ensureValidSlug(slug: string): string {
|
|
66
|
+
if (!isValidPathname(slug)) {
|
|
67
|
+
throw new Error(
|
|
68
|
+
`We couldn't compute a valid slug for document with ID "${baseID}" in "${sourceDirName}" directory.
|
|
51
69
|
The slug we computed looks invalid: ${slug}.
|
|
52
|
-
Maybe your slug
|
|
53
|
-
By using
|
|
70
|
+
Maybe your slug front matter is incorrect or there are special characters in the file path?
|
|
71
|
+
By using front matter to set a custom slug, you should be able to fix this error:
|
|
54
72
|
|
|
55
|
-
Example =>
|
|
56
73
|
---
|
|
57
74
|
slug: /my/customDocPath
|
|
58
75
|
---
|
|
59
76
|
`,
|
|
60
|
-
|
|
77
|
+
);
|
|
78
|
+
}
|
|
79
|
+
return slug;
|
|
61
80
|
}
|
|
62
81
|
|
|
63
|
-
return
|
|
82
|
+
return ensureValidSlug(computeSlug());
|
|
64
83
|
}
|
package/src/tags.ts
ADDED
|
@@ -0,0 +1,20 @@
|
|
|
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 _ from 'lodash';
|
|
9
|
+
import {groupTaggedItems} from '@docusaurus/utils';
|
|
10
|
+
import type {VersionTags} from './types';
|
|
11
|
+
import type {DocMetadata} from '@docusaurus/plugin-content-docs';
|
|
12
|
+
|
|
13
|
+
export function getVersionTags(docs: DocMetadata[]): VersionTags {
|
|
14
|
+
const groups = groupTaggedItems(docs, (doc) => doc.tags);
|
|
15
|
+
return _.mapValues(groups, (group) => ({
|
|
16
|
+
label: group.tag.label,
|
|
17
|
+
docIds: group.items.map((item) => item.id),
|
|
18
|
+
permalink: group.tag.permalink,
|
|
19
|
+
}));
|
|
20
|
+
}
|