-
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
-
122
-
123
-
124
-
125
-
126
-
127
-
128
-
129
-
130
-
131
-
132
// SPDX-FileCopyrightText: 2024 Shota FUJI <pockawoooh@gmail.com>
//
// SPDX-License-Identifier: Apache-2.0
import type {
DirectoryReader,
FileReader,
} from "../filesystem_reader/interface.ts";
import type { MetadataParser } from "../metadata_parser/interface.ts";
import type {
BuildParameters,
Document,
DocumentDirectory,
DocumentTree,
TreeBuilder,
} from "./interface.ts";
import { assertDocumentTreeIsValid } from "./assert.ts";
export interface MultiLocaleTreeBuilderConfig {
defaultLocale?: string;
/**
* Callback function to be invoked on every file and directory.
* If this function returned true, the file or directory is skipped and does not
* appear on the resulted document tree.
*/
ignore?(fileOrDirectory: FileReader | DirectoryReader): boolean;
}
export class MultiLocaleTreeBuilder implements TreeBuilder {
#config: MultiLocaleTreeBuilderConfig;
constructor(config: MultiLocaleTreeBuilderConfig = {}) {
this.#config = config;
}
async build(
{ fileSystemReader, metadataParser }: BuildParameters,
): Promise<DocumentTree> {
const root = await fileSystemReader.getRootDirectory();
const nodes = await root.read();
const map = new Map<string, Array<Document | DocumentDirectory>>();
for (const node of nodes) {
if (this.#config.ignore && this.#config.ignore(node)) {
// TODO: Debug log
continue;
}
if (node.type === "file") {
// TODO: Warning instead?
throw new Error(
`You can't have a regular file at top-level directory, found "${node.name}".`,
);
}
const locale = node.name;
const children = await node.read();
const entries = await Promise.all(
children.map((child) => this.#build(child, metadataParser)),
);
map.set(
locale,
entries.filter((entry): entry is NonNullable<typeof entry> => !!entry),
);
}
const firstLocale = map.keys().next().value;
if (typeof firstLocale !== "string") {
throw new Error("No locale directories found.");
}
if (this.#config.defaultLocale && !map.has(this.#config.defaultLocale)) {
throw new Error(
`Received defaultLocale=${this.#config.defaultLocale}, however we couldn't find that locale (found ${
Array.from(map.keys()).join(", ")
}).`,
);
}
const tree: DocumentTree = {
defaultLocale: this.#config.defaultLocale || firstLocale,
locales: map,
};
assertDocumentTreeIsValid(tree);
return tree;
}
async #build(
node: FileReader | DirectoryReader,
metadataParser: MetadataParser,
): Promise<DocumentDirectory | Document | null> {
if (this.#config.ignore && this.#config.ignore(node)) {
// TODO: Debug log
return null;
}
const metadata = await metadataParser.parse(node);
// This SHOULD have check for `metadata.skip` being `true`. However, a bug
// (or "feature") in TypeScript breaks type-narrowing by doing so.
if ("skip" in metadata) {
// TODO: Debug log
return null;
}
if (node.type === "file") {
return {
metadata,
file: node,
};
}
const children = await node.read();
const entries = (await Promise.all(
children.map((child) => this.#build(child, metadataParser)),
)).filter((child): child is NonNullable<typeof child> => !!child);
return {
metadata,
directory: node,
entries,
};
}
}