@ukwhatn/wikidot 1.0.9 → 4.0.2
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/README.md +227 -79
- package/dist/errors.cjs +114 -0
- package/dist/errors.d.cts +94 -0
- package/dist/errors.d.ts +94 -0
- package/dist/errors.js +81 -0
- package/dist/index.cjs +4624 -0
- package/dist/index.d.cts +2106 -0
- package/dist/index.d.ts +2106 -1
- package/dist/index.js +2651 -6
- package/dist/shared/index-7dqqxq7x.js +105 -0
- package/dist/shared/index-f2eh3ykk.js +172 -0
- package/dist/shared/index-kka6e8cb.js +627 -0
- package/dist/shared/index-ytknx2hn.js +980 -0
- package/package.json +64 -26
- package/Makefile +0 -13
- package/dist/common/exceptions.d.ts +0 -95
- package/dist/common/exceptions.js +0 -146
- package/dist/common/exceptions.js.map +0 -1
- package/dist/common/index.d.ts +0 -1
- package/dist/common/index.js +0 -6
- package/dist/common/index.js.map +0 -1
- package/dist/common/logger.d.ts +0 -2
- package/dist/common/logger.js +0 -19
- package/dist/common/logger.js.map +0 -1
- package/dist/connector/ajax.d.ts +0 -142
- package/dist/connector/ajax.js +0 -259
- package/dist/connector/ajax.js.map +0 -1
- package/dist/connector/api.d.ts +0 -11
- package/dist/connector/api.js +0 -17
- package/dist/connector/api.js.map +0 -1
- package/dist/connector/index.d.ts +0 -0
- package/dist/connector/index.js +0 -2
- package/dist/connector/index.js.map +0 -1
- package/dist/index.js.map +0 -1
- package/dist/module/auth.d.ts +0 -8
- package/dist/module/auth.js +0 -77
- package/dist/module/auth.js.map +0 -1
- package/dist/module/client.d.ts +0 -39
- package/dist/module/client.js +0 -97
- package/dist/module/client.js.map +0 -1
- package/dist/module/index.d.ts +0 -0
- package/dist/module/index.js +0 -2
- package/dist/module/index.js.map +0 -1
- package/dist/module/page.d.ts +0 -102
- package/dist/module/page.js +0 -395
- package/dist/module/page.js.map +0 -1
- package/dist/module/pageRevision.d.ts +0 -29
- package/dist/module/pageRevision.js +0 -114
- package/dist/module/pageRevision.js.map +0 -1
- package/dist/module/pageSource.d.ts +0 -7
- package/dist/module/pageSource.js +0 -11
- package/dist/module/pageSource.js.map +0 -1
- package/dist/module/pageVote.d.ts +0 -14
- package/dist/module/pageVote.js +0 -20
- package/dist/module/pageVote.js.map +0 -1
- package/dist/module/privateMessage.d.ts +0 -29
- package/dist/module/privateMessage.js +0 -128
- package/dist/module/privateMessage.js.map +0 -1
- package/dist/module/site.d.ts +0 -32
- package/dist/module/site.js +0 -115
- package/dist/module/site.js.map +0 -1
- package/dist/module/siteApplication.d.ts +0 -14
- package/dist/module/siteApplication.js +0 -96
- package/dist/module/siteApplication.js.map +0 -1
- package/dist/module/user.d.ts +0 -56
- package/dist/module/user.js +0 -115
- package/dist/module/user.js.map +0 -1
- package/dist/util/index.d.ts +0 -3
- package/dist/util/index.js +0 -10
- package/dist/util/index.js.map +0 -1
- package/dist/util/parser/index.d.ts +0 -1
- package/dist/util/parser/index.js +0 -18
- package/dist/util/parser/index.js.map +0 -1
- package/dist/util/parser/odate.d.ts +0 -12
- package/dist/util/parser/odate.js +0 -25
- package/dist/util/parser/odate.js.map +0 -1
- package/dist/util/parser/user.d.ts +0 -16
- package/dist/util/parser/user.js +0 -36
- package/dist/util/parser/user.js.map +0 -1
- package/dist/util/quickModule.d.ts +0 -96
- package/dist/util/quickModule.js +0 -137
- package/dist/util/quickModule.js.map +0 -1
- package/dist/util/requestUtil.d.ts +0 -26
- package/dist/util/requestUtil.js +0 -69
- package/dist/util/requestUtil.js.map +0 -1
- package/dist/util/stringUtil.d.ts +0 -21
- package/dist/util/stringUtil.js +0 -64
- package/dist/util/stringUtil.js.map +0 -1
- package/dist/util/table/charTable.d.ts +0 -3
- package/dist/util/table/charTable.js +0 -472
- package/dist/util/table/charTable.js.map +0 -1
- package/dist/util/table/index.d.ts +0 -1
- package/dist/util/table/index.js +0 -18
- package/dist/util/table/index.js.map +0 -1
package/dist/index.js
CHANGED
|
@@ -1,6 +1,2651 @@
|
|
|
1
|
-
|
|
2
|
-
|
|
3
|
-
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
|
|
1
|
+
import {
|
|
2
|
+
AnonymousUser,
|
|
3
|
+
DeletedUser,
|
|
4
|
+
ForumCategory,
|
|
5
|
+
ForumCategoryCollection,
|
|
6
|
+
ForumPost,
|
|
7
|
+
ForumPostCollection,
|
|
8
|
+
ForumThread,
|
|
9
|
+
ForumThreadCollection,
|
|
10
|
+
GuestUser,
|
|
11
|
+
Logger,
|
|
12
|
+
RequireLogin,
|
|
13
|
+
WikidotUser,
|
|
14
|
+
consoleHandler,
|
|
15
|
+
getLogger,
|
|
16
|
+
logger,
|
|
17
|
+
nullHandler,
|
|
18
|
+
parseOdate,
|
|
19
|
+
parseUser,
|
|
20
|
+
setupConsoleHandler
|
|
21
|
+
} from "./shared/index-ytknx2hn.js";
|
|
22
|
+
import {
|
|
23
|
+
PageRevision,
|
|
24
|
+
PageRevisionCollection
|
|
25
|
+
} from "./shared/index-f2eh3ykk.js";
|
|
26
|
+
import {
|
|
27
|
+
User,
|
|
28
|
+
UserCollection
|
|
29
|
+
} from "./shared/index-kka6e8cb.js";
|
|
30
|
+
import {
|
|
31
|
+
AMCError,
|
|
32
|
+
AMCHttpError,
|
|
33
|
+
ForbiddenError,
|
|
34
|
+
LoginRequiredError,
|
|
35
|
+
NoElementError,
|
|
36
|
+
NotFoundException,
|
|
37
|
+
ResponseDataError,
|
|
38
|
+
SessionCreateError,
|
|
39
|
+
SessionError,
|
|
40
|
+
TargetError,
|
|
41
|
+
TargetExistsError,
|
|
42
|
+
UnexpectedError,
|
|
43
|
+
WikidotError,
|
|
44
|
+
WikidotStatusError,
|
|
45
|
+
__legacyDecorateClassTS,
|
|
46
|
+
__require,
|
|
47
|
+
__toESM,
|
|
48
|
+
combineResults,
|
|
49
|
+
fromPromise,
|
|
50
|
+
wdErr,
|
|
51
|
+
wdErrAsync,
|
|
52
|
+
wdOk,
|
|
53
|
+
wdOkAsync
|
|
54
|
+
} from "./shared/index-7dqqxq7x.js";
|
|
55
|
+
// src/connector/amc-client.ts
|
|
56
|
+
import ky from "ky";
|
|
57
|
+
import pLimit from "p-limit";
|
|
58
|
+
|
|
59
|
+
// src/connector/amc-config.ts
|
|
60
|
+
var DEFAULT_AMC_CONFIG = {
|
|
61
|
+
timeout: 20000,
|
|
62
|
+
retryLimit: 3,
|
|
63
|
+
retryInterval: 1000,
|
|
64
|
+
maxBackoff: 60000,
|
|
65
|
+
backoffFactor: 2,
|
|
66
|
+
semaphoreLimit: 10
|
|
67
|
+
};
|
|
68
|
+
var WIKIDOT_TOKEN7 = "123456";
|
|
69
|
+
var DEFAULT_HTTP_STATUS_CODE = 999;
|
|
70
|
+
|
|
71
|
+
// src/connector/amc-header.ts
|
|
72
|
+
class AMCHeader {
|
|
73
|
+
cookies;
|
|
74
|
+
contentType;
|
|
75
|
+
userAgent;
|
|
76
|
+
referer;
|
|
77
|
+
constructor(options) {
|
|
78
|
+
this.contentType = options?.contentType ?? "application/x-www-form-urlencoded; charset=UTF-8";
|
|
79
|
+
this.userAgent = options?.userAgent ?? "WikidotTS";
|
|
80
|
+
this.referer = options?.referer ?? "https://www.wikidot.com/";
|
|
81
|
+
this.cookies = new Map([["wikidot_token7", "123456"]]);
|
|
82
|
+
}
|
|
83
|
+
setCookie(name, value) {
|
|
84
|
+
this.cookies.set(name, value);
|
|
85
|
+
}
|
|
86
|
+
deleteCookie(name) {
|
|
87
|
+
this.cookies.delete(name);
|
|
88
|
+
}
|
|
89
|
+
getCookie(name) {
|
|
90
|
+
return this.cookies.get(name);
|
|
91
|
+
}
|
|
92
|
+
getHeaders() {
|
|
93
|
+
const cookieString = Array.from(this.cookies.entries()).map(([name, value]) => `${name}=${value}`).join("; ");
|
|
94
|
+
return {
|
|
95
|
+
"Content-Type": this.contentType,
|
|
96
|
+
"User-Agent": this.userAgent,
|
|
97
|
+
Referer: this.referer,
|
|
98
|
+
Cookie: cookieString
|
|
99
|
+
};
|
|
100
|
+
}
|
|
101
|
+
}
|
|
102
|
+
|
|
103
|
+
// src/connector/amc-types.ts
|
|
104
|
+
import { z } from "zod";
|
|
105
|
+
var baseSchema = z.object({
|
|
106
|
+
status: z.string(),
|
|
107
|
+
body: z.string().optional(),
|
|
108
|
+
message: z.string().optional()
|
|
109
|
+
});
|
|
110
|
+
var amcResponseSchema = baseSchema.passthrough();
|
|
111
|
+
function isSuccessResponse(response) {
|
|
112
|
+
return response.status === "ok";
|
|
113
|
+
}
|
|
114
|
+
|
|
115
|
+
// src/connector/amc-client.ts
|
|
116
|
+
function maskSensitiveData(body) {
|
|
117
|
+
const masked = { ...body };
|
|
118
|
+
const sensitiveKeys = ["password", "login", "WIKIDOT_SESSION_ID", "wikidot_token7"];
|
|
119
|
+
for (const key of sensitiveKeys) {
|
|
120
|
+
if (key in masked) {
|
|
121
|
+
masked[key] = "***MASKED***";
|
|
122
|
+
}
|
|
123
|
+
}
|
|
124
|
+
return masked;
|
|
125
|
+
}
|
|
126
|
+
function calculateBackoff(retryCount, baseInterval, backoffFactor, maxBackoff) {
|
|
127
|
+
const backoff = baseInterval * backoffFactor ** (retryCount - 1);
|
|
128
|
+
const jitter = Math.random() * backoff * 0.1;
|
|
129
|
+
return Math.min(backoff + jitter, maxBackoff);
|
|
130
|
+
}
|
|
131
|
+
function sleep(ms) {
|
|
132
|
+
return new Promise((resolve) => setTimeout(resolve, ms));
|
|
133
|
+
}
|
|
134
|
+
|
|
135
|
+
class AMCClient {
|
|
136
|
+
ky;
|
|
137
|
+
limit;
|
|
138
|
+
header;
|
|
139
|
+
config;
|
|
140
|
+
domain;
|
|
141
|
+
sslCache = new Map;
|
|
142
|
+
constructor(config = {}, domain = "wikidot.com") {
|
|
143
|
+
this.config = { ...DEFAULT_AMC_CONFIG, ...config };
|
|
144
|
+
this.domain = domain;
|
|
145
|
+
this.header = new AMCHeader;
|
|
146
|
+
this.limit = pLimit(this.config.semaphoreLimit);
|
|
147
|
+
this.ky = ky.create({
|
|
148
|
+
timeout: this.config.timeout,
|
|
149
|
+
retry: 0
|
|
150
|
+
});
|
|
151
|
+
this.sslCache.set("www", true);
|
|
152
|
+
}
|
|
153
|
+
checkSiteSSL(siteName) {
|
|
154
|
+
const cached = this.sslCache.get(siteName);
|
|
155
|
+
if (cached !== undefined) {
|
|
156
|
+
return wdOkAsync(cached);
|
|
157
|
+
}
|
|
158
|
+
if (siteName === "www") {
|
|
159
|
+
return wdOkAsync(true);
|
|
160
|
+
}
|
|
161
|
+
return fromPromise((async () => {
|
|
162
|
+
const response = await fetch(`http://${siteName}.${this.domain}`, {
|
|
163
|
+
method: "GET",
|
|
164
|
+
redirect: "manual"
|
|
165
|
+
});
|
|
166
|
+
if (response.status === 404) {
|
|
167
|
+
throw new NotFoundException(`Site is not found: ${siteName}.${this.domain}`);
|
|
168
|
+
}
|
|
169
|
+
const isSSL = response.status === 301 && response.headers.get("Location")?.startsWith("https") === true;
|
|
170
|
+
this.sslCache.set(siteName, isSSL);
|
|
171
|
+
return isSSL;
|
|
172
|
+
})(), (error) => {
|
|
173
|
+
if (error instanceof WikidotError) {
|
|
174
|
+
return error;
|
|
175
|
+
}
|
|
176
|
+
return new UnexpectedError(`Failed to check SSL for ${siteName}: ${String(error)}`);
|
|
177
|
+
});
|
|
178
|
+
}
|
|
179
|
+
request(bodies, siteName = "www", sslSupported) {
|
|
180
|
+
return this.requestWithOptions(bodies, {
|
|
181
|
+
siteName,
|
|
182
|
+
sslSupported,
|
|
183
|
+
returnExceptions: false
|
|
184
|
+
});
|
|
185
|
+
}
|
|
186
|
+
requestWithOptions(bodies, options = {}) {
|
|
187
|
+
const { siteName = "www", sslSupported, returnExceptions = false } = options;
|
|
188
|
+
return fromPromise((async () => {
|
|
189
|
+
let ssl = sslSupported;
|
|
190
|
+
if (ssl === undefined) {
|
|
191
|
+
const sslResult = await this.checkSiteSSL(siteName);
|
|
192
|
+
if (sslResult.isErr()) {
|
|
193
|
+
throw sslResult.error;
|
|
194
|
+
}
|
|
195
|
+
ssl = sslResult.value;
|
|
196
|
+
}
|
|
197
|
+
const protocol = ssl ? "https" : "http";
|
|
198
|
+
const url = `${protocol}://${siteName}.${this.domain}/ajax-module-connector.php`;
|
|
199
|
+
const results = await Promise.all(bodies.map((body) => this.limit(() => this.singleRequest(body, url))));
|
|
200
|
+
if (returnExceptions) {
|
|
201
|
+
return results.map((r) => {
|
|
202
|
+
if (r.isOk()) {
|
|
203
|
+
return r.value;
|
|
204
|
+
}
|
|
205
|
+
return r.error;
|
|
206
|
+
});
|
|
207
|
+
}
|
|
208
|
+
const firstError = results.find((r) => r.isErr());
|
|
209
|
+
if (firstError?.isErr()) {
|
|
210
|
+
throw firstError.error;
|
|
211
|
+
}
|
|
212
|
+
return results.map((r) => {
|
|
213
|
+
if (r.isOk()) {
|
|
214
|
+
return r.value;
|
|
215
|
+
}
|
|
216
|
+
throw new UnexpectedError("Unexpected error in result processing");
|
|
217
|
+
});
|
|
218
|
+
})(), (error) => {
|
|
219
|
+
if (error instanceof WikidotError) {
|
|
220
|
+
return error;
|
|
221
|
+
}
|
|
222
|
+
return new UnexpectedError(`AMC request failed: ${String(error)}`);
|
|
223
|
+
});
|
|
224
|
+
}
|
|
225
|
+
async singleRequest(body, url) {
|
|
226
|
+
let retryCount = 0;
|
|
227
|
+
while (true) {
|
|
228
|
+
try {
|
|
229
|
+
const requestBody = { ...body, wikidot_token7: WIKIDOT_TOKEN7 };
|
|
230
|
+
const formData = new URLSearchParams;
|
|
231
|
+
for (const [key, value] of Object.entries(requestBody)) {
|
|
232
|
+
if (value !== undefined) {
|
|
233
|
+
formData.append(key, String(value));
|
|
234
|
+
}
|
|
235
|
+
}
|
|
236
|
+
const response = await this.ky.post(url, {
|
|
237
|
+
headers: this.header.getHeaders(),
|
|
238
|
+
body: formData.toString()
|
|
239
|
+
});
|
|
240
|
+
let responseData;
|
|
241
|
+
try {
|
|
242
|
+
responseData = await response.json();
|
|
243
|
+
} catch {
|
|
244
|
+
return wdErrAsync(new ResponseDataError(`AMC responded with non-JSON data: ${await response.text()}`));
|
|
245
|
+
}
|
|
246
|
+
const parseResult = amcResponseSchema.safeParse(responseData);
|
|
247
|
+
if (!parseResult.success) {
|
|
248
|
+
return wdErrAsync(new ResponseDataError(`Invalid AMC response format: ${parseResult.error.message}`));
|
|
249
|
+
}
|
|
250
|
+
const amcResponse = parseResult.data;
|
|
251
|
+
if (amcResponse.status === "try_again") {
|
|
252
|
+
retryCount++;
|
|
253
|
+
if (retryCount >= this.config.retryLimit) {
|
|
254
|
+
return wdErrAsync(new WikidotStatusError("AMC responded with try_again", "try_again"));
|
|
255
|
+
}
|
|
256
|
+
const backoff = calculateBackoff(retryCount, this.config.retryInterval, this.config.backoffFactor, this.config.maxBackoff);
|
|
257
|
+
await sleep(backoff);
|
|
258
|
+
continue;
|
|
259
|
+
}
|
|
260
|
+
if (amcResponse.status === "no_permission") {
|
|
261
|
+
const targetStr = body.moduleName ? `moduleName: ${body.moduleName}` : body.action ? `action: ${body.action}/${body.event ?? ""}` : "unknown";
|
|
262
|
+
return wdErrAsync(new ForbiddenError(`Your account has no permission to perform this action: ${targetStr}`));
|
|
263
|
+
}
|
|
264
|
+
if (amcResponse.status !== "ok") {
|
|
265
|
+
return wdErrAsync(new WikidotStatusError(`AMC responded with error status: "${amcResponse.status}"`, amcResponse.status));
|
|
266
|
+
}
|
|
267
|
+
return wdOkAsync(amcResponse);
|
|
268
|
+
} catch (error) {
|
|
269
|
+
retryCount++;
|
|
270
|
+
if (retryCount >= this.config.retryLimit) {
|
|
271
|
+
const statusCode = error instanceof Error && "response" in error ? error.response?.status ?? DEFAULT_HTTP_STATUS_CODE : DEFAULT_HTTP_STATUS_CODE;
|
|
272
|
+
return wdErrAsync(new AMCHttpError(`AMC HTTP request failed: ${String(error)}`, statusCode));
|
|
273
|
+
}
|
|
274
|
+
const backoff = calculateBackoff(retryCount, this.config.retryInterval, this.config.backoffFactor, this.config.maxBackoff);
|
|
275
|
+
await sleep(backoff);
|
|
276
|
+
}
|
|
277
|
+
}
|
|
278
|
+
}
|
|
279
|
+
}
|
|
280
|
+
// src/connector/auth.ts
|
|
281
|
+
var LOGIN_URL = "https://www.wikidot.com/default--flow/login__LoginPopupScreen";
|
|
282
|
+
function login(client, username, password) {
|
|
283
|
+
return fromPromise((async () => {
|
|
284
|
+
const formData = new URLSearchParams({
|
|
285
|
+
login: username,
|
|
286
|
+
password,
|
|
287
|
+
action: "Login2Action",
|
|
288
|
+
event: "login"
|
|
289
|
+
});
|
|
290
|
+
const response = await fetch(LOGIN_URL, {
|
|
291
|
+
method: "POST",
|
|
292
|
+
headers: client.amcClient.header.getHeaders(),
|
|
293
|
+
body: formData.toString()
|
|
294
|
+
});
|
|
295
|
+
if (!response.ok) {
|
|
296
|
+
throw new SessionCreateError(`Login attempt failed due to HTTP status code: ${response.status}`);
|
|
297
|
+
}
|
|
298
|
+
const body = await response.text();
|
|
299
|
+
if (body.includes("The login and password do not match")) {
|
|
300
|
+
throw new SessionCreateError("Login attempt failed due to invalid username or password");
|
|
301
|
+
}
|
|
302
|
+
const cookies = response.headers.get("Set-Cookie");
|
|
303
|
+
if (!cookies) {
|
|
304
|
+
throw new SessionCreateError("Login attempt failed due to missing cookies");
|
|
305
|
+
}
|
|
306
|
+
const sessionIdMatch = cookies.match(/WIKIDOT_SESSION_ID=([^;]+)/);
|
|
307
|
+
if (!sessionIdMatch?.[1]) {
|
|
308
|
+
throw new SessionCreateError("Login attempt failed due to missing WIKIDOT_SESSION_ID cookie");
|
|
309
|
+
}
|
|
310
|
+
client.amcClient.header.setCookie("WIKIDOT_SESSION_ID", sessionIdMatch[1]);
|
|
311
|
+
})(), (error) => {
|
|
312
|
+
if (error instanceof SessionCreateError) {
|
|
313
|
+
return error;
|
|
314
|
+
}
|
|
315
|
+
return new SessionCreateError(`Login failed: ${String(error)}`);
|
|
316
|
+
});
|
|
317
|
+
}
|
|
318
|
+
function logout(client) {
|
|
319
|
+
return client.amcClient.request([
|
|
320
|
+
{
|
|
321
|
+
moduleName: "Empty",
|
|
322
|
+
action: "Login2Action",
|
|
323
|
+
event: "logout"
|
|
324
|
+
}
|
|
325
|
+
]).map(() => {
|
|
326
|
+
client.amcClient.header.deleteCookie("WIKIDOT_SESSION_ID");
|
|
327
|
+
return;
|
|
328
|
+
}).orElse(() => {
|
|
329
|
+
client.amcClient.header.deleteCookie("WIKIDOT_SESSION_ID");
|
|
330
|
+
return wdOkAsync(undefined);
|
|
331
|
+
});
|
|
332
|
+
}
|
|
333
|
+
// src/module/private-message/private-message.ts
|
|
334
|
+
import * as cheerio from "cheerio";
|
|
335
|
+
class PrivateMessage {
|
|
336
|
+
client;
|
|
337
|
+
id;
|
|
338
|
+
sender;
|
|
339
|
+
recipient;
|
|
340
|
+
subject;
|
|
341
|
+
body;
|
|
342
|
+
createdAt;
|
|
343
|
+
constructor(data) {
|
|
344
|
+
this.client = data.client;
|
|
345
|
+
this.id = data.id;
|
|
346
|
+
this.sender = data.sender;
|
|
347
|
+
this.recipient = data.recipient;
|
|
348
|
+
this.subject = data.subject;
|
|
349
|
+
this.body = data.body;
|
|
350
|
+
this.createdAt = data.createdAt;
|
|
351
|
+
}
|
|
352
|
+
static fromId(client, messageId) {
|
|
353
|
+
return fromPromise((async () => {
|
|
354
|
+
const result = await PrivateMessageCollection.fromIds(client, [messageId]);
|
|
355
|
+
if (result.isErr()) {
|
|
356
|
+
throw result.error;
|
|
357
|
+
}
|
|
358
|
+
const message = result.value[0];
|
|
359
|
+
if (!message) {
|
|
360
|
+
throw new NoElementError(`Message not found: ${messageId}`);
|
|
361
|
+
}
|
|
362
|
+
return message;
|
|
363
|
+
})(), (error) => {
|
|
364
|
+
if (error instanceof ForbiddenError || error instanceof NoElementError) {
|
|
365
|
+
return error;
|
|
366
|
+
}
|
|
367
|
+
return new UnexpectedError(`Failed to get message: ${String(error)}`);
|
|
368
|
+
});
|
|
369
|
+
}
|
|
370
|
+
static send(client, recipient, subject, body) {
|
|
371
|
+
const loginResult = client.requireLogin();
|
|
372
|
+
if (loginResult.isErr()) {
|
|
373
|
+
return fromPromise(Promise.reject(loginResult.error), () => new LoginRequiredError("Login required to send message"));
|
|
374
|
+
}
|
|
375
|
+
return fromPromise((async () => {
|
|
376
|
+
const result = await client.amcClient.request([
|
|
377
|
+
{
|
|
378
|
+
source: body,
|
|
379
|
+
subject,
|
|
380
|
+
to_user_id: recipient.id,
|
|
381
|
+
action: "DashboardMessageAction",
|
|
382
|
+
event: "send",
|
|
383
|
+
moduleName: "Empty"
|
|
384
|
+
}
|
|
385
|
+
]);
|
|
386
|
+
if (result.isErr()) {
|
|
387
|
+
throw result.error;
|
|
388
|
+
}
|
|
389
|
+
})(), (error) => new UnexpectedError(`Failed to send message: ${String(error)}`));
|
|
390
|
+
}
|
|
391
|
+
toString() {
|
|
392
|
+
return `PrivateMessage(id=${this.id}, sender=${this.sender}, recipient=${this.recipient}, subject=${this.subject})`;
|
|
393
|
+
}
|
|
394
|
+
}
|
|
395
|
+
|
|
396
|
+
class PrivateMessageCollection extends Array {
|
|
397
|
+
client;
|
|
398
|
+
constructor(client, messages) {
|
|
399
|
+
super();
|
|
400
|
+
this.client = client;
|
|
401
|
+
if (messages) {
|
|
402
|
+
this.push(...messages);
|
|
403
|
+
}
|
|
404
|
+
}
|
|
405
|
+
findById(id) {
|
|
406
|
+
return this.find((message) => message.id === id);
|
|
407
|
+
}
|
|
408
|
+
static fromIds(client, messageIds) {
|
|
409
|
+
const loginResult = client.requireLogin();
|
|
410
|
+
if (loginResult.isErr()) {
|
|
411
|
+
return fromPromise(Promise.reject(loginResult.error), () => new LoginRequiredError("Login required to get messages"));
|
|
412
|
+
}
|
|
413
|
+
return fromPromise((async () => {
|
|
414
|
+
const bodies = messageIds.map((messageId) => ({
|
|
415
|
+
item: messageId,
|
|
416
|
+
moduleName: "dashboard/messages/DMViewMessageModule"
|
|
417
|
+
}));
|
|
418
|
+
const result = await client.amcClient.request(bodies);
|
|
419
|
+
if (result.isErr()) {
|
|
420
|
+
throw result.error;
|
|
421
|
+
}
|
|
422
|
+
const messages = [];
|
|
423
|
+
for (let i = 0;i < messageIds.length; i++) {
|
|
424
|
+
const response = result.value[i];
|
|
425
|
+
const messageId = messageIds[i];
|
|
426
|
+
if (!response || messageId === undefined)
|
|
427
|
+
continue;
|
|
428
|
+
const html = String(response.body ?? "");
|
|
429
|
+
const $ = cheerio.load(html);
|
|
430
|
+
const printuserElems = $("div.pmessage div.header span.printuser");
|
|
431
|
+
if (printuserElems.length < 2) {
|
|
432
|
+
throw new ForbiddenError(`Failed to get message: ${messageId}`);
|
|
433
|
+
}
|
|
434
|
+
const senderElem = $(printuserElems[0]);
|
|
435
|
+
const recipientElem = $(printuserElems[1]);
|
|
436
|
+
const sender = parseUser(client, senderElem);
|
|
437
|
+
const recipient = parseUser(client, recipientElem);
|
|
438
|
+
const subjectElem = $("div.pmessage div.header span.subject");
|
|
439
|
+
const subject = subjectElem.text().trim();
|
|
440
|
+
const bodyElem = $("div.pmessage div.body");
|
|
441
|
+
const body = bodyElem.text().trim();
|
|
442
|
+
const odateElem = $("div.header span.odate");
|
|
443
|
+
const createdAt = odateElem.length > 0 ? parseOdate(odateElem) ?? new Date(0) : new Date(0);
|
|
444
|
+
messages.push(new PrivateMessage({
|
|
445
|
+
client,
|
|
446
|
+
id: messageId,
|
|
447
|
+
sender,
|
|
448
|
+
recipient,
|
|
449
|
+
subject,
|
|
450
|
+
body,
|
|
451
|
+
createdAt
|
|
452
|
+
}));
|
|
453
|
+
}
|
|
454
|
+
return new PrivateMessageCollection(client, messages);
|
|
455
|
+
})(), (error) => {
|
|
456
|
+
if (error instanceof ForbiddenError || error instanceof LoginRequiredError) {
|
|
457
|
+
return error;
|
|
458
|
+
}
|
|
459
|
+
return new UnexpectedError(`Failed to get messages: ${String(error)}`);
|
|
460
|
+
});
|
|
461
|
+
}
|
|
462
|
+
static acquireFromModule(client, moduleName) {
|
|
463
|
+
const loginResult = client.requireLogin();
|
|
464
|
+
if (loginResult.isErr()) {
|
|
465
|
+
return fromPromise(Promise.reject(loginResult.error), () => new LoginRequiredError("Login required to get messages"));
|
|
466
|
+
}
|
|
467
|
+
return fromPromise((async () => {
|
|
468
|
+
const firstResult = await client.amcClient.request([{ moduleName }]);
|
|
469
|
+
if (firstResult.isErr()) {
|
|
470
|
+
throw firstResult.error;
|
|
471
|
+
}
|
|
472
|
+
const firstResponse = firstResult.value[0];
|
|
473
|
+
if (!firstResponse) {
|
|
474
|
+
throw new NoElementError("Empty response");
|
|
475
|
+
}
|
|
476
|
+
const firstHtml = String(firstResponse.body ?? "");
|
|
477
|
+
const $first = cheerio.load(firstHtml);
|
|
478
|
+
const pagerTargets = $first("div.pager span.target");
|
|
479
|
+
let maxPage = 1;
|
|
480
|
+
if (pagerTargets.length > 2) {
|
|
481
|
+
const lastPageText = $first(pagerTargets[pagerTargets.length - 2]).text().trim();
|
|
482
|
+
maxPage = Number.parseInt(lastPageText, 10) || 1;
|
|
483
|
+
}
|
|
484
|
+
const messageIds = [];
|
|
485
|
+
if (maxPage > 1) {
|
|
486
|
+
const bodies = [];
|
|
487
|
+
for (let page = 1;page <= maxPage; page++) {
|
|
488
|
+
bodies.push({ page, moduleName });
|
|
489
|
+
}
|
|
490
|
+
const additionalResults = await client.amcClient.request(bodies);
|
|
491
|
+
if (additionalResults.isErr()) {
|
|
492
|
+
throw additionalResults.error;
|
|
493
|
+
}
|
|
494
|
+
for (const response of additionalResults.value) {
|
|
495
|
+
const html = String(response?.body ?? "");
|
|
496
|
+
const $ = cheerio.load(html);
|
|
497
|
+
$("tr.message").each((_i, elem) => {
|
|
498
|
+
const dataHref = $(elem).attr("data-href") ?? "";
|
|
499
|
+
const idMatch = dataHref.match(/\/(\d+)$/);
|
|
500
|
+
if (idMatch?.[1]) {
|
|
501
|
+
messageIds.push(Number.parseInt(idMatch[1], 10));
|
|
502
|
+
}
|
|
503
|
+
});
|
|
504
|
+
}
|
|
505
|
+
} else {
|
|
506
|
+
$first("tr.message").each((_i, elem) => {
|
|
507
|
+
const dataHref = $first(elem).attr("data-href") ?? "";
|
|
508
|
+
const idMatch = dataHref.match(/\/(\d+)$/);
|
|
509
|
+
if (idMatch?.[1]) {
|
|
510
|
+
messageIds.push(Number.parseInt(idMatch[1], 10));
|
|
511
|
+
}
|
|
512
|
+
});
|
|
513
|
+
}
|
|
514
|
+
const messagesResult = await PrivateMessageCollection.fromIds(client, messageIds);
|
|
515
|
+
if (messagesResult.isErr()) {
|
|
516
|
+
throw messagesResult.error;
|
|
517
|
+
}
|
|
518
|
+
return messagesResult.value;
|
|
519
|
+
})(), (error) => {
|
|
520
|
+
if (error instanceof ForbiddenError || error instanceof LoginRequiredError || error instanceof NoElementError) {
|
|
521
|
+
return error;
|
|
522
|
+
}
|
|
523
|
+
return new UnexpectedError(`Failed to acquire messages: ${String(error)}`);
|
|
524
|
+
});
|
|
525
|
+
}
|
|
526
|
+
}
|
|
527
|
+
|
|
528
|
+
class PrivateMessageInbox extends PrivateMessageCollection {
|
|
529
|
+
static acquire(client) {
|
|
530
|
+
return fromPromise((async () => {
|
|
531
|
+
const result = await PrivateMessageCollection.acquireFromModule(client, "dashboard/messages/DMInboxModule");
|
|
532
|
+
if (result.isErr()) {
|
|
533
|
+
throw result.error;
|
|
534
|
+
}
|
|
535
|
+
const inbox = new PrivateMessageInbox(client);
|
|
536
|
+
inbox.push(...result.value);
|
|
537
|
+
return inbox;
|
|
538
|
+
})(), (error) => {
|
|
539
|
+
if (error instanceof ForbiddenError || error instanceof LoginRequiredError) {
|
|
540
|
+
return error;
|
|
541
|
+
}
|
|
542
|
+
return new UnexpectedError(`Failed to acquire inbox: ${String(error)}`);
|
|
543
|
+
});
|
|
544
|
+
}
|
|
545
|
+
}
|
|
546
|
+
|
|
547
|
+
class PrivateMessageSentBox extends PrivateMessageCollection {
|
|
548
|
+
static acquire(client) {
|
|
549
|
+
return fromPromise((async () => {
|
|
550
|
+
const result = await PrivateMessageCollection.acquireFromModule(client, "dashboard/messages/DMSentModule");
|
|
551
|
+
if (result.isErr()) {
|
|
552
|
+
throw result.error;
|
|
553
|
+
}
|
|
554
|
+
const sentBox = new PrivateMessageSentBox(client);
|
|
555
|
+
sentBox.push(...result.value);
|
|
556
|
+
return sentBox;
|
|
557
|
+
})(), (error) => {
|
|
558
|
+
if (error instanceof ForbiddenError || error instanceof LoginRequiredError) {
|
|
559
|
+
return error;
|
|
560
|
+
}
|
|
561
|
+
return new UnexpectedError(`Failed to acquire sent box: ${String(error)}`);
|
|
562
|
+
});
|
|
563
|
+
}
|
|
564
|
+
}
|
|
565
|
+
// src/module/client/accessors/pm-accessor.ts
|
|
566
|
+
class PrivateMessageAccessor {
|
|
567
|
+
client;
|
|
568
|
+
constructor(client) {
|
|
569
|
+
this.client = client;
|
|
570
|
+
}
|
|
571
|
+
get(id) {
|
|
572
|
+
return PrivateMessage.fromId(this.client, id);
|
|
573
|
+
}
|
|
574
|
+
getMessages(ids) {
|
|
575
|
+
return PrivateMessageCollection.fromIds(this.client, ids);
|
|
576
|
+
}
|
|
577
|
+
inbox() {
|
|
578
|
+
return PrivateMessageInbox.acquire(this.client);
|
|
579
|
+
}
|
|
580
|
+
sentBox() {
|
|
581
|
+
return PrivateMessageSentBox.acquire(this.client);
|
|
582
|
+
}
|
|
583
|
+
send(recipient, subject, body) {
|
|
584
|
+
return PrivateMessage.send(this.client, recipient, subject, body);
|
|
585
|
+
}
|
|
586
|
+
}
|
|
587
|
+
// src/module/site/accessors/forum-accessor.ts
|
|
588
|
+
class ForumAccessor {
|
|
589
|
+
site;
|
|
590
|
+
constructor(site) {
|
|
591
|
+
this.site = site;
|
|
592
|
+
}
|
|
593
|
+
getCategories() {
|
|
594
|
+
return ForumCategoryCollection.acquireAll(this.site);
|
|
595
|
+
}
|
|
596
|
+
getThread(threadId) {
|
|
597
|
+
return ForumThread.getFromId(this.site, threadId);
|
|
598
|
+
}
|
|
599
|
+
getThreads(threadIds) {
|
|
600
|
+
return ForumThreadCollection.acquireFromThreadIds(this.site, threadIds);
|
|
601
|
+
}
|
|
602
|
+
}
|
|
603
|
+
// src/util/quick-module.ts
|
|
604
|
+
import { z as z2 } from "zod";
|
|
605
|
+
var quickModuleUserResponseSchema = z2.object({
|
|
606
|
+
users: z2.union([
|
|
607
|
+
z2.array(z2.object({
|
|
608
|
+
user_id: z2.union([z2.string(), z2.number()]),
|
|
609
|
+
name: z2.string()
|
|
610
|
+
})),
|
|
611
|
+
z2.literal(false)
|
|
612
|
+
])
|
|
613
|
+
});
|
|
614
|
+
var quickModulePageResponseSchema = z2.object({
|
|
615
|
+
pages: z2.union([
|
|
616
|
+
z2.array(z2.object({
|
|
617
|
+
title: z2.string(),
|
|
618
|
+
unix_name: z2.string()
|
|
619
|
+
})),
|
|
620
|
+
z2.literal(false)
|
|
621
|
+
])
|
|
622
|
+
});
|
|
623
|
+
async function requestQuickModule(moduleName, siteId, query) {
|
|
624
|
+
const url = `https://www.wikidot.com/quickmodule.php?module=${moduleName}&s=${siteId}&q=${encodeURIComponent(query)}`;
|
|
625
|
+
const response = await fetch(url, {
|
|
626
|
+
method: "GET",
|
|
627
|
+
headers: {
|
|
628
|
+
Accept: "application/json"
|
|
629
|
+
}
|
|
630
|
+
});
|
|
631
|
+
if (response.status === 500) {
|
|
632
|
+
throw new NotFoundException(`Site not found: siteId=${siteId}`);
|
|
633
|
+
}
|
|
634
|
+
if (!response.ok) {
|
|
635
|
+
throw new UnexpectedError(`QuickModule request failed: ${response.status}`);
|
|
636
|
+
}
|
|
637
|
+
return response.json();
|
|
638
|
+
}
|
|
639
|
+
function memberLookup(siteId, query) {
|
|
640
|
+
return fromPromise((async () => {
|
|
641
|
+
const data = await requestQuickModule("MemberLookupQModule", siteId, query);
|
|
642
|
+
const parsed = quickModuleUserResponseSchema.parse(data);
|
|
643
|
+
if (parsed.users === false) {
|
|
644
|
+
return [];
|
|
645
|
+
}
|
|
646
|
+
return parsed.users.map((user) => ({
|
|
647
|
+
id: typeof user.user_id === "string" ? Number.parseInt(user.user_id, 10) : user.user_id,
|
|
648
|
+
name: user.name
|
|
649
|
+
}));
|
|
650
|
+
})(), (error) => {
|
|
651
|
+
if (error instanceof NotFoundException)
|
|
652
|
+
return error;
|
|
653
|
+
return new UnexpectedError(`Member lookup failed: ${String(error)}`);
|
|
654
|
+
});
|
|
655
|
+
}
|
|
656
|
+
function userLookup(siteId, query) {
|
|
657
|
+
return fromPromise((async () => {
|
|
658
|
+
const data = await requestQuickModule("UserLookupQModule", siteId, query);
|
|
659
|
+
const parsed = quickModuleUserResponseSchema.parse(data);
|
|
660
|
+
if (parsed.users === false) {
|
|
661
|
+
return [];
|
|
662
|
+
}
|
|
663
|
+
return parsed.users.map((user) => ({
|
|
664
|
+
id: typeof user.user_id === "string" ? Number.parseInt(user.user_id, 10) : user.user_id,
|
|
665
|
+
name: user.name
|
|
666
|
+
}));
|
|
667
|
+
})(), (error) => {
|
|
668
|
+
if (error instanceof NotFoundException)
|
|
669
|
+
return error;
|
|
670
|
+
return new UnexpectedError(`User lookup failed: ${String(error)}`);
|
|
671
|
+
});
|
|
672
|
+
}
|
|
673
|
+
function pageLookup(siteId, query) {
|
|
674
|
+
return fromPromise((async () => {
|
|
675
|
+
const data = await requestQuickModule("PageLookupQModule", siteId, query);
|
|
676
|
+
const parsed = quickModulePageResponseSchema.parse(data);
|
|
677
|
+
if (parsed.pages === false) {
|
|
678
|
+
return [];
|
|
679
|
+
}
|
|
680
|
+
return parsed.pages.map((page) => ({
|
|
681
|
+
title: page.title,
|
|
682
|
+
unixName: page.unix_name
|
|
683
|
+
}));
|
|
684
|
+
})(), (error) => {
|
|
685
|
+
if (error instanceof NotFoundException)
|
|
686
|
+
return error;
|
|
687
|
+
return new UnexpectedError(`Page lookup failed: ${String(error)}`);
|
|
688
|
+
});
|
|
689
|
+
}
|
|
690
|
+
var QuickModule = {
|
|
691
|
+
memberLookup,
|
|
692
|
+
userLookup,
|
|
693
|
+
pageLookup
|
|
694
|
+
};
|
|
695
|
+
|
|
696
|
+
// src/module/site/site-application.ts
|
|
697
|
+
import * as cheerio2 from "cheerio";
|
|
698
|
+
class SiteApplication {
|
|
699
|
+
site;
|
|
700
|
+
user;
|
|
701
|
+
text;
|
|
702
|
+
constructor(data) {
|
|
703
|
+
this.site = data.site;
|
|
704
|
+
this.user = data.user;
|
|
705
|
+
this.text = data.text;
|
|
706
|
+
}
|
|
707
|
+
static acquireAll(site) {
|
|
708
|
+
const loginResult = site.client.requireLogin();
|
|
709
|
+
if (loginResult.isErr()) {
|
|
710
|
+
return fromPromise(Promise.reject(loginResult.error), () => new LoginRequiredError("Login required to get applications"));
|
|
711
|
+
}
|
|
712
|
+
return fromPromise((async () => {
|
|
713
|
+
const result = await site.amcRequest([
|
|
714
|
+
{ moduleName: "managesite/ManageSiteMembersApplicationsModule" }
|
|
715
|
+
]);
|
|
716
|
+
if (result.isErr()) {
|
|
717
|
+
throw result.error;
|
|
718
|
+
}
|
|
719
|
+
const response = result.value[0];
|
|
720
|
+
if (!response) {
|
|
721
|
+
throw new UnexpectedError("Empty response");
|
|
722
|
+
}
|
|
723
|
+
const html = String(response.body ?? "");
|
|
724
|
+
if (html.includes("WIKIDOT.page.listeners.loginClick(event)")) {
|
|
725
|
+
throw new ForbiddenError("You are not allowed to access this page");
|
|
726
|
+
}
|
|
727
|
+
const $ = cheerio2.load(html);
|
|
728
|
+
const applications = [];
|
|
729
|
+
const userElements = $("h3 span.printuser").toArray();
|
|
730
|
+
const textWrapperElements = $("table").toArray();
|
|
731
|
+
if (userElements.length !== textWrapperElements.length) {
|
|
732
|
+
throw new UnexpectedError("Length of user_elements and text_wrapper_elements are different");
|
|
733
|
+
}
|
|
734
|
+
for (let i = 0;i < userElements.length; i++) {
|
|
735
|
+
const userElement = userElements[i];
|
|
736
|
+
const textWrapperElement = textWrapperElements[i];
|
|
737
|
+
if (!userElement || !textWrapperElement)
|
|
738
|
+
continue;
|
|
739
|
+
const user = parseUser(site.client, $(userElement));
|
|
740
|
+
const textElement = $(textWrapperElement).find("td").eq(1);
|
|
741
|
+
const text = textElement.text().trim();
|
|
742
|
+
applications.push(new SiteApplication({ site, user, text }));
|
|
743
|
+
}
|
|
744
|
+
return applications;
|
|
745
|
+
})(), (error) => {
|
|
746
|
+
if (error instanceof ForbiddenError || error instanceof LoginRequiredError) {
|
|
747
|
+
return error;
|
|
748
|
+
}
|
|
749
|
+
return new UnexpectedError(`Failed to get applications: ${String(error)}`);
|
|
750
|
+
});
|
|
751
|
+
}
|
|
752
|
+
process(action) {
|
|
753
|
+
return fromPromise((async () => {
|
|
754
|
+
const result = await this.site.amcRequest([
|
|
755
|
+
{
|
|
756
|
+
action: "ManageSiteMembershipAction",
|
|
757
|
+
event: "acceptApplication",
|
|
758
|
+
user_id: this.user.id,
|
|
759
|
+
text: `your application has been ${action}ed`,
|
|
760
|
+
type: action,
|
|
761
|
+
moduleName: "Empty"
|
|
762
|
+
}
|
|
763
|
+
]);
|
|
764
|
+
if (result.isErr()) {
|
|
765
|
+
const error = result.error;
|
|
766
|
+
if (error instanceof WikidotStatusError && error.statusCode === "no_application") {
|
|
767
|
+
throw new NotFoundException(`Application not found: ${this.user.name}`);
|
|
768
|
+
}
|
|
769
|
+
throw error;
|
|
770
|
+
}
|
|
771
|
+
})(), (error) => {
|
|
772
|
+
if (error instanceof NotFoundException || error instanceof LoginRequiredError) {
|
|
773
|
+
return error;
|
|
774
|
+
}
|
|
775
|
+
return new UnexpectedError(`Failed to process application: ${String(error)}`);
|
|
776
|
+
});
|
|
777
|
+
}
|
|
778
|
+
accept() {
|
|
779
|
+
return this.process("accept");
|
|
780
|
+
}
|
|
781
|
+
decline() {
|
|
782
|
+
return this.process("decline");
|
|
783
|
+
}
|
|
784
|
+
toString() {
|
|
785
|
+
return `SiteApplication(user=${this.user.name}, site=${this.site.unixName}, text=${this.text})`;
|
|
786
|
+
}
|
|
787
|
+
}
|
|
788
|
+
__legacyDecorateClassTS([
|
|
789
|
+
RequireLogin
|
|
790
|
+
], SiteApplication.prototype, "process", null);
|
|
791
|
+
|
|
792
|
+
// src/module/site/site-member.ts
|
|
793
|
+
import * as cheerio3 from "cheerio";
|
|
794
|
+
class SiteMember {
|
|
795
|
+
site;
|
|
796
|
+
user;
|
|
797
|
+
joinedAt;
|
|
798
|
+
constructor(data) {
|
|
799
|
+
this.site = data.site;
|
|
800
|
+
this.user = data.user;
|
|
801
|
+
this.joinedAt = data.joinedAt;
|
|
802
|
+
}
|
|
803
|
+
static parse(site, html) {
|
|
804
|
+
const $ = cheerio3.load(html);
|
|
805
|
+
const members = [];
|
|
806
|
+
$("table tr").each((_i, row) => {
|
|
807
|
+
const tds = $(row).find("td");
|
|
808
|
+
const userElem = $(tds[0]).find(".printuser");
|
|
809
|
+
if (userElem.length === 0) {
|
|
810
|
+
return;
|
|
811
|
+
}
|
|
812
|
+
const user = parseUser(site.client, userElem);
|
|
813
|
+
let joinedAt = null;
|
|
814
|
+
if (tds.length >= 2) {
|
|
815
|
+
const odateElem = $(tds[1]).find(".odate");
|
|
816
|
+
if (odateElem.length > 0) {
|
|
817
|
+
joinedAt = parseOdate(odateElem);
|
|
818
|
+
}
|
|
819
|
+
}
|
|
820
|
+
members.push(new SiteMember({ site, user, joinedAt }));
|
|
821
|
+
});
|
|
822
|
+
return members;
|
|
823
|
+
}
|
|
824
|
+
static getMembers(site, group = "") {
|
|
825
|
+
return fromPromise((async () => {
|
|
826
|
+
const members = [];
|
|
827
|
+
const firstResult = await site.amcRequest([
|
|
828
|
+
{
|
|
829
|
+
moduleName: "membership/MembersListModule",
|
|
830
|
+
page: 1,
|
|
831
|
+
group
|
|
832
|
+
}
|
|
833
|
+
]);
|
|
834
|
+
if (firstResult.isErr()) {
|
|
835
|
+
throw firstResult.error;
|
|
836
|
+
}
|
|
837
|
+
const firstResponse = firstResult.value[0];
|
|
838
|
+
if (!firstResponse) {
|
|
839
|
+
throw new UnexpectedError("Empty response");
|
|
840
|
+
}
|
|
841
|
+
const firstHtml = String(firstResponse.body ?? "");
|
|
842
|
+
members.push(...SiteMember.parse(site, firstHtml));
|
|
843
|
+
const $first = cheerio3.load(firstHtml);
|
|
844
|
+
const pagerLinks = $first("div.pager a");
|
|
845
|
+
if (pagerLinks.length < 2) {
|
|
846
|
+
return members;
|
|
847
|
+
}
|
|
848
|
+
const lastPageText = $first(pagerLinks[pagerLinks.length - 2]).text().trim();
|
|
849
|
+
const lastPage = Number.parseInt(lastPageText, 10) || 1;
|
|
850
|
+
if (lastPage <= 1) {
|
|
851
|
+
return members;
|
|
852
|
+
}
|
|
853
|
+
const bodies = [];
|
|
854
|
+
for (let page = 2;page <= lastPage; page++) {
|
|
855
|
+
bodies.push({
|
|
856
|
+
moduleName: "membership/MembersListModule",
|
|
857
|
+
page,
|
|
858
|
+
group
|
|
859
|
+
});
|
|
860
|
+
}
|
|
861
|
+
const additionalResults = await site.amcRequest(bodies);
|
|
862
|
+
if (additionalResults.isErr()) {
|
|
863
|
+
throw additionalResults.error;
|
|
864
|
+
}
|
|
865
|
+
for (const response of additionalResults.value) {
|
|
866
|
+
const html = String(response?.body ?? "");
|
|
867
|
+
members.push(...SiteMember.parse(site, html));
|
|
868
|
+
}
|
|
869
|
+
return members;
|
|
870
|
+
})(), (error) => new UnexpectedError(`Failed to get members: ${String(error)}`));
|
|
871
|
+
}
|
|
872
|
+
changeGroup(event) {
|
|
873
|
+
return fromPromise((async () => {
|
|
874
|
+
const result = await this.site.amcRequest([
|
|
875
|
+
{
|
|
876
|
+
action: "ManageSiteMembershipAction",
|
|
877
|
+
event,
|
|
878
|
+
user_id: this.user.id,
|
|
879
|
+
moduleName: ""
|
|
880
|
+
}
|
|
881
|
+
]);
|
|
882
|
+
if (result.isErr()) {
|
|
883
|
+
const error = result.error;
|
|
884
|
+
if (error instanceof WikidotStatusError) {
|
|
885
|
+
if (error.statusCode === "not_already") {
|
|
886
|
+
throw new TargetError(`User is not moderator/admin: ${this.user.name}`);
|
|
887
|
+
}
|
|
888
|
+
if (error.statusCode === "already_admin" || error.statusCode === "already_moderator") {
|
|
889
|
+
throw new TargetError(`User is already ${error.statusCode.replace("already_", "")}: ${this.user.name}`);
|
|
890
|
+
}
|
|
891
|
+
}
|
|
892
|
+
throw error;
|
|
893
|
+
}
|
|
894
|
+
})(), (error) => {
|
|
895
|
+
if (error instanceof TargetError || error instanceof LoginRequiredError) {
|
|
896
|
+
return error;
|
|
897
|
+
}
|
|
898
|
+
return new UnexpectedError(`Failed to change member group: ${String(error)}`);
|
|
899
|
+
});
|
|
900
|
+
}
|
|
901
|
+
toModerator() {
|
|
902
|
+
return this.changeGroup("toModerators");
|
|
903
|
+
}
|
|
904
|
+
removeModerator() {
|
|
905
|
+
return this.changeGroup("removeModerator");
|
|
906
|
+
}
|
|
907
|
+
toAdmin() {
|
|
908
|
+
return this.changeGroup("toAdmins");
|
|
909
|
+
}
|
|
910
|
+
removeAdmin() {
|
|
911
|
+
return this.changeGroup("removeAdmin");
|
|
912
|
+
}
|
|
913
|
+
toString() {
|
|
914
|
+
return `SiteMember(user=${this.user.name}, site=${this.site.unixName})`;
|
|
915
|
+
}
|
|
916
|
+
}
|
|
917
|
+
__legacyDecorateClassTS([
|
|
918
|
+
RequireLogin
|
|
919
|
+
], SiteMember.prototype, "changeGroup", null);
|
|
920
|
+
|
|
921
|
+
// src/module/site/accessors/member-accessor.ts
|
|
922
|
+
class MemberAccessor {
|
|
923
|
+
site;
|
|
924
|
+
constructor(site) {
|
|
925
|
+
this.site = site;
|
|
926
|
+
}
|
|
927
|
+
getAll() {
|
|
928
|
+
return SiteMember.getMembers(this.site, "");
|
|
929
|
+
}
|
|
930
|
+
getModerators() {
|
|
931
|
+
return SiteMember.getMembers(this.site, "moderators");
|
|
932
|
+
}
|
|
933
|
+
getAdmins() {
|
|
934
|
+
return SiteMember.getMembers(this.site, "admins");
|
|
935
|
+
}
|
|
936
|
+
getApplications() {
|
|
937
|
+
return SiteApplication.acquireAll(this.site);
|
|
938
|
+
}
|
|
939
|
+
lookup(query) {
|
|
940
|
+
return QuickModule.memberLookup(this.site.id, query);
|
|
941
|
+
}
|
|
942
|
+
invite(user, text) {
|
|
943
|
+
return fromPromise((async () => {
|
|
944
|
+
const result = await this.site.amcRequest([
|
|
945
|
+
{
|
|
946
|
+
action: "ManageSiteMembershipAction",
|
|
947
|
+
event: "inviteMember",
|
|
948
|
+
user_id: user.id,
|
|
949
|
+
text,
|
|
950
|
+
moduleName: "Empty"
|
|
951
|
+
}
|
|
952
|
+
]);
|
|
953
|
+
if (result.isErr()) {
|
|
954
|
+
const error = result.error;
|
|
955
|
+
if (error instanceof WikidotStatusError) {
|
|
956
|
+
if (error.statusCode === "already_invited") {
|
|
957
|
+
throw new TargetError(`User is already invited to ${this.site.unixName}: ${user.name}`);
|
|
958
|
+
}
|
|
959
|
+
if (error.statusCode === "already_member") {
|
|
960
|
+
throw new TargetError(`User is already a member of ${this.site.unixName}: ${user.name}`);
|
|
961
|
+
}
|
|
962
|
+
}
|
|
963
|
+
throw error;
|
|
964
|
+
}
|
|
965
|
+
})(), (error) => {
|
|
966
|
+
if (error instanceof TargetError || error instanceof LoginRequiredError) {
|
|
967
|
+
return error;
|
|
968
|
+
}
|
|
969
|
+
return new UnexpectedError(`Failed to invite user: ${String(error)}`);
|
|
970
|
+
});
|
|
971
|
+
}
|
|
972
|
+
}
|
|
973
|
+
__legacyDecorateClassTS([
|
|
974
|
+
RequireLogin
|
|
975
|
+
], MemberAccessor.prototype, "invite", null);
|
|
976
|
+
// src/module/page/page.ts
|
|
977
|
+
import * as cheerio6 from "cheerio";
|
|
978
|
+
import { z as z3 } from "zod";
|
|
979
|
+
|
|
980
|
+
// src/module/page/page-file.ts
|
|
981
|
+
import * as cheerio4 from "cheerio";
|
|
982
|
+
class PageFile {
|
|
983
|
+
page;
|
|
984
|
+
id;
|
|
985
|
+
name;
|
|
986
|
+
url;
|
|
987
|
+
mimeType;
|
|
988
|
+
size;
|
|
989
|
+
constructor(data) {
|
|
990
|
+
this.page = data.page;
|
|
991
|
+
this.id = data.id;
|
|
992
|
+
this.name = data.name;
|
|
993
|
+
this.url = data.url;
|
|
994
|
+
this.mimeType = data.mimeType;
|
|
995
|
+
this.size = data.size;
|
|
996
|
+
}
|
|
997
|
+
toString() {
|
|
998
|
+
return `PageFile(id=${this.id}, name=${this.name}, size=${this.size})`;
|
|
999
|
+
}
|
|
1000
|
+
}
|
|
1001
|
+
|
|
1002
|
+
class PageFileCollection extends Array {
|
|
1003
|
+
page;
|
|
1004
|
+
constructor(page, files) {
|
|
1005
|
+
super();
|
|
1006
|
+
this.page = page;
|
|
1007
|
+
if (files) {
|
|
1008
|
+
this.push(...files);
|
|
1009
|
+
}
|
|
1010
|
+
}
|
|
1011
|
+
findById(id) {
|
|
1012
|
+
return this.find((file) => file.id === id);
|
|
1013
|
+
}
|
|
1014
|
+
findByName(name) {
|
|
1015
|
+
return this.find((file) => file.name === name);
|
|
1016
|
+
}
|
|
1017
|
+
static parseSize(sizeText) {
|
|
1018
|
+
const text = sizeText.trim();
|
|
1019
|
+
if (text.includes("Bytes")) {
|
|
1020
|
+
return Math.floor(Number.parseFloat(text.replace("Bytes", "").trim()));
|
|
1021
|
+
}
|
|
1022
|
+
if (text.includes("kB")) {
|
|
1023
|
+
return Math.floor(Number.parseFloat(text.replace("kB", "").trim()) * 1000);
|
|
1024
|
+
}
|
|
1025
|
+
if (text.includes("MB")) {
|
|
1026
|
+
return Math.floor(Number.parseFloat(text.replace("MB", "").trim()) * 1e6);
|
|
1027
|
+
}
|
|
1028
|
+
if (text.includes("GB")) {
|
|
1029
|
+
return Math.floor(Number.parseFloat(text.replace("GB", "").trim()) * 1e9);
|
|
1030
|
+
}
|
|
1031
|
+
return 0;
|
|
1032
|
+
}
|
|
1033
|
+
static acquire(page) {
|
|
1034
|
+
if (page.id === null) {
|
|
1035
|
+
return fromPromise(Promise.reject(new Error("Page ID not acquired")), () => new UnexpectedError("Page ID must be acquired before getting files"));
|
|
1036
|
+
}
|
|
1037
|
+
const pageId = page.id;
|
|
1038
|
+
return fromPromise((async () => {
|
|
1039
|
+
const result = await page.site.amcRequest([
|
|
1040
|
+
{
|
|
1041
|
+
moduleName: "files/PageFilesModule",
|
|
1042
|
+
page_id: pageId
|
|
1043
|
+
}
|
|
1044
|
+
]);
|
|
1045
|
+
if (result.isErr()) {
|
|
1046
|
+
throw result.error;
|
|
1047
|
+
}
|
|
1048
|
+
const response = result.value[0];
|
|
1049
|
+
if (!response) {
|
|
1050
|
+
throw new UnexpectedError("Empty response");
|
|
1051
|
+
}
|
|
1052
|
+
const html = String(response.body ?? "");
|
|
1053
|
+
const $ = cheerio4.load(html);
|
|
1054
|
+
const filesTable = $("table.page-files");
|
|
1055
|
+
if (filesTable.length === 0) {
|
|
1056
|
+
return new PageFileCollection(page, []);
|
|
1057
|
+
}
|
|
1058
|
+
const files = [];
|
|
1059
|
+
filesTable.find("tbody tr[id^='file-row-']").each((_i, row) => {
|
|
1060
|
+
const rowId = $(row).attr("id");
|
|
1061
|
+
if (!rowId)
|
|
1062
|
+
return;
|
|
1063
|
+
const fileId = Number.parseInt(rowId.replace("file-row-", ""), 10);
|
|
1064
|
+
const tds = $(row).find("td");
|
|
1065
|
+
if (tds.length < 3)
|
|
1066
|
+
return;
|
|
1067
|
+
const linkElem = $(tds[0]).find("a");
|
|
1068
|
+
if (linkElem.length === 0)
|
|
1069
|
+
return;
|
|
1070
|
+
const name = linkElem.text().trim();
|
|
1071
|
+
const href = linkElem.attr("href") ?? "";
|
|
1072
|
+
const url = `${page.site.getBaseUrl()}${href}`;
|
|
1073
|
+
const mimeElem = $(tds[1]).find("span");
|
|
1074
|
+
const mimeType = mimeElem.attr("title") ?? "";
|
|
1075
|
+
const sizeText = $(tds[2]).text().trim();
|
|
1076
|
+
const size = PageFileCollection.parseSize(sizeText);
|
|
1077
|
+
files.push(new PageFile({
|
|
1078
|
+
page,
|
|
1079
|
+
id: fileId,
|
|
1080
|
+
name,
|
|
1081
|
+
url,
|
|
1082
|
+
mimeType,
|
|
1083
|
+
size
|
|
1084
|
+
}));
|
|
1085
|
+
});
|
|
1086
|
+
return new PageFileCollection(page, files);
|
|
1087
|
+
})(), (error) => new UnexpectedError(`Failed to acquire files: ${String(error)}`));
|
|
1088
|
+
}
|
|
1089
|
+
}
|
|
1090
|
+
|
|
1091
|
+
// src/module/page/page-meta.ts
|
|
1092
|
+
import * as cheerio5 from "cheerio";
|
|
1093
|
+
class PageMeta {
|
|
1094
|
+
page;
|
|
1095
|
+
name;
|
|
1096
|
+
content;
|
|
1097
|
+
constructor(data) {
|
|
1098
|
+
this.page = data.page;
|
|
1099
|
+
this.name = data.name;
|
|
1100
|
+
this.content = data.content;
|
|
1101
|
+
}
|
|
1102
|
+
update(content) {
|
|
1103
|
+
return PageMetaCollection.setMeta(this.page, this.name, content);
|
|
1104
|
+
}
|
|
1105
|
+
delete() {
|
|
1106
|
+
return PageMetaCollection.deleteMeta(this.page, this.name);
|
|
1107
|
+
}
|
|
1108
|
+
toString() {
|
|
1109
|
+
return `PageMeta(name=${this.name}, content=${this.content})`;
|
|
1110
|
+
}
|
|
1111
|
+
}
|
|
1112
|
+
|
|
1113
|
+
class PageMetaCollection extends Array {
|
|
1114
|
+
page;
|
|
1115
|
+
constructor(page, metas) {
|
|
1116
|
+
super();
|
|
1117
|
+
this.page = page;
|
|
1118
|
+
if (metas) {
|
|
1119
|
+
this.push(...metas);
|
|
1120
|
+
}
|
|
1121
|
+
}
|
|
1122
|
+
findByName(name) {
|
|
1123
|
+
return this.find((meta) => meta.name === name);
|
|
1124
|
+
}
|
|
1125
|
+
static acquire(page) {
|
|
1126
|
+
return fromPromise((async () => {
|
|
1127
|
+
const result = await page.site.amcRequest([
|
|
1128
|
+
{
|
|
1129
|
+
moduleName: "edit/EditMetaModule",
|
|
1130
|
+
page_id: page.id
|
|
1131
|
+
}
|
|
1132
|
+
]);
|
|
1133
|
+
if (result.isErr()) {
|
|
1134
|
+
throw result.error;
|
|
1135
|
+
}
|
|
1136
|
+
const response = result.value[0];
|
|
1137
|
+
if (!response) {
|
|
1138
|
+
throw new NoElementError("Empty response");
|
|
1139
|
+
}
|
|
1140
|
+
const html = String(response.body ?? "");
|
|
1141
|
+
const $ = cheerio5.load(html);
|
|
1142
|
+
const metas = [];
|
|
1143
|
+
$("table.meta-table tr").each((_i, elem) => {
|
|
1144
|
+
const $row = $(elem);
|
|
1145
|
+
const $cells = $row.find("td");
|
|
1146
|
+
if ($cells.length < 2)
|
|
1147
|
+
return;
|
|
1148
|
+
const name = $($cells[0]).text().trim();
|
|
1149
|
+
const content = $($cells[1]).text().trim();
|
|
1150
|
+
if (name) {
|
|
1151
|
+
metas.push(new PageMeta({
|
|
1152
|
+
page,
|
|
1153
|
+
name,
|
|
1154
|
+
content
|
|
1155
|
+
}));
|
|
1156
|
+
}
|
|
1157
|
+
});
|
|
1158
|
+
return new PageMetaCollection(page, metas);
|
|
1159
|
+
})(), (error) => {
|
|
1160
|
+
if (error instanceof NoElementError) {
|
|
1161
|
+
return error;
|
|
1162
|
+
}
|
|
1163
|
+
return new UnexpectedError(`Failed to acquire page metas: ${String(error)}`);
|
|
1164
|
+
});
|
|
1165
|
+
}
|
|
1166
|
+
static setMeta(page, name, content) {
|
|
1167
|
+
const loginResult = page.site.client.requireLogin();
|
|
1168
|
+
if (loginResult.isErr()) {
|
|
1169
|
+
return fromPromise(Promise.reject(loginResult.error), () => new LoginRequiredError("Login required to set meta tag"));
|
|
1170
|
+
}
|
|
1171
|
+
return fromPromise((async () => {
|
|
1172
|
+
const result = await page.site.amcRequest([
|
|
1173
|
+
{
|
|
1174
|
+
action: "WikiPageAction",
|
|
1175
|
+
event: "saveMetaTag",
|
|
1176
|
+
moduleName: "Empty",
|
|
1177
|
+
page_id: page.id,
|
|
1178
|
+
meta_name: name,
|
|
1179
|
+
meta_content: content
|
|
1180
|
+
}
|
|
1181
|
+
]);
|
|
1182
|
+
if (result.isErr()) {
|
|
1183
|
+
throw result.error;
|
|
1184
|
+
}
|
|
1185
|
+
})(), (error) => {
|
|
1186
|
+
if (error instanceof LoginRequiredError) {
|
|
1187
|
+
return error;
|
|
1188
|
+
}
|
|
1189
|
+
return new UnexpectedError(`Failed to set meta tag: ${String(error)}`);
|
|
1190
|
+
});
|
|
1191
|
+
}
|
|
1192
|
+
static deleteMeta(page, name) {
|
|
1193
|
+
const loginResult = page.site.client.requireLogin();
|
|
1194
|
+
if (loginResult.isErr()) {
|
|
1195
|
+
return fromPromise(Promise.reject(loginResult.error), () => new LoginRequiredError("Login required to delete meta tag"));
|
|
1196
|
+
}
|
|
1197
|
+
return fromPromise((async () => {
|
|
1198
|
+
const result = await page.site.amcRequest([
|
|
1199
|
+
{
|
|
1200
|
+
action: "WikiPageAction",
|
|
1201
|
+
event: "deleteMetaTag",
|
|
1202
|
+
moduleName: "Empty",
|
|
1203
|
+
page_id: page.id,
|
|
1204
|
+
meta_name: name
|
|
1205
|
+
}
|
|
1206
|
+
]);
|
|
1207
|
+
if (result.isErr()) {
|
|
1208
|
+
throw result.error;
|
|
1209
|
+
}
|
|
1210
|
+
})(), (error) => {
|
|
1211
|
+
if (error instanceof LoginRequiredError) {
|
|
1212
|
+
return error;
|
|
1213
|
+
}
|
|
1214
|
+
return new UnexpectedError(`Failed to delete meta tag: ${String(error)}`);
|
|
1215
|
+
});
|
|
1216
|
+
}
|
|
1217
|
+
}
|
|
1218
|
+
|
|
1219
|
+
// src/module/page/page-source.ts
|
|
1220
|
+
class PageSource {
|
|
1221
|
+
page;
|
|
1222
|
+
wikiText;
|
|
1223
|
+
constructor(data) {
|
|
1224
|
+
this.page = data.page;
|
|
1225
|
+
this.wikiText = data.wikiText;
|
|
1226
|
+
}
|
|
1227
|
+
toString() {
|
|
1228
|
+
return `PageSource(page=${this.page.fullname}, length=${this.wikiText.length})`;
|
|
1229
|
+
}
|
|
1230
|
+
}
|
|
1231
|
+
|
|
1232
|
+
// src/module/page/page-vote.ts
|
|
1233
|
+
class PageVote {
|
|
1234
|
+
page;
|
|
1235
|
+
user;
|
|
1236
|
+
value;
|
|
1237
|
+
constructor(data) {
|
|
1238
|
+
this.page = data.page;
|
|
1239
|
+
this.user = data.user;
|
|
1240
|
+
this.value = data.value;
|
|
1241
|
+
}
|
|
1242
|
+
toString() {
|
|
1243
|
+
return `PageVote(user=${this.user.name}, value=${this.value})`;
|
|
1244
|
+
}
|
|
1245
|
+
}
|
|
1246
|
+
|
|
1247
|
+
class PageVoteCollection extends Array {
|
|
1248
|
+
page;
|
|
1249
|
+
constructor(page, votes) {
|
|
1250
|
+
super();
|
|
1251
|
+
this.page = page;
|
|
1252
|
+
if (votes) {
|
|
1253
|
+
this.push(...votes);
|
|
1254
|
+
}
|
|
1255
|
+
}
|
|
1256
|
+
findByUser(user) {
|
|
1257
|
+
return this.find((vote) => vote.user.id === user.id);
|
|
1258
|
+
}
|
|
1259
|
+
}
|
|
1260
|
+
|
|
1261
|
+
// src/module/page/search-query.ts
|
|
1262
|
+
var DEFAULT_PER_PAGE = 250;
|
|
1263
|
+
var DEFAULT_MODULE_BODY = [
|
|
1264
|
+
"fullname",
|
|
1265
|
+
"category",
|
|
1266
|
+
"name",
|
|
1267
|
+
"title",
|
|
1268
|
+
"created_at",
|
|
1269
|
+
"created_by_linked",
|
|
1270
|
+
"updated_at",
|
|
1271
|
+
"updated_by_linked",
|
|
1272
|
+
"commented_at",
|
|
1273
|
+
"commented_by_linked",
|
|
1274
|
+
"parent_fullname",
|
|
1275
|
+
"comments",
|
|
1276
|
+
"size",
|
|
1277
|
+
"children",
|
|
1278
|
+
"rating_votes",
|
|
1279
|
+
"rating",
|
|
1280
|
+
"rating_percent",
|
|
1281
|
+
"revisions",
|
|
1282
|
+
"tags",
|
|
1283
|
+
"_tags"
|
|
1284
|
+
];
|
|
1285
|
+
|
|
1286
|
+
class SearchPagesQuery {
|
|
1287
|
+
pagetype;
|
|
1288
|
+
category;
|
|
1289
|
+
tags;
|
|
1290
|
+
parent;
|
|
1291
|
+
linkTo;
|
|
1292
|
+
createdAt;
|
|
1293
|
+
updatedAt;
|
|
1294
|
+
createdBy;
|
|
1295
|
+
rating;
|
|
1296
|
+
votes;
|
|
1297
|
+
name;
|
|
1298
|
+
fullname;
|
|
1299
|
+
range;
|
|
1300
|
+
order;
|
|
1301
|
+
offset;
|
|
1302
|
+
limit;
|
|
1303
|
+
perPage;
|
|
1304
|
+
separate;
|
|
1305
|
+
wrapper;
|
|
1306
|
+
constructor(params = {}) {
|
|
1307
|
+
this.pagetype = params.pagetype ?? "*";
|
|
1308
|
+
this.category = params.category ?? "*";
|
|
1309
|
+
this.tags = params.tags ?? null;
|
|
1310
|
+
this.parent = params.parent ?? null;
|
|
1311
|
+
this.linkTo = params.linkTo ?? null;
|
|
1312
|
+
this.createdAt = params.createdAt ?? null;
|
|
1313
|
+
this.updatedAt = params.updatedAt ?? null;
|
|
1314
|
+
this.createdBy = params.createdBy ?? null;
|
|
1315
|
+
this.rating = params.rating ?? null;
|
|
1316
|
+
this.votes = params.votes ?? null;
|
|
1317
|
+
this.name = params.name ?? null;
|
|
1318
|
+
this.fullname = params.fullname ?? null;
|
|
1319
|
+
this.range = params.range ?? null;
|
|
1320
|
+
this.order = params.order ?? "created_at desc";
|
|
1321
|
+
this.offset = params.offset ?? 0;
|
|
1322
|
+
this.limit = params.limit ?? null;
|
|
1323
|
+
this.perPage = params.perPage ?? DEFAULT_PER_PAGE;
|
|
1324
|
+
this.separate = params.separate ?? "no";
|
|
1325
|
+
this.wrapper = params.wrapper ?? "no";
|
|
1326
|
+
}
|
|
1327
|
+
asDict() {
|
|
1328
|
+
const result = {};
|
|
1329
|
+
if (this.pagetype !== "*")
|
|
1330
|
+
result.pagetype = this.pagetype;
|
|
1331
|
+
if (this.category !== "*")
|
|
1332
|
+
result.category = this.category;
|
|
1333
|
+
if (this.tags !== null) {
|
|
1334
|
+
result.tags = Array.isArray(this.tags) ? this.tags.join(" ") : this.tags;
|
|
1335
|
+
}
|
|
1336
|
+
if (this.parent !== null)
|
|
1337
|
+
result.parent = this.parent;
|
|
1338
|
+
if (this.linkTo !== null)
|
|
1339
|
+
result.link_to = this.linkTo;
|
|
1340
|
+
if (this.createdAt !== null)
|
|
1341
|
+
result.created_at = this.createdAt;
|
|
1342
|
+
if (this.updatedAt !== null)
|
|
1343
|
+
result.updated_at = this.updatedAt;
|
|
1344
|
+
if (this.createdBy !== null) {
|
|
1345
|
+
result.created_by = typeof this.createdBy === "string" ? this.createdBy : this.createdBy.name;
|
|
1346
|
+
}
|
|
1347
|
+
if (this.rating !== null)
|
|
1348
|
+
result.rating = this.rating;
|
|
1349
|
+
if (this.votes !== null)
|
|
1350
|
+
result.votes = this.votes;
|
|
1351
|
+
if (this.name !== null)
|
|
1352
|
+
result.name = this.name;
|
|
1353
|
+
if (this.fullname !== null)
|
|
1354
|
+
result.fullname = this.fullname;
|
|
1355
|
+
if (this.range !== null)
|
|
1356
|
+
result.range = this.range;
|
|
1357
|
+
result.order = this.order;
|
|
1358
|
+
result.offset = this.offset;
|
|
1359
|
+
if (this.limit !== null)
|
|
1360
|
+
result.limit = this.limit;
|
|
1361
|
+
result.perPage = this.perPage;
|
|
1362
|
+
result.separate = this.separate;
|
|
1363
|
+
result.wrapper = this.wrapper;
|
|
1364
|
+
return result;
|
|
1365
|
+
}
|
|
1366
|
+
}
|
|
1367
|
+
|
|
1368
|
+
// src/module/page/page.ts
|
|
1369
|
+
var pageParamsSchema = z3.object({
|
|
1370
|
+
fullname: z3.string().default(""),
|
|
1371
|
+
name: z3.string().default(""),
|
|
1372
|
+
category: z3.string().default(""),
|
|
1373
|
+
title: z3.string().default(""),
|
|
1374
|
+
children_count: z3.number().default(0),
|
|
1375
|
+
comments_count: z3.number().default(0),
|
|
1376
|
+
size: z3.number().default(0),
|
|
1377
|
+
rating: z3.number().default(0),
|
|
1378
|
+
votes_count: z3.number().default(0),
|
|
1379
|
+
rating_percent: z3.number().nullable().default(null),
|
|
1380
|
+
revisions_count: z3.number().default(0),
|
|
1381
|
+
parent_fullname: z3.string().nullable().default(null),
|
|
1382
|
+
tags: z3.array(z3.string()).default([]),
|
|
1383
|
+
created_by: z3.custom().nullable().default(null),
|
|
1384
|
+
created_at: z3.date().nullable().default(null),
|
|
1385
|
+
updated_by: z3.custom().nullable().default(null),
|
|
1386
|
+
updated_at: z3.date().nullable().default(null),
|
|
1387
|
+
commented_by: z3.custom().nullable().default(null),
|
|
1388
|
+
commented_at: z3.date().nullable().default(null)
|
|
1389
|
+
});
|
|
1390
|
+
|
|
1391
|
+
class Page {
|
|
1392
|
+
site;
|
|
1393
|
+
fullname;
|
|
1394
|
+
name;
|
|
1395
|
+
category;
|
|
1396
|
+
title;
|
|
1397
|
+
childrenCount;
|
|
1398
|
+
commentsCount;
|
|
1399
|
+
size;
|
|
1400
|
+
rating;
|
|
1401
|
+
votesCount;
|
|
1402
|
+
ratingPercent;
|
|
1403
|
+
revisionsCount;
|
|
1404
|
+
parentFullname;
|
|
1405
|
+
tags;
|
|
1406
|
+
createdBy;
|
|
1407
|
+
createdAt;
|
|
1408
|
+
updatedBy;
|
|
1409
|
+
updatedAt;
|
|
1410
|
+
commentedBy;
|
|
1411
|
+
commentedAt;
|
|
1412
|
+
_id = null;
|
|
1413
|
+
_source = null;
|
|
1414
|
+
_revisions = null;
|
|
1415
|
+
_votes = null;
|
|
1416
|
+
constructor(data) {
|
|
1417
|
+
this.site = data.site;
|
|
1418
|
+
this.fullname = data.fullname;
|
|
1419
|
+
this.name = data.name;
|
|
1420
|
+
this.category = data.category;
|
|
1421
|
+
this.title = data.title;
|
|
1422
|
+
this.childrenCount = data.childrenCount;
|
|
1423
|
+
this.commentsCount = data.commentsCount;
|
|
1424
|
+
this.size = data.size;
|
|
1425
|
+
this.rating = data.rating;
|
|
1426
|
+
this.votesCount = data.votesCount;
|
|
1427
|
+
this.ratingPercent = data.ratingPercent;
|
|
1428
|
+
this.revisionsCount = data.revisionsCount;
|
|
1429
|
+
this.parentFullname = data.parentFullname;
|
|
1430
|
+
this.tags = data.tags;
|
|
1431
|
+
this.createdBy = data.createdBy;
|
|
1432
|
+
this.createdAt = data.createdAt;
|
|
1433
|
+
this.updatedBy = data.updatedBy;
|
|
1434
|
+
this.updatedAt = data.updatedAt;
|
|
1435
|
+
this.commentedBy = data.commentedBy;
|
|
1436
|
+
this.commentedAt = data.commentedAt;
|
|
1437
|
+
}
|
|
1438
|
+
getUrl() {
|
|
1439
|
+
return `${this.site.getBaseUrl()}/${this.fullname}`;
|
|
1440
|
+
}
|
|
1441
|
+
isIdAcquired() {
|
|
1442
|
+
return this._id !== null;
|
|
1443
|
+
}
|
|
1444
|
+
get id() {
|
|
1445
|
+
return this._id;
|
|
1446
|
+
}
|
|
1447
|
+
set id(value) {
|
|
1448
|
+
this._id = value;
|
|
1449
|
+
}
|
|
1450
|
+
get source() {
|
|
1451
|
+
return this._source;
|
|
1452
|
+
}
|
|
1453
|
+
set source(value) {
|
|
1454
|
+
this._source = value;
|
|
1455
|
+
}
|
|
1456
|
+
get revisions() {
|
|
1457
|
+
return this._revisions;
|
|
1458
|
+
}
|
|
1459
|
+
set revisions(value) {
|
|
1460
|
+
this._revisions = value;
|
|
1461
|
+
}
|
|
1462
|
+
get votes() {
|
|
1463
|
+
return this._votes;
|
|
1464
|
+
}
|
|
1465
|
+
set votes(value) {
|
|
1466
|
+
this._votes = value;
|
|
1467
|
+
}
|
|
1468
|
+
get latestRevision() {
|
|
1469
|
+
if (!this._revisions || this._revisions.length === 0)
|
|
1470
|
+
return;
|
|
1471
|
+
return this._revisions.reduce((max, rev) => rev.revNo > max.revNo ? rev : max);
|
|
1472
|
+
}
|
|
1473
|
+
requireId(operation) {
|
|
1474
|
+
if (this._id === null) {
|
|
1475
|
+
return {
|
|
1476
|
+
ok: false,
|
|
1477
|
+
error: fromPromise(Promise.reject(new Error("Page ID not acquired")), () => new UnexpectedError(`Page ID must be acquired before ${operation}`))
|
|
1478
|
+
};
|
|
1479
|
+
}
|
|
1480
|
+
return { ok: true, id: this._id };
|
|
1481
|
+
}
|
|
1482
|
+
destroy() {
|
|
1483
|
+
const idCheck = this.requireId("deletion");
|
|
1484
|
+
if (!idCheck.ok)
|
|
1485
|
+
return idCheck.error;
|
|
1486
|
+
return fromPromise((async () => {
|
|
1487
|
+
const result = await this.site.amcRequest([
|
|
1488
|
+
{
|
|
1489
|
+
action: "WikiPageAction",
|
|
1490
|
+
event: "deletePage",
|
|
1491
|
+
page_id: this._id,
|
|
1492
|
+
moduleName: "Empty"
|
|
1493
|
+
}
|
|
1494
|
+
]);
|
|
1495
|
+
if (result.isErr()) {
|
|
1496
|
+
throw result.error;
|
|
1497
|
+
}
|
|
1498
|
+
})(), (error) => new UnexpectedError(`Failed to delete page: ${String(error)}`));
|
|
1499
|
+
}
|
|
1500
|
+
commitTags() {
|
|
1501
|
+
const idCheck = this.requireId("saving tags");
|
|
1502
|
+
if (!idCheck.ok)
|
|
1503
|
+
return idCheck.error;
|
|
1504
|
+
return fromPromise((async () => {
|
|
1505
|
+
const result = await this.site.amcRequest([
|
|
1506
|
+
{
|
|
1507
|
+
tags: this.tags.join(" "),
|
|
1508
|
+
action: "WikiPageAction",
|
|
1509
|
+
event: "saveTags",
|
|
1510
|
+
pageId: this._id,
|
|
1511
|
+
moduleName: "Empty"
|
|
1512
|
+
}
|
|
1513
|
+
]);
|
|
1514
|
+
if (result.isErr()) {
|
|
1515
|
+
throw result.error;
|
|
1516
|
+
}
|
|
1517
|
+
})(), (error) => new UnexpectedError(`Failed to save tags: ${String(error)}`));
|
|
1518
|
+
}
|
|
1519
|
+
setParent(parentFullname) {
|
|
1520
|
+
const idCheck = this.requireId("setting parent");
|
|
1521
|
+
if (!idCheck.ok)
|
|
1522
|
+
return idCheck.error;
|
|
1523
|
+
return fromPromise((async () => {
|
|
1524
|
+
const result = await this.site.amcRequest([
|
|
1525
|
+
{
|
|
1526
|
+
action: "WikiPageAction",
|
|
1527
|
+
event: "setParentPage",
|
|
1528
|
+
moduleName: "Empty",
|
|
1529
|
+
pageId: String(this._id),
|
|
1530
|
+
parentName: parentFullname ?? ""
|
|
1531
|
+
}
|
|
1532
|
+
]);
|
|
1533
|
+
if (result.isErr()) {
|
|
1534
|
+
throw result.error;
|
|
1535
|
+
}
|
|
1536
|
+
this.parentFullname = parentFullname;
|
|
1537
|
+
})(), (error) => new UnexpectedError(`Failed to set parent: ${String(error)}`));
|
|
1538
|
+
}
|
|
1539
|
+
vote(value) {
|
|
1540
|
+
const idCheck = this.requireId("voting");
|
|
1541
|
+
if (!idCheck.ok)
|
|
1542
|
+
return idCheck.error;
|
|
1543
|
+
return fromPromise((async () => {
|
|
1544
|
+
const result = await this.site.amcRequest([
|
|
1545
|
+
{
|
|
1546
|
+
action: "RateAction",
|
|
1547
|
+
event: "ratePage",
|
|
1548
|
+
moduleName: "Empty",
|
|
1549
|
+
pageId: this._id,
|
|
1550
|
+
points: value,
|
|
1551
|
+
force: "yes"
|
|
1552
|
+
}
|
|
1553
|
+
]);
|
|
1554
|
+
if (result.isErr()) {
|
|
1555
|
+
throw result.error;
|
|
1556
|
+
}
|
|
1557
|
+
const response = result.value[0];
|
|
1558
|
+
if (!response) {
|
|
1559
|
+
throw new UnexpectedError("Empty response from vote request");
|
|
1560
|
+
}
|
|
1561
|
+
const newRating = Number.parseInt(String(response.points ?? this.rating), 10);
|
|
1562
|
+
this.rating = newRating;
|
|
1563
|
+
return newRating;
|
|
1564
|
+
})(), (error) => new UnexpectedError(`Failed to vote: ${String(error)}`));
|
|
1565
|
+
}
|
|
1566
|
+
cancelVote() {
|
|
1567
|
+
const idCheck = this.requireId("canceling vote");
|
|
1568
|
+
if (!idCheck.ok)
|
|
1569
|
+
return idCheck.error;
|
|
1570
|
+
return fromPromise((async () => {
|
|
1571
|
+
const result = await this.site.amcRequest([
|
|
1572
|
+
{
|
|
1573
|
+
action: "RateAction",
|
|
1574
|
+
event: "cancelVote",
|
|
1575
|
+
moduleName: "Empty",
|
|
1576
|
+
pageId: this._id
|
|
1577
|
+
}
|
|
1578
|
+
]);
|
|
1579
|
+
if (result.isErr()) {
|
|
1580
|
+
throw result.error;
|
|
1581
|
+
}
|
|
1582
|
+
const response = result.value[0];
|
|
1583
|
+
if (!response) {
|
|
1584
|
+
throw new UnexpectedError("Empty response from cancel vote request");
|
|
1585
|
+
}
|
|
1586
|
+
const newRating = Number.parseInt(String(response.points ?? this.rating), 10);
|
|
1587
|
+
this.rating = newRating;
|
|
1588
|
+
return newRating;
|
|
1589
|
+
})(), (error) => new UnexpectedError(`Failed to cancel vote: ${String(error)}`));
|
|
1590
|
+
}
|
|
1591
|
+
edit(options) {
|
|
1592
|
+
const idCheck = this.requireId("editing");
|
|
1593
|
+
if (!idCheck.ok)
|
|
1594
|
+
return idCheck.error;
|
|
1595
|
+
const pageId = idCheck.id;
|
|
1596
|
+
return fromPromise((async () => {
|
|
1597
|
+
let currentSource = options.source;
|
|
1598
|
+
if (currentSource === undefined) {
|
|
1599
|
+
const existingSource = this._source;
|
|
1600
|
+
if (existingSource !== null) {
|
|
1601
|
+
currentSource = existingSource.wikiText;
|
|
1602
|
+
} else {
|
|
1603
|
+
const sourceResult = await PageCollection.acquirePageSources(this.site, [this]);
|
|
1604
|
+
if (sourceResult.isErr()) {
|
|
1605
|
+
throw sourceResult.error;
|
|
1606
|
+
}
|
|
1607
|
+
currentSource = this._source?.wikiText ?? "";
|
|
1608
|
+
}
|
|
1609
|
+
}
|
|
1610
|
+
const result = await PageCollection.createOrEdit(this.site, this.fullname, {
|
|
1611
|
+
pageId,
|
|
1612
|
+
title: options.title ?? this.title,
|
|
1613
|
+
source: currentSource,
|
|
1614
|
+
comment: options.comment ?? "",
|
|
1615
|
+
forceEdit: options.forceEdit ?? false
|
|
1616
|
+
});
|
|
1617
|
+
if (result.isErr()) {
|
|
1618
|
+
throw result.error;
|
|
1619
|
+
}
|
|
1620
|
+
})(), (error) => {
|
|
1621
|
+
if (error instanceof LoginRequiredError || error instanceof ForbiddenError) {
|
|
1622
|
+
return error;
|
|
1623
|
+
}
|
|
1624
|
+
return new UnexpectedError(`Failed to edit page: ${String(error)}`);
|
|
1625
|
+
});
|
|
1626
|
+
}
|
|
1627
|
+
rename(newFullname) {
|
|
1628
|
+
const idCheck = this.requireId("renaming");
|
|
1629
|
+
if (!idCheck.ok)
|
|
1630
|
+
return idCheck.error;
|
|
1631
|
+
return fromPromise((async () => {
|
|
1632
|
+
const result = await this.site.amcRequest([
|
|
1633
|
+
{
|
|
1634
|
+
action: "WikiPageAction",
|
|
1635
|
+
event: "renamePage",
|
|
1636
|
+
moduleName: "Empty",
|
|
1637
|
+
page_id: this._id,
|
|
1638
|
+
new_name: newFullname
|
|
1639
|
+
}
|
|
1640
|
+
]);
|
|
1641
|
+
if (result.isErr()) {
|
|
1642
|
+
throw result.error;
|
|
1643
|
+
}
|
|
1644
|
+
Object.assign(this, {
|
|
1645
|
+
fullname: newFullname,
|
|
1646
|
+
category: newFullname.includes(":") ? newFullname.split(":")[0] : "_default",
|
|
1647
|
+
name: newFullname.includes(":") ? newFullname.split(":")[1] : newFullname
|
|
1648
|
+
});
|
|
1649
|
+
})(), (error) => new UnexpectedError(`Failed to rename page: ${String(error)}`));
|
|
1650
|
+
}
|
|
1651
|
+
getFiles() {
|
|
1652
|
+
return PageFileCollection.acquire(this);
|
|
1653
|
+
}
|
|
1654
|
+
getDiscussion() {
|
|
1655
|
+
const idCheck = this.requireId("getting discussion");
|
|
1656
|
+
if (!idCheck.ok)
|
|
1657
|
+
return idCheck.error;
|
|
1658
|
+
const pageId = idCheck.id;
|
|
1659
|
+
return fromPromise((async () => {
|
|
1660
|
+
const result = await this.site.amcRequest([
|
|
1661
|
+
{
|
|
1662
|
+
moduleName: "forum/ForumCommentsListModule",
|
|
1663
|
+
pageId
|
|
1664
|
+
}
|
|
1665
|
+
]);
|
|
1666
|
+
if (result.isErr()) {
|
|
1667
|
+
throw result.error;
|
|
1668
|
+
}
|
|
1669
|
+
const response = result.value[0];
|
|
1670
|
+
if (!response) {
|
|
1671
|
+
return null;
|
|
1672
|
+
}
|
|
1673
|
+
const html = String(response.body ?? "");
|
|
1674
|
+
const match = html.match(/WIKIDOT\.modules\.ForumViewThreadModule\.vars\.threadId\s*=\s*(\d+)/);
|
|
1675
|
+
if (!match?.[1]) {
|
|
1676
|
+
return null;
|
|
1677
|
+
}
|
|
1678
|
+
const threadId = Number.parseInt(match[1], 10);
|
|
1679
|
+
const { ForumThread: ForumThread2 } = await import("./shared/index-ytknx2hn.js");
|
|
1680
|
+
const threadResult = await ForumThread2.getFromId(this.site, threadId);
|
|
1681
|
+
if (threadResult.isErr()) {
|
|
1682
|
+
throw threadResult.error;
|
|
1683
|
+
}
|
|
1684
|
+
return threadResult.value;
|
|
1685
|
+
})(), (error) => new UnexpectedError(`Failed to get discussion: ${String(error)}`));
|
|
1686
|
+
}
|
|
1687
|
+
getMetas() {
|
|
1688
|
+
const idCheck = this.requireId("getting metas");
|
|
1689
|
+
if (!idCheck.ok) {
|
|
1690
|
+
return idCheck.error;
|
|
1691
|
+
}
|
|
1692
|
+
return PageMetaCollection.acquire(this);
|
|
1693
|
+
}
|
|
1694
|
+
setMeta(name, content) {
|
|
1695
|
+
const idCheck = this.requireId("setting meta");
|
|
1696
|
+
if (!idCheck.ok) {
|
|
1697
|
+
return idCheck.error;
|
|
1698
|
+
}
|
|
1699
|
+
return PageMetaCollection.setMeta(this, name, content);
|
|
1700
|
+
}
|
|
1701
|
+
deleteMeta(name) {
|
|
1702
|
+
const idCheck = this.requireId("deleting meta");
|
|
1703
|
+
if (!idCheck.ok) {
|
|
1704
|
+
return idCheck.error;
|
|
1705
|
+
}
|
|
1706
|
+
return PageMetaCollection.deleteMeta(this, name);
|
|
1707
|
+
}
|
|
1708
|
+
toString() {
|
|
1709
|
+
return `Page(fullname=${this.fullname}, title=${this.title})`;
|
|
1710
|
+
}
|
|
1711
|
+
}
|
|
1712
|
+
__legacyDecorateClassTS([
|
|
1713
|
+
RequireLogin
|
|
1714
|
+
], Page.prototype, "destroy", null);
|
|
1715
|
+
__legacyDecorateClassTS([
|
|
1716
|
+
RequireLogin
|
|
1717
|
+
], Page.prototype, "commitTags", null);
|
|
1718
|
+
__legacyDecorateClassTS([
|
|
1719
|
+
RequireLogin
|
|
1720
|
+
], Page.prototype, "setParent", null);
|
|
1721
|
+
__legacyDecorateClassTS([
|
|
1722
|
+
RequireLogin
|
|
1723
|
+
], Page.prototype, "vote", null);
|
|
1724
|
+
__legacyDecorateClassTS([
|
|
1725
|
+
RequireLogin
|
|
1726
|
+
], Page.prototype, "cancelVote", null);
|
|
1727
|
+
__legacyDecorateClassTS([
|
|
1728
|
+
RequireLogin
|
|
1729
|
+
], Page.prototype, "edit", null);
|
|
1730
|
+
__legacyDecorateClassTS([
|
|
1731
|
+
RequireLogin
|
|
1732
|
+
], Page.prototype, "rename", null);
|
|
1733
|
+
|
|
1734
|
+
class PageCollection extends Array {
|
|
1735
|
+
site;
|
|
1736
|
+
constructor(site, pages) {
|
|
1737
|
+
super();
|
|
1738
|
+
this.site = site;
|
|
1739
|
+
if (pages) {
|
|
1740
|
+
this.push(...pages);
|
|
1741
|
+
}
|
|
1742
|
+
}
|
|
1743
|
+
findByFullname(fullname) {
|
|
1744
|
+
return this.find((page) => page.fullname === fullname);
|
|
1745
|
+
}
|
|
1746
|
+
getPageIds() {
|
|
1747
|
+
return PageCollection.acquirePageIds(this.site, this);
|
|
1748
|
+
}
|
|
1749
|
+
getPageSources() {
|
|
1750
|
+
return PageCollection.acquirePageSources(this.site, this);
|
|
1751
|
+
}
|
|
1752
|
+
getPageRevisions() {
|
|
1753
|
+
return PageCollection.acquirePageRevisions(this.site, this);
|
|
1754
|
+
}
|
|
1755
|
+
getPageVotes() {
|
|
1756
|
+
return PageCollection.acquirePageVotes(this.site, this);
|
|
1757
|
+
}
|
|
1758
|
+
static acquirePageIds(site, pages) {
|
|
1759
|
+
return fromPromise((async () => {
|
|
1760
|
+
const targetPages = pages.filter((page) => !page.isIdAcquired());
|
|
1761
|
+
if (targetPages.length === 0) {
|
|
1762
|
+
return new PageCollection(site, pages);
|
|
1763
|
+
}
|
|
1764
|
+
const responses = await Promise.all(targetPages.map(async (page) => {
|
|
1765
|
+
const url = `${page.getUrl()}/norender/true/noredirect/true`;
|
|
1766
|
+
const response = await fetch(url, {
|
|
1767
|
+
headers: site.client.amcClient.header.getHeaders()
|
|
1768
|
+
});
|
|
1769
|
+
return { page, response };
|
|
1770
|
+
}));
|
|
1771
|
+
for (const { page, response } of responses) {
|
|
1772
|
+
const text = await response.text();
|
|
1773
|
+
const match = text.match(/WIKIREQUEST\.info\.pageId\s*=\s*(\d+);/);
|
|
1774
|
+
if (!match?.[1]) {
|
|
1775
|
+
throw new NoElementError(`Cannot find page id: ${page.fullname}`);
|
|
1776
|
+
}
|
|
1777
|
+
page.id = Number.parseInt(match[1], 10);
|
|
1778
|
+
}
|
|
1779
|
+
return new PageCollection(site, pages);
|
|
1780
|
+
})(), (error) => {
|
|
1781
|
+
if (error instanceof NoElementError)
|
|
1782
|
+
return error;
|
|
1783
|
+
return new UnexpectedError(`Failed to acquire page IDs: ${String(error)}`);
|
|
1784
|
+
});
|
|
1785
|
+
}
|
|
1786
|
+
static acquirePageSources(site, pages) {
|
|
1787
|
+
return fromPromise((async () => {
|
|
1788
|
+
const targetPages = pages.filter((page) => page.source === null && page.id !== null);
|
|
1789
|
+
if (targetPages.length === 0) {
|
|
1790
|
+
return new PageCollection(site, pages);
|
|
1791
|
+
}
|
|
1792
|
+
const result = await site.amcRequest(targetPages.map((page) => ({
|
|
1793
|
+
moduleName: "viewsource/ViewSourceModule",
|
|
1794
|
+
page_id: page.id
|
|
1795
|
+
})));
|
|
1796
|
+
if (result.isErr()) {
|
|
1797
|
+
throw result.error;
|
|
1798
|
+
}
|
|
1799
|
+
for (let i = 0;i < targetPages.length; i++) {
|
|
1800
|
+
const page = targetPages[i];
|
|
1801
|
+
const response = result.value[i];
|
|
1802
|
+
if (!page || !response)
|
|
1803
|
+
continue;
|
|
1804
|
+
const body = String(response.body ?? "").replace(/ /g, " ");
|
|
1805
|
+
const $ = cheerio6.load(body);
|
|
1806
|
+
const sourceElement = $("div.page-source");
|
|
1807
|
+
if (sourceElement.length === 0) {
|
|
1808
|
+
throw new NoElementError(`Cannot find source element for page: ${page.fullname}`);
|
|
1809
|
+
}
|
|
1810
|
+
const wikiText = sourceElement.text().trim().replace(/^\t/, "");
|
|
1811
|
+
page.source = new PageSource({ page, wikiText });
|
|
1812
|
+
}
|
|
1813
|
+
return new PageCollection(site, pages);
|
|
1814
|
+
})(), (error) => {
|
|
1815
|
+
if (error instanceof NoElementError)
|
|
1816
|
+
return error;
|
|
1817
|
+
return new UnexpectedError(`Failed to acquire page sources: ${String(error)}`);
|
|
1818
|
+
});
|
|
1819
|
+
}
|
|
1820
|
+
static acquirePageRevisions(site, pages) {
|
|
1821
|
+
return fromPromise((async () => {
|
|
1822
|
+
const targetPages = pages.filter((page) => page.revisions === null && page.id !== null);
|
|
1823
|
+
if (targetPages.length === 0) {
|
|
1824
|
+
return new PageCollection(site, pages);
|
|
1825
|
+
}
|
|
1826
|
+
const result = await site.amcRequest(targetPages.map((page) => ({
|
|
1827
|
+
moduleName: "history/PageRevisionListModule",
|
|
1828
|
+
page_id: page.id,
|
|
1829
|
+
options: { all: true },
|
|
1830
|
+
perpage: 1e8
|
|
1831
|
+
})));
|
|
1832
|
+
if (result.isErr()) {
|
|
1833
|
+
throw result.error;
|
|
1834
|
+
}
|
|
1835
|
+
const { PageRevision: PageRevision2 } = await import("./shared/index-f2eh3ykk.js");
|
|
1836
|
+
for (let i = 0;i < targetPages.length; i++) {
|
|
1837
|
+
const page = targetPages[i];
|
|
1838
|
+
const response = result.value[i];
|
|
1839
|
+
if (!page || !response)
|
|
1840
|
+
continue;
|
|
1841
|
+
const body = String(response.body ?? "");
|
|
1842
|
+
const $ = cheerio6.load(body);
|
|
1843
|
+
const revisions = [];
|
|
1844
|
+
$("table.page-history > tr[id^=revision-row-]").each((_j, revElement) => {
|
|
1845
|
+
const $rev = $(revElement);
|
|
1846
|
+
const revIdAttr = $rev.attr("id");
|
|
1847
|
+
if (!revIdAttr)
|
|
1848
|
+
return;
|
|
1849
|
+
const revId = Number.parseInt(revIdAttr.replace("revision-row-", ""), 10);
|
|
1850
|
+
if (Number.isNaN(revId))
|
|
1851
|
+
return;
|
|
1852
|
+
const $tds = $rev.find("td");
|
|
1853
|
+
if ($tds.length < 7)
|
|
1854
|
+
return;
|
|
1855
|
+
const revNoText = $tds.eq(0).text().trim().replace(/\.$/, "");
|
|
1856
|
+
const revNo = Number.parseInt(revNoText, 10);
|
|
1857
|
+
if (Number.isNaN(revNo))
|
|
1858
|
+
return;
|
|
1859
|
+
const $createdByElem = $tds.eq(4).find("span.printuser");
|
|
1860
|
+
if ($createdByElem.length === 0)
|
|
1861
|
+
return;
|
|
1862
|
+
const createdBy = parseUser(site.client, $createdByElem);
|
|
1863
|
+
const $createdAtElem = $tds.eq(5).find("span.odate");
|
|
1864
|
+
if ($createdAtElem.length === 0)
|
|
1865
|
+
return;
|
|
1866
|
+
const createdAt = parseOdate($createdAtElem) ?? new Date;
|
|
1867
|
+
const comment = $tds.eq(6).text().trim();
|
|
1868
|
+
revisions.push(new PageRevision2({
|
|
1869
|
+
page,
|
|
1870
|
+
id: revId,
|
|
1871
|
+
revNo,
|
|
1872
|
+
createdBy,
|
|
1873
|
+
createdAt,
|
|
1874
|
+
comment
|
|
1875
|
+
}));
|
|
1876
|
+
});
|
|
1877
|
+
page.revisions = new PageRevisionCollection(page, revisions);
|
|
1878
|
+
}
|
|
1879
|
+
return new PageCollection(site, pages);
|
|
1880
|
+
})(), (error) => new UnexpectedError(`Failed to acquire page revisions: ${String(error)}`));
|
|
1881
|
+
}
|
|
1882
|
+
static acquirePageVotes(site, pages) {
|
|
1883
|
+
return fromPromise((async () => {
|
|
1884
|
+
const targetPages = pages.filter((page) => page.votes === null && page.id !== null);
|
|
1885
|
+
if (targetPages.length === 0) {
|
|
1886
|
+
return new PageCollection(site, pages);
|
|
1887
|
+
}
|
|
1888
|
+
const result = await site.amcRequest(targetPages.map((page) => ({
|
|
1889
|
+
moduleName: "pagerate/WhoRatedPageModule",
|
|
1890
|
+
pageId: page.id
|
|
1891
|
+
})));
|
|
1892
|
+
if (result.isErr()) {
|
|
1893
|
+
throw result.error;
|
|
1894
|
+
}
|
|
1895
|
+
for (let i = 0;i < targetPages.length; i++) {
|
|
1896
|
+
const page = targetPages[i];
|
|
1897
|
+
const response = result.value[i];
|
|
1898
|
+
if (!page || !response)
|
|
1899
|
+
continue;
|
|
1900
|
+
const body = String(response.body ?? "");
|
|
1901
|
+
const $ = cheerio6.load(body);
|
|
1902
|
+
const $userElems = $("span.printuser");
|
|
1903
|
+
const $valueElems = $("span[style^='color']");
|
|
1904
|
+
if ($userElems.length !== $valueElems.length) {
|
|
1905
|
+
throw new UnexpectedError("User and value count mismatch in votes");
|
|
1906
|
+
}
|
|
1907
|
+
const votes = [];
|
|
1908
|
+
$userElems.each((j, userElem) => {
|
|
1909
|
+
const $user = $(userElem);
|
|
1910
|
+
const $value = $valueElems.eq(j);
|
|
1911
|
+
const user = parseUser(site.client, $user);
|
|
1912
|
+
const valueText = $value.text().trim();
|
|
1913
|
+
let value;
|
|
1914
|
+
if (valueText === "+") {
|
|
1915
|
+
value = 1;
|
|
1916
|
+
} else if (valueText === "-") {
|
|
1917
|
+
value = -1;
|
|
1918
|
+
} else {
|
|
1919
|
+
value = Number.parseInt(valueText, 10) || 0;
|
|
1920
|
+
}
|
|
1921
|
+
votes.push(new PageVote({ page, user, value }));
|
|
1922
|
+
});
|
|
1923
|
+
page.votes = new PageVoteCollection(page, votes);
|
|
1924
|
+
}
|
|
1925
|
+
return new PageCollection(site, pages);
|
|
1926
|
+
})(), (error) => new UnexpectedError(`Failed to acquire page votes: ${String(error)}`));
|
|
1927
|
+
}
|
|
1928
|
+
static parse(site, htmlBody, _parseUser) {
|
|
1929
|
+
const pages = [];
|
|
1930
|
+
htmlBody("div.page").each((_i, pageElement) => {
|
|
1931
|
+
const $page = htmlBody(pageElement);
|
|
1932
|
+
const pageParams = {};
|
|
1933
|
+
const is5StarRating = $page.find("span.rating span.page-rate-list-pages-start").length > 0;
|
|
1934
|
+
$page.find("span.set").each((_j, setElement) => {
|
|
1935
|
+
const $set = htmlBody(setElement);
|
|
1936
|
+
const keyElement = $set.find("span.name");
|
|
1937
|
+
if (keyElement.length === 0)
|
|
1938
|
+
return;
|
|
1939
|
+
let key = keyElement.text().trim();
|
|
1940
|
+
const valueElement = $set.find("span.value");
|
|
1941
|
+
let value = null;
|
|
1942
|
+
if (valueElement.length === 0) {
|
|
1943
|
+
value = null;
|
|
1944
|
+
} else if (["created_at", "updated_at", "commented_at"].includes(key)) {
|
|
1945
|
+
const odateElement = valueElement.find("span.odate");
|
|
1946
|
+
if (odateElement.length > 0) {
|
|
1947
|
+
const timestamp = odateElement.attr("class")?.match(/time_(\d+)/)?.[1];
|
|
1948
|
+
value = timestamp ? new Date(Number.parseInt(timestamp, 10) * 1000) : null;
|
|
1949
|
+
}
|
|
1950
|
+
} else if (["created_by_linked", "updated_by_linked", "commented_by_linked"].includes(key)) {
|
|
1951
|
+
const printuserElement = valueElement.find("span.printuser");
|
|
1952
|
+
if (printuserElement.length > 0) {
|
|
1953
|
+
value = _parseUser(printuserElement);
|
|
1954
|
+
}
|
|
1955
|
+
} else if (["tags", "_tags"].includes(key)) {
|
|
1956
|
+
value = valueElement.text().split(/\s+/).filter(Boolean);
|
|
1957
|
+
} else if (["rating_votes", "comments", "size", "revisions"].includes(key)) {
|
|
1958
|
+
value = Number.parseInt(valueElement.text().trim(), 10) || 0;
|
|
1959
|
+
} else if (key === "rating") {
|
|
1960
|
+
const ratingText = valueElement.text().trim();
|
|
1961
|
+
value = is5StarRating ? Number.parseFloat(ratingText) || 0 : Number.parseInt(ratingText, 10) || 0;
|
|
1962
|
+
} else if (key === "rating_percent") {
|
|
1963
|
+
if (is5StarRating) {
|
|
1964
|
+
value = (Number.parseFloat(valueElement.text().trim()) || 0) / 100;
|
|
1965
|
+
} else {
|
|
1966
|
+
value = null;
|
|
1967
|
+
}
|
|
1968
|
+
} else {
|
|
1969
|
+
value = valueElement.text().trim();
|
|
1970
|
+
}
|
|
1971
|
+
if (key.includes("_linked")) {
|
|
1972
|
+
key = key.replace("_linked", "");
|
|
1973
|
+
} else if (["comments", "children", "revisions"].includes(key)) {
|
|
1974
|
+
key = `${key}_count`;
|
|
1975
|
+
} else if (key === "rating_votes") {
|
|
1976
|
+
key = "votes_count";
|
|
1977
|
+
}
|
|
1978
|
+
pageParams[key] = value;
|
|
1979
|
+
});
|
|
1980
|
+
const tags = Array.isArray(pageParams.tags) ? pageParams.tags : [];
|
|
1981
|
+
const hiddenTags = Array.isArray(pageParams._tags) ? pageParams._tags : [];
|
|
1982
|
+
pageParams.tags = [...tags, ...hiddenTags];
|
|
1983
|
+
const parsed = pageParamsSchema.parse(pageParams);
|
|
1984
|
+
pages.push(new Page({
|
|
1985
|
+
site,
|
|
1986
|
+
fullname: parsed.fullname,
|
|
1987
|
+
name: parsed.name,
|
|
1988
|
+
category: parsed.category,
|
|
1989
|
+
title: parsed.title,
|
|
1990
|
+
childrenCount: parsed.children_count,
|
|
1991
|
+
commentsCount: parsed.comments_count,
|
|
1992
|
+
size: parsed.size,
|
|
1993
|
+
rating: parsed.rating,
|
|
1994
|
+
votesCount: parsed.votes_count,
|
|
1995
|
+
ratingPercent: parsed.rating_percent,
|
|
1996
|
+
revisionsCount: parsed.revisions_count,
|
|
1997
|
+
parentFullname: parsed.parent_fullname,
|
|
1998
|
+
tags: parsed.tags,
|
|
1999
|
+
createdBy: parsed.created_by,
|
|
2000
|
+
createdAt: parsed.created_at ?? new Date,
|
|
2001
|
+
updatedBy: parsed.updated_by,
|
|
2002
|
+
updatedAt: parsed.updated_at ?? new Date,
|
|
2003
|
+
commentedBy: parsed.commented_by,
|
|
2004
|
+
commentedAt: parsed.commented_at
|
|
2005
|
+
}));
|
|
2006
|
+
});
|
|
2007
|
+
return new PageCollection(site, pages);
|
|
2008
|
+
}
|
|
2009
|
+
static searchPages(site, parseUser2, query = null) {
|
|
2010
|
+
return fromPromise((async () => {
|
|
2011
|
+
const q = query ?? new SearchPagesQuery;
|
|
2012
|
+
const queryDict = q.asDict();
|
|
2013
|
+
const moduleBody = `[[div class="page"]]
|
|
2014
|
+
${DEFAULT_MODULE_BODY.map((key) => `[[span class="set ${key}"]][[span class="name"]] ${key} [[/span]][[span class="value"]] %%${key}%% [[/span]][[/span]]`).join("")}
|
|
2015
|
+
[[/div]]`;
|
|
2016
|
+
const requestBody = {
|
|
2017
|
+
...queryDict,
|
|
2018
|
+
moduleName: "list/ListPagesModule",
|
|
2019
|
+
module_body: moduleBody
|
|
2020
|
+
};
|
|
2021
|
+
const result = await site.amcRequest([requestBody]);
|
|
2022
|
+
if (result.isErr()) {
|
|
2023
|
+
if (result.error.message.includes("not_ok")) {
|
|
2024
|
+
throw new ForbiddenError("Failed to get pages, target site may be private");
|
|
2025
|
+
}
|
|
2026
|
+
throw result.error;
|
|
2027
|
+
}
|
|
2028
|
+
const firstResponse = result.value[0];
|
|
2029
|
+
const body = String(firstResponse?.body ?? "");
|
|
2030
|
+
const $first = cheerio6.load(body);
|
|
2031
|
+
let total = 1;
|
|
2032
|
+
const htmlBodies = [$first];
|
|
2033
|
+
const pagerElement = $first("div.pager");
|
|
2034
|
+
if (pagerElement.length > 0) {
|
|
2035
|
+
const lastPagerElements = $first("div.pager span.target");
|
|
2036
|
+
if (lastPagerElements.length >= 2) {
|
|
2037
|
+
const secondLastPager = $first(lastPagerElements[lastPagerElements.length - 2]);
|
|
2038
|
+
const lastPagerLink = secondLastPager.find("a");
|
|
2039
|
+
if (lastPagerLink.length > 0) {
|
|
2040
|
+
total = Number.parseInt(lastPagerLink.text().trim(), 10) || 1;
|
|
2041
|
+
}
|
|
2042
|
+
}
|
|
2043
|
+
}
|
|
2044
|
+
if (total > 1) {
|
|
2045
|
+
const additionalBodies = [];
|
|
2046
|
+
for (let i = 1;i < total; i++) {
|
|
2047
|
+
additionalBodies.push({
|
|
2048
|
+
...queryDict,
|
|
2049
|
+
moduleName: "list/ListPagesModule",
|
|
2050
|
+
module_body: moduleBody,
|
|
2051
|
+
offset: i * (q.perPage ?? DEFAULT_PER_PAGE)
|
|
2052
|
+
});
|
|
2053
|
+
}
|
|
2054
|
+
const additionalResults = await site.amcRequest(additionalBodies);
|
|
2055
|
+
if (additionalResults.isErr()) {
|
|
2056
|
+
throw additionalResults.error;
|
|
2057
|
+
}
|
|
2058
|
+
for (const response of additionalResults.value) {
|
|
2059
|
+
const respBody = String(response?.body ?? "");
|
|
2060
|
+
htmlBodies.push(cheerio6.load(respBody));
|
|
2061
|
+
}
|
|
2062
|
+
}
|
|
2063
|
+
const pages = [];
|
|
2064
|
+
for (const $html of htmlBodies) {
|
|
2065
|
+
const parsed = PageCollection.parse(site, $html, parseUser2);
|
|
2066
|
+
pages.push(...parsed);
|
|
2067
|
+
}
|
|
2068
|
+
return new PageCollection(site, pages);
|
|
2069
|
+
})(), (error) => {
|
|
2070
|
+
if (error instanceof ForbiddenError || error instanceof NotFoundException) {
|
|
2071
|
+
return error;
|
|
2072
|
+
}
|
|
2073
|
+
return new UnexpectedError(`Failed to search pages: ${String(error)}`);
|
|
2074
|
+
});
|
|
2075
|
+
}
|
|
2076
|
+
static createOrEdit(site, fullname, options = {}) {
|
|
2077
|
+
const loginResult = site.client.requireLogin();
|
|
2078
|
+
if (loginResult.isErr()) {
|
|
2079
|
+
return fromPromise(Promise.reject(loginResult.error), () => new LoginRequiredError("Login required to create/edit page"));
|
|
2080
|
+
}
|
|
2081
|
+
return fromPromise((async () => {
|
|
2082
|
+
const {
|
|
2083
|
+
pageId = null,
|
|
2084
|
+
title = "",
|
|
2085
|
+
source = "",
|
|
2086
|
+
comment = "",
|
|
2087
|
+
forceEdit = false,
|
|
2088
|
+
raiseOnExists = false
|
|
2089
|
+
} = options;
|
|
2090
|
+
const lockRequestBody = {
|
|
2091
|
+
mode: "page",
|
|
2092
|
+
wiki_page: fullname,
|
|
2093
|
+
moduleName: "edit/PageEditModule"
|
|
2094
|
+
};
|
|
2095
|
+
if (forceEdit) {
|
|
2096
|
+
lockRequestBody.force_lock = "yes";
|
|
2097
|
+
}
|
|
2098
|
+
const lockResult = await site.amcRequest([lockRequestBody]);
|
|
2099
|
+
if (lockResult.isErr()) {
|
|
2100
|
+
throw lockResult.error;
|
|
2101
|
+
}
|
|
2102
|
+
const lockResponse = lockResult.value[0];
|
|
2103
|
+
if (lockResponse?.locked || lockResponse?.other_locks) {
|
|
2104
|
+
throw new UnexpectedError(`Page ${fullname} is locked or other locks exist`);
|
|
2105
|
+
}
|
|
2106
|
+
const isExist = "page_revision_id" in (lockResponse ?? {});
|
|
2107
|
+
if (raiseOnExists && isExist) {
|
|
2108
|
+
throw new TargetExistsError(`Page ${fullname} already exists`);
|
|
2109
|
+
}
|
|
2110
|
+
if (isExist && pageId === null) {
|
|
2111
|
+
throw new UnexpectedError("page_id must be specified when editing existing page");
|
|
2112
|
+
}
|
|
2113
|
+
const lockId = String(lockResponse?.lock_id ?? "");
|
|
2114
|
+
const lockSecret = String(lockResponse?.lock_secret ?? "");
|
|
2115
|
+
const pageRevisionId = String(lockResponse?.page_revision_id ?? "");
|
|
2116
|
+
const editRequestBody = {
|
|
2117
|
+
action: "WikiPageAction",
|
|
2118
|
+
event: "savePage",
|
|
2119
|
+
moduleName: "Empty",
|
|
2120
|
+
mode: "page",
|
|
2121
|
+
lock_id: lockId,
|
|
2122
|
+
lock_secret: lockSecret,
|
|
2123
|
+
revision_id: pageRevisionId,
|
|
2124
|
+
wiki_page: fullname,
|
|
2125
|
+
page_id: pageId ?? "",
|
|
2126
|
+
title,
|
|
2127
|
+
source,
|
|
2128
|
+
comments: comment
|
|
2129
|
+
};
|
|
2130
|
+
const editResult = await site.amcRequest([editRequestBody]);
|
|
2131
|
+
if (editResult.isErr()) {
|
|
2132
|
+
throw editResult.error;
|
|
2133
|
+
}
|
|
2134
|
+
})(), (error) => {
|
|
2135
|
+
if (error instanceof TargetExistsError || error instanceof LoginRequiredError) {
|
|
2136
|
+
return error;
|
|
2137
|
+
}
|
|
2138
|
+
return new UnexpectedError(`Failed to create/edit page: ${String(error)}`);
|
|
2139
|
+
});
|
|
2140
|
+
}
|
|
2141
|
+
}
|
|
2142
|
+
// src/module/page/site-change.ts
|
|
2143
|
+
import * as cheerio7 from "cheerio";
|
|
2144
|
+
class SiteChange {
|
|
2145
|
+
site;
|
|
2146
|
+
pageFullname;
|
|
2147
|
+
pageTitle;
|
|
2148
|
+
revisionNo;
|
|
2149
|
+
changedBy;
|
|
2150
|
+
changedAt;
|
|
2151
|
+
flags;
|
|
2152
|
+
comment;
|
|
2153
|
+
constructor(data) {
|
|
2154
|
+
this.site = data.site;
|
|
2155
|
+
this.pageFullname = data.pageFullname;
|
|
2156
|
+
this.pageTitle = data.pageTitle;
|
|
2157
|
+
this.revisionNo = data.revisionNo;
|
|
2158
|
+
this.changedBy = data.changedBy;
|
|
2159
|
+
this.changedAt = data.changedAt;
|
|
2160
|
+
this.flags = data.flags;
|
|
2161
|
+
this.comment = data.comment;
|
|
2162
|
+
}
|
|
2163
|
+
getPageUrl() {
|
|
2164
|
+
return `${this.site.getBaseUrl()}/${this.pageFullname}`;
|
|
2165
|
+
}
|
|
2166
|
+
toString() {
|
|
2167
|
+
return `SiteChange(page=${this.pageFullname}, rev=${this.revisionNo}, by=${this.changedBy})`;
|
|
2168
|
+
}
|
|
2169
|
+
}
|
|
2170
|
+
|
|
2171
|
+
class SiteChangeCollection extends Array {
|
|
2172
|
+
site;
|
|
2173
|
+
constructor(site, changes) {
|
|
2174
|
+
super();
|
|
2175
|
+
this.site = site;
|
|
2176
|
+
if (changes) {
|
|
2177
|
+
this.push(...changes);
|
|
2178
|
+
}
|
|
2179
|
+
}
|
|
2180
|
+
static acquire(site, options) {
|
|
2181
|
+
const perPage = options?.perPage ?? 20;
|
|
2182
|
+
const page = options?.page ?? 1;
|
|
2183
|
+
const limit = options?.limit;
|
|
2184
|
+
return fromPromise((async () => {
|
|
2185
|
+
const result = await site.amcRequest([
|
|
2186
|
+
{
|
|
2187
|
+
moduleName: "changes/SiteChangesListModule",
|
|
2188
|
+
perpage: perPage,
|
|
2189
|
+
page
|
|
2190
|
+
}
|
|
2191
|
+
]);
|
|
2192
|
+
if (result.isErr()) {
|
|
2193
|
+
throw result.error;
|
|
2194
|
+
}
|
|
2195
|
+
const response = result.value[0];
|
|
2196
|
+
if (!response) {
|
|
2197
|
+
throw new NoElementError("Empty response");
|
|
2198
|
+
}
|
|
2199
|
+
const html = String(response.body ?? "");
|
|
2200
|
+
const $ = cheerio7.load(html);
|
|
2201
|
+
const changes = [];
|
|
2202
|
+
$("table.wiki-content-table tr").each((_i, elem) => {
|
|
2203
|
+
const $row = $(elem);
|
|
2204
|
+
const $cells = $row.find("td");
|
|
2205
|
+
if ($cells.length < 4)
|
|
2206
|
+
return;
|
|
2207
|
+
const pageLink = $($cells[0]).find("a");
|
|
2208
|
+
const href = pageLink.attr("href") ?? "";
|
|
2209
|
+
const pageFullname = href.replace(/^\//, "").split("/")[0] ?? "";
|
|
2210
|
+
const pageTitle = pageLink.text().trim();
|
|
2211
|
+
const revText = $($cells[1]).text().trim();
|
|
2212
|
+
const revMatch = revText.match(/(\d+)/);
|
|
2213
|
+
const revisionNo = revMatch?.[1] ? Number.parseInt(revMatch[1], 10) : 0;
|
|
2214
|
+
const flagsCell = $($cells[2]);
|
|
2215
|
+
const flags = [];
|
|
2216
|
+
flagsCell.find("span").each((_j, flagElem) => {
|
|
2217
|
+
const flagClass = $(flagElem).attr("class") ?? "";
|
|
2218
|
+
if (flagClass.includes("spantip")) {
|
|
2219
|
+
const title = $(flagElem).attr("title") ?? "";
|
|
2220
|
+
if (title)
|
|
2221
|
+
flags.push(title);
|
|
2222
|
+
}
|
|
2223
|
+
});
|
|
2224
|
+
const infoCell = $($cells[3]);
|
|
2225
|
+
const userElem = infoCell.find("span.printuser");
|
|
2226
|
+
const changedBy = userElem.length > 0 ? parseUser(site.client, userElem) : null;
|
|
2227
|
+
const odateElem = infoCell.find("span.odate");
|
|
2228
|
+
const changedAt = odateElem.length > 0 ? parseOdate(odateElem) : null;
|
|
2229
|
+
const commentElem = infoCell.find("span.comments");
|
|
2230
|
+
const comment = commentElem.text().trim().replace(/^[""]|[""]$/g, "");
|
|
2231
|
+
changes.push(new SiteChange({
|
|
2232
|
+
site,
|
|
2233
|
+
pageFullname,
|
|
2234
|
+
pageTitle,
|
|
2235
|
+
revisionNo,
|
|
2236
|
+
changedBy,
|
|
2237
|
+
changedAt,
|
|
2238
|
+
flags,
|
|
2239
|
+
comment
|
|
2240
|
+
}));
|
|
2241
|
+
});
|
|
2242
|
+
const limitedChanges = limit !== undefined ? changes.slice(0, limit) : changes;
|
|
2243
|
+
return new SiteChangeCollection(site, limitedChanges);
|
|
2244
|
+
})(), (error) => {
|
|
2245
|
+
if (error instanceof NoElementError) {
|
|
2246
|
+
return error;
|
|
2247
|
+
}
|
|
2248
|
+
return new UnexpectedError(`Failed to acquire site changes: ${String(error)}`);
|
|
2249
|
+
});
|
|
2250
|
+
}
|
|
2251
|
+
}
|
|
2252
|
+
// src/module/site/accessors/page-accessor.ts
|
|
2253
|
+
class PageAccessor {
|
|
2254
|
+
site;
|
|
2255
|
+
constructor(site) {
|
|
2256
|
+
this.site = site;
|
|
2257
|
+
}
|
|
2258
|
+
get(unixName) {
|
|
2259
|
+
return fromPromise((async () => {
|
|
2260
|
+
const query = new SearchPagesQuery({ fullname: unixName });
|
|
2261
|
+
const userParser = parseUser.bind(null, this.site.client);
|
|
2262
|
+
const result = await PageCollection.searchPages(this.site, userParser, query);
|
|
2263
|
+
if (result.isErr()) {
|
|
2264
|
+
throw result.error;
|
|
2265
|
+
}
|
|
2266
|
+
return result.value.length > 0 ? result.value[0] ?? null : null;
|
|
2267
|
+
})(), (error) => {
|
|
2268
|
+
if (error instanceof NoElementError)
|
|
2269
|
+
return error;
|
|
2270
|
+
return new UnexpectedError(`Failed to get page: ${String(error)}`);
|
|
2271
|
+
});
|
|
2272
|
+
}
|
|
2273
|
+
create(fullname, options = {}) {
|
|
2274
|
+
return PageCollection.createOrEdit(this.site, fullname, {
|
|
2275
|
+
title: options.title,
|
|
2276
|
+
source: options.source,
|
|
2277
|
+
comment: options.comment,
|
|
2278
|
+
forceEdit: options.forceEdit
|
|
2279
|
+
});
|
|
2280
|
+
}
|
|
2281
|
+
}
|
|
2282
|
+
// src/module/site/accessors/pages-accessor.ts
|
|
2283
|
+
class PagesAccessor {
|
|
2284
|
+
site;
|
|
2285
|
+
constructor(site) {
|
|
2286
|
+
this.site = site;
|
|
2287
|
+
}
|
|
2288
|
+
search(params) {
|
|
2289
|
+
return fromPromise((async () => {
|
|
2290
|
+
const query = new SearchPagesQuery(params);
|
|
2291
|
+
const userParser = parseUser.bind(null, this.site.client);
|
|
2292
|
+
const result = await PageCollection.searchPages(this.site, userParser, query);
|
|
2293
|
+
if (result.isErr()) {
|
|
2294
|
+
throw result.error;
|
|
2295
|
+
}
|
|
2296
|
+
return result.value;
|
|
2297
|
+
})(), (error) => new UnexpectedError(`Failed to search pages: ${String(error)}`));
|
|
2298
|
+
}
|
|
2299
|
+
all() {
|
|
2300
|
+
return this.search({});
|
|
2301
|
+
}
|
|
2302
|
+
getRecentChanges(options) {
|
|
2303
|
+
return SiteChangeCollection.acquire(this.site, options);
|
|
2304
|
+
}
|
|
2305
|
+
}
|
|
2306
|
+
// src/module/site/site.ts
|
|
2307
|
+
import * as cheerio8 from "cheerio";
|
|
2308
|
+
class Site {
|
|
2309
|
+
client;
|
|
2310
|
+
id;
|
|
2311
|
+
title;
|
|
2312
|
+
unixName;
|
|
2313
|
+
domain;
|
|
2314
|
+
sslSupported;
|
|
2315
|
+
_page = null;
|
|
2316
|
+
_pages = null;
|
|
2317
|
+
_forum = null;
|
|
2318
|
+
_member = null;
|
|
2319
|
+
constructor(client, data) {
|
|
2320
|
+
this.client = client;
|
|
2321
|
+
this.id = data.id;
|
|
2322
|
+
this.title = data.title;
|
|
2323
|
+
this.unixName = data.unixName;
|
|
2324
|
+
this.domain = data.domain;
|
|
2325
|
+
this.sslSupported = data.sslSupported;
|
|
2326
|
+
}
|
|
2327
|
+
get page() {
|
|
2328
|
+
if (!this._page) {
|
|
2329
|
+
this._page = new PageAccessor(this);
|
|
2330
|
+
}
|
|
2331
|
+
return this._page;
|
|
2332
|
+
}
|
|
2333
|
+
get pages() {
|
|
2334
|
+
if (!this._pages) {
|
|
2335
|
+
this._pages = new PagesAccessor(this);
|
|
2336
|
+
}
|
|
2337
|
+
return this._pages;
|
|
2338
|
+
}
|
|
2339
|
+
get forum() {
|
|
2340
|
+
if (!this._forum) {
|
|
2341
|
+
this._forum = new ForumAccessor(this);
|
|
2342
|
+
}
|
|
2343
|
+
return this._forum;
|
|
2344
|
+
}
|
|
2345
|
+
get member() {
|
|
2346
|
+
if (!this._member) {
|
|
2347
|
+
this._member = new MemberAccessor(this);
|
|
2348
|
+
}
|
|
2349
|
+
return this._member;
|
|
2350
|
+
}
|
|
2351
|
+
getBaseUrl() {
|
|
2352
|
+
const protocol = this.sslSupported ? "https" : "http";
|
|
2353
|
+
return `${protocol}://${this.domain}`;
|
|
2354
|
+
}
|
|
2355
|
+
amcRequest(bodies) {
|
|
2356
|
+
return this.client.amcClient.request(bodies, this.unixName, this.sslSupported);
|
|
2357
|
+
}
|
|
2358
|
+
amcRequestSingle(body) {
|
|
2359
|
+
return fromPromise((async () => {
|
|
2360
|
+
const result = await this.amcRequest([body]);
|
|
2361
|
+
if (result.isErr()) {
|
|
2362
|
+
throw result.error;
|
|
2363
|
+
}
|
|
2364
|
+
const response = result.value[0];
|
|
2365
|
+
if (!response) {
|
|
2366
|
+
throw new UnexpectedError("AMC request returned empty response");
|
|
2367
|
+
}
|
|
2368
|
+
return response;
|
|
2369
|
+
})(), (error) => {
|
|
2370
|
+
if (error instanceof UnexpectedError) {
|
|
2371
|
+
return error;
|
|
2372
|
+
}
|
|
2373
|
+
return new UnexpectedError(`AMC request failed: ${String(error)}`);
|
|
2374
|
+
});
|
|
2375
|
+
}
|
|
2376
|
+
static fromUnixName(client, unixName) {
|
|
2377
|
+
return fromPromise((async () => {
|
|
2378
|
+
const url = `https://${unixName}.wikidot.com`;
|
|
2379
|
+
const response = await fetch(url, {
|
|
2380
|
+
headers: client.amcClient.header.getHeaders()
|
|
2381
|
+
});
|
|
2382
|
+
if (!response.ok) {
|
|
2383
|
+
if (response.status === 404) {
|
|
2384
|
+
throw new NotFoundException(`Site not found: ${unixName}`);
|
|
2385
|
+
}
|
|
2386
|
+
throw new UnexpectedError(`Failed to fetch site: ${response.status}`);
|
|
2387
|
+
}
|
|
2388
|
+
const html = await response.text();
|
|
2389
|
+
const $ = cheerio8.load(html);
|
|
2390
|
+
const scripts = $("script").toArray();
|
|
2391
|
+
let siteId = null;
|
|
2392
|
+
let siteUnixName = null;
|
|
2393
|
+
let domain = null;
|
|
2394
|
+
let title = null;
|
|
2395
|
+
for (const script of scripts) {
|
|
2396
|
+
const content = $(script).html();
|
|
2397
|
+
if (!content || !content.includes("WIKIREQUEST")) {
|
|
2398
|
+
continue;
|
|
2399
|
+
}
|
|
2400
|
+
const siteIdMatch = content.match(/WIKIREQUEST\.info\.siteId\s*=\s*(\d+)/);
|
|
2401
|
+
if (siteIdMatch?.[1]) {
|
|
2402
|
+
siteId = Number.parseInt(siteIdMatch[1], 10);
|
|
2403
|
+
}
|
|
2404
|
+
const siteUnixNameMatch = content.match(/WIKIREQUEST\.info\.siteUnixName\s*=\s*["']([^"']+)["']/);
|
|
2405
|
+
if (siteUnixNameMatch?.[1]) {
|
|
2406
|
+
siteUnixName = siteUnixNameMatch[1];
|
|
2407
|
+
}
|
|
2408
|
+
const domainMatch = content.match(/WIKIREQUEST\.info\.domain\s*=\s*["']([^"']+)["']/);
|
|
2409
|
+
if (domainMatch?.[1]) {
|
|
2410
|
+
domain = domainMatch[1];
|
|
2411
|
+
}
|
|
2412
|
+
}
|
|
2413
|
+
title = $("title").text().trim() || null;
|
|
2414
|
+
if (title?.endsWith(" - Wikidot")) {
|
|
2415
|
+
title = title.slice(0, -10).trim();
|
|
2416
|
+
}
|
|
2417
|
+
if (siteId === null) {
|
|
2418
|
+
throw new NoElementError("Site ID not found in WIKIREQUEST");
|
|
2419
|
+
}
|
|
2420
|
+
if (siteUnixName === null) {
|
|
2421
|
+
siteUnixName = unixName;
|
|
2422
|
+
}
|
|
2423
|
+
if (domain === null) {
|
|
2424
|
+
domain = `${unixName}.wikidot.com`;
|
|
2425
|
+
}
|
|
2426
|
+
if (title === null) {
|
|
2427
|
+
title = unixName;
|
|
2428
|
+
}
|
|
2429
|
+
const sslSupported = true;
|
|
2430
|
+
return new Site(client, {
|
|
2431
|
+
id: siteId,
|
|
2432
|
+
title,
|
|
2433
|
+
unixName: siteUnixName,
|
|
2434
|
+
domain,
|
|
2435
|
+
sslSupported
|
|
2436
|
+
});
|
|
2437
|
+
})(), (error) => {
|
|
2438
|
+
if (error instanceof NotFoundException || error instanceof NoElementError) {
|
|
2439
|
+
return error;
|
|
2440
|
+
}
|
|
2441
|
+
return new UnexpectedError(`Failed to get site: ${String(error)}`);
|
|
2442
|
+
});
|
|
2443
|
+
}
|
|
2444
|
+
toString() {
|
|
2445
|
+
return `Site(id=${this.id}, unixName=${this.unixName}, title=${this.title})`;
|
|
2446
|
+
}
|
|
2447
|
+
}
|
|
2448
|
+
// src/module/client/accessors/site-accessor.ts
|
|
2449
|
+
class SiteAccessor {
|
|
2450
|
+
client;
|
|
2451
|
+
constructor(client) {
|
|
2452
|
+
this.client = client;
|
|
2453
|
+
}
|
|
2454
|
+
get(unixName) {
|
|
2455
|
+
return Site.fromUnixName(this.client, unixName);
|
|
2456
|
+
}
|
|
2457
|
+
}
|
|
2458
|
+
// src/module/client/accessors/user-accessor.ts
|
|
2459
|
+
class UserAccessor {
|
|
2460
|
+
client;
|
|
2461
|
+
constructor(client) {
|
|
2462
|
+
this.client = client;
|
|
2463
|
+
}
|
|
2464
|
+
get(name, options = {}) {
|
|
2465
|
+
const { raiseWhenNotFound = false } = options;
|
|
2466
|
+
return User.fromName(this.client, name).andThen((user) => {
|
|
2467
|
+
if (user === null && raiseWhenNotFound) {
|
|
2468
|
+
return wdErrAsync(new NotFoundException(`User not found: ${name}`));
|
|
2469
|
+
}
|
|
2470
|
+
return wdOkAsync(user);
|
|
2471
|
+
});
|
|
2472
|
+
}
|
|
2473
|
+
getMany(names, options = {}) {
|
|
2474
|
+
const { raiseWhenNotFound = false } = options;
|
|
2475
|
+
return User.fromNames(this.client, names).andThen((collection) => {
|
|
2476
|
+
if (raiseWhenNotFound) {
|
|
2477
|
+
const notFoundNames = [];
|
|
2478
|
+
for (let i = 0;i < names.length; i++) {
|
|
2479
|
+
const name = names[i];
|
|
2480
|
+
if (collection[i] === null && name !== undefined) {
|
|
2481
|
+
notFoundNames.push(name);
|
|
2482
|
+
}
|
|
2483
|
+
}
|
|
2484
|
+
if (notFoundNames.length > 0) {
|
|
2485
|
+
return wdErrAsync(new NotFoundException(`Users not found: ${notFoundNames.join(", ")}`));
|
|
2486
|
+
}
|
|
2487
|
+
}
|
|
2488
|
+
return wdOkAsync(collection);
|
|
2489
|
+
});
|
|
2490
|
+
}
|
|
2491
|
+
}
|
|
2492
|
+
// src/module/client/client.ts
|
|
2493
|
+
class Client {
|
|
2494
|
+
amcClient;
|
|
2495
|
+
domain;
|
|
2496
|
+
user;
|
|
2497
|
+
site;
|
|
2498
|
+
privateMessage;
|
|
2499
|
+
_username;
|
|
2500
|
+
_me = null;
|
|
2501
|
+
constructor(amcClient, domain, username = null) {
|
|
2502
|
+
this.amcClient = amcClient;
|
|
2503
|
+
this.domain = domain;
|
|
2504
|
+
this._username = username;
|
|
2505
|
+
this.user = new UserAccessor(this);
|
|
2506
|
+
this.site = new SiteAccessor(this);
|
|
2507
|
+
this.privateMessage = new PrivateMessageAccessor(this);
|
|
2508
|
+
}
|
|
2509
|
+
get username() {
|
|
2510
|
+
return this._username;
|
|
2511
|
+
}
|
|
2512
|
+
get me() {
|
|
2513
|
+
return this._me;
|
|
2514
|
+
}
|
|
2515
|
+
static create(options = {}) {
|
|
2516
|
+
const { username, password, domain = "wikidot.com", amcConfig = {} } = options;
|
|
2517
|
+
const amcClient = new AMCClient(amcConfig, domain);
|
|
2518
|
+
if (username && password) {
|
|
2519
|
+
return fromPromise((async () => {
|
|
2520
|
+
const client = new Client(amcClient, domain, username);
|
|
2521
|
+
const loginResult = await login(client, username, password);
|
|
2522
|
+
if (loginResult.isErr()) {
|
|
2523
|
+
throw loginResult.error;
|
|
2524
|
+
}
|
|
2525
|
+
const { User: User2 } = await import("./shared/index-kka6e8cb.js");
|
|
2526
|
+
const userResult = await User2.fromName(client, username);
|
|
2527
|
+
if (userResult.isOk() && userResult.value) {
|
|
2528
|
+
client._me = userResult.value;
|
|
2529
|
+
}
|
|
2530
|
+
return client;
|
|
2531
|
+
})(), (error) => {
|
|
2532
|
+
if (error instanceof LoginRequiredError) {
|
|
2533
|
+
return error;
|
|
2534
|
+
}
|
|
2535
|
+
return new LoginRequiredError(`Failed to create client: ${String(error)}`);
|
|
2536
|
+
});
|
|
2537
|
+
}
|
|
2538
|
+
return wdOkAsync(new Client(amcClient, domain));
|
|
2539
|
+
}
|
|
2540
|
+
static createAnonymous(options = {}) {
|
|
2541
|
+
const { domain = "wikidot.com", amcConfig = {} } = options;
|
|
2542
|
+
const amcClient = new AMCClient(amcConfig, domain);
|
|
2543
|
+
return new Client(amcClient, domain);
|
|
2544
|
+
}
|
|
2545
|
+
isLoggedIn() {
|
|
2546
|
+
return this._username !== null;
|
|
2547
|
+
}
|
|
2548
|
+
requireLogin() {
|
|
2549
|
+
if (!this.isLoggedIn()) {
|
|
2550
|
+
return wdErr(new LoginRequiredError);
|
|
2551
|
+
}
|
|
2552
|
+
return wdOk(undefined);
|
|
2553
|
+
}
|
|
2554
|
+
close() {
|
|
2555
|
+
if (this.isLoggedIn()) {
|
|
2556
|
+
return logout(this).map(() => {
|
|
2557
|
+
this._username = null;
|
|
2558
|
+
return;
|
|
2559
|
+
});
|
|
2560
|
+
}
|
|
2561
|
+
return wdOkAsync(undefined);
|
|
2562
|
+
}
|
|
2563
|
+
}
|
|
2564
|
+
export {
|
|
2565
|
+
wdOkAsync,
|
|
2566
|
+
wdOk,
|
|
2567
|
+
wdErrAsync,
|
|
2568
|
+
wdErr,
|
|
2569
|
+
userLookup,
|
|
2570
|
+
setupConsoleHandler,
|
|
2571
|
+
parseUser,
|
|
2572
|
+
parseOdate,
|
|
2573
|
+
pageLookup,
|
|
2574
|
+
nullHandler,
|
|
2575
|
+
memberLookup,
|
|
2576
|
+
maskSensitiveData,
|
|
2577
|
+
logout,
|
|
2578
|
+
login,
|
|
2579
|
+
logger,
|
|
2580
|
+
isSuccessResponse,
|
|
2581
|
+
getLogger,
|
|
2582
|
+
fromPromise,
|
|
2583
|
+
consoleHandler,
|
|
2584
|
+
combineResults,
|
|
2585
|
+
amcResponseSchema,
|
|
2586
|
+
WikidotUser,
|
|
2587
|
+
WikidotStatusError,
|
|
2588
|
+
WikidotError,
|
|
2589
|
+
UserCollection,
|
|
2590
|
+
UserAccessor,
|
|
2591
|
+
User,
|
|
2592
|
+
UnexpectedError,
|
|
2593
|
+
TargetExistsError,
|
|
2594
|
+
TargetError,
|
|
2595
|
+
SiteMember,
|
|
2596
|
+
SiteChangeCollection,
|
|
2597
|
+
SiteChange,
|
|
2598
|
+
SiteApplication,
|
|
2599
|
+
SiteAccessor,
|
|
2600
|
+
Site,
|
|
2601
|
+
SessionError,
|
|
2602
|
+
SessionCreateError,
|
|
2603
|
+
SearchPagesQuery,
|
|
2604
|
+
ResponseDataError,
|
|
2605
|
+
QuickModule,
|
|
2606
|
+
PrivateMessageSentBox,
|
|
2607
|
+
PrivateMessageInbox,
|
|
2608
|
+
PrivateMessageCollection,
|
|
2609
|
+
PrivateMessageAccessor,
|
|
2610
|
+
PrivateMessage,
|
|
2611
|
+
PagesAccessor,
|
|
2612
|
+
PageVoteCollection,
|
|
2613
|
+
PageVote,
|
|
2614
|
+
PageSource,
|
|
2615
|
+
PageRevisionCollection,
|
|
2616
|
+
PageRevision,
|
|
2617
|
+
PageMetaCollection,
|
|
2618
|
+
PageMeta,
|
|
2619
|
+
PageFileCollection,
|
|
2620
|
+
PageFile,
|
|
2621
|
+
PageCollection,
|
|
2622
|
+
PageAccessor,
|
|
2623
|
+
Page,
|
|
2624
|
+
NotFoundException,
|
|
2625
|
+
NoElementError,
|
|
2626
|
+
MemberAccessor,
|
|
2627
|
+
LoginRequiredError,
|
|
2628
|
+
Logger,
|
|
2629
|
+
GuestUser,
|
|
2630
|
+
ForumThreadCollection,
|
|
2631
|
+
ForumThread,
|
|
2632
|
+
ForumPostCollection,
|
|
2633
|
+
ForumPost,
|
|
2634
|
+
ForumCategoryCollection,
|
|
2635
|
+
ForumCategory,
|
|
2636
|
+
ForumAccessor,
|
|
2637
|
+
ForbiddenError,
|
|
2638
|
+
DeletedUser,
|
|
2639
|
+
DEFAULT_PER_PAGE,
|
|
2640
|
+
DEFAULT_MODULE_BODY,
|
|
2641
|
+
DEFAULT_AMC_CONFIG,
|
|
2642
|
+
Client,
|
|
2643
|
+
AnonymousUser,
|
|
2644
|
+
AMCHttpError,
|
|
2645
|
+
AMCHeader,
|
|
2646
|
+
AMCError,
|
|
2647
|
+
AMCClient
|
|
2648
|
+
};
|
|
2649
|
+
|
|
2650
|
+
//# debugId=1A0AF3AA4D5CAB4D64756E2164756E21
|
|
2651
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["src/connector/amc-client.ts", "src/connector/amc-config.ts", "src/connector/amc-header.ts", "src/connector/amc-types.ts", "src/connector/auth.ts", "src/module/private-message/private-message.ts", "src/module/client/accessors/pm-accessor.ts", "src/module/site/accessors/forum-accessor.ts", "src/util/quick-module.ts", "src/module/site/site-application.ts", "src/module/site/site-member.ts", "src/module/site/accessors/member-accessor.ts", "src/module/page/page.ts", "src/module/page/page-file.ts", "src/module/page/page-meta.ts", "src/module/page/page-source.ts", "src/module/page/page-vote.ts", "src/module/page/search-query.ts", "src/module/page/site-change.ts", "src/module/site/accessors/page-accessor.ts", "src/module/site/accessors/pages-accessor.ts", "src/module/site/site.ts", "src/module/client/accessors/site-accessor.ts", "src/module/client/accessors/user-accessor.ts", "src/module/client/client.ts"],
  "sourcesContent": [
    "import ky, { type KyInstance } from 'ky';\nimport pLimit, { type LimitFunction } from 'p-limit';\nimport {\n  AMCHttpError,\n  ForbiddenError,\n  NotFoundException,\n  ResponseDataError,\n  UnexpectedError,\n  WikidotError,\n  WikidotStatusError,\n} from '../common/errors';\nimport { fromPromise, type WikidotResultAsync, wdErrAsync, wdOkAsync } from '../common/types';\nimport {\n  type AMCConfig,\n  DEFAULT_AMC_CONFIG,\n  DEFAULT_HTTP_STATUS_CODE,\n  WIKIDOT_TOKEN7,\n} from './amc-config';\nimport { AMCHeader } from './amc-header';\nimport { type AMCRequestBody, type AMCResponse, amcResponseSchema } from './amc-types';\n\n/**\n * 機密情報をマスクする（ログ出力用）\n * @param body - マスク対象のリクエストボディ\n * @returns マスクされたボディ\n */\nexport function maskSensitiveData(body: AMCRequestBody): Record<string, unknown> {\n  const masked = { ...body };\n  const sensitiveKeys = ['password', 'login', 'WIKIDOT_SESSION_ID', 'wikidot_token7'];\n  for (const key of sensitiveKeys) {\n    if (key in masked) {\n      masked[key] = '***MASKED***';\n    }\n  }\n  return masked;\n}\n\n/**\n * 指数バックオフ間隔を計算する（ジッター付き）\n * @param retryCount - 現在のリトライ回数（1から開始）\n * @param baseInterval - 基本間隔（ミリ秒）\n * @param backoffFactor - バックオフ係数\n * @param maxBackoff - 最大バックオフ間隔（ミリ秒）\n * @returns 計算されたバックオフ間隔（ミリ秒）\n */\nfunction calculateBackoff(\n  retryCount: number,\n  baseInterval: number,\n  backoffFactor: number,\n  maxBackoff: number\n): number {\n  const backoff = baseInterval * backoffFactor ** (retryCount - 1);\n  const jitter = Math.random() * backoff * 0.1;\n  return Math.min(backoff + jitter, maxBackoff);\n}\n\n/**\n * 指定時間待機する\n * @param ms - 待機時間（ミリ秒）\n */\nfunction sleep(ms: number): Promise<void> {\n  return new Promise((resolve) => setTimeout(resolve, ms));\n}\n\n/**\n * AMCリクエストオプション\n */\nexport interface AMCRequestOptions {\n  /** サイト名（デフォルト: www） */\n  siteName?: string;\n  /** SSL対応（省略時は自動検出） */\n  sslSupported?: boolean;\n  /** エラーを例外として投げずに結果に含める（デフォルト: false） */\n  returnExceptions?: boolean;\n}\n\n/**\n * Ajax Module Connectorクライアント\n * Wikidot AMCエンドポイントへのリクエストを管理する\n */\nexport class AMCClient {\n  /** kyインスタンス */\n  private readonly ky: KyInstance;\n\n  /** 並列リクエスト制限 */\n  private readonly limit: LimitFunction;\n\n  /** ヘッダー管理 */\n  public readonly header: AMCHeader;\n\n  /** 設定 */\n  public readonly config: AMCConfig;\n\n  /** ベースドメイン */\n  public readonly domain: string;\n\n  /** SSL対応状況のキャッシュ */\n  private sslCache: Map<string, boolean> = new Map();\n\n  /**\n   * @param config - AMC設定（省略時はデフォルト値）\n   * @param domain - ベースドメイン（デフォルト: wikidot.com）\n   */\n  constructor(config: Partial<AMCConfig> = {}, domain = 'wikidot.com') {\n    this.config = { ...DEFAULT_AMC_CONFIG, ...config };\n    this.domain = domain;\n    this.header = new AMCHeader();\n    this.limit = pLimit(this.config.semaphoreLimit);\n\n    this.ky = ky.create({\n      timeout: this.config.timeout,\n      retry: 0, // 手動でリトライを制御\n    });\n\n    // wwwは常にSSL対応\n    this.sslCache.set('www', true);\n  }\n\n  /**\n   * サイトの存在とSSL対応状況を確認する\n   * @param siteName - サイト名\n   * @returns SSL対応状況（true: HTTPS、false: HTTP）\n   */\n  checkSiteSSL(siteName: string): WikidotResultAsync<boolean> {\n    // キャッシュに存在すればそれを返す\n    const cached = this.sslCache.get(siteName);\n    if (cached !== undefined) {\n      return wdOkAsync(cached);\n    }\n\n    // wwwは常にSSL対応\n    if (siteName === 'www') {\n      return wdOkAsync(true);\n    }\n\n    return fromPromise(\n      (async () => {\n        const response = await fetch(`http://${siteName}.${this.domain}`, {\n          method: 'GET',\n          redirect: 'manual',\n        });\n\n        // 404の場合はサイトが存在しない\n        if (response.status === 404) {\n          throw new NotFoundException(`Site is not found: ${siteName}.${this.domain}`);\n        }\n\n        // 301リダイレクトでhttpsに向かう場合はSSL対応\n        const isSSL =\n          response.status === 301 && response.headers.get('Location')?.startsWith('https') === true;\n\n        // キャッシュに保存\n        this.sslCache.set(siteName, isSSL);\n        return isSSL;\n      })(),\n      (error) => {\n        if (error instanceof WikidotError) {\n          return error;\n        }\n        return new UnexpectedError(`Failed to check SSL for ${siteName}: ${String(error)}`);\n      }\n    );\n  }\n\n  /**\n   * AMCリクエストを実行する\n   * @param bodies - リクエストボディ配列\n   * @param siteName - サイト名（省略時はwww）\n   * @param sslSupported - SSL対応（省略時は自動検出）\n   * @returns レスポンス配列\n   */\n  request(\n    bodies: AMCRequestBody[],\n    siteName = 'www',\n    sslSupported?: boolean\n  ): WikidotResultAsync<AMCResponse[]> {\n    return this.requestWithOptions(bodies, {\n      siteName,\n      sslSupported,\n      returnExceptions: false,\n    }) as WikidotResultAsync<AMCResponse[]>;\n  }\n\n  /**\n   * AMCリクエストを実行する（オプション指定版）\n   * @param bodies - リクエストボディ配列\n   * @param options - リクエストオプション\n   * @returns レスポンス配列（returnExceptionsがtrueの場合はエラーも含む）\n   */\n  requestWithOptions(\n    bodies: AMCRequestBody[],\n    options: AMCRequestOptions = {}\n  ): WikidotResultAsync<(AMCResponse | WikidotError)[]> {\n    const { siteName = 'www', sslSupported, returnExceptions = false } = options;\n\n    return fromPromise(\n      (async () => {\n        // SSL対応状況を取得\n        let ssl = sslSupported;\n        if (ssl === undefined) {\n          const sslResult = await this.checkSiteSSL(siteName);\n          if (sslResult.isErr()) {\n            throw sslResult.error;\n          }\n          ssl = sslResult.value;\n        }\n\n        const protocol = ssl ? 'https' : 'http';\n        const url = `${protocol}://${siteName}.${this.domain}/ajax-module-connector.php`;\n\n        // 並列でリクエストを実行\n        const results = await Promise.all(\n          bodies.map((body) => this.limit(() => this.singleRequest(body, url)))\n        );\n\n        if (returnExceptions) {\n          // エラーも含めてすべての結果を返す\n          return results.map((r) => {\n            if (r.isOk()) {\n              return r.value;\n            }\n            return r.error;\n          });\n        }\n\n        // エラーがあれば最初のエラーをスロー\n        const firstError = results.find((r) => r.isErr());\n        if (firstError?.isErr()) {\n          throw firstError.error;\n        }\n\n        return results.map((r) => {\n          if (r.isOk()) {\n            return r.value;\n          }\n          throw new UnexpectedError('Unexpected error in result processing');\n        });\n      })(),\n      (error) => {\n        if (error instanceof WikidotError) {\n          return error;\n        }\n        return new UnexpectedError(`AMC request failed: ${String(error)}`);\n      }\n    );\n  }\n\n  /**\n   * 単一リクエストを実行する内部メソッド\n   * @param body - リクエストボディ\n   * @param url - リクエストURL\n   * @returns レスポンス\n   */\n  private async singleRequest(\n    body: AMCRequestBody,\n    url: string\n  ): Promise<WikidotResultAsync<AMCResponse>> {\n    let retryCount = 0;\n\n    while (true) {\n      try {\n        // wikidot_token7を追加\n        const requestBody = { ...body, wikidot_token7: WIKIDOT_TOKEN7 };\n\n        // URLエンコードされたボディを作成\n        const formData = new URLSearchParams();\n        for (const [key, value] of Object.entries(requestBody)) {\n          if (value !== undefined) {\n            formData.append(key, String(value));\n          }\n        }\n\n        const response = await this.ky.post(url, {\n          headers: this.header.getHeaders(),\n          body: formData.toString(),\n        });\n\n        // JSONとしてパース\n        let responseData: unknown;\n        try {\n          responseData = await response.json();\n        } catch {\n          return wdErrAsync(\n            new ResponseDataError(`AMC responded with non-JSON data: ${await response.text()}`)\n          );\n        }\n\n        // zodでバリデーション\n        const parseResult = amcResponseSchema.safeParse(responseData);\n        if (!parseResult.success) {\n          return wdErrAsync(\n            new ResponseDataError(`Invalid AMC response format: ${parseResult.error.message}`)\n          );\n        }\n\n        const amcResponse = parseResult.data;\n\n        // try_againの場合はリトライ\n        if (amcResponse.status === 'try_again') {\n          retryCount++;\n          if (retryCount >= this.config.retryLimit) {\n            return wdErrAsync(new WikidotStatusError('AMC responded with try_again', 'try_again'));\n          }\n          const backoff = calculateBackoff(\n            retryCount,\n            this.config.retryInterval,\n            this.config.backoffFactor,\n            this.config.maxBackoff\n          );\n          await sleep(backoff);\n          continue;\n        }\n\n        // no_permissionの場合はForbiddenError\n        if (amcResponse.status === 'no_permission') {\n          const targetStr = body.moduleName\n            ? `moduleName: ${body.moduleName}`\n            : body.action\n              ? `action: ${body.action}/${body.event ?? ''}`\n              : 'unknown';\n          return wdErrAsync(\n            new ForbiddenError(\n              `Your account has no permission to perform this action: ${targetStr}`\n            )\n          );\n        }\n\n        // okでない場合はエラー\n        if (amcResponse.status !== 'ok') {\n          return wdErrAsync(\n            new WikidotStatusError(\n              `AMC responded with error status: \"${amcResponse.status}\"`,\n              amcResponse.status\n            )\n          );\n        }\n\n        return wdOkAsync(amcResponse);\n      } catch (error) {\n        // HTTPエラーの場合はリトライ\n        retryCount++;\n        if (retryCount >= this.config.retryLimit) {\n          const statusCode =\n            error instanceof Error && 'response' in error\n              ? ((error as { response?: { status?: number } }).response?.status ??\n                DEFAULT_HTTP_STATUS_CODE)\n              : DEFAULT_HTTP_STATUS_CODE;\n          return wdErrAsync(\n            new AMCHttpError(`AMC HTTP request failed: ${String(error)}`, statusCode)\n          );\n        }\n\n        const backoff = calculateBackoff(\n          retryCount,\n          this.config.retryInterval,\n          this.config.backoffFactor,\n          this.config.maxBackoff\n        );\n        await sleep(backoff);\n      }\n    }\n  }\n}\n",
    "/**\n * Ajax Module Connector設定\n */\nexport interface AMCConfig {\n  /** リクエストタイムアウト（ミリ秒） */\n  timeout: number;\n\n  /** リトライ上限回数 */\n  retryLimit: number;\n\n  /** リトライ基本間隔（ミリ秒） */\n  retryInterval: number;\n\n  /** 最大バックオフ（ミリ秒） */\n  maxBackoff: number;\n\n  /** バックオフ係数 */\n  backoffFactor: number;\n\n  /** 最大並列リクエスト数 */\n  semaphoreLimit: number;\n}\n\n/** デフォルトAMC設定 */\nexport const DEFAULT_AMC_CONFIG: AMCConfig = {\n  timeout: 20000,\n  retryLimit: 3,\n  retryInterval: 1000,\n  maxBackoff: 60000,\n  backoffFactor: 2.0,\n  semaphoreLimit: 10,\n};\n\n/**\n * Wikidotが要求する固定トークン値\n * これはWikidotのフロントエンドから取得される固定値で、\n * セキュリティトークンではなく単なる識別子として使用される\n */\nexport const WIKIDOT_TOKEN7 = '123456';\n\n/**\n * HTTPエラー時のフォールバックステータスコード\n * レスポンスからステータスコードを取得できない場合に使用\n */\nexport const DEFAULT_HTTP_STATUS_CODE = 999;\n",
    "/**\n * AMCリクエストヘッダーを管理するクラス\n */\nexport class AMCHeader {\n  private cookies: Map<string, string>;\n  private contentType: string;\n  private userAgent: string;\n  private referer: string;\n\n  /**\n   * @param options - ヘッダー初期化オプション\n   */\n  constructor(options?: { contentType?: string; userAgent?: string; referer?: string }) {\n    this.contentType = options?.contentType ?? 'application/x-www-form-urlencoded; charset=UTF-8';\n    this.userAgent = options?.userAgent ?? 'WikidotTS';\n    this.referer = options?.referer ?? 'https://www.wikidot.com/';\n    this.cookies = new Map([['wikidot_token7', '123456']]);\n  }\n\n  /**\n   * Cookieを設定する\n   * @param name - Cookie名\n   * @param value - Cookie値\n   */\n  setCookie(name: string, value: string): void {\n    this.cookies.set(name, value);\n  }\n\n  /**\n   * Cookieを削除する\n   * @param name - Cookie名\n   */\n  deleteCookie(name: string): void {\n    this.cookies.delete(name);\n  }\n\n  /**\n   * Cookieを取得する\n   * @param name - Cookie名\n   * @returns Cookie値、存在しない場合はundefined\n   */\n  getCookie(name: string): string | undefined {\n    return this.cookies.get(name);\n  }\n\n  /**\n   * HTTPヘッダーオブジェクトを取得する\n   * @returns ヘッダー辞書\n   */\n  getHeaders(): Record<string, string> {\n    const cookieString = Array.from(this.cookies.entries())\n      .map(([name, value]) => `${name}=${value}`)\n      .join('; ');\n\n    return {\n      'Content-Type': this.contentType,\n      'User-Agent': this.userAgent,\n      Referer: this.referer,\n      Cookie: cookieString,\n    };\n  }\n}\n",
    "import { z } from 'zod';\n\n/**\n * AMCリクエストボディの値型\n */\nexport type AMCRequestBodyValue =\n  | string\n  | number\n  | boolean\n  | null\n  | undefined\n  | Record<string, unknown>\n  | AMCRequestBodyValue[];\n\n/**\n * AMCリクエストボディの型定義\n */\nexport interface AMCRequestBody {\n  moduleName?: string;\n  action?: string;\n  event?: string;\n  [key: string]: AMCRequestBodyValue;\n}\n\n/**\n * AMCレスポンスのベーススキーマ\n */\nconst baseSchema: z.ZodObject<{\n  status: z.ZodString;\n  body: z.ZodOptional<z.ZodString>;\n  message: z.ZodOptional<z.ZodString>;\n}> = z.object({\n  status: z.string(),\n  body: z.string().optional(),\n  message: z.string().optional(),\n});\n\n/**\n * AMCレスポンススキーマ\n */\nexport const amcResponseSchema: z.ZodType<z.infer<typeof baseSchema> & Record<string, unknown>> =\n  baseSchema.passthrough();\n\n/**\n * AMCレスポンス型\n */\nexport type AMCResponse = z.infer<typeof amcResponseSchema>;\n\n/**\n * 成功したAMCレスポンス\n */\nexport interface AMCSuccessResponse {\n  status: 'ok';\n  body: string;\n  [key: string]: unknown;\n}\n\n/**\n * AMCレスポンスが成功かどうかを判定する型ガード\n * @param response - AMCレスポンス\n * @returns 成功レスポンスの場合true\n */\nexport function isSuccessResponse(response: AMCResponse): response is AMCSuccessResponse {\n  return response.status === 'ok';\n}\n",
    "import { SessionCreateError } from '../common/errors';\nimport { fromPromise, type WikidotResultAsync, wdOkAsync } from '../common/types';\nimport type { AuthClientContext } from '../module/types';\n\nconst LOGIN_URL = 'https://www.wikidot.com/default--flow/login__LoginPopupScreen';\n\n/**\n * ユーザー名とパスワードでWikidotにログインする\n * @param client - クライアントコンテキスト（AMCClientを持つオブジェクト）\n * @param username - ユーザー名\n * @param password - パスワード\n * @returns 成功時はvoid、失敗時はSessionCreateError\n */\nexport function login(\n  client: AuthClientContext,\n  username: string,\n  password: string\n): WikidotResultAsync<void> {\n  return fromPromise(\n    (async () => {\n      const formData = new URLSearchParams({\n        login: username,\n        password: password,\n        action: 'Login2Action',\n        event: 'login',\n      });\n\n      const response = await fetch(LOGIN_URL, {\n        method: 'POST',\n        headers: client.amcClient.header.getHeaders(),\n        body: formData.toString(),\n      });\n\n      // Check status code\n      if (!response.ok) {\n        throw new SessionCreateError(\n          `Login attempt failed due to HTTP status code: ${response.status}`\n        );\n      }\n\n      // Check body for error message\n      const body = await response.text();\n      if (body.includes('The login and password do not match')) {\n        throw new SessionCreateError('Login attempt failed due to invalid username or password');\n      }\n\n      // Check cookies\n      const cookies = response.headers.get('Set-Cookie');\n      if (!cookies) {\n        throw new SessionCreateError('Login attempt failed due to missing cookies');\n      }\n\n      // Extract WIKIDOT_SESSION_ID from cookies\n      const sessionIdMatch = cookies.match(/WIKIDOT_SESSION_ID=([^;]+)/);\n      if (!sessionIdMatch?.[1]) {\n        throw new SessionCreateError(\n          'Login attempt failed due to missing WIKIDOT_SESSION_ID cookie'\n        );\n      }\n\n      // Set session cookie\n      client.amcClient.header.setCookie('WIKIDOT_SESSION_ID', sessionIdMatch[1]);\n    })(),\n    (error) => {\n      if (error instanceof SessionCreateError) {\n        return error;\n      }\n      return new SessionCreateError(`Login failed: ${String(error)}`);\n    }\n  );\n}\n\n/**\n * ログアウトする\n * @param client - クライアントコンテキスト（AMCClientを持つオブジェクト）\n * @returns 成功時はvoid\n */\nexport function logout(client: AuthClientContext): WikidotResultAsync<void> {\n  // Try to logout via AMC, then always remove session cookie\n  return client.amcClient\n    .request([\n      {\n        moduleName: 'Empty',\n        action: 'Login2Action',\n        event: 'logout',\n      },\n    ])\n    .map(() => {\n      // Logout succeeded, remove session cookie\n      client.amcClient.header.deleteCookie('WIKIDOT_SESSION_ID');\n      return undefined;\n    })\n    .orElse(() => {\n      // Even if logout request fails, we still want to clear the session locally\n      client.amcClient.header.deleteCookie('WIKIDOT_SESSION_ID');\n      return wdOkAsync(undefined);\n    });\n}\n",
    "import * as cheerio from 'cheerio';\nimport {\n  ForbiddenError,\n  LoginRequiredError,\n  NoElementError,\n  UnexpectedError,\n} from '../../common/errors';\nimport { fromPromise, type WikidotResultAsync } from '../../common/types';\nimport { parseOdate, parseUser } from '../../util/parser';\nimport type { Client } from '../client';\nimport type { AbstractUser } from '../user';\nimport type { User } from '../user/user';\n\n/**\n * プライベートメッセージデータ\n */\nexport interface PrivateMessageData {\n  client: Client;\n  id: number;\n  sender: AbstractUser;\n  recipient: AbstractUser;\n  subject: string;\n  body: string;\n  createdAt: Date;\n}\n\n/**\n * プライベートメッセージ\n */\nexport class PrivateMessage {\n  public readonly client: Client;\n  public readonly id: number;\n  public readonly sender: AbstractUser;\n  public readonly recipient: AbstractUser;\n  public readonly subject: string;\n  public readonly body: string;\n  public readonly createdAt: Date;\n\n  constructor(data: PrivateMessageData) {\n    this.client = data.client;\n    this.id = data.id;\n    this.sender = data.sender;\n    this.recipient = data.recipient;\n    this.subject = data.subject;\n    this.body = data.body;\n    this.createdAt = data.createdAt;\n  }\n\n  /**\n   * メッセージIDからメッセージを取得する\n   * @param client - クライアント\n   * @param messageId - メッセージID\n   * @returns プライベートメッセージ\n   */\n  static fromId(client: Client, messageId: number): WikidotResultAsync<PrivateMessage> {\n    return fromPromise(\n      (async () => {\n        const result = await PrivateMessageCollection.fromIds(client, [messageId]);\n        if (result.isErr()) {\n          throw result.error;\n        }\n        const message = result.value[0];\n        if (!message) {\n          throw new NoElementError(`Message not found: ${messageId}`);\n        }\n        return message;\n      })(),\n      (error) => {\n        if (error instanceof ForbiddenError || error instanceof NoElementError) {\n          return error;\n        }\n        return new UnexpectedError(`Failed to get message: ${String(error)}`);\n      }\n    );\n  }\n\n  /**\n   * プライベートメッセージを送信する\n   * @param client - クライアント\n   * @param recipient - 受信者\n   * @param subject - 件名\n   * @param body - 本文\n   */\n  static send(\n    client: Client,\n    recipient: User,\n    subject: string,\n    body: string\n  ): WikidotResultAsync<void> {\n    const loginResult = client.requireLogin();\n    if (loginResult.isErr()) {\n      return fromPromise(\n        Promise.reject(loginResult.error),\n        () => new LoginRequiredError('Login required to send message')\n      );\n    }\n\n    return fromPromise(\n      (async () => {\n        const result = await client.amcClient.request([\n          {\n            source: body,\n            subject,\n            to_user_id: recipient.id,\n            action: 'DashboardMessageAction',\n            event: 'send',\n            moduleName: 'Empty',\n          },\n        ]);\n        if (result.isErr()) {\n          throw result.error;\n        }\n      })(),\n      (error) => new UnexpectedError(`Failed to send message: ${String(error)}`)\n    );\n  }\n\n  toString(): string {\n    return `PrivateMessage(id=${this.id}, sender=${this.sender}, recipient=${this.recipient}, subject=${this.subject})`;\n  }\n}\n\n/**\n * プライベートメッセージコレクション\n */\nexport class PrivateMessageCollection extends Array<PrivateMessage> {\n  public readonly client: Client;\n\n  constructor(client: Client, messages?: PrivateMessage[]) {\n    super();\n    this.client = client;\n    if (messages) {\n      this.push(...messages);\n    }\n  }\n\n  /**\n   * IDで検索\n   */\n  findById(id: number): PrivateMessage | undefined {\n    return this.find((message) => message.id === id);\n  }\n\n  /**\n   * メッセージIDのリストからメッセージを取得する\n   */\n  static fromIds(\n    client: Client,\n    messageIds: number[]\n  ): WikidotResultAsync<PrivateMessageCollection> {\n    const loginResult = client.requireLogin();\n    if (loginResult.isErr()) {\n      return fromPromise(\n        Promise.reject(loginResult.error),\n        () => new LoginRequiredError('Login required to get messages')\n      );\n    }\n\n    return fromPromise(\n      (async () => {\n        const bodies = messageIds.map((messageId) => ({\n          item: messageId,\n          moduleName: 'dashboard/messages/DMViewMessageModule',\n        }));\n\n        const result = await client.amcClient.request(bodies);\n        if (result.isErr()) {\n          throw result.error;\n        }\n\n        const messages: PrivateMessage[] = [];\n\n        for (let i = 0; i < messageIds.length; i++) {\n          const response = result.value[i];\n          const messageId = messageIds[i];\n          if (!response || messageId === undefined) continue;\n\n          const html = String(response.body ?? '');\n          const $ = cheerio.load(html);\n\n          // ユーザー情報を取得\n          const printuserElems = $('div.pmessage div.header span.printuser');\n          if (printuserElems.length < 2) {\n            throw new ForbiddenError(`Failed to get message: ${messageId}`);\n          }\n\n          const senderElem = $(printuserElems[0]);\n          const recipientElem = $(printuserElems[1]);\n\n          const sender = parseUser(client, senderElem);\n          const recipient = parseUser(client, recipientElem);\n\n          // 件名\n          const subjectElem = $('div.pmessage div.header span.subject');\n          const subject = subjectElem.text().trim();\n\n          // 本文\n          const bodyElem = $('div.pmessage div.body');\n          const body = bodyElem.text().trim();\n\n          // 日時\n          const odateElem = $('div.header span.odate');\n          const createdAt =\n            odateElem.length > 0 ? (parseOdate(odateElem) ?? new Date(0)) : new Date(0);\n\n          messages.push(\n            new PrivateMessage({\n              client,\n              id: messageId,\n              sender,\n              recipient,\n              subject,\n              body,\n              createdAt,\n            })\n          );\n        }\n\n        return new PrivateMessageCollection(client, messages);\n      })(),\n      (error) => {\n        if (error instanceof ForbiddenError || error instanceof LoginRequiredError) {\n          return error;\n        }\n        return new UnexpectedError(`Failed to get messages: ${String(error)}`);\n      }\n    );\n  }\n\n  /**\n   * モジュールからメッセージを取得する内部メソッド\n   */\n  protected static acquireFromModule(\n    client: Client,\n    moduleName: string\n  ): WikidotResultAsync<PrivateMessageCollection> {\n    const loginResult = client.requireLogin();\n    if (loginResult.isErr()) {\n      return fromPromise(\n        Promise.reject(loginResult.error),\n        () => new LoginRequiredError('Login required to get messages')\n      );\n    }\n\n    return fromPromise(\n      (async () => {\n        // ページャー取得\n        const firstResult = await client.amcClient.request([{ moduleName }]);\n        if (firstResult.isErr()) {\n          throw firstResult.error;\n        }\n\n        const firstResponse = firstResult.value[0];\n        if (!firstResponse) {\n          throw new NoElementError('Empty response');\n        }\n\n        const firstHtml = String(firstResponse.body ?? '');\n        const $first = cheerio.load(firstHtml);\n\n        // ページ数を取得\n        const pagerTargets = $first('div.pager span.target');\n        let maxPage = 1;\n        if (pagerTargets.length > 2) {\n          const lastPageText = $first(pagerTargets[pagerTargets.length - 2])\n            .text()\n            .trim();\n          maxPage = Number.parseInt(lastPageText, 10) || 1;\n        }\n\n        // 全ページからメッセージIDを取得\n        const messageIds: number[] = [];\n\n        if (maxPage > 1) {\n          const bodies = [];\n          for (let page = 1; page <= maxPage; page++) {\n            bodies.push({ page, moduleName });\n          }\n          const additionalResults = await client.amcClient.request(bodies);\n          if (additionalResults.isErr()) {\n            throw additionalResults.error;\n          }\n\n          for (const response of additionalResults.value) {\n            const html = String(response?.body ?? '');\n            const $ = cheerio.load(html);\n            $('tr.message').each((_i, elem) => {\n              const dataHref = $(elem).attr('data-href') ?? '';\n              const idMatch = dataHref.match(/\\/(\\d+)$/);\n              if (idMatch?.[1]) {\n                messageIds.push(Number.parseInt(idMatch[1], 10));\n              }\n            });\n          }\n        } else {\n          $first('tr.message').each((_i, elem) => {\n            const dataHref = $first(elem).attr('data-href') ?? '';\n            const idMatch = dataHref.match(/\\/(\\d+)$/);\n            if (idMatch?.[1]) {\n              messageIds.push(Number.parseInt(idMatch[1], 10));\n            }\n          });\n        }\n\n        // メッセージを取得\n        const messagesResult = await PrivateMessageCollection.fromIds(client, messageIds);\n        if (messagesResult.isErr()) {\n          throw messagesResult.error;\n        }\n\n        return messagesResult.value;\n      })(),\n      (error) => {\n        if (\n          error instanceof ForbiddenError ||\n          error instanceof LoginRequiredError ||\n          error instanceof NoElementError\n        ) {\n          return error;\n        }\n        return new UnexpectedError(`Failed to acquire messages: ${String(error)}`);\n      }\n    );\n  }\n}\n\n/**\n * 受信箱\n */\nexport class PrivateMessageInbox extends PrivateMessageCollection {\n  /**\n   * 受信箱のメッセージをすべて取得する\n   */\n  static acquire(client: Client): WikidotResultAsync<PrivateMessageInbox> {\n    return fromPromise(\n      (async () => {\n        const result = await PrivateMessageCollection.acquireFromModule(\n          client,\n          'dashboard/messages/DMInboxModule'\n        );\n        if (result.isErr()) {\n          throw result.error;\n        }\n        const inbox = new PrivateMessageInbox(client);\n        inbox.push(...result.value);\n        return inbox;\n      })(),\n      (error) => {\n        if (error instanceof ForbiddenError || error instanceof LoginRequiredError) {\n          return error;\n        }\n        return new UnexpectedError(`Failed to acquire inbox: ${String(error)}`);\n      }\n    );\n  }\n}\n\n/**\n * 送信箱\n */\nexport class PrivateMessageSentBox extends PrivateMessageCollection {\n  /**\n   * 送信箱のメッセージをすべて取得する\n   */\n  static acquire(client: Client): WikidotResultAsync<PrivateMessageSentBox> {\n    return fromPromise(\n      (async () => {\n        const result = await PrivateMessageCollection.acquireFromModule(\n          client,\n          'dashboard/messages/DMSentModule'\n        );\n        if (result.isErr()) {\n          throw result.error;\n        }\n        const sentBox = new PrivateMessageSentBox(client);\n        sentBox.push(...result.value);\n        return sentBox;\n      })(),\n      (error) => {\n        if (error instanceof ForbiddenError || error instanceof LoginRequiredError) {\n          return error;\n        }\n        return new UnexpectedError(`Failed to acquire sent box: ${String(error)}`);\n      }\n    );\n  }\n}\n",
    "import type { WikidotResultAsync } from '../../../common/types';\nimport {\n  PrivateMessage,\n  PrivateMessageCollection,\n  PrivateMessageInbox,\n  PrivateMessageSentBox,\n} from '../../private-message';\nimport type { User } from '../../user/user';\nimport type { Client } from '../client';\n\n/**\n * プライベートメッセージ操作アクセサ\n */\nexport class PrivateMessageAccessor {\n  public readonly client: Client;\n\n  constructor(client: Client) {\n    this.client = client;\n  }\n\n  /**\n   * メッセージIDからメッセージを取得する\n   * @param id - メッセージID\n   * @returns メッセージオブジェクト\n   */\n  get(id: number): WikidotResultAsync<PrivateMessage> {\n    return PrivateMessage.fromId(this.client, id);\n  }\n\n  /**\n   * 複数のメッセージIDからメッセージを取得する\n   * @param ids - メッセージID配列\n   * @returns メッセージコレクション\n   */\n  getMessages(ids: number[]): WikidotResultAsync<PrivateMessageCollection> {\n    return PrivateMessageCollection.fromIds(this.client, ids);\n  }\n\n  /**\n   * 受信箱のメッセージ一覧を取得する\n   * @returns 受信箱\n   */\n  inbox(): WikidotResultAsync<PrivateMessageInbox> {\n    return PrivateMessageInbox.acquire(this.client);\n  }\n\n  /**\n   * 送信箱のメッセージ一覧を取得する\n   * @returns 送信箱\n   */\n  sentBox(): WikidotResultAsync<PrivateMessageSentBox> {\n    return PrivateMessageSentBox.acquire(this.client);\n  }\n\n  /**\n   * プライベートメッセージを送信する\n   * @param recipient - 受信者\n   * @param subject - 件名\n   * @param body - 本文\n   */\n  send(recipient: User, subject: string, body: string): WikidotResultAsync<void> {\n    return PrivateMessage.send(this.client, recipient, subject, body);\n  }\n}\n\nexport { PrivateMessage, PrivateMessageCollection, PrivateMessageInbox, PrivateMessageSentBox };\n",
    "import type { WikidotResultAsync } from '../../../common/types';\nimport {\n  ForumCategory,\n  ForumCategoryCollection,\n  ForumPost,\n  ForumThread,\n  ForumThreadCollection,\n} from '../../forum';\nimport type { Site } from '../site';\n\n/**\n * フォーラム操作アクセサ\n */\nexport class ForumAccessor {\n  public readonly site: Site;\n\n  constructor(site: Site) {\n    this.site = site;\n  }\n\n  /**\n   * フォーラムカテゴリ一覧を取得\n   * @returns カテゴリ一覧\n   */\n  getCategories(): WikidotResultAsync<ForumCategoryCollection> {\n    return ForumCategoryCollection.acquireAll(this.site);\n  }\n\n  /**\n   * スレッドを取得\n   * @param threadId - スレッドID\n   * @returns スレッド\n   */\n  getThread(threadId: number): WikidotResultAsync<ForumThread> {\n    return ForumThread.getFromId(this.site, threadId);\n  }\n\n  /**\n   * 複数スレッドを取得\n   * @param threadIds - スレッドID配列\n   * @returns スレッドコレクション\n   */\n  getThreads(threadIds: number[]): WikidotResultAsync<ForumThreadCollection> {\n    return ForumThreadCollection.acquireFromThreadIds(this.site, threadIds);\n  }\n}\n\nexport { ForumCategory, ForumCategoryCollection, ForumThread, ForumThreadCollection, ForumPost };\n",
    "/**\n * QuickModule - Wikidot軽量API\n *\n * quickmodule.phpエンドポイントを使用した検索機能\n */\n\nimport { z } from 'zod';\nimport { NotFoundException, UnexpectedError } from '../common/errors';\nimport { fromPromise, type WikidotResultAsync } from '../common/types';\n\n/**\n * QuickModuleモジュール名\n */\nexport type QuickModuleName = 'MemberLookupQModule' | 'UserLookupQModule' | 'PageLookupQModule';\n\n/**\n * QuickModuleユーザー情報\n */\nexport interface QMCUser {\n  id: number;\n  name: string;\n}\n\n/**\n * QuickModuleページ情報\n */\nexport interface QMCPage {\n  title: string;\n  unixName: string;\n}\n\n/**\n * QuickModuleレスポンススキーマ\n */\nconst quickModuleUserResponseSchema = z.object({\n  users: z.union([\n    z.array(\n      z.object({\n        user_id: z.union([z.string(), z.number()]),\n        name: z.string(),\n      })\n    ),\n    z.literal(false),\n  ]),\n});\n\nconst quickModulePageResponseSchema = z.object({\n  pages: z.union([\n    z.array(\n      z.object({\n        title: z.string(),\n        unix_name: z.string(),\n      })\n    ),\n    z.literal(false),\n  ]),\n});\n\n/**\n * QuickModuleエンドポイントにリクエストを送信\n */\nasync function requestQuickModule(\n  moduleName: QuickModuleName,\n  siteId: number,\n  query: string\n): Promise<unknown> {\n  const url = `https://www.wikidot.com/quickmodule.php?module=${moduleName}&s=${siteId}&q=${encodeURIComponent(query)}`;\n\n  const response = await fetch(url, {\n    method: 'GET',\n    headers: {\n      Accept: 'application/json',\n    },\n  });\n\n  if (response.status === 500) {\n    throw new NotFoundException(`Site not found: siteId=${siteId}`);\n  }\n\n  if (!response.ok) {\n    throw new UnexpectedError(`QuickModule request failed: ${response.status}`);\n  }\n\n  return response.json();\n}\n\n/**\n * サイトメンバーを検索\n * @param siteId - サイトID\n * @param query - 検索クエリ（ユーザー名の一部）\n * @returns マッチしたユーザー一覧\n */\nexport function memberLookup(siteId: number, query: string): WikidotResultAsync<QMCUser[]> {\n  return fromPromise(\n    (async () => {\n      const data = await requestQuickModule('MemberLookupQModule', siteId, query);\n      const parsed = quickModuleUserResponseSchema.parse(data);\n\n      if (parsed.users === false) {\n        return [];\n      }\n\n      return parsed.users.map((user) => ({\n        id: typeof user.user_id === 'string' ? Number.parseInt(user.user_id, 10) : user.user_id,\n        name: user.name,\n      }));\n    })(),\n    (error) => {\n      if (error instanceof NotFoundException) return error;\n      return new UnexpectedError(`Member lookup failed: ${String(error)}`);\n    }\n  );\n}\n\n/**\n * Wikidot全体からユーザーを検索\n * @param siteId - サイトID（任意のサイトIDで可）\n * @param query - 検索クエリ（ユーザー名の一部）\n * @returns マッチしたユーザー一覧\n */\nexport function userLookup(siteId: number, query: string): WikidotResultAsync<QMCUser[]> {\n  return fromPromise(\n    (async () => {\n      const data = await requestQuickModule('UserLookupQModule', siteId, query);\n      const parsed = quickModuleUserResponseSchema.parse(data);\n\n      if (parsed.users === false) {\n        return [];\n      }\n\n      return parsed.users.map((user) => ({\n        id: typeof user.user_id === 'string' ? Number.parseInt(user.user_id, 10) : user.user_id,\n        name: user.name,\n      }));\n    })(),\n    (error) => {\n      if (error instanceof NotFoundException) return error;\n      return new UnexpectedError(`User lookup failed: ${String(error)}`);\n    }\n  );\n}\n\n/**\n * サイト内のページを検索\n * @param siteId - サイトID\n * @param query - 検索クエリ（ページ名の一部）\n * @returns マッチしたページ一覧\n */\nexport function pageLookup(siteId: number, query: string): WikidotResultAsync<QMCPage[]> {\n  return fromPromise(\n    (async () => {\n      const data = await requestQuickModule('PageLookupQModule', siteId, query);\n      const parsed = quickModulePageResponseSchema.parse(data);\n\n      if (parsed.pages === false) {\n        return [];\n      }\n\n      return parsed.pages.map((page) => ({\n        title: page.title,\n        unixName: page.unix_name,\n      }));\n    })(),\n    (error) => {\n      if (error instanceof NotFoundException) return error;\n      return new UnexpectedError(`Page lookup failed: ${String(error)}`);\n    }\n  );\n}\n\n/**\n * QuickModule API（後方互換性のため維持）\n * @deprecated 代わりに個別の関数（memberLookup, userLookup, pageLookup）を使用してください\n */\nexport const QuickModule: {\n  memberLookup: typeof memberLookup;\n  userLookup: typeof userLookup;\n  pageLookup: typeof pageLookup;\n} = {\n  memberLookup: memberLookup,\n  userLookup: userLookup,\n  pageLookup: pageLookup,\n};\n",
    "import * as cheerio from 'cheerio';\nimport { RequireLogin } from '../../common/decorators';\nimport {\n  ForbiddenError,\n  LoginRequiredError,\n  NotFoundException,\n  UnexpectedError,\n  WikidotStatusError,\n} from '../../common/errors';\nimport { fromPromise, type WikidotResultAsync } from '../../common/types';\nimport { parseUser } from '../../util/parser';\nimport type { AbstractUser } from '../user';\nimport type { Site } from './site';\n\n/**\n * サイト参加申請データ\n */\nexport interface SiteApplicationData {\n  site: Site;\n  user: AbstractUser;\n  text: string;\n}\n\n/**\n * サイト参加申請\n */\nexport class SiteApplication {\n  public readonly site: Site;\n  public readonly user: AbstractUser;\n  public readonly text: string;\n\n  constructor(data: SiteApplicationData) {\n    this.site = data.site;\n    this.user = data.user;\n    this.text = data.text;\n  }\n\n  /**\n   * 未処理の参加申請をすべて取得する\n   * @param site - 対象サイト\n   */\n  static acquireAll(site: Site): WikidotResultAsync<SiteApplication[]> {\n    const loginResult = site.client.requireLogin();\n    if (loginResult.isErr()) {\n      return fromPromise(\n        Promise.reject(loginResult.error),\n        () => new LoginRequiredError('Login required to get applications')\n      );\n    }\n\n    return fromPromise(\n      (async () => {\n        const result = await site.amcRequest([\n          { moduleName: 'managesite/ManageSiteMembersApplicationsModule' },\n        ]);\n\n        if (result.isErr()) {\n          throw result.error;\n        }\n\n        const response = result.value[0];\n        if (!response) {\n          throw new UnexpectedError('Empty response');\n        }\n\n        const html = String(response.body ?? '');\n\n        // 権限チェック\n        if (html.includes('WIKIDOT.page.listeners.loginClick(event)')) {\n          throw new ForbiddenError('You are not allowed to access this page');\n        }\n\n        const $ = cheerio.load(html);\n        const applications: SiteApplication[] = [];\n\n        const userElements = $('h3 span.printuser').toArray();\n        const textWrapperElements = $('table').toArray();\n\n        if (userElements.length !== textWrapperElements.length) {\n          throw new UnexpectedError(\n            'Length of user_elements and text_wrapper_elements are different'\n          );\n        }\n\n        for (let i = 0; i < userElements.length; i++) {\n          const userElement = userElements[i];\n          const textWrapperElement = textWrapperElements[i];\n\n          if (!userElement || !textWrapperElement) continue;\n\n          const user = parseUser(site.client, $(userElement));\n          const textElement = $(textWrapperElement).find('td').eq(1);\n          const text = textElement.text().trim();\n\n          applications.push(new SiteApplication({ site, user, text }));\n        }\n\n        return applications;\n      })(),\n      (error) => {\n        if (error instanceof ForbiddenError || error instanceof LoginRequiredError) {\n          return error;\n        }\n        return new UnexpectedError(`Failed to get applications: ${String(error)}`);\n      }\n    );\n  }\n\n  /**\n   * 申請を処理する内部メソッド\n   */\n  @RequireLogin\n  private process(action: 'accept' | 'decline'): WikidotResultAsync<void> {\n    return fromPromise(\n      (async () => {\n        const result = await this.site.amcRequest([\n          {\n            action: 'ManageSiteMembershipAction',\n            event: 'acceptApplication',\n            user_id: this.user.id,\n            text: `your application has been ${action}ed`,\n            type: action,\n            moduleName: 'Empty',\n          },\n        ]);\n        if (result.isErr()) {\n          const error = result.error;\n          if (error instanceof WikidotStatusError && error.statusCode === 'no_application') {\n            throw new NotFoundException(`Application not found: ${this.user.name}`);\n          }\n          throw error;\n        }\n      })(),\n      (error) => {\n        if (error instanceof NotFoundException || error instanceof LoginRequiredError) {\n          return error;\n        }\n        return new UnexpectedError(`Failed to process application: ${String(error)}`);\n      }\n    );\n  }\n\n  /**\n   * 参加申請を承認する\n   */\n  accept(): WikidotResultAsync<void> {\n    return this.process('accept');\n  }\n\n  /**\n   * 参加申請を拒否する\n   */\n  decline(): WikidotResultAsync<void> {\n    return this.process('decline');\n  }\n\n  toString(): string {\n    return `SiteApplication(user=${this.user.name}, site=${this.site.unixName}, text=${this.text})`;\n  }\n}\n",
    "import * as cheerio from 'cheerio';\nimport { RequireLogin } from '../../common/decorators';\nimport {\n  LoginRequiredError,\n  TargetError,\n  UnexpectedError,\n  WikidotStatusError,\n} from '../../common/errors';\nimport { fromPromise, type WikidotResultAsync } from '../../common/types';\nimport { parseOdate, parseUser } from '../../util/parser';\nimport type { AbstractUser } from '../user';\nimport type { Site } from './site';\n\n/**\n * サイトメンバーデータ\n */\nexport interface SiteMemberData {\n  site: Site;\n  user: AbstractUser;\n  joinedAt: Date | null;\n}\n\n/**\n * サイトメンバー\n */\nexport class SiteMember {\n  public readonly site: Site;\n  public readonly user: AbstractUser;\n  public readonly joinedAt: Date | null;\n\n  constructor(data: SiteMemberData) {\n    this.site = data.site;\n    this.user = data.user;\n    this.joinedAt = data.joinedAt;\n  }\n\n  /**\n   * HTMLからメンバー情報をパースする\n   */\n  private static parse(site: Site, html: string): SiteMember[] {\n    const $ = cheerio.load(html);\n    const members: SiteMember[] = [];\n\n    $('table tr').each((_i, row) => {\n      const tds = $(row).find('td');\n      const userElem = $(tds[0]).find('.printuser');\n\n      if (userElem.length === 0) {\n        return;\n      }\n\n      const user = parseUser(site.client, userElem);\n\n      // 2つ目のtdがあれば加入日時\n      let joinedAt: Date | null = null;\n      if (tds.length >= 2) {\n        const odateElem = $(tds[1]).find('.odate');\n        if (odateElem.length > 0) {\n          joinedAt = parseOdate(odateElem);\n        }\n      }\n\n      members.push(new SiteMember({ site, user, joinedAt }));\n    });\n\n    return members;\n  }\n\n  /**\n   * サイトメンバー一覧を取得する\n   * @param site - 対象サイト\n   * @param group - グループ（\"admins\", \"moderators\", または空文字で全メンバー）\n   */\n  static getMembers(\n    site: Site,\n    group: 'admins' | 'moderators' | '' = ''\n  ): WikidotResultAsync<SiteMember[]> {\n    return fromPromise(\n      (async () => {\n        const members: SiteMember[] = [];\n\n        // 最初のページを取得\n        const firstResult = await site.amcRequest([\n          {\n            moduleName: 'membership/MembersListModule',\n            page: 1,\n            group,\n          },\n        ]);\n\n        if (firstResult.isErr()) {\n          throw firstResult.error;\n        }\n\n        const firstResponse = firstResult.value[0];\n        if (!firstResponse) {\n          throw new UnexpectedError('Empty response');\n        }\n\n        const firstHtml = String(firstResponse.body ?? '');\n        members.push(...SiteMember.parse(site, firstHtml));\n\n        // ページャーを確認\n        const $first = cheerio.load(firstHtml);\n        const pagerLinks = $first('div.pager a');\n        if (pagerLinks.length < 2) {\n          return members;\n        }\n\n        const lastPageText = $first(pagerLinks[pagerLinks.length - 2])\n          .text()\n          .trim();\n        const lastPage = Number.parseInt(lastPageText, 10) || 1;\n        if (lastPage <= 1) {\n          return members;\n        }\n\n        // 残りのページを取得\n        const bodies = [];\n        for (let page = 2; page <= lastPage; page++) {\n          bodies.push({\n            moduleName: 'membership/MembersListModule',\n            page,\n            group,\n          });\n        }\n\n        const additionalResults = await site.amcRequest(bodies);\n        if (additionalResults.isErr()) {\n          throw additionalResults.error;\n        }\n\n        for (const response of additionalResults.value) {\n          const html = String(response?.body ?? '');\n          members.push(...SiteMember.parse(site, html));\n        }\n\n        return members;\n      })(),\n      (error) => new UnexpectedError(`Failed to get members: ${String(error)}`)\n    );\n  }\n\n  /**\n   * グループ変更の内部メソッド\n   */\n  @RequireLogin\n  private changeGroup(\n    event: 'toModerators' | 'removeModerator' | 'toAdmins' | 'removeAdmin'\n  ): WikidotResultAsync<void> {\n    return fromPromise(\n      (async () => {\n        const result = await this.site.amcRequest([\n          {\n            action: 'ManageSiteMembershipAction',\n            event,\n            user_id: this.user.id,\n            moduleName: '',\n          },\n        ]);\n        if (result.isErr()) {\n          const error = result.error;\n          if (error instanceof WikidotStatusError) {\n            if (error.statusCode === 'not_already') {\n              throw new TargetError(`User is not moderator/admin: ${this.user.name}`);\n            }\n            if (error.statusCode === 'already_admin' || error.statusCode === 'already_moderator') {\n              throw new TargetError(\n                `User is already ${error.statusCode.replace('already_', '')}: ${this.user.name}`\n              );\n            }\n          }\n          throw error;\n        }\n      })(),\n      (error) => {\n        if (error instanceof TargetError || error instanceof LoginRequiredError) {\n          return error;\n        }\n        return new UnexpectedError(`Failed to change member group: ${String(error)}`);\n      }\n    );\n  }\n\n  /**\n   * モデレーターに昇格\n   */\n  toModerator(): WikidotResultAsync<void> {\n    return this.changeGroup('toModerators');\n  }\n\n  /**\n   * モデレーター権限を削除\n   */\n  removeModerator(): WikidotResultAsync<void> {\n    return this.changeGroup('removeModerator');\n  }\n\n  /**\n   * 管理者に昇格\n   */\n  toAdmin(): WikidotResultAsync<void> {\n    return this.changeGroup('toAdmins');\n  }\n\n  /**\n   * 管理者権限を削除\n   */\n  removeAdmin(): WikidotResultAsync<void> {\n    return this.changeGroup('removeAdmin');\n  }\n\n  toString(): string {\n    return `SiteMember(user=${this.user.name}, site=${this.site.unixName})`;\n  }\n}\n",
    "import { RequireLogin } from '../../../common/decorators';\nimport {\n  LoginRequiredError,\n  TargetError,\n  UnexpectedError,\n  WikidotStatusError,\n} from '../../../common/errors';\nimport { fromPromise, type WikidotResultAsync } from '../../../common/types';\nimport { type QMCUser, QuickModule } from '../../../util/quick-module';\nimport type { User } from '../../user/user';\nimport type { Site } from '../site';\nimport { SiteApplication } from '../site-application';\nimport { SiteMember } from '../site-member';\n\n/**\n * サイトメンバー操作アクセサ\n */\nexport class MemberAccessor {\n  public readonly site: Site;\n\n  constructor(site: Site) {\n    this.site = site;\n  }\n\n  /**\n   * 全メンバーを取得する\n   * @returns メンバー一覧\n   */\n  getAll(): WikidotResultAsync<SiteMember[]> {\n    return SiteMember.getMembers(this.site, '');\n  }\n\n  /**\n   * モデレーター一覧を取得する\n   * @returns モデレーター一覧\n   */\n  getModerators(): WikidotResultAsync<SiteMember[]> {\n    return SiteMember.getMembers(this.site, 'moderators');\n  }\n\n  /**\n   * 管理者一覧を取得する\n   * @returns 管理者一覧\n   */\n  getAdmins(): WikidotResultAsync<SiteMember[]> {\n    return SiteMember.getMembers(this.site, 'admins');\n  }\n\n  /**\n   * 未処理の参加申請を取得する\n   * @returns 参加申請一覧\n   */\n  getApplications(): WikidotResultAsync<SiteApplication[]> {\n    return SiteApplication.acquireAll(this.site);\n  }\n\n  /**\n   * メンバーを検索する\n   * @param query - 検索クエリ（ユーザー名の一部）\n   * @returns マッチしたユーザー一覧（QMCUser形式）\n   */\n  lookup(query: string): WikidotResultAsync<QMCUser[]> {\n    return QuickModule.memberLookup(this.site.id, query);\n  }\n\n  /**\n   * ユーザーをサイトに招待する\n   * @param user - 招待するユーザー\n   * @param text - 招待メッセージ\n   */\n  @RequireLogin\n  invite(user: User, text: string): WikidotResultAsync<void> {\n    return fromPromise(\n      (async () => {\n        const result = await this.site.amcRequest([\n          {\n            action: 'ManageSiteMembershipAction',\n            event: 'inviteMember',\n            user_id: user.id,\n            text,\n            moduleName: 'Empty',\n          },\n        ]);\n        if (result.isErr()) {\n          const error = result.error;\n          if (error instanceof WikidotStatusError) {\n            if (error.statusCode === 'already_invited') {\n              throw new TargetError(\n                `User is already invited to ${this.site.unixName}: ${user.name}`\n              );\n            }\n            if (error.statusCode === 'already_member') {\n              throw new TargetError(\n                `User is already a member of ${this.site.unixName}: ${user.name}`\n              );\n            }\n          }\n          throw error;\n        }\n      })(),\n      (error) => {\n        if (error instanceof TargetError || error instanceof LoginRequiredError) {\n          return error;\n        }\n        return new UnexpectedError(`Failed to invite user: ${String(error)}`);\n      }\n    );\n  }\n}\n\nexport { SiteMember, SiteApplication };\n",
    "import type { Cheerio } from 'cheerio';\nimport * as cheerio from 'cheerio';\nimport type { AnyNode } from 'domhandler';\nimport { z } from 'zod';\nimport { RequireLogin } from '../../common/decorators';\nimport {\n  ForbiddenError,\n  LoginRequiredError,\n  NoElementError,\n  NotFoundException,\n  TargetExistsError,\n  UnexpectedError,\n} from '../../common/errors';\nimport { fromPromise, type WikidotResultAsync } from '../../common/types';\nimport type { AMCRequestBody } from '../../connector';\nimport { parseOdate, parseUser } from '../../util/parser';\nimport type { Site } from '../site';\nimport type { AbstractUser } from '../user';\nimport { PageFileCollection } from './page-file';\nimport { PageMetaCollection } from './page-meta';\nimport { type PageRevision, PageRevisionCollection } from './page-revision';\nimport { PageSource } from './page-source';\nimport { PageVote, PageVoteCollection } from './page-vote';\nimport { DEFAULT_MODULE_BODY, DEFAULT_PER_PAGE, SearchPagesQuery } from './search-query';\n\n/**\n * ListPagesModuleパース結果のスキーマ\n * 型安全性のためにZodを使用してパース結果を検証\n */\nconst pageParamsSchema = z.object({\n  fullname: z.string().default(''),\n  name: z.string().default(''),\n  category: z.string().default(''),\n  title: z.string().default(''),\n  children_count: z.number().default(0),\n  comments_count: z.number().default(0),\n  size: z.number().default(0),\n  rating: z.number().default(0),\n  votes_count: z.number().default(0),\n  rating_percent: z.number().nullable().default(null),\n  revisions_count: z.number().default(0),\n  parent_fullname: z.string().nullable().default(null),\n  tags: z.array(z.string()).default([]),\n  created_by: z.custom<AbstractUser>().nullable().default(null),\n  created_at: z.date().nullable().default(null),\n  updated_by: z.custom<AbstractUser>().nullable().default(null),\n  updated_at: z.date().nullable().default(null),\n  commented_by: z.custom<AbstractUser>().nullable().default(null),\n  commented_at: z.date().nullable().default(null),\n});\n\n/**\n * ページデータ\n */\nexport interface PageData {\n  site: Site;\n  fullname: string;\n  name: string;\n  category: string;\n  title: string;\n  childrenCount: number;\n  commentsCount: number;\n  size: number;\n  rating: number;\n  votesCount: number;\n  ratingPercent: number | null;\n  revisionsCount: number;\n  parentFullname: string | null;\n  tags: string[];\n  createdBy: AbstractUser | null;\n  createdAt: Date;\n  updatedBy: AbstractUser | null;\n  updatedAt: Date;\n  commentedBy: AbstractUser | null;\n  commentedAt: Date | null;\n}\n\n/**\n * Wikidotページ\n */\nexport class Page {\n  public readonly site: Site;\n  public readonly fullname: string;\n  public readonly name: string;\n  public readonly category: string;\n  public title: string;\n  public childrenCount: number;\n  public commentsCount: number;\n  public size: number;\n  public rating: number;\n  public votesCount: number;\n  public ratingPercent: number | null;\n  public revisionsCount: number;\n  public parentFullname: string | null;\n  public tags: string[];\n  public readonly createdBy: AbstractUser | null;\n  public readonly createdAt: Date;\n  public updatedBy: AbstractUser | null;\n  public updatedAt: Date;\n  public commentedBy: AbstractUser | null;\n  public commentedAt: Date | null;\n\n  private _id: number | null = null;\n  private _source: PageSource | null = null;\n  private _revisions: PageRevisionCollection | null = null;\n  private _votes: PageVoteCollection | null = null;\n\n  constructor(data: PageData) {\n    this.site = data.site;\n    this.fullname = data.fullname;\n    this.name = data.name;\n    this.category = data.category;\n    this.title = data.title;\n    this.childrenCount = data.childrenCount;\n    this.commentsCount = data.commentsCount;\n    this.size = data.size;\n    this.rating = data.rating;\n    this.votesCount = data.votesCount;\n    this.ratingPercent = data.ratingPercent;\n    this.revisionsCount = data.revisionsCount;\n    this.parentFullname = data.parentFullname;\n    this.tags = data.tags;\n    this.createdBy = data.createdBy;\n    this.createdAt = data.createdAt;\n    this.updatedBy = data.updatedBy;\n    this.updatedAt = data.updatedAt;\n    this.commentedBy = data.commentedBy;\n    this.commentedAt = data.commentedAt;\n  }\n\n  /**\n   * ページURLを取得\n   */\n  getUrl(): string {\n    return `${this.site.getBaseUrl()}/${this.fullname}`;\n  }\n\n  /**\n   * ページIDが取得済みかどうか\n   */\n  isIdAcquired(): boolean {\n    return this._id !== null;\n  }\n\n  /**\n   * ページIDを取得\n   */\n  get id(): number | null {\n    return this._id;\n  }\n\n  /**\n   * ページIDを設定\n   */\n  set id(value: number | null) {\n    this._id = value;\n  }\n\n  /**\n   * ソースコードを取得\n   */\n  get source(): PageSource | null {\n    return this._source;\n  }\n\n  /**\n   * ソースコードを設定\n   */\n  set source(value: PageSource | null) {\n    this._source = value;\n  }\n\n  /**\n   * リビジョン履歴を取得\n   */\n  get revisions(): PageRevisionCollection | null {\n    return this._revisions;\n  }\n\n  /**\n   * リビジョン履歴を設定\n   */\n  set revisions(value: PageRevisionCollection | null) {\n    this._revisions = value;\n  }\n\n  /**\n   * 投票情報を取得\n   */\n  get votes(): PageVoteCollection | null {\n    return this._votes;\n  }\n\n  /**\n   * 投票情報を設定\n   */\n  set votes(value: PageVoteCollection | null) {\n    this._votes = value;\n  }\n\n  /**\n   * 最新リビジョンを取得\n   */\n  get latestRevision(): PageRevision | undefined {\n    if (!this._revisions || this._revisions.length === 0) return undefined;\n    return this._revisions.reduce((max, rev) => (rev.revNo > max.revNo ? rev : max));\n  }\n\n  /**\n   * ページIDが必須の操作で、IDがない場合にエラーを返すヘルパー\n   * @param operation - 操作名（エラーメッセージ用）\n   * @returns ページIDまたはエラーResult\n   */\n  private requireId(\n    operation: string\n  ): { ok: true; id: number } | { ok: false; error: WikidotResultAsync<never> } {\n    if (this._id === null) {\n      return {\n        ok: false,\n        error: fromPromise(\n          Promise.reject(new Error('Page ID not acquired')),\n          () => new UnexpectedError(`Page ID must be acquired before ${operation}`)\n        ),\n      };\n    }\n    return { ok: true, id: this._id };\n  }\n\n  /**\n   * ページを削除する\n   */\n  @RequireLogin\n  destroy(): WikidotResultAsync<void> {\n    const idCheck = this.requireId('deletion');\n    if (!idCheck.ok) return idCheck.error;\n\n    return fromPromise(\n      (async () => {\n        const result = await this.site.amcRequest([\n          {\n            action: 'WikiPageAction',\n            event: 'deletePage',\n            page_id: this._id,\n            moduleName: 'Empty',\n          },\n        ]);\n        if (result.isErr()) {\n          throw result.error;\n        }\n      })(),\n      (error) => new UnexpectedError(`Failed to delete page: ${String(error)}`)\n    );\n  }\n\n  /**\n   * タグを保存する\n   */\n  @RequireLogin\n  commitTags(): WikidotResultAsync<void> {\n    const idCheck = this.requireId('saving tags');\n    if (!idCheck.ok) return idCheck.error;\n\n    return fromPromise(\n      (async () => {\n        const result = await this.site.amcRequest([\n          {\n            tags: this.tags.join(' '),\n            action: 'WikiPageAction',\n            event: 'saveTags',\n            pageId: this._id,\n            moduleName: 'Empty',\n          },\n        ]);\n        if (result.isErr()) {\n          throw result.error;\n        }\n      })(),\n      (error) => new UnexpectedError(`Failed to save tags: ${String(error)}`)\n    );\n  }\n\n  /**\n   * 親ページを設定する\n   * @param parentFullname - 親ページのフルネーム（nullで解除）\n   */\n  @RequireLogin\n  setParent(parentFullname: string | null): WikidotResultAsync<void> {\n    const idCheck = this.requireId('setting parent');\n    if (!idCheck.ok) return idCheck.error;\n\n    return fromPromise(\n      (async () => {\n        const result = await this.site.amcRequest([\n          {\n            action: 'WikiPageAction',\n            event: 'setParentPage',\n            moduleName: 'Empty',\n            pageId: String(this._id),\n            parentName: parentFullname ?? '',\n          },\n        ]);\n        if (result.isErr()) {\n          throw result.error;\n        }\n        this.parentFullname = parentFullname;\n      })(),\n      (error) => new UnexpectedError(`Failed to set parent: ${String(error)}`)\n    );\n  }\n\n  /**\n   * ページに投票する\n   * @param value - 投票値\n   * @returns 新しいレーティング\n   */\n  @RequireLogin\n  vote(value: number): WikidotResultAsync<number> {\n    const idCheck = this.requireId('voting');\n    if (!idCheck.ok) return idCheck.error;\n\n    return fromPromise(\n      (async () => {\n        const result = await this.site.amcRequest([\n          {\n            action: 'RateAction',\n            event: 'ratePage',\n            moduleName: 'Empty',\n            pageId: this._id,\n            points: value,\n            force: 'yes',\n          },\n        ]);\n        if (result.isErr()) {\n          throw result.error;\n        }\n        const response = result.value[0];\n        if (!response) {\n          throw new UnexpectedError('Empty response from vote request');\n        }\n        const newRating = Number.parseInt(String(response.points ?? this.rating), 10);\n        this.rating = newRating;\n        return newRating;\n      })(),\n      (error) => new UnexpectedError(`Failed to vote: ${String(error)}`)\n    );\n  }\n\n  /**\n   * 投票をキャンセルする\n   * @returns 新しいレーティング\n   */\n  @RequireLogin\n  cancelVote(): WikidotResultAsync<number> {\n    const idCheck = this.requireId('canceling vote');\n    if (!idCheck.ok) return idCheck.error;\n\n    return fromPromise(\n      (async () => {\n        const result = await this.site.amcRequest([\n          {\n            action: 'RateAction',\n            event: 'cancelVote',\n            moduleName: 'Empty',\n            pageId: this._id,\n          },\n        ]);\n        if (result.isErr()) {\n          throw result.error;\n        }\n        const response = result.value[0];\n        if (!response) {\n          throw new UnexpectedError('Empty response from cancel vote request');\n        }\n        const newRating = Number.parseInt(String(response.points ?? this.rating), 10);\n        this.rating = newRating;\n        return newRating;\n      })(),\n      (error) => new UnexpectedError(`Failed to cancel vote: ${String(error)}`)\n    );\n  }\n\n  /**\n   * ページを編集する\n   * @param options - 編集オプション\n   */\n  @RequireLogin\n  edit(options: {\n    title?: string;\n    source?: string;\n    comment?: string;\n    forceEdit?: boolean;\n  }): WikidotResultAsync<void> {\n    const idCheck = this.requireId('editing');\n    if (!idCheck.ok) return idCheck.error;\n\n    const pageId = idCheck.id;\n\n    return fromPromise(\n      (async () => {\n        // 現在のソースを取得（指定がない場合）\n        let currentSource = options.source;\n        if (currentSource === undefined) {\n          const existingSource = this._source;\n          if (existingSource !== null) {\n            currentSource = existingSource.wikiText;\n          } else {\n            // ソースを取得\n            const sourceResult = await PageCollection.acquirePageSources(this.site, [this]);\n            if (sourceResult.isErr()) {\n              throw sourceResult.error;\n            }\n            // acquirePageSources後、this._sourceにセットされる\n            currentSource = this._source?.wikiText ?? '';\n          }\n        }\n\n        const result = await PageCollection.createOrEdit(this.site, this.fullname, {\n          pageId,\n          title: options.title ?? this.title,\n          source: currentSource,\n          comment: options.comment ?? '',\n          forceEdit: options.forceEdit ?? false,\n        });\n        if (result.isErr()) {\n          throw result.error;\n        }\n      })(),\n      (error) => {\n        if (error instanceof LoginRequiredError || error instanceof ForbiddenError) {\n          return error;\n        }\n        return new UnexpectedError(`Failed to edit page: ${String(error)}`);\n      }\n    );\n  }\n\n  /**\n   * ページ名を変更する\n   * @param newFullname - 新しいフルネーム\n   */\n  @RequireLogin\n  rename(newFullname: string): WikidotResultAsync<void> {\n    const idCheck = this.requireId('renaming');\n    if (!idCheck.ok) return idCheck.error;\n\n    return fromPromise(\n      (async () => {\n        const result = await this.site.amcRequest([\n          {\n            action: 'WikiPageAction',\n            event: 'renamePage',\n            moduleName: 'Empty',\n            page_id: this._id,\n            new_name: newFullname,\n          },\n        ]);\n        if (result.isErr()) {\n          throw result.error;\n        }\n        // プロパティを更新（readonlyなのでObject.assignで）\n        Object.assign(this, {\n          fullname: newFullname,\n          category: newFullname.includes(':') ? newFullname.split(':')[0] : '_default',\n          name: newFullname.includes(':') ? newFullname.split(':')[1] : newFullname,\n        });\n      })(),\n      (error) => new UnexpectedError(`Failed to rename page: ${String(error)}`)\n    );\n  }\n\n  /**\n   * ページに添付されたファイル一覧を取得する\n   */\n  getFiles(): WikidotResultAsync<PageFileCollection> {\n    return PageFileCollection.acquire(this);\n  }\n\n  /**\n   * ページのディスカッションスレッドを取得する\n   */\n  getDiscussion(): WikidotResultAsync<import('../forum').ForumThread | null> {\n    const idCheck = this.requireId('getting discussion');\n    if (!idCheck.ok) return idCheck.error;\n\n    const pageId = idCheck.id;\n\n    return fromPromise(\n      (async () => {\n        const result = await this.site.amcRequest([\n          {\n            moduleName: 'forum/ForumCommentsListModule',\n            pageId,\n          },\n        ]);\n\n        if (result.isErr()) {\n          throw result.error;\n        }\n\n        const response = result.value[0];\n        if (!response) {\n          return null;\n        }\n\n        const html = String(response.body ?? '');\n        // スレッドIDを抽出\n        const match = html.match(\n          /WIKIDOT\\.modules\\.ForumViewThreadModule\\.vars\\.threadId\\s*=\\s*(\\d+)/\n        );\n        if (!match?.[1]) {\n          return null;\n        }\n\n        const threadId = Number.parseInt(match[1], 10);\n\n        // ForumThreadを取得\n        const { ForumThread } = await import('../forum');\n        const threadResult = await ForumThread.getFromId(this.site, threadId);\n        if (threadResult.isErr()) {\n          throw threadResult.error;\n        }\n        return threadResult.value;\n      })(),\n      (error) => new UnexpectedError(`Failed to get discussion: ${String(error)}`)\n    );\n  }\n\n  /**\n   * ページのメタタグ一覧を取得する\n   * @returns メタタグコレクション\n   */\n  getMetas(): WikidotResultAsync<PageMetaCollection> {\n    const idCheck = this.requireId('getting metas');\n    if (!idCheck.ok) {\n      return idCheck.error;\n    }\n\n    return PageMetaCollection.acquire(this);\n  }\n\n  /**\n   * メタタグを設定する\n   * @param name - メタタグ名\n   * @param content - メタタグの値\n   */\n  setMeta(name: string, content: string): WikidotResultAsync<void> {\n    const idCheck = this.requireId('setting meta');\n    if (!idCheck.ok) {\n      return idCheck.error;\n    }\n\n    return PageMetaCollection.setMeta(this, name, content);\n  }\n\n  /**\n   * メタタグを削除する\n   * @param name - メタタグ名\n   */\n  deleteMeta(name: string): WikidotResultAsync<void> {\n    const idCheck = this.requireId('deleting meta');\n    if (!idCheck.ok) {\n      return idCheck.error;\n    }\n\n    return PageMetaCollection.deleteMeta(this, name);\n  }\n\n  toString(): string {\n    return `Page(fullname=${this.fullname}, title=${this.title})`;\n  }\n}\n\n/**\n * ページコレクション\n */\nexport class PageCollection extends Array<Page> {\n  public readonly site: Site;\n\n  constructor(site: Site, pages?: Page[]) {\n    super();\n    this.site = site;\n    if (pages) {\n      this.push(...pages);\n    }\n  }\n\n  /**\n   * フルネームで検索\n   * @param fullname - ページのフルネーム\n   * @returns ページ（存在しない場合はundefined）\n   */\n  findByFullname(fullname: string): Page | undefined {\n    return this.find((page) => page.fullname === fullname);\n  }\n\n  /**\n   * ページIDを一括取得\n   */\n  getPageIds(): WikidotResultAsync<PageCollection> {\n    return PageCollection.acquirePageIds(this.site, this);\n  }\n\n  /**\n   * ページソースを一括取得\n   */\n  getPageSources(): WikidotResultAsync<PageCollection> {\n    return PageCollection.acquirePageSources(this.site, this);\n  }\n\n  /**\n   * ページリビジョンを一括取得\n   */\n  getPageRevisions(): WikidotResultAsync<PageCollection> {\n    return PageCollection.acquirePageRevisions(this.site, this);\n  }\n\n  /**\n   * ページ投票を一括取得\n   */\n  getPageVotes(): WikidotResultAsync<PageCollection> {\n    return PageCollection.acquirePageVotes(this.site, this);\n  }\n\n  /**\n   * ページIDを一括取得する内部メソッド\n   */\n  static acquirePageIds(site: Site, pages: Page[]): WikidotResultAsync<PageCollection> {\n    return fromPromise(\n      (async () => {\n        const targetPages = pages.filter((page) => !page.isIdAcquired());\n\n        if (targetPages.length === 0) {\n          return new PageCollection(site, pages);\n        }\n\n        // norender, noredirectでアクセス\n        const responses = await Promise.all(\n          targetPages.map(async (page) => {\n            const url = `${page.getUrl()}/norender/true/noredirect/true`;\n            const response = await fetch(url, {\n              headers: site.client.amcClient.header.getHeaders(),\n            });\n            return { page, response };\n          })\n        );\n\n        for (const { page, response } of responses) {\n          const text = await response.text();\n          const match = text.match(/WIKIREQUEST\\.info\\.pageId\\s*=\\s*(\\d+);/);\n          if (!match?.[1]) {\n            throw new NoElementError(`Cannot find page id: ${page.fullname}`);\n          }\n          page.id = Number.parseInt(match[1], 10);\n        }\n\n        return new PageCollection(site, pages);\n      })(),\n      (error) => {\n        if (error instanceof NoElementError) return error;\n        return new UnexpectedError(`Failed to acquire page IDs: ${String(error)}`);\n      }\n    );\n  }\n\n  /**\n   * ページソースを一括取得する内部メソッド\n   */\n  static acquirePageSources(site: Site, pages: Page[]): WikidotResultAsync<PageCollection> {\n    return fromPromise(\n      (async () => {\n        const targetPages = pages.filter((page) => page.source === null && page.id !== null);\n\n        if (targetPages.length === 0) {\n          return new PageCollection(site, pages);\n        }\n\n        const result = await site.amcRequest(\n          targetPages.map((page) => ({\n            moduleName: 'viewsource/ViewSourceModule',\n            page_id: page.id,\n          }))\n        );\n\n        if (result.isErr()) {\n          throw result.error;\n        }\n\n        for (let i = 0; i < targetPages.length; i++) {\n          const page = targetPages[i];\n          const response = result.value[i];\n          if (!page || !response) continue;\n          const body = String(response.body ?? '').replace(/&nbsp;/g, ' ');\n          const $ = cheerio.load(body);\n          const sourceElement = $('div.page-source');\n          if (sourceElement.length === 0) {\n            throw new NoElementError(`Cannot find source element for page: ${page.fullname}`);\n          }\n          const wikiText = sourceElement.text().trim().replace(/^\\t/, '');\n          page.source = new PageSource({ page, wikiText });\n        }\n\n        return new PageCollection(site, pages);\n      })(),\n      (error) => {\n        if (error instanceof NoElementError) return error;\n        return new UnexpectedError(`Failed to acquire page sources: ${String(error)}`);\n      }\n    );\n  }\n\n  /**\n   * ページリビジョンを一括取得する内部メソッド\n   */\n  static acquirePageRevisions(site: Site, pages: Page[]): WikidotResultAsync<PageCollection> {\n    return fromPromise(\n      (async () => {\n        const targetPages = pages.filter((page) => page.revisions === null && page.id !== null);\n\n        if (targetPages.length === 0) {\n          return new PageCollection(site, pages);\n        }\n\n        const result = await site.amcRequest(\n          targetPages.map((page) => ({\n            moduleName: 'history/PageRevisionListModule',\n            page_id: page.id,\n            options: { all: true },\n            perpage: 100000000,\n          }))\n        );\n\n        if (result.isErr()) {\n          throw result.error;\n        }\n\n        // リビジョンをパース\n        const { PageRevision } = await import('./page-revision');\n        for (let i = 0; i < targetPages.length; i++) {\n          const page = targetPages[i];\n          const response = result.value[i];\n          if (!page || !response) continue;\n\n          const body = String(response.body ?? '');\n          const $ = cheerio.load(body);\n          const revisions: PageRevision[] = [];\n\n          $('table.page-history > tr[id^=revision-row-]').each((_j, revElement) => {\n            const $rev = $(revElement);\n            const revIdAttr = $rev.attr('id');\n            if (!revIdAttr) return;\n\n            const revId = Number.parseInt(revIdAttr.replace('revision-row-', ''), 10);\n            if (Number.isNaN(revId)) return;\n\n            const $tds = $rev.find('td');\n            if ($tds.length < 7) return;\n\n            const revNoText = $tds.eq(0).text().trim().replace(/\\.$/, '');\n            const revNo = Number.parseInt(revNoText, 10);\n            if (Number.isNaN(revNo)) return;\n\n            const $createdByElem = $tds.eq(4).find('span.printuser');\n            if ($createdByElem.length === 0) return;\n            const createdBy = parseUser(site.client, $createdByElem as Cheerio<AnyNode>);\n\n            const $createdAtElem = $tds.eq(5).find('span.odate');\n            if ($createdAtElem.length === 0) return;\n            const createdAt = parseOdate($createdAtElem as Cheerio<AnyNode>) ?? new Date();\n\n            const comment = $tds.eq(6).text().trim();\n\n            revisions.push(\n              new PageRevision({\n                page,\n                id: revId,\n                revNo,\n                createdBy,\n                createdAt,\n                comment,\n              })\n            );\n          });\n\n          page.revisions = new PageRevisionCollection(page, revisions);\n        }\n\n        return new PageCollection(site, pages);\n      })(),\n      (error) => new UnexpectedError(`Failed to acquire page revisions: ${String(error)}`)\n    );\n  }\n\n  /**\n   * ページ投票を一括取得する内部メソッド\n   */\n  static acquirePageVotes(site: Site, pages: Page[]): WikidotResultAsync<PageCollection> {\n    return fromPromise(\n      (async () => {\n        const targetPages = pages.filter((page) => page.votes === null && page.id !== null);\n\n        if (targetPages.length === 0) {\n          return new PageCollection(site, pages);\n        }\n\n        const result = await site.amcRequest(\n          targetPages.map((page) => ({\n            moduleName: 'pagerate/WhoRatedPageModule',\n            pageId: page.id,\n          }))\n        );\n\n        if (result.isErr()) {\n          throw result.error;\n        }\n\n        // 投票をパース\n        for (let i = 0; i < targetPages.length; i++) {\n          const page = targetPages[i];\n          const response = result.value[i];\n          if (!page || !response) continue;\n\n          const body = String(response.body ?? '');\n          const $ = cheerio.load(body);\n\n          const $userElems = $('span.printuser');\n          const $valueElems = $(\"span[style^='color']\");\n\n          if ($userElems.length !== $valueElems.length) {\n            throw new UnexpectedError('User and value count mismatch in votes');\n          }\n\n          const votes: PageVote[] = [];\n          $userElems.each((j, userElem) => {\n            const $user = $(userElem);\n            const $value = $valueElems.eq(j);\n\n            const user = parseUser(site.client, $user as Cheerio<AnyNode>);\n            const valueText = $value.text().trim();\n\n            let value: number;\n            if (valueText === '+') {\n              value = 1;\n            } else if (valueText === '-') {\n              value = -1;\n            } else {\n              value = Number.parseInt(valueText, 10) || 0;\n            }\n\n            votes.push(new PageVote({ page, user, value }));\n          });\n\n          page.votes = new PageVoteCollection(page, votes);\n        }\n\n        return new PageCollection(site, pages);\n      })(),\n      (error) => new UnexpectedError(`Failed to acquire page votes: ${String(error)}`)\n    );\n  }\n\n  /**\n   * ListPagesModuleレスポンスをパース\n   */\n  static parse(\n    site: Site,\n    htmlBody: cheerio.CheerioAPI,\n    _parseUser: (element: cheerio.Cheerio<AnyNode>) => AbstractUser\n  ): PageCollection {\n    const pages: Page[] = [];\n\n    htmlBody('div.page').each((_i, pageElement) => {\n      const $page = htmlBody(pageElement);\n      const pageParams: Record<string, unknown> = {};\n\n      // 5つ星レーティング判定\n      const is5StarRating = $page.find('span.rating span.page-rate-list-pages-start').length > 0;\n\n      // 各値を取得\n      $page.find('span.set').each((_j, setElement) => {\n        const $set = htmlBody(setElement);\n        const keyElement = $set.find('span.name');\n        if (keyElement.length === 0) return;\n\n        let key = keyElement.text().trim();\n        const valueElement = $set.find('span.value');\n\n        let value: unknown = null;\n\n        if (valueElement.length === 0) {\n          value = null;\n        } else if (['created_at', 'updated_at', 'commented_at'].includes(key)) {\n          const odateElement = valueElement.find('span.odate');\n          if (odateElement.length > 0) {\n            const timestamp = odateElement.attr('class')?.match(/time_(\\d+)/)?.[1];\n            value = timestamp ? new Date(Number.parseInt(timestamp, 10) * 1000) : null;\n          }\n        } else if (\n          ['created_by_linked', 'updated_by_linked', 'commented_by_linked'].includes(key)\n        ) {\n          const printuserElement = valueElement.find('span.printuser');\n          if (printuserElement.length > 0) {\n            value = _parseUser(printuserElement);\n          }\n        } else if (['tags', '_tags'].includes(key)) {\n          value = valueElement.text().split(/\\s+/).filter(Boolean);\n        } else if (['rating_votes', 'comments', 'size', 'revisions'].includes(key)) {\n          value = Number.parseInt(valueElement.text().trim(), 10) || 0;\n        } else if (key === 'rating') {\n          const ratingText = valueElement.text().trim();\n          value = is5StarRating\n            ? Number.parseFloat(ratingText) || 0\n            : Number.parseInt(ratingText, 10) || 0;\n        } else if (key === 'rating_percent') {\n          if (is5StarRating) {\n            value = (Number.parseFloat(valueElement.text().trim()) || 0) / 100;\n          } else {\n            value = null;\n          }\n        } else {\n          value = valueElement.text().trim();\n        }\n\n        // キー変換\n        if (key.includes('_linked')) {\n          key = key.replace('_linked', '');\n        } else if (['comments', 'children', 'revisions'].includes(key)) {\n          key = `${key}_count`;\n        } else if (key === 'rating_votes') {\n          key = 'votes_count';\n        }\n\n        pageParams[key] = value;\n      });\n\n      // タグ統合\n      const tags = Array.isArray(pageParams.tags) ? pageParams.tags : [];\n      const hiddenTags = Array.isArray(pageParams._tags) ? pageParams._tags : [];\n      pageParams.tags = [...tags, ...hiddenTags];\n\n      // Zodスキーマで検証・デフォルト値適用\n      const parsed = pageParamsSchema.parse(pageParams);\n\n      // Pageオブジェクト作成\n      pages.push(\n        new Page({\n          site,\n          fullname: parsed.fullname,\n          name: parsed.name,\n          category: parsed.category,\n          title: parsed.title,\n          childrenCount: parsed.children_count,\n          commentsCount: parsed.comments_count,\n          size: parsed.size,\n          rating: parsed.rating,\n          votesCount: parsed.votes_count,\n          ratingPercent: parsed.rating_percent,\n          revisionsCount: parsed.revisions_count,\n          parentFullname: parsed.parent_fullname,\n          tags: parsed.tags,\n          createdBy: parsed.created_by,\n          createdAt: parsed.created_at ?? new Date(),\n          updatedBy: parsed.updated_by,\n          updatedAt: parsed.updated_at ?? new Date(),\n          commentedBy: parsed.commented_by,\n          commentedAt: parsed.commented_at,\n        })\n      );\n    });\n\n    return new PageCollection(site, pages);\n  }\n\n  /**\n   * ページ検索\n   */\n  static searchPages(\n    site: Site,\n    parseUser: (element: cheerio.Cheerio<AnyNode>) => AbstractUser,\n    query: SearchPagesQuery | null = null\n  ): WikidotResultAsync<PageCollection> {\n    return fromPromise(\n      (async () => {\n        const q = query ?? new SearchPagesQuery();\n        const queryDict = q.asDict();\n\n        // モジュールボディ生成\n        const moduleBody = `[[div class=\"page\"]]\\n${DEFAULT_MODULE_BODY.map(\n          (key) =>\n            `[[span class=\"set ${key}\"]][[span class=\"name\"]] ${key} [[/span]][[span class=\"value\"]] %%${key}%% [[/span]][[/span]]`\n        ).join('')}\\n[[/div]]`;\n\n        const requestBody = {\n          ...queryDict,\n          moduleName: 'list/ListPagesModule',\n          module_body: moduleBody,\n        };\n\n        const result = await site.amcRequest([requestBody]);\n        if (result.isErr()) {\n          if (result.error.message.includes('not_ok')) {\n            throw new ForbiddenError('Failed to get pages, target site may be private');\n          }\n          throw result.error;\n        }\n\n        const firstResponse = result.value[0];\n        const body = String(firstResponse?.body ?? '');\n        const $first = cheerio.load(body);\n\n        let total = 1;\n        const htmlBodies: cheerio.CheerioAPI[] = [$first];\n\n        // ページネーション確認\n        const pagerElement = $first('div.pager');\n        if (pagerElement.length > 0) {\n          const lastPagerElements = $first('div.pager span.target');\n          if (lastPagerElements.length >= 2) {\n            const secondLastPager = $first(lastPagerElements[lastPagerElements.length - 2]);\n            const lastPagerLink = secondLastPager.find('a');\n            if (lastPagerLink.length > 0) {\n              total = Number.parseInt(lastPagerLink.text().trim(), 10) || 1;\n            }\n          }\n        }\n\n        // 追加ページ取得\n        if (total > 1) {\n          const additionalBodies: AMCRequestBody[] = [];\n          for (let i = 1; i < total; i++) {\n            additionalBodies.push({\n              ...queryDict,\n              moduleName: 'list/ListPagesModule',\n              module_body: moduleBody,\n              offset: i * (q.perPage ?? DEFAULT_PER_PAGE),\n            } as AMCRequestBody);\n          }\n\n          const additionalResults = await site.amcRequest(additionalBodies);\n          if (additionalResults.isErr()) {\n            throw additionalResults.error;\n          }\n\n          for (const response of additionalResults.value) {\n            const respBody = String(response?.body ?? '');\n            htmlBodies.push(cheerio.load(respBody));\n          }\n        }\n\n        // パース\n        const pages: Page[] = [];\n        for (const $html of htmlBodies) {\n          const parsed = PageCollection.parse(site, $html, parseUser);\n          pages.push(...parsed);\n        }\n\n        return new PageCollection(site, pages);\n      })(),\n      (error) => {\n        if (error instanceof ForbiddenError || error instanceof NotFoundException) {\n          return error;\n        }\n        return new UnexpectedError(`Failed to search pages: ${String(error)}`);\n      }\n    );\n  }\n\n  /**\n   * ページを作成または編集\n   */\n  static createOrEdit(\n    site: Site,\n    fullname: string,\n    options: {\n      pageId?: number | null;\n      title?: string;\n      source?: string;\n      comment?: string;\n      forceEdit?: boolean;\n      raiseOnExists?: boolean;\n    } = {}\n  ): WikidotResultAsync<void> {\n    const loginResult = site.client.requireLogin();\n    if (loginResult.isErr()) {\n      return fromPromise(\n        Promise.reject(loginResult.error),\n        () => new LoginRequiredError('Login required to create/edit page')\n      );\n    }\n\n    return fromPromise(\n      (async () => {\n        const {\n          pageId = null,\n          title = '',\n          source = '',\n          comment = '',\n          forceEdit = false,\n          raiseOnExists = false,\n        } = options;\n\n        // ページロック取得\n        const lockRequestBody: AMCRequestBody = {\n          mode: 'page',\n          wiki_page: fullname,\n          moduleName: 'edit/PageEditModule',\n        };\n        if (forceEdit) {\n          lockRequestBody.force_lock = 'yes';\n        }\n\n        const lockResult = await site.amcRequest([lockRequestBody]);\n        if (lockResult.isErr()) {\n          throw lockResult.error;\n        }\n\n        const lockResponse = lockResult.value[0];\n        if (lockResponse?.locked || lockResponse?.other_locks) {\n          throw new UnexpectedError(`Page ${fullname} is locked or other locks exist`);\n        }\n\n        const isExist = 'page_revision_id' in (lockResponse ?? {});\n\n        if (raiseOnExists && isExist) {\n          throw new TargetExistsError(`Page ${fullname} already exists`);\n        }\n\n        if (isExist && pageId === null) {\n          throw new UnexpectedError('page_id must be specified when editing existing page');\n        }\n\n        const lockId = String(lockResponse?.lock_id ?? '');\n        const lockSecret = String(lockResponse?.lock_secret ?? '');\n        const pageRevisionId = String(lockResponse?.page_revision_id ?? '');\n\n        // ページ保存\n        const editRequestBody: AMCRequestBody = {\n          action: 'WikiPageAction',\n          event: 'savePage',\n          moduleName: 'Empty',\n          mode: 'page',\n          lock_id: lockId,\n          lock_secret: lockSecret,\n          revision_id: pageRevisionId,\n          wiki_page: fullname,\n          page_id: pageId ?? '',\n          title,\n          source,\n          comments: comment,\n        };\n\n        const editResult = await site.amcRequest([editRequestBody]);\n        if (editResult.isErr()) {\n          throw editResult.error;\n        }\n      })(),\n      (error) => {\n        if (error instanceof TargetExistsError || error instanceof LoginRequiredError) {\n          return error;\n        }\n        return new UnexpectedError(`Failed to create/edit page: ${String(error)}`);\n      }\n    );\n  }\n}\n\nexport { SearchPagesQuery };\n",
    "import * as cheerio from 'cheerio';\nimport { UnexpectedError } from '../../common/errors';\nimport { fromPromise, type WikidotResultAsync } from '../../common/types';\nimport type { Page } from './page';\n\n/**\n * ページファイルデータ\n */\nexport interface PageFileData {\n  page: Page;\n  id: number;\n  name: string;\n  url: string;\n  mimeType: string;\n  size: number;\n}\n\n/**\n * ページ添付ファイル\n */\nexport class PageFile {\n  public readonly page: Page;\n  public readonly id: number;\n  public readonly name: string;\n  public readonly url: string;\n  public readonly mimeType: string;\n  public readonly size: number;\n\n  constructor(data: PageFileData) {\n    this.page = data.page;\n    this.id = data.id;\n    this.name = data.name;\n    this.url = data.url;\n    this.mimeType = data.mimeType;\n    this.size = data.size;\n  }\n\n  toString(): string {\n    return `PageFile(id=${this.id}, name=${this.name}, size=${this.size})`;\n  }\n}\n\n/**\n * ページファイルコレクション\n */\nexport class PageFileCollection extends Array<PageFile> {\n  public readonly page: Page;\n\n  constructor(page: Page, files?: PageFile[]) {\n    super();\n    this.page = page;\n    if (files) {\n      this.push(...files);\n    }\n  }\n\n  /**\n   * IDで検索\n   */\n  findById(id: number): PageFile | undefined {\n    return this.find((file) => file.id === id);\n  }\n\n  /**\n   * 名前で検索\n   */\n  findByName(name: string): PageFile | undefined {\n    return this.find((file) => file.name === name);\n  }\n\n  /**\n   * サイズ文字列をバイト数に変換\n   */\n  private static parseSize(sizeText: string): number {\n    const text = sizeText.trim();\n    if (text.includes('Bytes')) {\n      return Math.floor(Number.parseFloat(text.replace('Bytes', '').trim()));\n    }\n    if (text.includes('kB')) {\n      return Math.floor(Number.parseFloat(text.replace('kB', '').trim()) * 1000);\n    }\n    if (text.includes('MB')) {\n      return Math.floor(Number.parseFloat(text.replace('MB', '').trim()) * 1000000);\n    }\n    if (text.includes('GB')) {\n      return Math.floor(Number.parseFloat(text.replace('GB', '').trim()) * 1000000000);\n    }\n    return 0;\n  }\n\n  /**\n   * ページに添付されたファイル一覧を取得する\n   */\n  static acquire(page: Page): WikidotResultAsync<PageFileCollection> {\n    if (page.id === null) {\n      return fromPromise(\n        Promise.reject(new Error('Page ID not acquired')),\n        () => new UnexpectedError('Page ID must be acquired before getting files')\n      );\n    }\n\n    const pageId = page.id;\n\n    return fromPromise(\n      (async () => {\n        const result = await page.site.amcRequest([\n          {\n            moduleName: 'files/PageFilesModule',\n            page_id: pageId,\n          },\n        ]);\n\n        if (result.isErr()) {\n          throw result.error;\n        }\n\n        const response = result.value[0];\n        if (!response) {\n          throw new UnexpectedError('Empty response');\n        }\n\n        const html = String(response.body ?? '');\n        const $ = cheerio.load(html);\n\n        const filesTable = $('table.page-files');\n        if (filesTable.length === 0) {\n          return new PageFileCollection(page, []);\n        }\n\n        const files: PageFile[] = [];\n\n        filesTable.find(\"tbody tr[id^='file-row-']\").each((_i, row) => {\n          const rowId = $(row).attr('id');\n          if (!rowId) return;\n\n          const fileId = Number.parseInt(rowId.replace('file-row-', ''), 10);\n          const tds = $(row).find('td');\n          if (tds.length < 3) return;\n\n          const linkElem = $(tds[0]).find('a');\n          if (linkElem.length === 0) return;\n\n          const name = linkElem.text().trim();\n          const href = linkElem.attr('href') ?? '';\n          const url = `${page.site.getBaseUrl()}${href}`;\n\n          const mimeElem = $(tds[1]).find('span');\n          const mimeType = mimeElem.attr('title') ?? '';\n\n          const sizeText = $(tds[2]).text().trim();\n          const size = PageFileCollection.parseSize(sizeText);\n\n          files.push(\n            new PageFile({\n              page,\n              id: fileId,\n              name,\n              url,\n              mimeType,\n              size,\n            })\n          );\n        });\n\n        return new PageFileCollection(page, files);\n      })(),\n      (error) => new UnexpectedError(`Failed to acquire files: ${String(error)}`)\n    );\n  }\n}\n",
    "import * as cheerio from 'cheerio';\nimport { LoginRequiredError, NoElementError, UnexpectedError } from '../../common/errors';\nimport { fromPromise, type WikidotResultAsync } from '../../common/types';\nimport type { PageRef } from '../types';\n\n/**\n * ページメタタグデータ\n */\nexport interface PageMetaData {\n  page: PageRef;\n  name: string;\n  content: string;\n}\n\n/**\n * ページメタタグ\n */\nexport class PageMeta {\n  public readonly page: PageRef;\n  public readonly name: string;\n  public content: string;\n\n  constructor(data: PageMetaData) {\n    this.page = data.page;\n    this.name = data.name;\n    this.content = data.content;\n  }\n\n  /**\n   * メタタグの値を更新する\n   * @param content - 新しい値\n   */\n  update(content: string): WikidotResultAsync<void> {\n    return PageMetaCollection.setMeta(this.page, this.name, content);\n  }\n\n  /**\n   * メタタグを削除する\n   */\n  delete(): WikidotResultAsync<void> {\n    return PageMetaCollection.deleteMeta(this.page, this.name);\n  }\n\n  toString(): string {\n    return `PageMeta(name=${this.name}, content=${this.content})`;\n  }\n}\n\n/**\n * ページメタタグコレクション\n */\nexport class PageMetaCollection extends Array<PageMeta> {\n  public readonly page: PageRef;\n\n  constructor(page: PageRef, metas?: PageMeta[]) {\n    super();\n    this.page = page;\n    if (metas) {\n      this.push(...metas);\n    }\n  }\n\n  /**\n   * 名前で検索\n   * @param name - メタタグ名\n   * @returns メタタグ（存在しない場合はundefined）\n   */\n  findByName(name: string): PageMeta | undefined {\n    return this.find((meta) => meta.name === name);\n  }\n\n  /**\n   * ページのメタタグを取得する\n   * @param page - ページ参照\n   * @returns メタタグコレクション\n   */\n  static acquire(page: PageRef): WikidotResultAsync<PageMetaCollection> {\n    return fromPromise(\n      (async () => {\n        const result = await page.site.amcRequest([\n          {\n            moduleName: 'edit/EditMetaModule',\n            page_id: page.id,\n          },\n        ]);\n\n        if (result.isErr()) {\n          throw result.error;\n        }\n\n        const response = result.value[0];\n        if (!response) {\n          throw new NoElementError('Empty response');\n        }\n\n        const html = String(response.body ?? '');\n        const $ = cheerio.load(html);\n        const metas: PageMeta[] = [];\n\n        // メタタグテーブルをパース\n        $('table.meta-table tr').each((_i, elem) => {\n          const $row = $(elem);\n          const $cells = $row.find('td');\n          if ($cells.length < 2) return;\n\n          const name = $($cells[0]).text().trim();\n          const content = $($cells[1]).text().trim();\n\n          if (name) {\n            metas.push(\n              new PageMeta({\n                page,\n                name,\n                content,\n              })\n            );\n          }\n        });\n\n        return new PageMetaCollection(page, metas);\n      })(),\n      (error) => {\n        if (error instanceof NoElementError) {\n          return error;\n        }\n        return new UnexpectedError(`Failed to acquire page metas: ${String(error)}`);\n      }\n    );\n  }\n\n  /**\n   * メタタグを設定する\n   * @param page - ページ参照\n   * @param name - メタタグ名\n   * @param content - メタタグの値\n   */\n  static setMeta(page: PageRef, name: string, content: string): WikidotResultAsync<void> {\n    const loginResult = page.site.client.requireLogin();\n    if (loginResult.isErr()) {\n      return fromPromise(\n        Promise.reject(loginResult.error),\n        () => new LoginRequiredError('Login required to set meta tag')\n      );\n    }\n\n    return fromPromise(\n      (async () => {\n        const result = await page.site.amcRequest([\n          {\n            action: 'WikiPageAction',\n            event: 'saveMetaTag',\n            moduleName: 'Empty',\n            page_id: page.id,\n            meta_name: name,\n            meta_content: content,\n          },\n        ]);\n\n        if (result.isErr()) {\n          throw result.error;\n        }\n      })(),\n      (error) => {\n        if (error instanceof LoginRequiredError) {\n          return error;\n        }\n        return new UnexpectedError(`Failed to set meta tag: ${String(error)}`);\n      }\n    );\n  }\n\n  /**\n   * メタタグを削除する\n   * @param page - ページ参照\n   * @param name - メタタグ名\n   */\n  static deleteMeta(page: PageRef, name: string): WikidotResultAsync<void> {\n    const loginResult = page.site.client.requireLogin();\n    if (loginResult.isErr()) {\n      return fromPromise(\n        Promise.reject(loginResult.error),\n        () => new LoginRequiredError('Login required to delete meta tag')\n      );\n    }\n\n    return fromPromise(\n      (async () => {\n        const result = await page.site.amcRequest([\n          {\n            action: 'WikiPageAction',\n            event: 'deleteMetaTag',\n            moduleName: 'Empty',\n            page_id: page.id,\n            meta_name: name,\n          },\n        ]);\n\n        if (result.isErr()) {\n          throw result.error;\n        }\n      })(),\n      (error) => {\n        if (error instanceof LoginRequiredError) {\n          return error;\n        }\n        return new UnexpectedError(`Failed to delete meta tag: ${String(error)}`);\n      }\n    );\n  }\n}\n",
    "import type { Page } from './page';\n\n/**\n * ページソースデータ\n */\nexport interface PageSourceData {\n  page: Page;\n  wikiText: string;\n}\n\n/**\n * ページのソースコード（Wikidot記法）\n */\nexport class PageSource {\n  /** ソースが属するページ */\n  public readonly page: Page;\n\n  /** ソースコード（Wikidot記法） */\n  public readonly wikiText: string;\n\n  constructor(data: PageSourceData) {\n    this.page = data.page;\n    this.wikiText = data.wikiText;\n  }\n\n  toString(): string {\n    return `PageSource(page=${this.page.fullname}, length=${this.wikiText.length})`;\n  }\n}\n",
    "import type { AbstractUser } from '../user';\nimport type { Page } from './page';\n\n/**\n * ページ投票データ\n */\nexport interface PageVoteData {\n  page: Page;\n  user: AbstractUser;\n  value: number;\n}\n\n/**\n * ページへの投票（レーティング）\n */\nexport class PageVote {\n  /** 投票が属するページ */\n  public readonly page: Page;\n\n  /** 投票したユーザー */\n  public readonly user: AbstractUser;\n\n  /** 投票値（+1/-1 または 数値） */\n  public readonly value: number;\n\n  constructor(data: PageVoteData) {\n    this.page = data.page;\n    this.user = data.user;\n    this.value = data.value;\n  }\n\n  toString(): string {\n    return `PageVote(user=${this.user.name}, value=${this.value})`;\n  }\n}\n\n/**\n * ページ投票コレクション\n */\nexport class PageVoteCollection extends Array<PageVote> {\n  public readonly page: Page;\n\n  constructor(page: Page, votes?: PageVote[]) {\n    super();\n    this.page = page;\n    if (votes) {\n      this.push(...votes);\n    }\n  }\n\n  /**\n   * ユーザーで検索\n   * @param user - 検索するユーザー\n   * @returns 投票（存在しない場合はundefined）\n   */\n  findByUser(user: AbstractUser): PageVote | undefined {\n    return this.find((vote) => vote.user.id === user.id);\n  }\n}\n",
    "import type { AbstractUser } from '../user';\n\n/**\n * ページ検索クエリのパラメータ\n */\nexport interface SearchPagesQueryParams {\n  /** ページタイプ（例: 'normal', 'admin'） */\n  pagetype?: string;\n  /** カテゴリ名 */\n  category?: string;\n  /** 検索対象タグ（AND条件） */\n  tags?: string | string[];\n  /** 親ページ名 */\n  parent?: string;\n  /** リンク先ページ名 */\n  linkTo?: string;\n  /** 作成日時条件 */\n  createdAt?: string;\n  /** 更新日時条件 */\n  updatedAt?: string;\n  /** 作成者 */\n  createdBy?: AbstractUser | string;\n  /** レーティング条件 */\n  rating?: string;\n  /** 投票数条件 */\n  votes?: string;\n  /** ページ名条件 */\n  name?: string;\n  /** フルネーム（完全一致） */\n  fullname?: string;\n  /** 範囲指定 */\n  range?: string;\n  /** ソート順（例: 'created_at desc'） */\n  order?: string;\n  /** 取得開始位置 */\n  offset?: number;\n  /** 取得件数制限 */\n  limit?: number;\n  /** 1ページあたり件数 */\n  perPage?: number;\n  /** 個別表示 */\n  separate?: string;\n  /** ラッパー表示 */\n  wrapper?: string;\n}\n\n/**\n * デフォルトの1ページあたり件数\n */\nexport const DEFAULT_PER_PAGE = 250;\n\n/**\n * デフォルトのモジュールボディフィールド\n */\nexport const DEFAULT_MODULE_BODY = [\n  'fullname',\n  'category',\n  'name',\n  'title',\n  'created_at',\n  'created_by_linked',\n  'updated_at',\n  'updated_by_linked',\n  'commented_at',\n  'commented_by_linked',\n  'parent_fullname',\n  'comments',\n  'size',\n  'children',\n  'rating_votes',\n  'rating',\n  'rating_percent',\n  'revisions',\n  'tags',\n  '_tags',\n] as const;\n\n/**\n * ページ検索クエリ\n */\nexport class SearchPagesQuery {\n  /** ページタイプ */\n  pagetype: string;\n  /** カテゴリ */\n  category: string;\n  /** タグ */\n  tags: string | string[] | null;\n  /** 親ページ */\n  parent: string | null;\n  /** リンク先 */\n  linkTo: string | null;\n  /** 作成日時条件 */\n  createdAt: string | null;\n  /** 更新日時条件 */\n  updatedAt: string | null;\n  /** 作成者 */\n  createdBy: AbstractUser | string | null;\n  /** レーティング条件 */\n  rating: string | null;\n  /** 投票数条件 */\n  votes: string | null;\n  /** ページ名条件 */\n  name: string | null;\n  /** フルネーム条件 */\n  fullname: string | null;\n  /** 範囲 */\n  range: string | null;\n  /** ソート順 */\n  order: string;\n  /** オフセット */\n  offset: number;\n  /** 取得件数制限 */\n  limit: number | null;\n  /** 1ページあたり件数 */\n  perPage: number;\n  /** 個別表示 */\n  separate: string;\n  /** ラッパー表示 */\n  wrapper: string;\n\n  constructor(params: SearchPagesQueryParams = {}) {\n    this.pagetype = params.pagetype ?? '*';\n    this.category = params.category ?? '*';\n    this.tags = params.tags ?? null;\n    this.parent = params.parent ?? null;\n    this.linkTo = params.linkTo ?? null;\n    this.createdAt = params.createdAt ?? null;\n    this.updatedAt = params.updatedAt ?? null;\n    this.createdBy = params.createdBy ?? null;\n    this.rating = params.rating ?? null;\n    this.votes = params.votes ?? null;\n    this.name = params.name ?? null;\n    this.fullname = params.fullname ?? null;\n    this.range = params.range ?? null;\n    this.order = params.order ?? 'created_at desc';\n    this.offset = params.offset ?? 0;\n    this.limit = params.limit ?? null;\n    this.perPage = params.perPage ?? DEFAULT_PER_PAGE;\n    this.separate = params.separate ?? 'no';\n    this.wrapper = params.wrapper ?? 'no';\n  }\n\n  /**\n   * 辞書形式に変換\n   */\n  asDict(): Record<string, unknown> {\n    const result: Record<string, unknown> = {};\n\n    if (this.pagetype !== '*') result.pagetype = this.pagetype;\n    if (this.category !== '*') result.category = this.category;\n    if (this.tags !== null) {\n      result.tags = Array.isArray(this.tags) ? this.tags.join(' ') : this.tags;\n    }\n    if (this.parent !== null) result.parent = this.parent;\n    if (this.linkTo !== null) result.link_to = this.linkTo;\n    if (this.createdAt !== null) result.created_at = this.createdAt;\n    if (this.updatedAt !== null) result.updated_at = this.updatedAt;\n    if (this.createdBy !== null) {\n      result.created_by = typeof this.createdBy === 'string' ? this.createdBy : this.createdBy.name;\n    }\n    if (this.rating !== null) result.rating = this.rating;\n    if (this.votes !== null) result.votes = this.votes;\n    if (this.name !== null) result.name = this.name;\n    if (this.fullname !== null) result.fullname = this.fullname;\n    if (this.range !== null) result.range = this.range;\n\n    result.order = this.order;\n    result.offset = this.offset;\n    if (this.limit !== null) result.limit = this.limit;\n    result.perPage = this.perPage;\n    result.separate = this.separate;\n    result.wrapper = this.wrapper;\n\n    return result;\n  }\n}\n",
    "import * as cheerio from 'cheerio';\nimport { NoElementError, UnexpectedError } from '../../common/errors';\nimport { fromPromise, type WikidotResultAsync } from '../../common/types';\nimport { parseOdate, parseUser } from '../../util/parser';\nimport type { Site } from '../site';\nimport type { AbstractUser } from '../user';\n\n/**\n * サイト変更履歴データ\n */\nexport interface SiteChangeData {\n  site: Site;\n  pageFullname: string;\n  pageTitle: string;\n  revisionNo: number;\n  changedBy: AbstractUser | null;\n  changedAt: Date | null;\n  flags: string[];\n  comment: string;\n}\n\n/**\n * サイト変更履歴\n */\nexport class SiteChange {\n  public readonly site: Site;\n  public readonly pageFullname: string;\n  public readonly pageTitle: string;\n  public readonly revisionNo: number;\n  public readonly changedBy: AbstractUser | null;\n  public readonly changedAt: Date | null;\n  public readonly flags: string[];\n  public readonly comment: string;\n\n  constructor(data: SiteChangeData) {\n    this.site = data.site;\n    this.pageFullname = data.pageFullname;\n    this.pageTitle = data.pageTitle;\n    this.revisionNo = data.revisionNo;\n    this.changedBy = data.changedBy;\n    this.changedAt = data.changedAt;\n    this.flags = data.flags;\n    this.comment = data.comment;\n  }\n\n  /**\n   * ページURL\n   */\n  getPageUrl(): string {\n    return `${this.site.getBaseUrl()}/${this.pageFullname}`;\n  }\n\n  toString(): string {\n    return `SiteChange(page=${this.pageFullname}, rev=${this.revisionNo}, by=${this.changedBy})`;\n  }\n}\n\n/**\n * サイト変更履歴コレクション\n */\nexport class SiteChangeCollection extends Array<SiteChange> {\n  public readonly site: Site;\n\n  constructor(site: Site, changes?: SiteChange[]) {\n    super();\n    this.site = site;\n    if (changes) {\n      this.push(...changes);\n    }\n  }\n\n  /**\n   * 最近の変更履歴を取得する\n   * @param site - サイト\n   * @param options - オプション\n   * @returns 変更履歴コレクション\n   */\n  static acquire(\n    site: Site,\n    options?: { perPage?: number; page?: number; limit?: number }\n  ): WikidotResultAsync<SiteChangeCollection> {\n    const perPage = options?.perPage ?? 20;\n    const page = options?.page ?? 1;\n    const limit = options?.limit;\n\n    return fromPromise(\n      (async () => {\n        const result = await site.amcRequest([\n          {\n            moduleName: 'changes/SiteChangesListModule',\n            perpage: perPage,\n            page: page,\n          },\n        ]);\n\n        if (result.isErr()) {\n          throw result.error;\n        }\n\n        const response = result.value[0];\n        if (!response) {\n          throw new NoElementError('Empty response');\n        }\n\n        const html = String(response.body ?? '');\n        const $ = cheerio.load(html);\n        const changes: SiteChange[] = [];\n\n        // テーブル行をパース\n        $('table.wiki-content-table tr').each((_i, elem) => {\n          const $row = $(elem);\n          const $cells = $row.find('td');\n          if ($cells.length < 4) return;\n\n          // ページリンク\n          const pageLink = $($cells[0]).find('a');\n          const href = pageLink.attr('href') ?? '';\n          const pageFullname = href.replace(/^\\//, '').split('/')[0] ?? '';\n          const pageTitle = pageLink.text().trim();\n\n          // リビジョン番号\n          const revText = $($cells[1]).text().trim();\n          const revMatch = revText.match(/(\\d+)/);\n          const revisionNo = revMatch?.[1] ? Number.parseInt(revMatch[1], 10) : 0;\n\n          // フラグ\n          const flagsCell = $($cells[2]);\n          const flags: string[] = [];\n          flagsCell.find('span').each((_j, flagElem) => {\n            const flagClass = $(flagElem).attr('class') ?? '';\n            if (flagClass.includes('spantip')) {\n              const title = $(flagElem).attr('title') ?? '';\n              if (title) flags.push(title);\n            }\n          });\n\n          // ユーザーと日時\n          const infoCell = $($cells[3]);\n          const userElem = infoCell.find('span.printuser');\n          const changedBy = userElem.length > 0 ? parseUser(site.client, userElem) : null;\n\n          const odateElem = infoCell.find('span.odate');\n          const changedAt = odateElem.length > 0 ? parseOdate(odateElem) : null;\n\n          // コメント\n          const commentElem = infoCell.find('span.comments');\n          const comment = commentElem\n            .text()\n            .trim()\n            .replace(/^[\"\"]|[\"\"]$/g, '');\n\n          changes.push(\n            new SiteChange({\n              site,\n              pageFullname,\n              pageTitle,\n              revisionNo,\n              changedBy,\n              changedAt,\n              flags,\n              comment,\n            })\n          );\n        });\n\n        // limitが指定されている場合は結果を制限\n        const limitedChanges = limit !== undefined ? changes.slice(0, limit) : changes;\n        return new SiteChangeCollection(site, limitedChanges);\n      })(),\n      (error) => {\n        if (error instanceof NoElementError) {\n          return error;\n        }\n        return new UnexpectedError(`Failed to acquire site changes: ${String(error)}`);\n      }\n    );\n  }\n}\n",
    "import { NoElementError, UnexpectedError } from '../../../common/errors';\nimport { fromPromise, type WikidotResultAsync } from '../../../common/types';\nimport { parseUser } from '../../../util/parser';\nimport { Page, PageCollection, SearchPagesQuery } from '../../page';\nimport type { Site } from '../site';\n\n/**\n * 単一ページ操作アクセサ\n */\nexport class PageAccessor {\n  public readonly site: Site;\n\n  constructor(site: Site) {\n    this.site = site;\n  }\n\n  /**\n   * UNIX名からページを取得する\n   * @param unixName - ページのUNIX名（例: 'scp-173'）\n   * @returns ページ（存在しない場合はnull）\n   */\n  get(unixName: string): WikidotResultAsync<Page | null> {\n    return fromPromise(\n      (async () => {\n        const query = new SearchPagesQuery({ fullname: unixName });\n        const userParser = parseUser.bind(null, this.site.client);\n\n        const result = await PageCollection.searchPages(this.site, userParser, query);\n        if (result.isErr()) {\n          throw result.error;\n        }\n\n        return result.value.length > 0 ? (result.value[0] ?? null) : null;\n      })(),\n      (error) => {\n        if (error instanceof NoElementError) return error;\n        return new UnexpectedError(`Failed to get page: ${String(error)}`);\n      }\n    );\n  }\n\n  /**\n   * ページを作成する\n   * @param fullname - ページのフルネーム（例: 'scp-173'）\n   * @param options - 作成オプション\n   * @returns void\n   */\n  create(\n    fullname: string,\n    options: {\n      title?: string;\n      source?: string;\n      comment?: string;\n      forceEdit?: boolean;\n    } = {}\n  ): WikidotResultAsync<void> {\n    return PageCollection.createOrEdit(this.site, fullname, {\n      title: options.title,\n      source: options.source,\n      comment: options.comment,\n      forceEdit: options.forceEdit,\n    });\n  }\n}\n\nexport { Page };\n",
    "import { UnexpectedError } from '../../../common/errors';\nimport { fromPromise, type WikidotResultAsync } from '../../../common/types';\nimport { parseUser } from '../../../util/parser';\nimport {\n  Page,\n  PageCollection,\n  SearchPagesQuery,\n  type SearchPagesQueryParams,\n  SiteChangeCollection,\n} from '../../page';\nimport type { Site } from '../site';\n\n/**\n * ページ一覧操作アクセサ\n */\nexport class PagesAccessor {\n  public readonly site: Site;\n\n  constructor(site: Site) {\n    this.site = site;\n  }\n\n  /**\n   * 条件に合うページを検索する\n   * @param params - 検索条件\n   * @returns ページコレクション\n   */\n  search(params?: SearchPagesQueryParams): WikidotResultAsync<PageCollection> {\n    return fromPromise(\n      (async () => {\n        const query = new SearchPagesQuery(params);\n        const userParser = parseUser.bind(null, this.site.client);\n\n        const result = await PageCollection.searchPages(this.site, userParser, query);\n        if (result.isErr()) {\n          throw result.error;\n        }\n\n        return result.value;\n      })(),\n      (error) => new UnexpectedError(`Failed to search pages: ${String(error)}`)\n    );\n  }\n\n  /**\n   * 全ページを取得する\n   * @returns ページコレクション\n   */\n  all(): WikidotResultAsync<PageCollection> {\n    return this.search({});\n  }\n\n  /**\n   * 最近の変更履歴を取得する\n   * @param options - オプション\n   * @param options.perPage - 1ページあたりの件数（デフォルト: 20）\n   * @param options.page - ページ番号（デフォルト: 1）\n   * @returns 変更履歴コレクション\n   */\n  getRecentChanges(options?: {\n    perPage?: number;\n    page?: number;\n  }): WikidotResultAsync<SiteChangeCollection> {\n    return SiteChangeCollection.acquire(this.site, options);\n  }\n}\n\nexport {\n  Page,\n  PageCollection,\n  SearchPagesQuery,\n  SiteChangeCollection,\n  type SearchPagesQueryParams,\n};\n",
    "import * as cheerio from 'cheerio';\nimport { NoElementError, NotFoundException, UnexpectedError } from '../../common/errors';\nimport { fromPromise, type WikidotResultAsync } from '../../common/types';\nimport type { AMCRequestBody, AMCResponse } from '../../connector';\nimport type { Client } from '../client/client';\nimport { ForumAccessor } from './accessors/forum-accessor';\nimport { MemberAccessor } from './accessors/member-accessor';\nimport { PageAccessor } from './accessors/page-accessor';\nimport { PagesAccessor } from './accessors/pages-accessor';\n\n/**\n * サイトデータ\n */\nexport interface SiteData {\n  id: number;\n  title: string;\n  unixName: string;\n  domain: string;\n  sslSupported: boolean;\n}\n\n/**\n * サイトクラス\n */\nexport class Site {\n  public readonly client: Client;\n\n  /** サイトID */\n  public readonly id: number;\n\n  /** サイトタイトル */\n  public readonly title: string;\n\n  /** UNIX名（例: scp-jp） */\n  public readonly unixName: string;\n\n  /** ドメイン */\n  public readonly domain: string;\n\n  /** SSL対応フラグ */\n  public readonly sslSupported: boolean;\n\n  /** ページアクセサ */\n  private _page: PageAccessor | null = null;\n\n  /** ページ一覧アクセサ */\n  private _pages: PagesAccessor | null = null;\n\n  /** フォーラムアクセサ */\n  private _forum: ForumAccessor | null = null;\n\n  /** メンバーアクセサ */\n  private _member: MemberAccessor | null = null;\n\n  constructor(client: Client, data: SiteData) {\n    this.client = client;\n    this.id = data.id;\n    this.title = data.title;\n    this.unixName = data.unixName;\n    this.domain = data.domain;\n    this.sslSupported = data.sslSupported;\n  }\n\n  /**\n   * ページアクセサを取得\n   */\n  get page(): PageAccessor {\n    if (!this._page) {\n      this._page = new PageAccessor(this);\n    }\n    return this._page;\n  }\n\n  /**\n   * ページ一覧アクセサを取得\n   */\n  get pages(): PagesAccessor {\n    if (!this._pages) {\n      this._pages = new PagesAccessor(this);\n    }\n    return this._pages;\n  }\n\n  /**\n   * フォーラムアクセサを取得\n   */\n  get forum(): ForumAccessor {\n    if (!this._forum) {\n      this._forum = new ForumAccessor(this);\n    }\n    return this._forum;\n  }\n\n  /**\n   * メンバーアクセサを取得\n   */\n  get member(): MemberAccessor {\n    if (!this._member) {\n      this._member = new MemberAccessor(this);\n    }\n    return this._member;\n  }\n\n  /**\n   * サイトのベースURLを取得\n   */\n  getBaseUrl(): string {\n    const protocol = this.sslSupported ? 'https' : 'http';\n    return `${protocol}://${this.domain}`;\n  }\n\n  /**\n   * サイトへのAMCリクエストを実行\n   * @param bodies - リクエストボディ配列\n   * @returns AMCレスポンス配列\n   */\n  amcRequest(bodies: AMCRequestBody[]): WikidotResultAsync<AMCResponse[]> {\n    return this.client.amcClient.request(bodies, this.unixName, this.sslSupported);\n  }\n\n  /**\n   * 単一のAMCリクエストを実行\n   * @param body - リクエストボディ\n   * @returns AMCレスポンス\n   */\n  amcRequestSingle(body: AMCRequestBody): WikidotResultAsync<AMCResponse> {\n    return fromPromise(\n      (async () => {\n        const result = await this.amcRequest([body]);\n        if (result.isErr()) {\n          throw result.error;\n        }\n        const response = result.value[0];\n        if (!response) {\n          throw new UnexpectedError('AMC request returned empty response');\n        }\n        return response;\n      })(),\n      (error) => {\n        if (error instanceof UnexpectedError) {\n          return error;\n        }\n        return new UnexpectedError(`AMC request failed: ${String(error)}`);\n      }\n    );\n  }\n\n  /**\n   * UNIX名からサイトを取得する\n   * @param client - クライアント\n   * @param unixName - サイトのUNIX名（例: 'scp-jp'）\n   * @returns サイト\n   */\n  static fromUnixName(client: Client, unixName: string): WikidotResultAsync<Site> {\n    return fromPromise(\n      (async () => {\n        // サイトページを取得\n        const url = `https://${unixName}.wikidot.com`;\n        const response = await fetch(url, {\n          headers: client.amcClient.header.getHeaders(),\n        });\n\n        if (!response.ok) {\n          if (response.status === 404) {\n            throw new NotFoundException(`Site not found: ${unixName}`);\n          }\n          throw new UnexpectedError(`Failed to fetch site: ${response.status}`);\n        }\n\n        const html = await response.text();\n        const $ = cheerio.load(html);\n\n        // WIKIREQUEST.info を解析\n        const scripts = $('script').toArray();\n        let siteId: number | null = null;\n        let siteUnixName: string | null = null;\n        let domain: string | null = null;\n        let title: string | null = null;\n\n        for (const script of scripts) {\n          const content = $(script).html();\n          if (!content || !content.includes('WIKIREQUEST')) {\n            continue;\n          }\n\n          // siteId\n          const siteIdMatch = content.match(/WIKIREQUEST\\.info\\.siteId\\s*=\\s*(\\d+)/);\n          if (siteIdMatch?.[1]) {\n            siteId = Number.parseInt(siteIdMatch[1], 10);\n          }\n\n          // siteUnixName\n          const siteUnixNameMatch = content.match(\n            /WIKIREQUEST\\.info\\.siteUnixName\\s*=\\s*[\"']([^\"']+)[\"']/\n          );\n          if (siteUnixNameMatch?.[1]) {\n            siteUnixName = siteUnixNameMatch[1];\n          }\n\n          // domain\n          const domainMatch = content.match(/WIKIREQUEST\\.info\\.domain\\s*=\\s*[\"']([^\"']+)[\"']/);\n          if (domainMatch?.[1]) {\n            domain = domainMatch[1];\n          }\n        }\n\n        // title from <title> tag\n        title = $('title').text().trim() || null;\n        // Remove \" - Wikidot\" suffix if present\n        if (title?.endsWith(' - Wikidot')) {\n          title = title.slice(0, -10).trim();\n        }\n\n        if (siteId === null) {\n          throw new NoElementError('Site ID not found in WIKIREQUEST');\n        }\n        if (siteUnixName === null) {\n          siteUnixName = unixName; // Fallback\n        }\n        if (domain === null) {\n          domain = `${unixName}.wikidot.com`; // Fallback\n        }\n        if (title === null) {\n          title = unixName; // Fallback\n        }\n\n        // SSL対応チェック（既にhttpsでアクセスできているので true）\n        const sslSupported = true;\n\n        return new Site(client, {\n          id: siteId,\n          title,\n          unixName: siteUnixName,\n          domain,\n          sslSupported,\n        });\n      })(),\n      (error) => {\n        if (error instanceof NotFoundException || error instanceof NoElementError) {\n          return error;\n        }\n        return new UnexpectedError(`Failed to get site: ${String(error)}`);\n      }\n    );\n  }\n\n  toString(): string {\n    return `Site(id=${this.id}, unixName=${this.unixName}, title=${this.title})`;\n  }\n}\n",
    "import type { WikidotResultAsync } from '../../../common/types';\nimport { Site } from '../../site';\nimport type { Client } from '../client';\n\n/**\n * サイト操作アクセサ\n */\nexport class SiteAccessor {\n  public readonly client: Client;\n\n  constructor(client: Client) {\n    this.client = client;\n  }\n\n  /**\n   * UNIX名からサイトを取得する\n   * @param unixName - サイトのUNIX名（例: 'scp-jp'）\n   * @returns サイトオブジェクト\n   */\n  get(unixName: string): WikidotResultAsync<Site> {\n    return Site.fromUnixName(this.client, unixName);\n  }\n}\n\nexport { Site };\n",
    "import { NotFoundException } from '../../../common/errors';\nimport { type WikidotResultAsync, wdErrAsync, wdOkAsync } from '../../../common/types';\nimport { User, type UserCollection } from '../../user';\nimport type { Client } from '../client';\n\n/**\n * ユーザー取得オプション\n */\nexport interface GetUserOptions {\n  /** ユーザーが見つからない場合にエラーを発生させる（デフォルト: false） */\n  raiseWhenNotFound?: boolean;\n}\n\n/**\n * ユーザー操作アクセサ\n */\nexport class UserAccessor {\n  public readonly client: Client;\n\n  constructor(client: Client) {\n    this.client = client;\n  }\n\n  /**\n   * ユーザー名からユーザーを取得する\n   * @param name - ユーザー名\n   * @param options - 取得オプション\n   * @returns ユーザー（存在しない場合はnull、raiseWhenNotFoundがtrueの場合はエラー）\n   */\n  get(name: string, options: GetUserOptions = {}): WikidotResultAsync<User | null> {\n    const { raiseWhenNotFound = false } = options;\n\n    return User.fromName(this.client, name).andThen((user) => {\n      if (user === null && raiseWhenNotFound) {\n        return wdErrAsync(new NotFoundException(`User not found: ${name}`));\n      }\n      return wdOkAsync(user);\n    });\n  }\n\n  /**\n   * 複数ユーザー名からユーザーを取得する\n   * @param names - ユーザー名配列\n   * @param options - 取得オプション\n   * @returns ユーザーコレクション（存在しないユーザーはnull、raiseWhenNotFoundがtrueの場合はエラー）\n   */\n  getMany(names: string[], options: GetUserOptions = {}): WikidotResultAsync<UserCollection> {\n    const { raiseWhenNotFound = false } = options;\n\n    return User.fromNames(this.client, names).andThen((collection) => {\n      if (raiseWhenNotFound) {\n        const notFoundNames: string[] = [];\n        for (let i = 0; i < names.length; i++) {\n          const name = names[i];\n          if (collection[i] === null && name !== undefined) {\n            notFoundNames.push(name);\n          }\n        }\n        if (notFoundNames.length > 0) {\n          return wdErrAsync(new NotFoundException(`Users not found: ${notFoundNames.join(', ')}`));\n        }\n      }\n      return wdOkAsync(collection);\n    });\n  }\n}\n",
    "import { LoginRequiredError } from '../../common/errors';\nimport {\n  fromPromise,\n  type WikidotResult,\n  type WikidotResultAsync,\n  wdErr,\n  wdOk,\n  wdOkAsync,\n} from '../../common/types';\nimport { AMCClient, type AMCConfig, login, logout } from '../../connector';\nimport type { User } from '../user/user';\nimport { PrivateMessageAccessor } from './accessors/pm-accessor';\nimport { SiteAccessor } from './accessors/site-accessor';\nimport { UserAccessor } from './accessors/user-accessor';\n\n/**\n * クライアント作成オプション\n */\nexport interface ClientOptions {\n  /** Wikidotユーザー名 */\n  username?: string;\n\n  /** Wikidotパスワード */\n  password?: string;\n\n  /** ベースドメイン（デフォルト: wikidot.com） */\n  domain?: string;\n\n  /** AMC設定オーバーライド */\n  amcConfig?: Partial<AMCConfig>;\n}\n\n/**\n * Wikidotクライアント\n * ライブラリのメインエントリポイント\n */\nexport class Client {\n  /** AMCクライアント */\n  public readonly amcClient: AMCClient;\n\n  /** ベースドメイン */\n  public readonly domain: string;\n\n  /** ユーザー操作アクセサ */\n  public readonly user: UserAccessor;\n\n  /** サイト操作アクセサ */\n  public readonly site: SiteAccessor;\n\n  /** プライベートメッセージ操作アクセサ */\n  public readonly privateMessage: PrivateMessageAccessor;\n\n  /** ログイン中のユーザー名 */\n  private _username: string | null;\n\n  /** ログイン中のユーザー */\n  private _me: User | null = null;\n\n  /**\n   * プライベートコンストラクタ\n   * createメソッドを使用してインスタンスを作成する\n   */\n  private constructor(amcClient: AMCClient, domain: string, username: string | null = null) {\n    this.amcClient = amcClient;\n    this.domain = domain;\n    this._username = username;\n\n    // アクセサを初期化\n    this.user = new UserAccessor(this);\n    this.site = new SiteAccessor(this);\n    this.privateMessage = new PrivateMessageAccessor(this);\n  }\n\n  /**\n   * ログイン中のユーザー名を取得\n   */\n  get username(): string | null {\n    return this._username;\n  }\n\n  /**\n   * ログイン中のユーザーを取得\n   * ログインしていない場合はnull\n   */\n  get me(): User | null {\n    return this._me;\n  }\n\n  /**\n   * クライアントを作成する\n   * @param options - クライアントオプション\n   * @returns クライアントインスタンス\n   */\n  static create(options: ClientOptions = {}): WikidotResultAsync<Client> {\n    const { username, password, domain = 'wikidot.com', amcConfig = {} } = options;\n\n    // AMCクライアントを作成\n    const amcClient = new AMCClient(amcConfig, domain);\n\n    // 認証情報がある場合はログイン\n    if (username && password) {\n      return fromPromise(\n        (async () => {\n          const client = new Client(amcClient, domain, username);\n          const loginResult = await login(client, username, password);\n          if (loginResult.isErr()) {\n            throw loginResult.error;\n          }\n\n          // ログイン中のユーザー情報を取得\n          const { User } = await import('../user/user');\n          const userResult = await User.fromName(client, username);\n          if (userResult.isOk() && userResult.value) {\n            client._me = userResult.value;\n          }\n\n          return client;\n        })(),\n        (error) => {\n          if (error instanceof LoginRequiredError) {\n            return error;\n          }\n          return new LoginRequiredError(`Failed to create client: ${String(error)}`);\n        }\n      );\n    }\n\n    // 認証なしのクライアントを返す\n    return wdOkAsync(new Client(amcClient, domain));\n  }\n\n  /**\n   * 未認証クライアントを作成する\n   * @param options - クライアントオプション（認証情報以外）\n   * @returns クライアントインスタンス\n   */\n  static createAnonymous(options: Omit<ClientOptions, 'username' | 'password'> = {}): Client {\n    const { domain = 'wikidot.com', amcConfig = {} } = options;\n    const amcClient = new AMCClient(amcConfig, domain);\n    return new Client(amcClient, domain);\n  }\n\n  /**\n   * ログイン状態を確認する\n   * @returns ログイン済みならtrue\n   */\n  isLoggedIn(): boolean {\n    return this._username !== null;\n  }\n\n  /**\n   * ログインを要求する\n   * ログイン済みでない場合はLoginRequiredErrorを返す\n   * @returns 成功時はvoid\n   */\n  requireLogin(): WikidotResult<void> {\n    if (!this.isLoggedIn()) {\n      return wdErr(new LoginRequiredError());\n    }\n    return wdOk(undefined);\n  }\n\n  /**\n   * クライアントをクローズする\n   * セッションがある場合はログアウトを試みる\n   */\n  close(): WikidotResultAsync<void> {\n    if (this.isLoggedIn()) {\n      return logout(this).map(() => {\n        this._username = null;\n        return undefined;\n      });\n    }\n    return wdOkAsync(undefined);\n  }\n}\n"
  ],
  "mappings": "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",
  "debugId": "1A0AF3AA4D5CAB4D64756E2164756E21",
  "names": []
}
|