erudit 3.0.0-dev.3 → 3.0.0-dev.5
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/app/app.vue +195 -172
- package/app/components/Loading.vue +23 -23
- package/app/components/SiteAside.vue +393 -382
- package/app/components/SiteMain.vue +35 -35
- package/app/components/ads/BannerTemplate.vue +51 -51
- package/app/components/ads/BottomBanner.vue +45 -45
- package/app/components/ads/LeftBanner.vue +50 -50
- package/app/components/aside/AsideListItem.vue +74 -74
- package/app/components/aside/AsideMajor.vue +56 -56
- package/app/components/aside/AsideMinor.vue +71 -71
- package/app/components/aside/major/PaneContentScroll.vue +23 -23
- package/app/components/aside/major/PaneSwitch.vue +54 -54
- package/app/components/aside/major/PaneSwitchButton.vue +63 -63
- package/app/components/aside/major/SiteInfo.vue +85 -85
- package/app/components/aside/major/panes/Language.vue +79 -79
- package/app/components/aside/major/panes/Pages.vue +34 -34
- package/app/components/aside/major/panes/Search.vue +11 -11
- package/app/components/aside/major/panes/nav/Nav.vue +87 -91
- package/app/components/aside/major/panes/nav/NavBook.vue +87 -86
- package/app/components/aside/major/panes/nav/NavBookLoading.vue +24 -24
- package/app/components/aside/major/panes/nav/NavGlobal.vue +16 -16
- package/app/components/aside/major/panes/nav/fnav/FNav.vue +105 -105
- package/app/components/aside/major/panes/nav/fnav/FNavBook.vue +32 -32
- package/app/components/aside/major/panes/nav/fnav/FNavFlags.vue +40 -40
- package/app/components/aside/major/panes/nav/fnav/FNavFolder.vue +60 -60
- package/app/components/aside/major/panes/nav/fnav/FNavItem.vue +34 -34
- package/app/components/aside/major/panes/nav/fnav/FNavSeparator.vue +80 -80
- package/app/components/aside/major/panes/nav/fnav/FNavTopic.vue +24 -24
- package/app/components/aside/major/panes/other/ItemContent.vue +29 -29
- package/app/components/aside/major/panes/other/ItemGenerator.vue +15 -15
- package/app/components/aside/major/panes/other/ItemTheme.vue +54 -54
- package/app/components/aside/major/panes/other/Other.vue +16 -16
- package/app/components/aside/minor/AsideMinorContributor.vue +5 -5
- package/app/components/aside/minor/AsideMinorNews.vue +11 -11
- package/app/components/aside/minor/AsideMinorPane.vue +15 -15
- package/app/components/aside/minor/AsideMinorTopLink.vue +67 -67
- package/app/components/aside/minor/Contribute.vue +145 -145
- package/app/components/aside/minor/content/AsideMinorContent.vue +92 -92
- package/app/components/aside/minor/topic/AsideMinorTopic.vue +32 -32
- package/app/components/aside/minor/topic/TopicContributors.vue +177 -177
- package/app/components/aside/minor/topic/TopicNav.vue +49 -49
- package/app/components/aside/minor/topic/TopicToc.vue +219 -203
- package/app/components/aside/minor/topic/TopicTocItem.vue +30 -31
- package/app/components/aside/utils/AsideOverlayPane.vue +40 -40
- package/app/components/bitran/BitranContent.vue +70 -63
- package/app/components/bitran/RenderWrapper.vue +10 -10
- package/app/components/contributor/ContributorAvatar.vue +43 -43
- package/app/components/contributor/ContributorListItem.vue +35 -35
- package/app/components/main/topic/MainTopic.vue +79 -79
- package/app/components/main/topic/TopicPartSwitch.vue +118 -118
- package/app/components/main/utils/Breadcrumb.vue +75 -75
- package/app/components/main/utils/ContentDecoration.vue +29 -29
- package/app/components/main/utils/ContentDescription.vue +19 -19
- package/app/components/main/utils/ContentPopover.vue +188 -176
- package/app/components/main/utils/ContentPopovers.vue +105 -105
- package/app/components/main/utils/ContentReferences.vue +70 -70
- package/app/components/main/utils/ContentSection.vue +45 -45
- package/app/components/main/utils/ContentTitle.vue +63 -39
- package/app/components/main/utils/reference/ReferenceGroup.vue +38 -38
- package/app/components/main/utils/reference/ReferenceItem.vue +68 -68
- package/app/components/main/utils/reference/ReferenceSource.vue +116 -116
- package/app/components/preview/Preview.vue +186 -177
- package/app/components/preview/PreviewDisplay.vue +139 -139
- package/app/components/preview/PreviewFooterAction.vue +73 -73
- package/app/components/preview/PreviewLoading.vue +14 -14
- package/app/components/preview/PreviewScreen.vue +141 -99
- package/app/components/preview/display/Alert.vue +50 -50
- package/app/components/preview/display/Custom.vue +18 -18
- package/app/components/preview/display/GenericLink.vue +48 -48
- package/app/components/preview/display/PageLink.vue +20 -20
- package/app/components/preview/display/Unique.vue +49 -49
- package/app/components/transition/Fade.vue +19 -19
- package/app/components/tree/TreeContainer.vue +11 -11
- package/app/components/tree/TreeItem.vue +89 -89
- package/app/composables/bitran.ts +129 -127
- package/app/composables/bitranContent.ts +39 -39
- package/app/composables/bitranLocation.ts +7 -7
- package/app/composables/contentData.ts +36 -36
- package/app/composables/contentPage.ts +157 -156
- package/app/composables/contentRoute.ts +45 -45
- package/app/composables/darkMagic.ts +24 -24
- package/app/composables/externalApi.ts +63 -63
- package/app/composables/favicon.ts +8 -8
- package/app/composables/formatText.ts +86 -86
- package/app/composables/majorPane.ts +60 -60
- package/app/composables/phrases.ts +81 -80
- package/app/composables/theme.ts +29 -29
- package/app/composables/url.ts +33 -33
- package/app/pages/_test/preview.vue +110 -110
- package/app/pages/article/[...articleId].vue +3 -3
- package/app/pages/book/[...bookId].vue +47 -47
- package/app/pages/group/[...groupId].vue +65 -65
- package/app/pages/index.vue +32 -32
- package/app/pages/members.vue +6 -6
- package/app/pages/practice/[...practice].vue +3 -3
- package/app/pages/summary/[...summaryId].vue +3 -3
- package/app/public/favicon/article.svg +9 -9
- package/app/public/favicon/default.svg +3 -3
- package/app/public/favicon/practice.svg +3 -3
- package/app/public/favicon/summary.svg +4 -4
- package/app/public/logotype.svg +2 -2
- package/app/scripts/_immediate.js +9 -9
- package/app/scripts/aside/index.ts +59 -59
- package/app/scripts/aside/major/nav.ts +26 -26
- package/app/scripts/aside/minor/state.ts +37 -37
- package/app/scripts/aside/minor/topic.ts +3 -3
- package/app/scripts/flag.ts +28 -28
- package/app/scripts/og.ts +27 -27
- package/app/scripts/preview/build.ts +73 -73
- package/app/scripts/preview/data/alert.ts +19 -19
- package/app/scripts/preview/data/custom.ts +8 -8
- package/app/scripts/preview/data/genericLink.ts +24 -24
- package/app/scripts/preview/data/pageLink.ts +22 -20
- package/app/scripts/preview/data/unique.ts +71 -71
- package/app/scripts/preview/data.ts +24 -24
- package/app/scripts/preview/display.ts +37 -37
- package/app/scripts/preview/footer.ts +9 -9
- package/app/scripts/preview/request.ts +51 -51
- package/app/scripts/preview/state.ts +63 -63
- package/app/styles/_immediate.css +7 -7
- package/app/styles/_util.scss +43 -43
- package/app/styles/_utils.scss +44 -44
- package/app/styles/app.scss +91 -91
- package/app/styles/def/_bp.scss +27 -27
- package/app/styles/def/_size.scss +7 -7
- package/app/styles/def/_z.scss +5 -5
- package/app/styles/normalize.scss +63 -63
- package/app/styles/partials/_darkMagic.scss +5 -5
- package/app/styles/partials/_fnav.scss +15 -15
- package/app/styles/partials/_preview.scss +5 -5
- package/globalPath.ts +21 -21
- package/globals/bitran.ts +2 -47
- package/globals/content.ts +22 -22
- package/globals/contributor.ts +5 -5
- package/globals/erudit.ts +5 -5
- package/globals/register.ts +18 -18
- package/languages/en.ts +95 -95
- package/languages/ru.ts +99 -99
- package/module/bitran.ts +35 -34
- package/module/config.ts +34 -34
- package/module/imports.ts +50 -46
- package/module/index.ts +47 -47
- package/module/logger.ts +10 -10
- package/module/paths.ts +22 -22
- package/module/restart.ts +61 -61
- package/nuxt.config.ts +138 -112
- package/package.json +7 -8
- package/server/api/aside/major/nav/bookIds.ts +5 -5
- package/server/api/aside/major/nav/bookNav/[...bookId].ts +20 -20
- package/server/api/aside/major/nav/global.ts +7 -7
- package/server/api/aside/minor/news.ts +7 -7
- package/server/api/aside/minor/path.ts +78 -78
- package/server/api/bitran/content/[location].ts +8 -8
- package/server/api/bitran/toc/[location].ts +7 -7
- package/server/api/content/data.ts +72 -72
- package/server/api/contributor/count.ts +6 -6
- package/server/api/fake/content.ts +11 -11
- package/server/api/fake/shared/languages.ts +12 -12
- package/server/api/language/functions.ts +12 -12
- package/server/api/language/phrase/[phraseId].ts +19 -19
- package/server/api/language/phraseIds.ts +8 -8
- package/server/api/preview/page/[...parts].ts +51 -51
- package/server/api/preview/unique/[location].ts +57 -57
- package/server/plugin/bitran/content.ts +187 -187
- package/server/plugin/bitran/location.ts +25 -25
- package/server/plugin/bitran/products/include.ts +230 -230
- package/server/plugin/bitran/products/link.ts +116 -116
- package/server/plugin/bitran/setup.ts +11 -9
- package/server/plugin/bitran/toc.ts +83 -83
- package/server/plugin/bitran/transpiler.ts +48 -46
- package/server/plugin/build/close.ts +10 -10
- package/server/plugin/build/jobs/content/builderArgs.ts +8 -8
- package/server/plugin/build/jobs/content/generic.ts +176 -176
- package/server/plugin/build/jobs/content/parse.ts +100 -100
- package/server/plugin/build/jobs/content/path.ts +6 -6
- package/server/plugin/build/jobs/content/type/book.ts +9 -9
- package/server/plugin/build/jobs/content/type/group.ts +37 -37
- package/server/plugin/build/jobs/content/type/topic.ts +36 -36
- package/server/plugin/build/jobs/contributors.ts +66 -66
- package/server/plugin/build/jobs/language.ts +36 -36
- package/server/plugin/build/jobs/nav.ts +210 -210
- package/server/plugin/build/process.ts +25 -25
- package/server/plugin/build/rebuild.ts +55 -55
- package/server/plugin/build/setup.ts +21 -21
- package/server/plugin/content/absoluteId.ts +94 -94
- package/server/plugin/content/context.ts +112 -112
- package/server/plugin/db/entities/Book.ts +7 -7
- package/server/plugin/db/entities/Content.ts +49 -49
- package/server/plugin/db/entities/Contribution.ts +10 -10
- package/server/plugin/db/entities/Contributor.ts +16 -16
- package/server/plugin/db/entities/Group.ts +14 -14
- package/server/plugin/db/entities/Hash.ts +15 -15
- package/server/plugin/db/entities/Topic.ts +20 -20
- package/server/plugin/db/entities/Unique.ts +21 -21
- package/server/plugin/db/setup.ts +34 -34
- package/server/plugin/global.ts +17 -18
- package/server/plugin/importer.ts +12 -12
- package/server/plugin/index.ts +9 -9
- package/server/plugin/logger.ts +23 -23
- package/server/plugin/nav/node.ts +26 -26
- package/server/plugin/nav/utils.ts +129 -129
- package/server/plugin/repository/book.ts +21 -21
- package/server/plugin/repository/content.ts +238 -238
- package/server/plugin/repository/contributor.ts +8 -8
- package/server/plugin/repository/frontNav.ts +148 -148
- package/server/plugin/repository/topic.ts +32 -32
- package/server/tsconfig.json +9 -9
- package/shared/aside/minor.ts +50 -50
- package/shared/asset.ts +15 -15
- package/shared/bitran/context.ts +8 -8
- package/shared/bitran/default.ts +46 -46
- package/shared/bitran/link/Link.vue +166 -167
- package/shared/bitran/link/factory.ts +24 -24
- package/shared/bitran/link/languages/en.ts +7 -7
- package/shared/bitran/link/languages/ru.ts +7 -7
- package/shared/bitran/link/renderer.ts +21 -21
- package/shared/bitran/link/shared.ts +17 -17
- package/shared/bitran/link/target.ts +134 -134
- package/shared/bitran/link/transpiler.ts +10 -10
- package/shared/bitran/location.ts +166 -166
- package/shared/bitran/toc.ts +8 -8
- package/shared/content/context.ts +9 -9
- package/shared/content/data/base.ts +32 -32
- package/shared/content/data/index.ts +5 -5
- package/shared/content/data/type/book.ts +5 -5
- package/shared/content/data/type/group.ts +6 -6
- package/shared/content/data/type/topic.ts +11 -11
- package/shared/content/previousNext.ts +9 -9
- package/shared/contributor.ts +5 -5
- package/shared/frontNav.ts +41 -41
- package/shared/icons.ts +38 -38
- package/shared/image.ts +5 -5
- package/shared/link.ts +25 -25
- package/shared/popover.ts +8 -0
- package/shared/types/language.ts +75 -75
- package/shared/utils/objectsEqual.ts +4 -4
- package/shared/utils/stringColor.ts +9 -9
- package/test/bitran/link/target.test.ts +141 -141
- package/test/bitran/location.test.ts +143 -143
- package/tsconfig.json +8 -8
- package/utils/stress.ts +9 -9
- package/app/components/main/utils/ContentFlag.vue +0 -15
- package/app/styles/default.scss +0 -83
|
@@ -1,210 +1,210 @@
|
|
|
1
|
-
import { globSync } from 'glob';
|
|
2
|
-
import chalk from 'chalk';
|
|
3
|
-
import { resolvePaths } from 'erudit-cog/kit';
|
|
4
|
-
import { contentTypes, topicParts, type ContentType } from 'erudit-cog/schema';
|
|
5
|
-
|
|
6
|
-
import { PROJECT_DIR } from '#erudit/globalPaths';
|
|
7
|
-
import { stress } from '@erudit/utils/stress';
|
|
8
|
-
import { debug, logger } from '@server/logger';
|
|
9
|
-
import {
|
|
10
|
-
createRootNode,
|
|
11
|
-
isRootNode,
|
|
12
|
-
type NavNode,
|
|
13
|
-
type RootNavNode,
|
|
14
|
-
} from '@server/nav/node';
|
|
15
|
-
import { ERUDIT_SERVER } from '../../global';
|
|
16
|
-
|
|
17
|
-
type Ids = Record<string, string>;
|
|
18
|
-
let ids: Ids;
|
|
19
|
-
|
|
20
|
-
const contentTargets = ERUDIT_SERVER.CONFIG?.contentTargets || [];
|
|
21
|
-
|
|
22
|
-
const nodePathRegexp = new RegExp(
|
|
23
|
-
`(?<pos>\\d+)(?<sep>-|\\+)(?<id>[\\w-]+)\\/(?<type>${contentTypes.join('|')})\\..*`,
|
|
24
|
-
);
|
|
25
|
-
|
|
26
|
-
export async function buildNav() {
|
|
27
|
-
debug.start('Building navigation tree...');
|
|
28
|
-
|
|
29
|
-
ids = {};
|
|
30
|
-
|
|
31
|
-
const rootNode = createRootNode();
|
|
32
|
-
rootNode.children = (await scanChildNodes(rootNode, false)).children;
|
|
33
|
-
ERUDIT_SERVER.NAV = rootNode.children ? rootNode : undefined;
|
|
34
|
-
|
|
35
|
-
const nodeCount = Object.values(ids).length;
|
|
36
|
-
|
|
37
|
-
if (nodeCount === 0) {
|
|
38
|
-
logger.warn('No content nodes detected! The site will be empty!');
|
|
39
|
-
} else {
|
|
40
|
-
if (ERUDIT_SERVER.CONFIG?.debug?.log) debugPrintNav(rootNode);
|
|
41
|
-
|
|
42
|
-
logger.success(
|
|
43
|
-
'Navigation tree built successfully!',
|
|
44
|
-
chalk.dim(`(${nodeCount})`),
|
|
45
|
-
);
|
|
46
|
-
}
|
|
47
|
-
}
|
|
48
|
-
|
|
49
|
-
async function scanChildNodes(
|
|
50
|
-
parent: NavNode | RootNavNode,
|
|
51
|
-
insideBook: boolean,
|
|
52
|
-
): Promise<{ children: NavNode[] | undefined; newIds: Ids }> {
|
|
53
|
-
const currentFsPath = isRootNode(parent) ? '' : parent.path;
|
|
54
|
-
|
|
55
|
-
const nodeFsPaths = globSync(
|
|
56
|
-
`${PROJECT_DIR}/content/${currentFsPath}/*/{${contentTypes.join(',')}}.{ts,js}`,
|
|
57
|
-
)
|
|
58
|
-
.sort()
|
|
59
|
-
.map((path) => resolvePaths(path));
|
|
60
|
-
|
|
61
|
-
let newIds: Ids = {};
|
|
62
|
-
const children: NavNode[] = [];
|
|
63
|
-
|
|
64
|
-
for (const nodeFsPath of nodeFsPaths) {
|
|
65
|
-
const pathParts:
|
|
66
|
-
| {
|
|
67
|
-
pos: string;
|
|
68
|
-
sep: string;
|
|
69
|
-
id: string;
|
|
70
|
-
type: ContentType;
|
|
71
|
-
}
|
|
72
|
-
| undefined = nodeFsPath.match(nodePathRegexp)?.groups as any;
|
|
73
|
-
|
|
74
|
-
if (!pathParts) continue; // Wrong path pattern
|
|
75
|
-
|
|
76
|
-
const nodePath = nodeFsPath
|
|
77
|
-
.replace(PROJECT_DIR + '/content/', '')
|
|
78
|
-
.split('/')
|
|
79
|
-
.slice(0, -1)
|
|
80
|
-
.join('/');
|
|
81
|
-
|
|
82
|
-
if (pathParts.type === 'book' && insideBook) {
|
|
83
|
-
logger.warn(
|
|
84
|
-
'Books inside books are not allowed!\n',
|
|
85
|
-
`Skipping ${stress(nodePath)} nav node!`,
|
|
86
|
-
);
|
|
87
|
-
continue;
|
|
88
|
-
}
|
|
89
|
-
|
|
90
|
-
if (!satisfiesContentTargets(nodePath)) continue; // Not a content target
|
|
91
|
-
|
|
92
|
-
const parentId = isRootNode(parent)
|
|
93
|
-
? ''
|
|
94
|
-
: parent.skip
|
|
95
|
-
? parent.id.split('/').slice(0, -1).join('/')
|
|
96
|
-
: parent.id;
|
|
97
|
-
|
|
98
|
-
// Regular id might not include parent id part if it is skipped
|
|
99
|
-
const id = parentId ? `${parentId}/${pathParts.id}` : pathParts.id;
|
|
100
|
-
|
|
101
|
-
if (ids[id]) {
|
|
102
|
-
logger.warn(
|
|
103
|
-
`Nav node ${stress(id)} ID collision!\n\n`,
|
|
104
|
-
`This ID belongs to ${stress(ids[id], chalk.greenBright)} nav node.\n`,
|
|
105
|
-
`Nav node ${stress(nodePath, chalk.redBright)} tries to use the same ID!\n`,
|
|
106
|
-
'Skipping nav node which causes collision!',
|
|
107
|
-
);
|
|
108
|
-
continue;
|
|
109
|
-
}
|
|
110
|
-
|
|
111
|
-
// Full id always includes parent id
|
|
112
|
-
const fullId = isRootNode(parent)
|
|
113
|
-
? pathParts.id
|
|
114
|
-
: `${parent.fullId}/${pathParts.id}`;
|
|
115
|
-
|
|
116
|
-
const skip = pathParts.sep === '+';
|
|
117
|
-
|
|
118
|
-
const childNode: NavNode = {
|
|
119
|
-
parent,
|
|
120
|
-
type: pathParts.type,
|
|
121
|
-
path: nodePath,
|
|
122
|
-
id,
|
|
123
|
-
fullId,
|
|
124
|
-
skip,
|
|
125
|
-
};
|
|
126
|
-
|
|
127
|
-
if (!validNode(childNode)) continue;
|
|
128
|
-
|
|
129
|
-
ids[id] = nodePath;
|
|
130
|
-
newIds[id] = nodePath;
|
|
131
|
-
|
|
132
|
-
const scanResult = await scanChildNodes(
|
|
133
|
-
childNode,
|
|
134
|
-
insideBook || childNode.type === 'book',
|
|
135
|
-
);
|
|
136
|
-
|
|
137
|
-
if (
|
|
138
|
-
['book', 'group'].includes(childNode.type) &&
|
|
139
|
-
!scanResult.children
|
|
140
|
-
) {
|
|
141
|
-
delete ids[id];
|
|
142
|
-
delete newIds[id];
|
|
143
|
-
for (const childNewId of Object.keys(scanResult.newIds))
|
|
144
|
-
delete ids[childNewId];
|
|
145
|
-
continue;
|
|
146
|
-
}
|
|
147
|
-
|
|
148
|
-
if (childNode.type === 'book') {
|
|
149
|
-
ERUDIT_SERVER.NAV_BOOKS ||= {};
|
|
150
|
-
ERUDIT_SERVER.NAV_BOOKS[id] = childNode;
|
|
151
|
-
}
|
|
152
|
-
|
|
153
|
-
childNode.children = scanResult.children;
|
|
154
|
-
children.push(childNode);
|
|
155
|
-
}
|
|
156
|
-
|
|
157
|
-
return {
|
|
158
|
-
children: children.length > 0 ? children : undefined,
|
|
159
|
-
newIds,
|
|
160
|
-
};
|
|
161
|
-
}
|
|
162
|
-
|
|
163
|
-
function satisfiesContentTargets(nodePath: string): boolean {
|
|
164
|
-
if (contentTargets.length === 0) return true;
|
|
165
|
-
|
|
166
|
-
for (const target of contentTargets)
|
|
167
|
-
if (nodePath.startsWith(target) || target.search(nodePath) === 0)
|
|
168
|
-
return true;
|
|
169
|
-
|
|
170
|
-
return false;
|
|
171
|
-
}
|
|
172
|
-
|
|
173
|
-
function validNode(node: NavNode): boolean {
|
|
174
|
-
switch (node.type) {
|
|
175
|
-
case 'topic':
|
|
176
|
-
const partPaths = globSync(
|
|
177
|
-
PROJECT_DIR +
|
|
178
|
-
`/content/${node.path}` +
|
|
179
|
-
`/{${topicParts.join(',')}}.bi`,
|
|
180
|
-
);
|
|
181
|
-
if (partPaths.length === 0) return false; // Topic is empty
|
|
182
|
-
break;
|
|
183
|
-
case 'book':
|
|
184
|
-
if (node.skip) {
|
|
185
|
-
logger.warn(
|
|
186
|
-
`Books can't skip their ID part!\n`,
|
|
187
|
-
`Skipping ${stress(node.path)} nav node!`,
|
|
188
|
-
);
|
|
189
|
-
return false;
|
|
190
|
-
}
|
|
191
|
-
break;
|
|
192
|
-
}
|
|
193
|
-
|
|
194
|
-
return true;
|
|
195
|
-
}
|
|
196
|
-
|
|
197
|
-
function debugPrintNav(node: RootNavNode) {
|
|
198
|
-
const logNode = (node: NavNode | RootNavNode, indent: number) => {
|
|
199
|
-
console.log(
|
|
200
|
-
isRootNode(node)
|
|
201
|
-
? chalk.dim('#root')
|
|
202
|
-
: `${' '.repeat(indent)}${node.id.split('/').pop()} ${chalk.dim(`[${node.type}${node.skip ? ', ' + chalk.yellow('skip') : ''}]`)}`,
|
|
203
|
-
);
|
|
204
|
-
|
|
205
|
-
if (node.children)
|
|
206
|
-
for (const child of node.children) logNode(child, indent + 1);
|
|
207
|
-
};
|
|
208
|
-
|
|
209
|
-
logNode(node, 0);
|
|
210
|
-
}
|
|
1
|
+
import { globSync } from 'glob';
|
|
2
|
+
import chalk from 'chalk';
|
|
3
|
+
import { resolvePaths } from 'erudit-cog/kit';
|
|
4
|
+
import { contentTypes, topicParts, type ContentType } from 'erudit-cog/schema';
|
|
5
|
+
|
|
6
|
+
import { PROJECT_DIR } from '#erudit/globalPaths';
|
|
7
|
+
import { stress } from '@erudit/utils/stress';
|
|
8
|
+
import { debug, logger } from '@server/logger';
|
|
9
|
+
import {
|
|
10
|
+
createRootNode,
|
|
11
|
+
isRootNode,
|
|
12
|
+
type NavNode,
|
|
13
|
+
type RootNavNode,
|
|
14
|
+
} from '@server/nav/node';
|
|
15
|
+
import { ERUDIT_SERVER } from '../../global';
|
|
16
|
+
|
|
17
|
+
type Ids = Record<string, string>;
|
|
18
|
+
let ids: Ids;
|
|
19
|
+
|
|
20
|
+
const contentTargets = ERUDIT_SERVER.CONFIG?.contentTargets || [];
|
|
21
|
+
|
|
22
|
+
const nodePathRegexp = new RegExp(
|
|
23
|
+
`(?<pos>\\d+)(?<sep>-|\\+)(?<id>[\\w-]+)\\/(?<type>${contentTypes.join('|')})\\..*`,
|
|
24
|
+
);
|
|
25
|
+
|
|
26
|
+
export async function buildNav() {
|
|
27
|
+
debug.start('Building navigation tree...');
|
|
28
|
+
|
|
29
|
+
ids = {};
|
|
30
|
+
|
|
31
|
+
const rootNode = createRootNode();
|
|
32
|
+
rootNode.children = (await scanChildNodes(rootNode, false)).children;
|
|
33
|
+
ERUDIT_SERVER.NAV = rootNode.children ? rootNode : undefined;
|
|
34
|
+
|
|
35
|
+
const nodeCount = Object.values(ids).length;
|
|
36
|
+
|
|
37
|
+
if (nodeCount === 0) {
|
|
38
|
+
logger.warn('No content nodes detected! The site will be empty!');
|
|
39
|
+
} else {
|
|
40
|
+
if (ERUDIT_SERVER.CONFIG?.debug?.log) debugPrintNav(rootNode);
|
|
41
|
+
|
|
42
|
+
logger.success(
|
|
43
|
+
'Navigation tree built successfully!',
|
|
44
|
+
chalk.dim(`(${nodeCount})`),
|
|
45
|
+
);
|
|
46
|
+
}
|
|
47
|
+
}
|
|
48
|
+
|
|
49
|
+
async function scanChildNodes(
|
|
50
|
+
parent: NavNode | RootNavNode,
|
|
51
|
+
insideBook: boolean,
|
|
52
|
+
): Promise<{ children: NavNode[] | undefined; newIds: Ids }> {
|
|
53
|
+
const currentFsPath = isRootNode(parent) ? '' : parent.path;
|
|
54
|
+
|
|
55
|
+
const nodeFsPaths = globSync(
|
|
56
|
+
`${PROJECT_DIR}/content/${currentFsPath}/*/{${contentTypes.join(',')}}.{ts,js}`,
|
|
57
|
+
)
|
|
58
|
+
.sort()
|
|
59
|
+
.map((path) => resolvePaths(path));
|
|
60
|
+
|
|
61
|
+
let newIds: Ids = {};
|
|
62
|
+
const children: NavNode[] = [];
|
|
63
|
+
|
|
64
|
+
for (const nodeFsPath of nodeFsPaths) {
|
|
65
|
+
const pathParts:
|
|
66
|
+
| {
|
|
67
|
+
pos: string;
|
|
68
|
+
sep: string;
|
|
69
|
+
id: string;
|
|
70
|
+
type: ContentType;
|
|
71
|
+
}
|
|
72
|
+
| undefined = nodeFsPath.match(nodePathRegexp)?.groups as any;
|
|
73
|
+
|
|
74
|
+
if (!pathParts) continue; // Wrong path pattern
|
|
75
|
+
|
|
76
|
+
const nodePath = nodeFsPath
|
|
77
|
+
.replace(PROJECT_DIR + '/content/', '')
|
|
78
|
+
.split('/')
|
|
79
|
+
.slice(0, -1)
|
|
80
|
+
.join('/');
|
|
81
|
+
|
|
82
|
+
if (pathParts.type === 'book' && insideBook) {
|
|
83
|
+
logger.warn(
|
|
84
|
+
'Books inside books are not allowed!\n',
|
|
85
|
+
`Skipping ${stress(nodePath)} nav node!`,
|
|
86
|
+
);
|
|
87
|
+
continue;
|
|
88
|
+
}
|
|
89
|
+
|
|
90
|
+
if (!satisfiesContentTargets(nodePath)) continue; // Not a content target
|
|
91
|
+
|
|
92
|
+
const parentId = isRootNode(parent)
|
|
93
|
+
? ''
|
|
94
|
+
: parent.skip
|
|
95
|
+
? parent.id.split('/').slice(0, -1).join('/')
|
|
96
|
+
: parent.id;
|
|
97
|
+
|
|
98
|
+
// Regular id might not include parent id part if it is skipped
|
|
99
|
+
const id = parentId ? `${parentId}/${pathParts.id}` : pathParts.id;
|
|
100
|
+
|
|
101
|
+
if (ids[id]) {
|
|
102
|
+
logger.warn(
|
|
103
|
+
`Nav node ${stress(id)} ID collision!\n\n`,
|
|
104
|
+
`This ID belongs to ${stress(ids[id], chalk.greenBright)} nav node.\n`,
|
|
105
|
+
`Nav node ${stress(nodePath, chalk.redBright)} tries to use the same ID!\n`,
|
|
106
|
+
'Skipping nav node which causes collision!',
|
|
107
|
+
);
|
|
108
|
+
continue;
|
|
109
|
+
}
|
|
110
|
+
|
|
111
|
+
// Full id always includes parent id
|
|
112
|
+
const fullId = isRootNode(parent)
|
|
113
|
+
? pathParts.id
|
|
114
|
+
: `${parent.fullId}/${pathParts.id}`;
|
|
115
|
+
|
|
116
|
+
const skip = pathParts.sep === '+';
|
|
117
|
+
|
|
118
|
+
const childNode: NavNode = {
|
|
119
|
+
parent,
|
|
120
|
+
type: pathParts.type,
|
|
121
|
+
path: nodePath,
|
|
122
|
+
id,
|
|
123
|
+
fullId,
|
|
124
|
+
skip,
|
|
125
|
+
};
|
|
126
|
+
|
|
127
|
+
if (!validNode(childNode)) continue;
|
|
128
|
+
|
|
129
|
+
ids[id] = nodePath;
|
|
130
|
+
newIds[id] = nodePath;
|
|
131
|
+
|
|
132
|
+
const scanResult = await scanChildNodes(
|
|
133
|
+
childNode,
|
|
134
|
+
insideBook || childNode.type === 'book',
|
|
135
|
+
);
|
|
136
|
+
|
|
137
|
+
if (
|
|
138
|
+
['book', 'group'].includes(childNode.type) &&
|
|
139
|
+
!scanResult.children
|
|
140
|
+
) {
|
|
141
|
+
delete ids[id];
|
|
142
|
+
delete newIds[id];
|
|
143
|
+
for (const childNewId of Object.keys(scanResult.newIds))
|
|
144
|
+
delete ids[childNewId];
|
|
145
|
+
continue;
|
|
146
|
+
}
|
|
147
|
+
|
|
148
|
+
if (childNode.type === 'book') {
|
|
149
|
+
ERUDIT_SERVER.NAV_BOOKS ||= {};
|
|
150
|
+
ERUDIT_SERVER.NAV_BOOKS[id] = childNode;
|
|
151
|
+
}
|
|
152
|
+
|
|
153
|
+
childNode.children = scanResult.children;
|
|
154
|
+
children.push(childNode);
|
|
155
|
+
}
|
|
156
|
+
|
|
157
|
+
return {
|
|
158
|
+
children: children.length > 0 ? children : undefined,
|
|
159
|
+
newIds,
|
|
160
|
+
};
|
|
161
|
+
}
|
|
162
|
+
|
|
163
|
+
function satisfiesContentTargets(nodePath: string): boolean {
|
|
164
|
+
if (contentTargets.length === 0) return true;
|
|
165
|
+
|
|
166
|
+
for (const target of contentTargets)
|
|
167
|
+
if (nodePath.startsWith(target) || target.search(nodePath) === 0)
|
|
168
|
+
return true;
|
|
169
|
+
|
|
170
|
+
return false;
|
|
171
|
+
}
|
|
172
|
+
|
|
173
|
+
function validNode(node: NavNode): boolean {
|
|
174
|
+
switch (node.type) {
|
|
175
|
+
case 'topic':
|
|
176
|
+
const partPaths = globSync(
|
|
177
|
+
PROJECT_DIR +
|
|
178
|
+
`/content/${node.path}` +
|
|
179
|
+
`/{${topicParts.join(',')}}.bi`,
|
|
180
|
+
);
|
|
181
|
+
if (partPaths.length === 0) return false; // Topic is empty
|
|
182
|
+
break;
|
|
183
|
+
case 'book':
|
|
184
|
+
if (node.skip) {
|
|
185
|
+
logger.warn(
|
|
186
|
+
`Books can't skip their ID part!\n`,
|
|
187
|
+
`Skipping ${stress(node.path)} nav node!`,
|
|
188
|
+
);
|
|
189
|
+
return false;
|
|
190
|
+
}
|
|
191
|
+
break;
|
|
192
|
+
}
|
|
193
|
+
|
|
194
|
+
return true;
|
|
195
|
+
}
|
|
196
|
+
|
|
197
|
+
function debugPrintNav(node: RootNavNode) {
|
|
198
|
+
const logNode = (node: NavNode | RootNavNode, indent: number) => {
|
|
199
|
+
console.log(
|
|
200
|
+
isRootNode(node)
|
|
201
|
+
? chalk.dim('#root')
|
|
202
|
+
: `${' '.repeat(indent)}${node.id.split('/').pop()} ${chalk.dim(`[${node.type}${node.skip ? ', ' + chalk.yellow('skip') : ''}]`)}`,
|
|
203
|
+
);
|
|
204
|
+
|
|
205
|
+
if (node.children)
|
|
206
|
+
for (const child of node.children) logNode(child, indent + 1);
|
|
207
|
+
};
|
|
208
|
+
|
|
209
|
+
logNode(node, 0);
|
|
210
|
+
}
|
|
@@ -1,25 +1,25 @@
|
|
|
1
|
-
import { debug, logger } from '@server/logger';
|
|
2
|
-
|
|
3
|
-
import { setup } from './setup';
|
|
4
|
-
import { setupLanguage } from './jobs/language';
|
|
5
|
-
import { buildContributors } from './jobs/contributors';
|
|
6
|
-
import { buildNav } from './jobs/nav';
|
|
7
|
-
import { buildContent } from './jobs/content/generic';
|
|
8
|
-
|
|
9
|
-
let initial = true;
|
|
10
|
-
|
|
11
|
-
export async function build() {
|
|
12
|
-
debug.start('Building data...');
|
|
13
|
-
|
|
14
|
-
if (initial) {
|
|
15
|
-
await setup();
|
|
16
|
-
initial = false;
|
|
17
|
-
}
|
|
18
|
-
|
|
19
|
-
await setupLanguage();
|
|
20
|
-
await buildContributors();
|
|
21
|
-
await buildNav();
|
|
22
|
-
await buildContent();
|
|
23
|
-
|
|
24
|
-
logger.success('Build successful!');
|
|
25
|
-
}
|
|
1
|
+
import { debug, logger } from '@server/logger';
|
|
2
|
+
|
|
3
|
+
import { setup } from './setup';
|
|
4
|
+
import { setupLanguage } from './jobs/language';
|
|
5
|
+
import { buildContributors } from './jobs/contributors';
|
|
6
|
+
import { buildNav } from './jobs/nav';
|
|
7
|
+
import { buildContent } from './jobs/content/generic';
|
|
8
|
+
|
|
9
|
+
let initial = true;
|
|
10
|
+
|
|
11
|
+
export async function build() {
|
|
12
|
+
debug.start('Building data...');
|
|
13
|
+
|
|
14
|
+
if (initial) {
|
|
15
|
+
await setup();
|
|
16
|
+
initial = false;
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
await setupLanguage();
|
|
20
|
+
await buildContributors();
|
|
21
|
+
await buildNav();
|
|
22
|
+
await buildContent();
|
|
23
|
+
|
|
24
|
+
logger.success('Build successful!');
|
|
25
|
+
}
|
|
@@ -1,55 +1,55 @@
|
|
|
1
|
-
import chokidar, { type FSWatcher } from 'chokidar';
|
|
2
|
-
|
|
3
|
-
import { ERUDIT_DIR, PROJECT_DIR } from '#erudit/globalPaths';
|
|
4
|
-
import { ERUDIT_SERVER } from '@server/global';
|
|
5
|
-
import { logger } from '@server/logger';
|
|
6
|
-
import { build } from '@server/build/process';
|
|
7
|
-
import { stress } from '@erudit/utils/stress';
|
|
8
|
-
|
|
9
|
-
const watchTargets: string[] = [
|
|
10
|
-
// Languages
|
|
11
|
-
ERUDIT_DIR + '/languages',
|
|
12
|
-
// Content directory
|
|
13
|
-
PROJECT_DIR + '/content',
|
|
14
|
-
// Contributors directory
|
|
15
|
-
PROJECT_DIR + '/contributors',
|
|
16
|
-
];
|
|
17
|
-
|
|
18
|
-
const ignoreTargets: string[] = [];
|
|
19
|
-
|
|
20
|
-
const rebuildDelay = 300;
|
|
21
|
-
|
|
22
|
-
//
|
|
23
|
-
//
|
|
24
|
-
//
|
|
25
|
-
|
|
26
|
-
let watcher: FSWatcher;
|
|
27
|
-
let rebuildTimeout: any;
|
|
28
|
-
let rebuildRequested = false;
|
|
29
|
-
|
|
30
|
-
export async function setupRebuildWatcher() {
|
|
31
|
-
if (watcher) return;
|
|
32
|
-
|
|
33
|
-
watcher = chokidar.watch(watchTargets, {
|
|
34
|
-
ignored: ignoreTargets,
|
|
35
|
-
ignoreInitial: true,
|
|
36
|
-
});
|
|
37
|
-
|
|
38
|
-
watcher.on('all', (event, path) =>
|
|
39
|
-
requestServerRebuild(`File change: ${stress(path)}`),
|
|
40
|
-
);
|
|
41
|
-
}
|
|
42
|
-
|
|
43
|
-
export async function requestServerRebuild(reason?: string) {
|
|
44
|
-
if (rebuildRequested) return;
|
|
45
|
-
|
|
46
|
-
clearTimeout(rebuildTimeout);
|
|
47
|
-
rebuildTimeout = setTimeout(async () => {
|
|
48
|
-
rebuildRequested = true;
|
|
49
|
-
await ERUDIT_SERVER.BUILD_PROMISE;
|
|
50
|
-
console.log('\n');
|
|
51
|
-
logger.info(`Rebuilding server data!${reason ? ` ${reason}` : ''}\n\n`);
|
|
52
|
-
await build();
|
|
53
|
-
rebuildRequested = false;
|
|
54
|
-
}, rebuildDelay);
|
|
55
|
-
}
|
|
1
|
+
import chokidar, { type FSWatcher } from 'chokidar';
|
|
2
|
+
|
|
3
|
+
import { ERUDIT_DIR, PROJECT_DIR } from '#erudit/globalPaths';
|
|
4
|
+
import { ERUDIT_SERVER } from '@server/global';
|
|
5
|
+
import { logger } from '@server/logger';
|
|
6
|
+
import { build } from '@server/build/process';
|
|
7
|
+
import { stress } from '@erudit/utils/stress';
|
|
8
|
+
|
|
9
|
+
const watchTargets: string[] = [
|
|
10
|
+
// Languages
|
|
11
|
+
ERUDIT_DIR + '/languages',
|
|
12
|
+
// Content directory
|
|
13
|
+
PROJECT_DIR + '/content',
|
|
14
|
+
// Contributors directory
|
|
15
|
+
PROJECT_DIR + '/contributors',
|
|
16
|
+
];
|
|
17
|
+
|
|
18
|
+
const ignoreTargets: string[] = [];
|
|
19
|
+
|
|
20
|
+
const rebuildDelay = 300;
|
|
21
|
+
|
|
22
|
+
//
|
|
23
|
+
//
|
|
24
|
+
//
|
|
25
|
+
|
|
26
|
+
let watcher: FSWatcher;
|
|
27
|
+
let rebuildTimeout: any;
|
|
28
|
+
let rebuildRequested = false;
|
|
29
|
+
|
|
30
|
+
export async function setupRebuildWatcher() {
|
|
31
|
+
if (watcher) return;
|
|
32
|
+
|
|
33
|
+
watcher = chokidar.watch(watchTargets, {
|
|
34
|
+
ignored: ignoreTargets,
|
|
35
|
+
ignoreInitial: true,
|
|
36
|
+
});
|
|
37
|
+
|
|
38
|
+
watcher.on('all', (event, path) =>
|
|
39
|
+
requestServerRebuild(`File change: ${stress(path)}`),
|
|
40
|
+
);
|
|
41
|
+
}
|
|
42
|
+
|
|
43
|
+
export async function requestServerRebuild(reason?: string) {
|
|
44
|
+
if (rebuildRequested) return;
|
|
45
|
+
|
|
46
|
+
clearTimeout(rebuildTimeout);
|
|
47
|
+
rebuildTimeout = setTimeout(async () => {
|
|
48
|
+
rebuildRequested = true;
|
|
49
|
+
await ERUDIT_SERVER.BUILD_PROMISE;
|
|
50
|
+
console.log('\n');
|
|
51
|
+
logger.info(`Rebuilding server data!${reason ? ` ${reason}` : ''}\n\n`);
|
|
52
|
+
await build();
|
|
53
|
+
rebuildRequested = false;
|
|
54
|
+
}, rebuildDelay);
|
|
55
|
+
}
|
|
@@ -1,21 +1,21 @@
|
|
|
1
|
-
import eruditConfig from '#erudit/config';
|
|
2
|
-
|
|
3
|
-
import { registerGlobals } from '@erudit/globals/register';
|
|
4
|
-
import { debug } from '@server/logger';
|
|
5
|
-
import { ERUDIT_SERVER } from '@server/global';
|
|
6
|
-
|
|
7
|
-
import { setupRebuildWatcher } from './rebuild';
|
|
8
|
-
import { setupDatabase } from '../db/setup';
|
|
9
|
-
import { setupBitranConfig } from '../bitran/setup';
|
|
10
|
-
|
|
11
|
-
export async function setup() {
|
|
12
|
-
debug.start('Running initial setup procedures...');
|
|
13
|
-
|
|
14
|
-
if (import.meta.dev) await setupRebuildWatcher();
|
|
15
|
-
|
|
16
|
-
registerGlobals();
|
|
17
|
-
ERUDIT_SERVER.CONFIG = eruditConfig;
|
|
18
|
-
|
|
19
|
-
await setupBitranConfig();
|
|
20
|
-
await setupDatabase();
|
|
21
|
-
}
|
|
1
|
+
import eruditConfig from '#erudit/config';
|
|
2
|
+
|
|
3
|
+
import { registerGlobals } from '@erudit/globals/register';
|
|
4
|
+
import { debug } from '@server/logger';
|
|
5
|
+
import { ERUDIT_SERVER } from '@server/global';
|
|
6
|
+
|
|
7
|
+
import { setupRebuildWatcher } from './rebuild';
|
|
8
|
+
import { setupDatabase } from '../db/setup';
|
|
9
|
+
import { setupBitranConfig } from '../bitran/setup';
|
|
10
|
+
|
|
11
|
+
export async function setup() {
|
|
12
|
+
debug.start('Running initial setup procedures...');
|
|
13
|
+
|
|
14
|
+
if (import.meta.dev) await setupRebuildWatcher();
|
|
15
|
+
|
|
16
|
+
registerGlobals();
|
|
17
|
+
ERUDIT_SERVER.CONFIG = eruditConfig;
|
|
18
|
+
|
|
19
|
+
await setupBitranConfig();
|
|
20
|
+
await setupDatabase();
|
|
21
|
+
}
|