tsondb 0.11.2 → 0.11.4
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/dist/src/bin/tsondb.js +5 -0
- package/dist/src/node/config.d.ts +33 -4
- package/dist/src/node/config.js +20 -2
- package/dist/src/node/server/api/git.js +1 -0
- package/dist/src/shared/utils/markdown/indentation.d.ts +2 -0
- package/dist/src/shared/utils/markdown/indentation.js +14 -0
- package/dist/src/shared/utils/markdown.d.ts +2 -1
- package/dist/src/shared/utils/markdown.js +62 -38
- package/dist/src/web/utils/BlockMarkdown.js +6 -7
- package/package.json +1 -1
package/dist/src/bin/tsondb.js
CHANGED
|
@@ -13,6 +13,7 @@ import { join } from "node:path";
|
|
|
13
13
|
import { cwd } from "node:process";
|
|
14
14
|
import { pathToFileURL } from "node:url";
|
|
15
15
|
import { parseArguments } from "simple-cli-args";
|
|
16
|
+
import { validateConfigForFormatting, validateConfigForGeneration, validateConfigForServer, validateConfigForTesting, } from "../node/config.js";
|
|
16
17
|
import { format, generateOutputs, serve, validate } from "../node/index.js";
|
|
17
18
|
const debug = Debug("tsondb:cli");
|
|
18
19
|
const passedArguments = parseArguments({
|
|
@@ -80,14 +81,17 @@ if (passedArguments.command === undefined) {
|
|
|
80
81
|
switch (passedArguments.command.name) {
|
|
81
82
|
case "generate":
|
|
82
83
|
debug(`running command: generate`);
|
|
84
|
+
validateConfigForGeneration(config);
|
|
83
85
|
await generateOutputs(config.schema, config.outputs);
|
|
84
86
|
break;
|
|
85
87
|
case "serve":
|
|
86
88
|
debug(`running command: serve`);
|
|
89
|
+
validateConfigForServer(config);
|
|
87
90
|
await serve(config.schema, config.dataRootPath, config.defaultLocales, config.homeLayoutSections, config.serverOptions, config.customStylesheetPath);
|
|
88
91
|
break;
|
|
89
92
|
case "validate":
|
|
90
93
|
debug(`running command: validate`);
|
|
94
|
+
validateConfigForTesting(config);
|
|
91
95
|
if (passedArguments.command.options?.checkReferentialIntegrity !== undefined) {
|
|
92
96
|
debug(`check referential integrity: ${passedArguments.command.options.checkReferentialIntegrity ? "yes" : "no"}`);
|
|
93
97
|
}
|
|
@@ -102,6 +106,7 @@ switch (passedArguments.command.name) {
|
|
|
102
106
|
break;
|
|
103
107
|
case "format":
|
|
104
108
|
debug(`running command: format`);
|
|
109
|
+
validateConfigForFormatting(config);
|
|
105
110
|
await format(config.schema, config.dataRootPath);
|
|
106
111
|
break;
|
|
107
112
|
}
|
|
@@ -1,12 +1,15 @@
|
|
|
1
1
|
import type { Output } from "../shared/output.ts";
|
|
2
2
|
import type { EntityDecl } from "./schema/index.ts";
|
|
3
3
|
import type { Schema } from "./schema/Schema.ts";
|
|
4
|
+
/**
|
|
5
|
+
* The main configuration type for TSONDB.
|
|
6
|
+
*/
|
|
4
7
|
export type Config = {
|
|
5
8
|
serverOptions?: ServerOptions;
|
|
6
9
|
schema: Schema;
|
|
7
|
-
outputs
|
|
8
|
-
defaultLocales
|
|
9
|
-
dataRootPath
|
|
10
|
+
outputs?: Output[];
|
|
11
|
+
defaultLocales?: string[];
|
|
12
|
+
dataRootPath?: string;
|
|
10
13
|
homeLayoutSections?: HomeLayoutSection[];
|
|
11
14
|
customStylesheetPath?: string;
|
|
12
15
|
};
|
|
@@ -18,4 +21,30 @@ export type HomeLayoutSection = {
|
|
|
18
21
|
comment?: string;
|
|
19
22
|
entities: EntityDecl[];
|
|
20
23
|
};
|
|
21
|
-
|
|
24
|
+
/**
|
|
25
|
+
* The configuration type required for generation commands.
|
|
26
|
+
*/
|
|
27
|
+
export type GenerationConfig = {
|
|
28
|
+
schema: Schema;
|
|
29
|
+
outputs: Output[];
|
|
30
|
+
};
|
|
31
|
+
export declare const validateConfigForGeneration: (config: Config) => asserts config is GenerationConfig;
|
|
32
|
+
/**
|
|
33
|
+
* The configuration type required for any commands that need to read data stored in the database.
|
|
34
|
+
*/
|
|
35
|
+
export type DataConfig = {
|
|
36
|
+
schema: Schema;
|
|
37
|
+
dataRootPath: string;
|
|
38
|
+
};
|
|
39
|
+
/**
|
|
40
|
+
* The configuration type required for running the server for the editor.
|
|
41
|
+
*/
|
|
42
|
+
export type ServerConfig = DataConfig & {
|
|
43
|
+
serverOptions?: ServerOptions;
|
|
44
|
+
defaultLocales: string[];
|
|
45
|
+
homeLayoutSections?: HomeLayoutSection[];
|
|
46
|
+
customStylesheetPath?: string;
|
|
47
|
+
};
|
|
48
|
+
export declare const validateConfigForServer: (config: Config) => asserts config is ServerConfig;
|
|
49
|
+
export declare const validateConfigForTesting: (config: Config) => asserts config is DataConfig;
|
|
50
|
+
export declare const validateConfigForFormatting: (config: Config) => asserts config is DataConfig;
|
package/dist/src/node/config.js
CHANGED
|
@@ -1,5 +1,23 @@
|
|
|
1
|
-
export const
|
|
2
|
-
if (config.
|
|
1
|
+
export const validateConfigForGeneration = config => {
|
|
2
|
+
if ((config.outputs?.length ?? 0) === 0) {
|
|
3
|
+
throw new Error("At least one output must be specified in the config.");
|
|
4
|
+
}
|
|
5
|
+
};
|
|
6
|
+
export const validateConfigForServer = config => {
|
|
7
|
+
if ((config.defaultLocales?.length ?? 0) === 0) {
|
|
3
8
|
throw new Error("At least one default locale must be specified in the config.");
|
|
4
9
|
}
|
|
10
|
+
if (config.dataRootPath === undefined) {
|
|
11
|
+
throw new Error("A data root path must be specified in the config.");
|
|
12
|
+
}
|
|
13
|
+
};
|
|
14
|
+
export const validateConfigForTesting = config => {
|
|
15
|
+
if (config.dataRootPath === undefined) {
|
|
16
|
+
throw new Error("A data root path must be specified in the config.");
|
|
17
|
+
}
|
|
18
|
+
};
|
|
19
|
+
export const validateConfigForFormatting = config => {
|
|
20
|
+
if (config.dataRootPath === undefined) {
|
|
21
|
+
throw new Error("A data root path must be specified in the config.");
|
|
22
|
+
}
|
|
5
23
|
};
|
|
@@ -228,6 +228,7 @@ gitApi.post("/pull", async (req, res) => {
|
|
|
228
228
|
const status = await req.git.status();
|
|
229
229
|
const remotes = await req.git.getRemotes();
|
|
230
230
|
await req.git.pull(remotes[0]?.name, status.current ?? undefined);
|
|
231
|
+
await reinit(req);
|
|
231
232
|
res.set("Content-Type", "text/plain");
|
|
232
233
|
res.status(200).send("Pull successful");
|
|
233
234
|
}
|
|
@@ -0,0 +1,14 @@
|
|
|
1
|
+
const emptyLinePattern = /^ *$$/;
|
|
2
|
+
export const detectIndentation = (text, excludeFirstLineForDetection) => {
|
|
3
|
+
const nonEmptyLines = text.split("\n").filter(line => !emptyLinePattern.test(line));
|
|
4
|
+
return nonEmptyLines[excludeFirstLineForDetection ? 1 : 0]?.match(/^ +/)?.[0].length ?? 0;
|
|
5
|
+
};
|
|
6
|
+
const mapLines = (text, lineMapper) => text.split("\n").map(lineMapper).join("\n");
|
|
7
|
+
export const removeIndentation = (text, excludeFirstLineForDetection, fixedIndentation) => {
|
|
8
|
+
const indentation = fixedIndentation ?? detectIndentation(text, excludeFirstLineForDetection);
|
|
9
|
+
if (indentation < 1) {
|
|
10
|
+
return text;
|
|
11
|
+
}
|
|
12
|
+
const regex = new RegExp(`^ {${indentation.toString()}}`);
|
|
13
|
+
return mapLines(text, line => line.replace(regex, ""));
|
|
14
|
+
};
|
|
@@ -1,4 +1,5 @@
|
|
|
1
1
|
import { chunk } from "./array.js";
|
|
2
|
+
import { detectIndentation, removeIndentation } from "./markdown/indentation.js";
|
|
2
3
|
import { omitUndefinedKeys } from "./object.js";
|
|
3
4
|
import { assertExhaustive } from "./typeSafety.js";
|
|
4
5
|
const codeRule = {
|
|
@@ -229,26 +230,56 @@ const nodesForTrailingWhitespace = (text) => {
|
|
|
229
230
|
return trailingWhitespace.length === 0 ? [] : [textNode(trailingWhitespace)];
|
|
230
231
|
};
|
|
231
232
|
const listRule = {
|
|
232
|
-
pattern: /^((?:(?:\d+\.|[-*]) [^\n]+?)(?:\n(?:\d+\.|[-*]) [^\n]+?)*)(\n{2,}|\s*$)/,
|
|
233
|
-
|
|
234
|
-
|
|
235
|
-
|
|
236
|
-
|
|
237
|
-
|
|
238
|
-
|
|
239
|
-
|
|
240
|
-
|
|
241
|
-
|
|
242
|
-
|
|
243
|
-
|
|
244
|
-
kind
|
|
245
|
-
|
|
246
|
-
|
|
247
|
-
|
|
248
|
-
|
|
249
|
-
|
|
250
|
-
|
|
251
|
-
|
|
233
|
+
// pattern: /^((?:(?:\d+\.|[-*]) [^\n]+?)(?:\n(?:\d+\.|[-*]) [^\n]+?)*)(\n{2,}|\s*$)/,
|
|
234
|
+
pattern: /^((?:(?:\d+\. +[^\n]+(?:\n\d+\. +[^\n]+|\n {2,}[^\n]+|\n)*)|(?:- +[^\n]+(?:\n- +[^\n]+|\n {2,}[^\n]+|\n)*)))(\n{2,}|\s*$)/,
|
|
235
|
+
map: result => {
|
|
236
|
+
const listItemPairs = (result[1] ?? "").split(/(?:^|\n)(?:\d+\.|-)/).slice(1);
|
|
237
|
+
const listItems = listItemPairs.map((itemText) => {
|
|
238
|
+
const [primaryContent = "", _separator, ...additionalContents] = itemText.split(/(\n *\n|\n(?= +(?:\d+\.|-) ))/);
|
|
239
|
+
const additionalContent = additionalContents.join("");
|
|
240
|
+
const primaryContentNodes = parseInlineMarkdown(primaryContent.trim());
|
|
241
|
+
const additionalContentNodes = additionalContent
|
|
242
|
+
? parseBlockMarkdown(removeIndentation(additionalContent))
|
|
243
|
+
: [];
|
|
244
|
+
if (additionalContentNodes.length < 2 &&
|
|
245
|
+
(additionalContentNodes[0] === undefined || additionalContentNodes[0].kind === "list")) {
|
|
246
|
+
return omitUndefinedKeys({
|
|
247
|
+
kind: "listItem",
|
|
248
|
+
inlineLabel: primaryContentNodes,
|
|
249
|
+
content: [additionalContentNodes[0]].filter(element => element !== undefined),
|
|
250
|
+
});
|
|
251
|
+
}
|
|
252
|
+
else {
|
|
253
|
+
return omitUndefinedKeys({
|
|
254
|
+
kind: "listItem",
|
|
255
|
+
content: additionalContentNodes,
|
|
256
|
+
});
|
|
257
|
+
}
|
|
258
|
+
});
|
|
259
|
+
return {
|
|
260
|
+
kind: "list",
|
|
261
|
+
ordered: /^\d+\. /.test(result[0]),
|
|
262
|
+
content: listItems,
|
|
263
|
+
};
|
|
264
|
+
},
|
|
265
|
+
mapHighlighting: result => {
|
|
266
|
+
const listItemPairs = chunk((result[1] ?? "").split(/(?:^|\n)(\d+\.|-)/).slice(1), 2);
|
|
267
|
+
const listItems = listItemPairs.flatMap(([marker = "", itemText = ""], itemIndex, itemArray) => {
|
|
268
|
+
const [primaryContent = "", separator, ...additionalContents] = itemText.split(/(\n *\n|\n(?= +(?:\d+\.|-) ))/);
|
|
269
|
+
const additionalContent = additionalContents.join("");
|
|
270
|
+
return [
|
|
271
|
+
{
|
|
272
|
+
kind: "listItemMarker",
|
|
273
|
+
content: marker,
|
|
274
|
+
},
|
|
275
|
+
...parseInlineMarkdownForSyntaxHighlighting(primaryContent),
|
|
276
|
+
...(separator ? [textNode(separator)] : []),
|
|
277
|
+
...addIndentationToSyntax(parseBlockMarkdownForSyntaxHighlighting(removeIndentation(additionalContent)), detectIndentation(additionalContent)),
|
|
278
|
+
...(itemIndex < itemArray.length - 1 ? [textNode("\n")] : []),
|
|
279
|
+
];
|
|
280
|
+
});
|
|
281
|
+
return [...listItems, ...nodesForTrailingWhitespace(result[2])];
|
|
282
|
+
},
|
|
252
283
|
};
|
|
253
284
|
const paragraphRule = {
|
|
254
285
|
pattern: /^((?:[^\n]+?)(?:\n[^\n]+?)*)(\n{2,}|\s*$)/,
|
|
@@ -444,10 +475,6 @@ const containerRule = {
|
|
|
444
475
|
...nodesForTrailingWhitespace(trailingWhitespace),
|
|
445
476
|
],
|
|
446
477
|
};
|
|
447
|
-
const removeIndentation = (text) => text
|
|
448
|
-
.split("\n")
|
|
449
|
-
.map(line => line.replace(/^ {2}/, ""))
|
|
450
|
-
.join("\n");
|
|
451
478
|
export const syntaxNodeToString = (node) => {
|
|
452
479
|
switch (node.kind) {
|
|
453
480
|
case "bold":
|
|
@@ -471,7 +498,7 @@ export const syntaxNodeToString = (node) => {
|
|
|
471
498
|
return assertExhaustive(node);
|
|
472
499
|
}
|
|
473
500
|
};
|
|
474
|
-
const addIndentationToSyntax = (nodes,
|
|
501
|
+
const addIndentationToSyntax = (nodes, indentation, excludeFirstLine = false) => nodes.reduce((accNodes, currentNode) => {
|
|
475
502
|
switch (currentNode.kind) {
|
|
476
503
|
case "bold":
|
|
477
504
|
case "italic":
|
|
@@ -482,7 +509,7 @@ const addIndentationToSyntax = (nodes, nextUpperNode) => nodes.reduce((accNodes,
|
|
|
482
509
|
...accNodes,
|
|
483
510
|
{
|
|
484
511
|
...currentNode,
|
|
485
|
-
content: addIndentationToSyntax(currentNode.content,
|
|
512
|
+
content: addIndentationToSyntax(currentNode.content, indentation, excludeFirstLine),
|
|
486
513
|
},
|
|
487
514
|
];
|
|
488
515
|
case "text":
|
|
@@ -493,15 +520,12 @@ const addIndentationToSyntax = (nodes, nextUpperNode) => nodes.reduce((accNodes,
|
|
|
493
520
|
case "sectionMarker":
|
|
494
521
|
case "footnoteMarker":
|
|
495
522
|
case "definitionMarker": {
|
|
496
|
-
const nextNode = nodes[index + 1] ?? nextUpperNode;
|
|
497
|
-
const currentContent = currentNode.content.endsWith("\n") &&
|
|
498
|
-
nextNode &&
|
|
499
|
-
/^[^\n]*?\S+?/.test(syntaxNodeToString(nextNode))
|
|
500
|
-
? currentNode.content + " "
|
|
501
|
-
: currentNode.content;
|
|
502
523
|
return [
|
|
503
524
|
...accNodes,
|
|
504
|
-
{
|
|
525
|
+
{
|
|
526
|
+
...currentNode,
|
|
527
|
+
content: currentNode.content.replace(/\n(?!\n)/g, "\n" + " ".repeat(indentation)),
|
|
528
|
+
},
|
|
505
529
|
];
|
|
506
530
|
}
|
|
507
531
|
case "footnoteRef":
|
|
@@ -509,18 +533,18 @@ const addIndentationToSyntax = (nodes, nextUpperNode) => nodes.reduce((accNodes,
|
|
|
509
533
|
default:
|
|
510
534
|
return assertExhaustive(currentNode);
|
|
511
535
|
}
|
|
512
|
-
}, []);
|
|
536
|
+
}, excludeFirstLine ? [] : [textNode(" ".repeat(indentation))]);
|
|
513
537
|
const footnoteRule = {
|
|
514
538
|
pattern: /^\[\^([a-zA-Z0-9]+?)\]: (.+?(?:\n(?: {2}.+)?)*)(\n{2,}|\s*$)/,
|
|
515
539
|
map: ([_match, label = "", content = "", _trailingWhitespace]) => ({
|
|
516
540
|
kind: "footnote",
|
|
517
541
|
label: label,
|
|
518
|
-
content: parseBlockMarkdown(removeIndentation(content)),
|
|
542
|
+
content: parseBlockMarkdown(removeIndentation(content, true)),
|
|
519
543
|
}),
|
|
520
544
|
mapHighlighting: ([_match, label = "", content = "", trailingWhitespace,]) => [
|
|
521
545
|
{ kind: "footnoteMarker", content: `[^${label}]:` },
|
|
522
546
|
textNode(" "),
|
|
523
|
-
...addIndentationToSyntax(parseBlockMarkdownForSyntaxHighlighting(removeIndentation(content))),
|
|
547
|
+
...addIndentationToSyntax(parseBlockMarkdownForSyntaxHighlighting(removeIndentation(content, true)), detectIndentation(content, true), true),
|
|
524
548
|
...nodesForTrailingWhitespace(trailingWhitespace),
|
|
525
549
|
],
|
|
526
550
|
};
|
|
@@ -537,7 +561,7 @@ const definitionListRule = {
|
|
|
537
561
|
const definitions = definitionsText
|
|
538
562
|
.split("\n:")
|
|
539
563
|
.slice(1)
|
|
540
|
-
.map(definition => parseBlockMarkdown(removeIndentation(definition.trim())));
|
|
564
|
+
.map(definition => parseBlockMarkdown(removeIndentation(definition.trim(), true)));
|
|
541
565
|
return {
|
|
542
566
|
kind: "definitionListItem",
|
|
543
567
|
terms,
|
|
@@ -562,7 +586,7 @@ const definitionListRule = {
|
|
|
562
586
|
.slice(1)
|
|
563
587
|
.flatMap((definition, defIndex, defArray) => [
|
|
564
588
|
{ kind: "definitionMarker", content: ":" },
|
|
565
|
-
...addIndentationToSyntax(parseBlockMarkdownForSyntaxHighlighting(removeIndentation(definition))),
|
|
589
|
+
...addIndentationToSyntax(parseBlockMarkdownForSyntaxHighlighting(removeIndentation(definition, true)), detectIndentation(definition, true), true),
|
|
566
590
|
...(defIndex < defArray.length - 1 ? [textNode("\n")] : []),
|
|
567
591
|
]);
|
|
568
592
|
return [...terms, textNode("\n"), ...definitions];
|
|
@@ -11,13 +11,12 @@ export const BlockMarkdown = ({ node, outerHeadingLevel = 0, insertBefore, footn
|
|
|
11
11
|
case "heading":
|
|
12
12
|
const Tag = `h${(node.level + outerHeadingLevel).toString()}`;
|
|
13
13
|
return (_jsxs(Tag, { children: [insertBefore, node.content.map((inline, ii) => (_jsx(InlineMarkdown, { node: inline }, ii)))] }));
|
|
14
|
-
case "list":
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
}
|
|
14
|
+
case "list": {
|
|
15
|
+
const Tag = node.ordered ? "ol" : "ul";
|
|
16
|
+
return (_jsxs(_Fragment, { children: [insertBefore, _jsx(Tag, { children: node.content.map((item, ii) => (_jsxs("li", { children: [item.inlineLabel && item.inlineLabel.length > 0
|
|
17
|
+
? item.inlineLabel.map((inline, iii) => (_jsx(InlineMarkdown, { node: inline }, iii)))
|
|
18
|
+
: null, item.content.map((content, iii) => (_jsx(BlockMarkdown, { node: content }, iii)))] }, ii))) })] }));
|
|
19
|
+
}
|
|
21
20
|
case "table":
|
|
22
21
|
return (_jsxs(_Fragment, { children: [insertBefore, _jsxs("table", { children: [node.caption !== undefined && (_jsx("caption", { children: node.caption.map((inline, ci) => (_jsx(InlineMarkdown, { node: inline }, ci))) })), _jsx("thead", { children: _jsx(TableRow, { columns: node.columns, cells: node.header, cellType: "th" }) }), checkTableRowsAreSections(node.rows) ? (node.rows.map((section, si) => (_jsxs("tbody", { children: [section.header && (_jsx(TableRow, { columns: node.columns, cells: section.header, cellType: "th" })), section.rows.map((row, ri) => (_jsx(TableRow, { columns: node.columns, cells: row.cells }, ri)))] }, si)))) : (_jsx("tbody", { children: node.rows.map((row, ri) => (_jsx(TableRow, { columns: node.columns, cells: row.cells }, ri))) }))] })] }));
|
|
23
22
|
case "container":
|