-
1
-
2
-
3
-
4
-
5
-
6
-
7
-
8
-
9
-
10
-
11
-
12
-
13
-
14
-
15
-
16
-
17
-
18
-
19
-
20
-
21
-
22
-
23
-
24
-
25
-
26
-
27
-
28
-
29
-
30
-
31
-
32
-
33
-
34
-
35
-
36
-
37
-
38
-
39
-
40
-
41
-
42
-
43
-
44
-
45
-
46
-
47
-
48
-
49
-
50
-
51
-
52
-
53
-
54
-
55
-
56
-
57
-
58
-
59
-
60
-
61
-
62
-
63
-
64
-
65
-
66
-
67
-
68
-
69
-
70
-
71
-
72
-
73
-
74
-
75
-
76
-
77
-
78
-
79
-
80
-
81
-
82
-
83
-
84
-
85
-
86
-
87
-
88
-
89
-
90
-
91
-
92
-
93
-
94
-
95
-
96
-
97
-
98
-
99
-
100
-
101
-
102
-
103
-
104
-
105
-
106
-
107
-
108
-
109
-
110
-
111
-
112
-
113
-
114
-
115
-
116
-
117
-
118
-
119
-
120
-
121
// SPDX-FileCopyrightText: 2024 Shota FUJI <pockawoooh@gmail.com>
//
// SPDX-License-Identifier: Apache-2.0
import type * as Mdast from "../../deps/esm.sh/mdast/types.ts";
import { SKIP, visit } from "../../deps/esm.sh/unist-util-visit/mod.ts";
import { definitions } from "../../deps/esm.sh/mdast-util-definitions/mod.ts";
import type { ParseParameters } from "../interface.ts";
import type { DocumentToken } from "../../types.ts";
import type { OfmWikilink } from "./mdast_util_ofm_wikilink.ts";
const SEPARATOR = "/";
const FRAGMENT_PREFIX = "#";
const IGNORE_REGEXP_PATTERN = /^([a-z0-9]+:\/\/|#)/i;
function setDocumentToken(node: Mdast.Node, token: DocumentToken): void {
node.data ??= {};
// @ts-expect-error: incorrect library type definition.
node.data.macanaDocumentToken = token;
}
function parseInternalLink(
link: string,
): { path: readonly string[]; fragments: readonly string[] } {
const [ref, ...fragments] = link.split(FRAGMENT_PREFIX);
return {
path: (ref || "").split(SEPARATOR),
fragments: fragments,
};
}
/**
* Searches document references and Marks thoese node by setting `macanaDocumentToken`
* with Document Token.
*
* This function mutates the Mdast tree in place.
*/
export async function macanaMarkDocumentToken(
tree: Mdast.Nodes | OfmWikilink,
getDocumentToken: ParseParameters["getDocumentToken"],
): Promise<void> {
const promises: Promise<unknown>[] = [];
const defs = definitions(tree as Mdast.Nodes);
visit(
tree,
(node) =>
node.type === "link" || node.type === "linkReference" ||
node.type === "ofmWikilink",
(node) => {
switch (node.type) {
case "ofmWikilink": {
const { path, fragments } = parseInternalLink(node.target);
const token = getDocumentToken(path, fragments);
if (token instanceof Promise) {
promises.push(token.then((t) => {
setDocumentToken(node, t);
}));
return SKIP;
}
setDocumentToken(node, token);
return SKIP;
}
case "link": {
if (IGNORE_REGEXP_PATTERN.test(node.url)) {
return SKIP;
}
const { path, fragments } = parseInternalLink(node.url);
const token = getDocumentToken(path, fragments);
if (token instanceof Promise) {
promises.push(token.then((t) => {
setDocumentToken(node, t);
}));
return SKIP;
}
setDocumentToken(node, token);
return SKIP;
}
case "linkReference": {
const def = defs(node.identifier);
if (!def) {
return;
}
if (IGNORE_REGEXP_PATTERN.test(def.url)) {
return SKIP;
}
const { path, fragments } = parseInternalLink(def.url);
const token = getDocumentToken(path, fragments);
if (token instanceof Promise) {
promises.push(token.then((t) => {
setDocumentToken(node, t);
}));
return SKIP;
}
setDocumentToken(node, token);
return SKIP;
}
}
},
);
if (promises.length > 0) {
await Promise.all(promises);
}
}