@dollhousemcp/mcp-server 2.0.29 → 2.0.30

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.
@@ -0,0 +1,126 @@
1
+ export interface PermissionHookMarker {
2
+ host: string;
3
+ scriptPath: string;
4
+ settingsPath?: string;
5
+ additionalPaths?: string[];
6
+ configured?: boolean;
7
+ assetsPrepared?: boolean;
8
+ installedAt: string;
9
+ }
10
+ export interface PermissionHookStatus {
11
+ installed: boolean;
12
+ configured?: boolean;
13
+ assetsPrepared?: boolean;
14
+ assetsCurrent?: boolean;
15
+ autoRepaired?: boolean;
16
+ needsRepair?: boolean;
17
+ repairError?: string;
18
+ host?: string;
19
+ scriptPath?: string;
20
+ settingsPath?: string;
21
+ additionalPaths?: string[];
22
+ }
23
+ export interface InstallPermissionHookResult {
24
+ supported: boolean;
25
+ installed: boolean;
26
+ configured: boolean;
27
+ assetsPrepared?: boolean;
28
+ host: string;
29
+ scriptPath?: string;
30
+ settingsPath?: string;
31
+ additionalPaths?: string[];
32
+ markerPath?: string;
33
+ backupPath?: string;
34
+ message: string;
35
+ }
36
+ export interface InstallPermissionHookOptions {
37
+ homeDir?: string;
38
+ sourceScriptPath?: string;
39
+ now?: Date;
40
+ }
41
+ export interface ReconcilePermissionHookOptions {
42
+ homeDir?: string;
43
+ sourceScriptPath?: string;
44
+ autoRepair?: boolean;
45
+ }
46
+ export interface PermissionHookAuditSummary {
47
+ installedHosts: string[];
48
+ currentHosts: string[];
49
+ repairedHosts: string[];
50
+ needsRepairHosts: string[];
51
+ lastStartupRepair: PermissionHookStartupRepairSummary | null;
52
+ }
53
+ export interface PermissionHookHealthSummary {
54
+ status: 'ok' | 'warning' | 'error';
55
+ message: string;
56
+ repairedCount: number;
57
+ needsRepairCount: number;
58
+ lastCheckedAt?: string;
59
+ }
60
+ export interface PermissionHookStartupRepairHostResult extends PermissionHookStatus {
61
+ host: string;
62
+ outcome: 'current' | 'repaired' | 'needs_repair' | 'not_installed' | 'error';
63
+ }
64
+ export interface PermissionHookStartupRepairSummary {
65
+ startedAt: string;
66
+ completedAt: string;
67
+ durationMs: number;
68
+ repairedCount: number;
69
+ needsRepairCount: number;
70
+ hostResults: PermissionHookStartupRepairHostResult[];
71
+ }
72
+ export interface HookAssetDescriptor {
73
+ kind: 'bridge' | 'port-helper' | 'config-helper' | 'wrapper';
74
+ sourcePath: string;
75
+ targetPath: string;
76
+ }
77
+ export interface HookAssetAuditResult {
78
+ assetsPrepared: boolean;
79
+ assetsCurrent: boolean;
80
+ staleAssets: HookAssetDescriptor[];
81
+ }
82
+ export declare const MANAGED_HOOK_WRAPPER_BASENAMES: {
83
+ readonly vscode: "pretooluse-vscode.sh";
84
+ readonly cursor: "pretooluse-cursor.sh";
85
+ readonly windsurf: "pretooluse-windsurf.sh";
86
+ readonly 'gemini-cli': "pretooluse-gemini.sh";
87
+ readonly codex: "pretooluse-codex.sh";
88
+ };
89
+ export declare const WRAPPER_HOOK_HOSTS: Array<keyof typeof MANAGED_HOOK_WRAPPER_BASENAMES>;
90
+ export declare const AUTO_REPAIRABLE_HOOK_HOSTS: readonly ["claude-code", ...("cursor" | "vscode" | "windsurf" | "gemini-cli" | "codex")[]];
91
+ export declare function isMissingFileError(error: unknown): boolean;
92
+ export declare function detectIndent(raw: string): number | string;
93
+ export declare function getPermissionHookScriptPath(homeDir?: string): string;
94
+ export declare function normalizeHookHost(host: string): string;
95
+ export declare function readHostSpecificHookStatus(homeDir: string, host: string): PermissionHookStatus;
96
+ export declare function collectHookMarkerPaths(homeDir: string): Set<string>;
97
+ export declare function collectHookMarkerPathsAsync(homeDir: string): Promise<Set<string>>;
98
+ export declare function summarizeMarkerStatuses(markerPaths: Iterable<string>): PermissionHookStatus;
99
+ export declare function getHookWrapperBasename(host: string): string | null;
100
+ export declare function getHookWrapperPath(host: string, homeDir?: string): string | null;
101
+ export declare function getHookSourcePath(host: string): string;
102
+ export declare function supportsManagedHookAssets(host: string): boolean;
103
+ export declare function getPrimaryHookScriptPath(host: string, homeDir?: string): string;
104
+ export declare function getManagedHookAssets(host: string, homeDir?: string, sourceScriptPath?: string): HookAssetDescriptor[];
105
+ export declare function auditHookAssets(host: string, homeDir?: string, sourceScriptPath?: string): Promise<HookAssetAuditResult>;
106
+ export declare function getPermissionHookMarkerPath(homeDir?: string, host?: string): string;
107
+ export declare function getClaudeHookSettingsPath(homeDir?: string): string;
108
+ export declare function getVsCodeHookSettingsPath(homeDir?: string): string;
109
+ export declare function getVsCodeUserSettingsPath(homeDir?: string): string;
110
+ export declare function getGeminiHookSettingsPath(homeDir?: string): string;
111
+ export declare function getCursorHookSettingsPath(homeDir?: string): string;
112
+ export declare function getWindsurfHookSettingsPath(homeDir?: string): string;
113
+ export declare function getCodexHookSettingsPath(homeDir?: string): string;
114
+ export declare function getCodexConfigPath(homeDir?: string): string;
115
+ export declare function normalizeHooksRoot(parsed: Record<string, unknown>): Record<string, unknown[]>;
116
+ export declare function ensureCommandHook(parsed: Record<string, unknown>, eventName: string, command: string, matcher: string, extraHookFields?: Record<string, unknown>): {
117
+ changed: boolean;
118
+ parsed: Record<string, unknown>;
119
+ };
120
+ export declare function readOptionalUtf8(filePath: string, fallback: string): Promise<string>;
121
+ export declare function writeBackupIfPresent(filePath: string, raw: string): Promise<string | undefined>;
122
+ export declare function writeHookMarker(homeDir: string, marker: PermissionHookMarker): Promise<string>;
123
+ export declare function installHookAssetsForHost(client: string, homeDir: string, sourceScriptPath?: string): Promise<{
124
+ scriptPath: string;
125
+ }>;
126
+ //# sourceMappingURL=permissionHookShared.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"permissionHookShared.d.ts","sourceRoot":"","sources":["../../src/utils/permissionHookShared.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,388 @@
1
+ import { existsSync, readFileSync, readdirSync, statSync } from 'node:fs';
2
+ import { access, chmod, copyFile, mkdir, readFile, readdir, writeFile } from 'node:fs/promises';
3
+ import { dirname, join } from 'node:path';
4
+ import { fileURLToPath } from 'node:url';
5
+ import { homedir, platform } from 'node:os';
6
+ import { UnicodeValidator } from '../security/validators/unicodeValidator.js';
7
+ import { logger } from './logger.js';
8
+ export const MANAGED_HOOK_WRAPPER_BASENAMES = {
9
+ 'vscode': 'pretooluse-vscode.sh',
10
+ 'cursor': 'pretooluse-cursor.sh',
11
+ 'windsurf': 'pretooluse-windsurf.sh',
12
+ 'gemini-cli': 'pretooluse-gemini.sh',
13
+ 'codex': 'pretooluse-codex.sh',
14
+ };
15
+ export const WRAPPER_HOOK_HOSTS = Object.keys(MANAGED_HOOK_WRAPPER_BASENAMES);
16
+ export const AUTO_REPAIRABLE_HOOK_HOSTS = ['claude-code', ...WRAPPER_HOOK_HOSTS];
17
+ function repoRootFromModule() {
18
+ const currentFile = fileURLToPath(import.meta.url);
19
+ return dirname(dirname(dirname(currentFile)));
20
+ }
21
+ export function isMissingFileError(error) {
22
+ return Boolean(error
23
+ && typeof error === 'object'
24
+ && 'code' in error
25
+ && error.code === 'ENOENT');
26
+ }
27
+ export function detectIndent(raw) {
28
+ for (const line of raw.split('\n')) {
29
+ if (line.length === 0 || line.startsWith('{') || line.startsWith('}'))
30
+ continue;
31
+ if (line.startsWith('\t'))
32
+ return '\t';
33
+ if (line.startsWith(' ')) {
34
+ const spaces = line.length - line.trimStart().length;
35
+ if (spaces >= 2)
36
+ return spaces;
37
+ }
38
+ }
39
+ return 2;
40
+ }
41
+ export function getPermissionHookScriptPath(homeDir = homedir()) {
42
+ return join(homeDir, '.dollhouse', 'hooks', 'pretooluse-dollhouse.sh');
43
+ }
44
+ function getPermissionHookRunDir(homeDir = homedir()) {
45
+ return join(homeDir, '.dollhouse', 'run');
46
+ }
47
+ export function normalizeHookHost(host) {
48
+ return UnicodeValidator.normalize(host).normalizedContent.trim().toLowerCase();
49
+ }
50
+ function isHookMarkerFilename(entry) {
51
+ return entry.startsWith('hook-installed-') && entry.endsWith('.json');
52
+ }
53
+ export function readHostSpecificHookStatus(homeDir, host) {
54
+ const normalized = normalizeHookHost(host);
55
+ const status = readMarkerStatus(getPermissionHookMarkerPath(homeDir, normalized));
56
+ if (status.installed || status.assetsPrepared) {
57
+ return status;
58
+ }
59
+ if (normalized === 'claude-code') {
60
+ return readMarkerStatus(getPermissionHookMarkerPath(homeDir));
61
+ }
62
+ return { installed: false };
63
+ }
64
+ export function collectHookMarkerPaths(homeDir) {
65
+ const markerPaths = new Set([getPermissionHookMarkerPath(homeDir)]);
66
+ const runDir = getPermissionHookRunDir(homeDir);
67
+ try {
68
+ for (const entry of readdirSync(runDir)) {
69
+ if (isHookMarkerFilename(entry)) {
70
+ markerPaths.add(join(runDir, entry));
71
+ }
72
+ }
73
+ }
74
+ catch {
75
+ // No run dir yet — fall through to default false.
76
+ }
77
+ return markerPaths;
78
+ }
79
+ export async function collectHookMarkerPathsAsync(homeDir) {
80
+ const markerPaths = new Set([getPermissionHookMarkerPath(homeDir)]);
81
+ const runDir = getPermissionHookRunDir(homeDir);
82
+ try {
83
+ for (const entry of await readdir(runDir)) {
84
+ if (isHookMarkerFilename(entry)) {
85
+ markerPaths.add(join(runDir, entry));
86
+ }
87
+ }
88
+ }
89
+ catch {
90
+ // No run dir yet — fall through to default false.
91
+ }
92
+ return markerPaths;
93
+ }
94
+ export function summarizeMarkerStatuses(markerPaths) {
95
+ let fallback = { installed: false };
96
+ for (const markerPath of markerPaths) {
97
+ const status = readMarkerStatus(markerPath);
98
+ if (status.installed)
99
+ return status;
100
+ if (!fallback.assetsPrepared && status.assetsPrepared)
101
+ fallback = status;
102
+ }
103
+ return fallback;
104
+ }
105
+ export function getHookWrapperBasename(host) {
106
+ const normalizedHost = normalizeHookHost(host);
107
+ return MANAGED_HOOK_WRAPPER_BASENAMES[normalizedHost] ?? null;
108
+ }
109
+ export function getHookWrapperPath(host, homeDir = homedir()) {
110
+ const basename = getHookWrapperBasename(host);
111
+ return basename ? join(homeDir, '.dollhouse', 'hooks', basename) : null;
112
+ }
113
+ export function getHookSourcePath(host) {
114
+ const root = repoRootFromModule();
115
+ const basename = getHookWrapperBasename(host);
116
+ return basename ? join(root, 'scripts', basename) : join(root, 'scripts', 'pretooluse-dollhouse.sh');
117
+ }
118
+ export function supportsManagedHookAssets(host) {
119
+ const normalized = normalizeHookHost(host);
120
+ return normalized === 'claude-code' || getHookWrapperBasename(normalized) !== null;
121
+ }
122
+ export function getPrimaryHookScriptPath(host, homeDir = homedir()) {
123
+ return getHookWrapperPath(host, homeDir) ?? getPermissionHookScriptPath(homeDir);
124
+ }
125
+ export function getManagedHookAssets(host, homeDir = homedir(), sourceScriptPath) {
126
+ const normalized = normalizeHookHost(host);
127
+ const hooksDir = dirname(getPermissionHookScriptPath(homeDir));
128
+ const assets = [
129
+ {
130
+ kind: 'bridge',
131
+ sourcePath: sourceScriptPath ?? join(repoRootFromModule(), 'scripts', 'pretooluse-dollhouse.sh'),
132
+ targetPath: getPermissionHookScriptPath(homeDir),
133
+ },
134
+ {
135
+ kind: 'port-helper',
136
+ sourcePath: join(repoRootFromModule(), 'scripts', 'permission-port-discovery.sh'),
137
+ targetPath: join(hooksDir, 'permission-port-discovery.sh'),
138
+ },
139
+ {
140
+ kind: 'config-helper',
141
+ sourcePath: join(repoRootFromModule(), 'scripts', 'permission-hook-config.sh'),
142
+ targetPath: join(hooksDir, 'permission-hook-config.sh'),
143
+ },
144
+ ];
145
+ const wrapperTargetPath = getHookWrapperPath(normalized, homeDir);
146
+ if (wrapperTargetPath) {
147
+ assets.push({
148
+ kind: 'wrapper',
149
+ sourcePath: getHookSourcePath(normalized),
150
+ targetPath: wrapperTargetPath,
151
+ });
152
+ }
153
+ return assets;
154
+ }
155
+ async function readOptionalUtf8File(filePath) {
156
+ try {
157
+ return await readFile(filePath, 'utf-8');
158
+ }
159
+ catch (error) {
160
+ if (isMissingFileError(error)) {
161
+ return undefined;
162
+ }
163
+ throw error;
164
+ }
165
+ }
166
+ export async function auditHookAssets(host, homeDir = homedir(), sourceScriptPath) {
167
+ const assets = getManagedHookAssets(host, homeDir, sourceScriptPath);
168
+ const staleAssets = [];
169
+ let assetsPrepared = true;
170
+ for (const asset of assets) {
171
+ const sourceRaw = await readFile(asset.sourcePath, 'utf-8');
172
+ const targetRaw = await readOptionalUtf8File(asset.targetPath);
173
+ if (targetRaw === undefined) {
174
+ assetsPrepared = false;
175
+ staleAssets.push(asset);
176
+ continue;
177
+ }
178
+ if (targetRaw !== sourceRaw) {
179
+ staleAssets.push(asset);
180
+ }
181
+ }
182
+ return {
183
+ assetsPrepared,
184
+ assetsCurrent: staleAssets.length === 0,
185
+ staleAssets,
186
+ };
187
+ }
188
+ export function getPermissionHookMarkerPath(homeDir = homedir(), host) {
189
+ if (!host) {
190
+ return join(getPermissionHookRunDir(homeDir), 'hook-installed.json');
191
+ }
192
+ return join(getPermissionHookRunDir(homeDir), `hook-installed-${normalizeHookHost(host)}.json`);
193
+ }
194
+ export function getClaudeHookSettingsPath(homeDir = homedir()) {
195
+ return join(homeDir, '.claude', 'settings.json');
196
+ }
197
+ export function getVsCodeHookSettingsPath(homeDir = homedir()) {
198
+ return join(homeDir, '.copilot', 'hooks', 'dollhouse-permissions.json');
199
+ }
200
+ export function getVsCodeUserSettingsPath(homeDir = homedir()) {
201
+ const currentPlatform = platform();
202
+ if (currentPlatform === 'darwin') {
203
+ return join(homeDir, 'Library', 'Application Support', 'Code', 'User', 'settings.json');
204
+ }
205
+ if (currentPlatform === 'win32') {
206
+ const appData = process.env.APPDATA || join(homeDir, 'AppData', 'Roaming');
207
+ return join(appData, 'Code', 'User', 'settings.json');
208
+ }
209
+ return join(homeDir, '.config', 'Code', 'User', 'settings.json');
210
+ }
211
+ export function getGeminiHookSettingsPath(homeDir = homedir()) {
212
+ return join(homeDir, '.gemini', 'settings.json');
213
+ }
214
+ export function getCursorHookSettingsPath(homeDir = homedir()) {
215
+ return join(homeDir, '.cursor', 'hooks.json');
216
+ }
217
+ export function getWindsurfHookSettingsPath(homeDir = homedir()) {
218
+ return join(homeDir, '.codeium', 'windsurf', 'hooks.json');
219
+ }
220
+ export function getCodexHookSettingsPath(homeDir = homedir()) {
221
+ return join(homeDir, '.codex', 'hooks.json');
222
+ }
223
+ export function getCodexConfigPath(homeDir = homedir()) {
224
+ return join(homeDir, '.codex', 'config.toml');
225
+ }
226
+ function toPermissionHookStatus(raw) {
227
+ if (typeof raw.host !== 'string' ||
228
+ typeof raw.scriptPath !== 'string') {
229
+ return { installed: false };
230
+ }
231
+ const scriptReady = existsSync(raw.scriptPath);
232
+ const settingsReady = !raw.settingsPath || existsSync(raw.settingsPath);
233
+ const additionalPathsReady = !raw.additionalPaths || raw.additionalPaths.every((path) => existsSync(path));
234
+ const assetsPrepared = (raw.assetsPrepared ?? raw.configured ?? true) && scriptReady;
235
+ const configured = (raw.configured ?? true) && scriptReady && settingsReady && additionalPathsReady;
236
+ return {
237
+ installed: configured,
238
+ configured,
239
+ assetsPrepared,
240
+ host: raw.host,
241
+ scriptPath: raw.scriptPath,
242
+ settingsPath: raw.settingsPath,
243
+ additionalPaths: raw.additionalPaths,
244
+ };
245
+ }
246
+ function readMarkerStatus(markerPath) {
247
+ try {
248
+ const raw = readFileSync(markerPath, 'utf-8');
249
+ return toPermissionHookStatus(JSON.parse(raw));
250
+ }
251
+ catch (error) {
252
+ if (!isMissingFileError(error)) {
253
+ logger.warn(`[Permissions] Failed to read hook marker at ${markerPath}: ${String(error)}`);
254
+ }
255
+ return { installed: false };
256
+ }
257
+ }
258
+ export function normalizeHooksRoot(parsed) {
259
+ const hooksValue = parsed.hooks;
260
+ if (!hooksValue || typeof hooksValue !== 'object' || Array.isArray(hooksValue)) {
261
+ parsed.hooks = {};
262
+ }
263
+ return parsed.hooks;
264
+ }
265
+ export function ensureCommandHook(parsed, eventName, command, matcher, extraHookFields = {}) {
266
+ const hooksRoot = normalizeHooksRoot(parsed);
267
+ const existingEntries = Array.isArray(hooksRoot[eventName])
268
+ ? hooksRoot[eventName].filter((entry) => typeof entry === 'object' && entry !== null)
269
+ : [];
270
+ hooksRoot[eventName] = existingEntries;
271
+ const commandExists = existingEntries.some((entry) => {
272
+ const hooks = Array.isArray(entry?.hooks) ? entry.hooks : [];
273
+ return hooks.some((hook) => hook?.type === 'command' && hook?.command === command);
274
+ });
275
+ if (commandExists) {
276
+ return { changed: false, parsed };
277
+ }
278
+ const wildcardEntry = existingEntries.find((entry) => (entry?.matcher === matcher || entry?.matcher === undefined) && Array.isArray(entry?.hooks));
279
+ if (wildcardEntry) {
280
+ const hooks = wildcardEntry.hooks;
281
+ hooks.push({
282
+ type: 'command',
283
+ command,
284
+ ...extraHookFields,
285
+ });
286
+ }
287
+ else {
288
+ existingEntries.push({
289
+ matcher,
290
+ hooks: [
291
+ {
292
+ type: 'command',
293
+ command,
294
+ ...extraHookFields,
295
+ },
296
+ ],
297
+ });
298
+ }
299
+ return { changed: true, parsed };
300
+ }
301
+ async function copyHookAsset(sourcePath, targetPath) {
302
+ await mkdir(dirname(targetPath), { recursive: true });
303
+ const sourceRaw = await readFile(sourcePath, 'utf-8');
304
+ let targetRaw;
305
+ try {
306
+ targetRaw = await readFile(targetPath, 'utf-8');
307
+ }
308
+ catch (error) {
309
+ if (!isMissingFileError(error)) {
310
+ throw error;
311
+ }
312
+ }
313
+ const changed = targetRaw === undefined || sourceRaw !== targetRaw;
314
+ if (changed) {
315
+ await copyFile(sourcePath, targetPath);
316
+ }
317
+ else {
318
+ await access(targetPath);
319
+ }
320
+ await chmod(targetPath, 0o755);
321
+ return changed;
322
+ }
323
+ export async function readOptionalUtf8(filePath, fallback) {
324
+ try {
325
+ return await readFile(filePath, 'utf-8');
326
+ }
327
+ catch (error) {
328
+ if (isMissingFileError(error)) {
329
+ return fallback;
330
+ }
331
+ throw error;
332
+ }
333
+ }
334
+ export async function writeBackupIfPresent(filePath, raw) {
335
+ if (!existsSync(filePath)) {
336
+ return undefined;
337
+ }
338
+ const backupPath = `${filePath}.dollhouse.bak`;
339
+ await writeFile(backupPath, raw, 'utf-8');
340
+ return backupPath;
341
+ }
342
+ export async function writeHookMarker(homeDir, marker) {
343
+ const markerPath = getPermissionHookMarkerPath(homeDir, marker.host);
344
+ await mkdir(dirname(markerPath), { recursive: true });
345
+ await writeFile(markerPath, JSON.stringify(marker, null, 2) + '\n', 'utf-8');
346
+ return markerPath;
347
+ }
348
+ export async function installHookAssetsForHost(client, homeDir, sourceScriptPath) {
349
+ const normalizedClient = normalizeHookHost(client);
350
+ const hooksDir = dirname(getPermissionHookScriptPath(homeDir));
351
+ const sharedTargetPath = getPermissionHookScriptPath(homeDir);
352
+ const sharedSourcePath = sourceScriptPath ?? join(repoRootFromModule(), 'scripts', 'pretooluse-dollhouse.sh');
353
+ const portHelperSourcePath = join(repoRootFromModule(), 'scripts', 'permission-port-discovery.sh');
354
+ const portHelperTargetPath = join(hooksDir, 'permission-port-discovery.sh');
355
+ const configHelperSourcePath = join(repoRootFromModule(), 'scripts', 'permission-hook-config.sh');
356
+ const configHelperTargetPath = join(hooksDir, 'permission-hook-config.sh');
357
+ const sharedStat = statSync(sharedSourcePath);
358
+ if (!sharedStat.isFile()) {
359
+ logger.warn(`[PermissionHooks] Shared hook bridge missing for ${normalizedClient}: ${sharedSourcePath}`);
360
+ throw new Error(`Permission hook source script not found: ${sharedSourcePath}`);
361
+ }
362
+ await copyHookAsset(sharedSourcePath, sharedTargetPath);
363
+ const portHelperStat = statSync(portHelperSourcePath);
364
+ if (!portHelperStat.isFile()) {
365
+ logger.warn(`[PermissionHooks] Port discovery helper missing for ${normalizedClient}: ${portHelperSourcePath}`);
366
+ throw new Error(`Permission hook helper script not found: ${portHelperSourcePath}`);
367
+ }
368
+ await copyHookAsset(portHelperSourcePath, portHelperTargetPath);
369
+ const configHelperStat = statSync(configHelperSourcePath);
370
+ if (!configHelperStat.isFile()) {
371
+ logger.warn(`[PermissionHooks] Config helper missing for ${normalizedClient}: ${configHelperSourcePath}`);
372
+ throw new Error(`Permission hook config helper not found: ${configHelperSourcePath}`);
373
+ }
374
+ await copyHookAsset(configHelperSourcePath, configHelperTargetPath);
375
+ const wrapperTargetPath = getHookWrapperPath(normalizedClient, homeDir);
376
+ if (!wrapperTargetPath) {
377
+ return { scriptPath: sharedTargetPath };
378
+ }
379
+ const wrapperSourcePath = getHookSourcePath(normalizedClient);
380
+ const wrapperStat = statSync(wrapperSourcePath);
381
+ if (!wrapperStat.isFile()) {
382
+ logger.warn(`[PermissionHooks] Wrapper hook script missing for ${normalizedClient}: ${wrapperSourcePath}`);
383
+ throw new Error(`Permission hook wrapper script not found: ${wrapperSourcePath}`);
384
+ }
385
+ await copyHookAsset(wrapperSourcePath, wrapperTargetPath);
386
+ return { scriptPath: wrapperTargetPath };
387
+ }
388
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"permissionHookShared.js","sourceRoot":"","sources":["../../src/utils/permissionHookShared.ts"],"names":[],"mappings":"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","sourcesContent":["import { existsSync, readFileSync, readdirSync, statSync } from 'node:fs';\nimport { access, chmod, copyFile, mkdir, readFile, readdir, writeFile } from 'node:fs/promises';\nimport { dirname, join } from 'node:path';\nimport { fileURLToPath } from 'node:url';\nimport { homedir, platform } from 'node:os';\nimport { UnicodeValidator } from '../security/validators/unicodeValidator.js';\nimport { logger } from './logger.js';\n\nexport interface PermissionHookMarker {\n  host: string;\n  scriptPath: string;\n  settingsPath?: string;\n  additionalPaths?: string[];\n  configured?: boolean;\n  assetsPrepared?: boolean;\n  installedAt: string;\n}\n\nexport interface PermissionHookStatus {\n  installed: boolean;\n  configured?: boolean;\n  assetsPrepared?: boolean;\n  assetsCurrent?: boolean;\n  autoRepaired?: boolean;\n  needsRepair?: boolean;\n  repairError?: string;\n  host?: string;\n  scriptPath?: string;\n  settingsPath?: string;\n  additionalPaths?: string[];\n}\n\nexport interface InstallPermissionHookResult {\n  supported: boolean;\n  installed: boolean;\n  configured: boolean;\n  assetsPrepared?: boolean;\n  host: string;\n  scriptPath?: string;\n  settingsPath?: string;\n  additionalPaths?: string[];\n  markerPath?: string;\n  backupPath?: string;\n  message: string;\n}\n\nexport interface InstallPermissionHookOptions {\n  homeDir?: string;\n  sourceScriptPath?: string;\n  now?: Date;\n}\n\nexport interface ReconcilePermissionHookOptions {\n  homeDir?: string;\n  sourceScriptPath?: string;\n  autoRepair?: boolean;\n}\n\nexport interface PermissionHookAuditSummary {\n  installedHosts: string[];\n  currentHosts: string[];\n  repairedHosts: string[];\n  needsRepairHosts: string[];\n  lastStartupRepair: PermissionHookStartupRepairSummary | null;\n}\n\nexport interface PermissionHookHealthSummary {\n  status: 'ok' | 'warning' | 'error';\n  message: string;\n  repairedCount: number;\n  needsRepairCount: number;\n  lastCheckedAt?: string;\n}\n\nexport interface PermissionHookStartupRepairHostResult extends PermissionHookStatus {\n  host: string;\n  outcome: 'current' | 'repaired' | 'needs_repair' | 'not_installed' | 'error';\n}\n\nexport interface PermissionHookStartupRepairSummary {\n  startedAt: string;\n  completedAt: string;\n  durationMs: number;\n  repairedCount: number;\n  needsRepairCount: number;\n  hostResults: PermissionHookStartupRepairHostResult[];\n}\n\nexport interface HookAssetDescriptor {\n  kind: 'bridge' | 'port-helper' | 'config-helper' | 'wrapper';\n  sourcePath: string;\n  targetPath: string;\n}\n\nexport interface HookAssetAuditResult {\n  assetsPrepared: boolean;\n  assetsCurrent: boolean;\n  staleAssets: HookAssetDescriptor[];\n}\n\nexport const MANAGED_HOOK_WRAPPER_BASENAMES = {\n  'vscode': 'pretooluse-vscode.sh',\n  'cursor': 'pretooluse-cursor.sh',\n  'windsurf': 'pretooluse-windsurf.sh',\n  'gemini-cli': 'pretooluse-gemini.sh',\n  'codex': 'pretooluse-codex.sh',\n} as const;\n\nexport const WRAPPER_HOOK_HOSTS = Object.keys(MANAGED_HOOK_WRAPPER_BASENAMES) as Array<keyof typeof MANAGED_HOOK_WRAPPER_BASENAMES>;\n\nexport const AUTO_REPAIRABLE_HOOK_HOSTS = ['claude-code', ...WRAPPER_HOOK_HOSTS] as const;\n\nfunction repoRootFromModule(): string {\n  const currentFile = fileURLToPath(import.meta.url);\n  return dirname(dirname(dirname(currentFile)));\n}\n\nexport function isMissingFileError(error: unknown): boolean {\n  return Boolean(\n    error\n    && typeof error === 'object'\n    && 'code' in error\n    && (error as { code?: string }).code === 'ENOENT',\n  );\n}\n\nexport function detectIndent(raw: string): number | string {\n  for (const line of raw.split('\\n')) {\n    if (line.length === 0 || line.startsWith('{') || line.startsWith('}')) continue;\n    if (line.startsWith('\\t')) return '\\t';\n    if (line.startsWith(' ')) {\n      const spaces = line.length - line.trimStart().length;\n      if (spaces >= 2) return spaces;\n    }\n  }\n  return 2;\n}\n\nexport function getPermissionHookScriptPath(homeDir = homedir()): string {\n  return join(homeDir, '.dollhouse', 'hooks', 'pretooluse-dollhouse.sh');\n}\n\nfunction getPermissionHookRunDir(homeDir = homedir()): string {\n  return join(homeDir, '.dollhouse', 'run');\n}\n\nexport function normalizeHookHost(host: string): string {\n  return UnicodeValidator.normalize(host).normalizedContent.trim().toLowerCase();\n}\n\nfunction isHookMarkerFilename(entry: string): boolean {\n  return entry.startsWith('hook-installed-') && entry.endsWith('.json');\n}\n\nexport function readHostSpecificHookStatus(homeDir: string, host: string): PermissionHookStatus {\n  const normalized = normalizeHookHost(host);\n  const status = readMarkerStatus(getPermissionHookMarkerPath(homeDir, normalized));\n  if (status.installed || status.assetsPrepared) {\n    return status;\n  }\n  if (normalized === 'claude-code') {\n    return readMarkerStatus(getPermissionHookMarkerPath(homeDir));\n  }\n  return { installed: false };\n}\n\nexport function collectHookMarkerPaths(homeDir: string): Set<string> {\n  const markerPaths = new Set<string>([getPermissionHookMarkerPath(homeDir)]);\n  const runDir = getPermissionHookRunDir(homeDir);\n  try {\n    for (const entry of readdirSync(runDir)) {\n      if (isHookMarkerFilename(entry)) {\n        markerPaths.add(join(runDir, entry));\n      }\n    }\n  } catch {\n    // No run dir yet — fall through to default false.\n  }\n  return markerPaths;\n}\n\nexport async function collectHookMarkerPathsAsync(homeDir: string): Promise<Set<string>> {\n  const markerPaths = new Set<string>([getPermissionHookMarkerPath(homeDir)]);\n  const runDir = getPermissionHookRunDir(homeDir);\n  try {\n    for (const entry of await readdir(runDir)) {\n      if (isHookMarkerFilename(entry)) {\n        markerPaths.add(join(runDir, entry));\n      }\n    }\n  } catch {\n    // No run dir yet — fall through to default false.\n  }\n  return markerPaths;\n}\n\nexport function summarizeMarkerStatuses(markerPaths: Iterable<string>): PermissionHookStatus {\n  let fallback: PermissionHookStatus = { installed: false };\n  for (const markerPath of markerPaths) {\n    const status = readMarkerStatus(markerPath);\n    if (status.installed) return status;\n    if (!fallback.assetsPrepared && status.assetsPrepared) fallback = status;\n  }\n  return fallback;\n}\n\nexport function getHookWrapperBasename(host: string): string | null {\n  const normalizedHost = normalizeHookHost(host);\n  return MANAGED_HOOK_WRAPPER_BASENAMES[normalizedHost as keyof typeof MANAGED_HOOK_WRAPPER_BASENAMES] ?? null;\n}\n\nexport function getHookWrapperPath(host: string, homeDir = homedir()): string | null {\n  const basename = getHookWrapperBasename(host);\n  return basename ? join(homeDir, '.dollhouse', 'hooks', basename) : null;\n}\n\nexport function getHookSourcePath(host: string): string {\n  const root = repoRootFromModule();\n  const basename = getHookWrapperBasename(host);\n  return basename ? join(root, 'scripts', basename) : join(root, 'scripts', 'pretooluse-dollhouse.sh');\n}\n\nexport function supportsManagedHookAssets(host: string): boolean {\n  const normalized = normalizeHookHost(host);\n  return normalized === 'claude-code' || getHookWrapperBasename(normalized) !== null;\n}\n\nexport function getPrimaryHookScriptPath(host: string, homeDir = homedir()): string {\n  return getHookWrapperPath(host, homeDir) ?? getPermissionHookScriptPath(homeDir);\n}\n\nexport function getManagedHookAssets(\n  host: string,\n  homeDir = homedir(),\n  sourceScriptPath?: string,\n): HookAssetDescriptor[] {\n  const normalized = normalizeHookHost(host);\n  const hooksDir = dirname(getPermissionHookScriptPath(homeDir));\n  const assets: HookAssetDescriptor[] = [\n    {\n      kind: 'bridge',\n      sourcePath: sourceScriptPath ?? join(repoRootFromModule(), 'scripts', 'pretooluse-dollhouse.sh'),\n      targetPath: getPermissionHookScriptPath(homeDir),\n    },\n    {\n      kind: 'port-helper',\n      sourcePath: join(repoRootFromModule(), 'scripts', 'permission-port-discovery.sh'),\n      targetPath: join(hooksDir, 'permission-port-discovery.sh'),\n    },\n    {\n      kind: 'config-helper',\n      sourcePath: join(repoRootFromModule(), 'scripts', 'permission-hook-config.sh'),\n      targetPath: join(hooksDir, 'permission-hook-config.sh'),\n    },\n  ];\n\n  const wrapperTargetPath = getHookWrapperPath(normalized, homeDir);\n  if (wrapperTargetPath) {\n    assets.push({\n      kind: 'wrapper',\n      sourcePath: getHookSourcePath(normalized),\n      targetPath: wrapperTargetPath,\n    });\n  }\n\n  return assets;\n}\n\nasync function readOptionalUtf8File(filePath: string): Promise<string | undefined> {\n  try {\n    return await readFile(filePath, 'utf-8');\n  } catch (error) {\n    if (isMissingFileError(error)) {\n      return undefined;\n    }\n    throw error;\n  }\n}\n\nexport async function auditHookAssets(\n  host: string,\n  homeDir = homedir(),\n  sourceScriptPath?: string,\n): Promise<HookAssetAuditResult> {\n  const assets = getManagedHookAssets(host, homeDir, sourceScriptPath);\n  const staleAssets: HookAssetDescriptor[] = [];\n  let assetsPrepared = true;\n\n  for (const asset of assets) {\n    const sourceRaw = await readFile(asset.sourcePath, 'utf-8');\n    const targetRaw = await readOptionalUtf8File(asset.targetPath);\n    if (targetRaw === undefined) {\n      assetsPrepared = false;\n      staleAssets.push(asset);\n      continue;\n    }\n    if (targetRaw !== sourceRaw) {\n      staleAssets.push(asset);\n    }\n  }\n\n  return {\n    assetsPrepared,\n    assetsCurrent: staleAssets.length === 0,\n    staleAssets,\n  };\n}\n\nexport function getPermissionHookMarkerPath(homeDir = homedir(), host?: string): string {\n  if (!host) {\n    return join(getPermissionHookRunDir(homeDir), 'hook-installed.json');\n  }\n  return join(getPermissionHookRunDir(homeDir), `hook-installed-${normalizeHookHost(host)}.json`);\n}\n\nexport function getClaudeHookSettingsPath(homeDir = homedir()): string {\n  return join(homeDir, '.claude', 'settings.json');\n}\n\nexport function getVsCodeHookSettingsPath(homeDir = homedir()): string {\n  return join(homeDir, '.copilot', 'hooks', 'dollhouse-permissions.json');\n}\n\nexport function getVsCodeUserSettingsPath(homeDir = homedir()): string {\n  const currentPlatform = platform();\n  if (currentPlatform === 'darwin') {\n    return join(homeDir, 'Library', 'Application Support', 'Code', 'User', 'settings.json');\n  }\n  if (currentPlatform === 'win32') {\n    const appData = process.env.APPDATA || join(homeDir, 'AppData', 'Roaming');\n    return join(appData, 'Code', 'User', 'settings.json');\n  }\n  return join(homeDir, '.config', 'Code', 'User', 'settings.json');\n}\n\nexport function getGeminiHookSettingsPath(homeDir = homedir()): string {\n  return join(homeDir, '.gemini', 'settings.json');\n}\n\nexport function getCursorHookSettingsPath(homeDir = homedir()): string {\n  return join(homeDir, '.cursor', 'hooks.json');\n}\n\nexport function getWindsurfHookSettingsPath(homeDir = homedir()): string {\n  return join(homeDir, '.codeium', 'windsurf', 'hooks.json');\n}\n\nexport function getCodexHookSettingsPath(homeDir = homedir()): string {\n  return join(homeDir, '.codex', 'hooks.json');\n}\n\nexport function getCodexConfigPath(homeDir = homedir()): string {\n  return join(homeDir, '.codex', 'config.toml');\n}\n\nfunction toPermissionHookStatus(raw: PermissionHookMarker): PermissionHookStatus {\n  if (\n    typeof raw.host !== 'string' ||\n    typeof raw.scriptPath !== 'string'\n  ) {\n    return { installed: false };\n  }\n\n  const scriptReady = existsSync(raw.scriptPath);\n  const settingsReady = !raw.settingsPath || existsSync(raw.settingsPath);\n  const additionalPathsReady = !raw.additionalPaths || raw.additionalPaths.every((path) => existsSync(path));\n  const assetsPrepared = (raw.assetsPrepared ?? raw.configured ?? true) && scriptReady;\n  const configured = (raw.configured ?? true) && scriptReady && settingsReady && additionalPathsReady;\n\n  return {\n    installed: configured,\n    configured,\n    assetsPrepared,\n    host: raw.host,\n    scriptPath: raw.scriptPath,\n    settingsPath: raw.settingsPath,\n    additionalPaths: raw.additionalPaths,\n  };\n}\n\nfunction readMarkerStatus(markerPath: string): PermissionHookStatus {\n  try {\n    const raw = readFileSync(markerPath, 'utf-8');\n    return toPermissionHookStatus(JSON.parse(raw) as PermissionHookMarker);\n  } catch (error) {\n    if (!isMissingFileError(error)) {\n      logger.warn(`[Permissions] Failed to read hook marker at ${markerPath}: ${String(error)}`);\n    }\n    return { installed: false };\n  }\n}\n\nexport function normalizeHooksRoot(parsed: Record<string, unknown>): Record<string, unknown[]> {\n  const hooksValue = parsed.hooks;\n  if (!hooksValue || typeof hooksValue !== 'object' || Array.isArray(hooksValue)) {\n    parsed.hooks = {};\n  }\n  return parsed.hooks as Record<string, unknown[]>;\n}\n\nexport function ensureCommandHook(\n  parsed: Record<string, unknown>,\n  eventName: string,\n  command: string,\n  matcher: string,\n  extraHookFields: Record<string, unknown> = {},\n): { changed: boolean; parsed: Record<string, unknown> } {\n  const hooksRoot = normalizeHooksRoot(parsed);\n  const existingEntries: Array<Record<string, unknown>> = Array.isArray(hooksRoot[eventName])\n    ? hooksRoot[eventName].filter((entry): entry is Record<string, unknown> => typeof entry === 'object' && entry !== null)\n    : [];\n  hooksRoot[eventName] = existingEntries;\n\n  const commandExists = existingEntries.some((entry) => {\n    const hooks = Array.isArray(entry?.hooks) ? entry.hooks as Array<Record<string, unknown>> : [];\n    return hooks.some((hook) => hook?.type === 'command' && hook?.command === command);\n  });\n  if (commandExists) {\n    return { changed: false, parsed };\n  }\n\n  const wildcardEntry = existingEntries.find((entry): entry is Record<string, unknown> =>\n    (entry?.matcher === matcher || entry?.matcher === undefined) && Array.isArray(entry?.hooks),\n  );\n\n  if (wildcardEntry) {\n    const hooks = wildcardEntry.hooks as Array<Record<string, unknown>>;\n    hooks.push({\n      type: 'command',\n      command,\n      ...extraHookFields,\n    });\n  } else {\n    existingEntries.push({\n      matcher,\n      hooks: [\n        {\n          type: 'command',\n          command,\n          ...extraHookFields,\n        },\n      ],\n    });\n  }\n\n  return { changed: true, parsed };\n}\n\nasync function copyHookAsset(sourcePath: string, targetPath: string): Promise<boolean> {\n  await mkdir(dirname(targetPath), { recursive: true });\n\n  const sourceRaw = await readFile(sourcePath, 'utf-8');\n\n  let targetRaw: string | undefined;\n  try {\n    targetRaw = await readFile(targetPath, 'utf-8');\n  } catch (error) {\n    if (!isMissingFileError(error)) {\n      throw error;\n    }\n  }\n  const changed = targetRaw === undefined || sourceRaw !== targetRaw;\n\n  if (changed) {\n    await copyFile(sourcePath, targetPath);\n  } else {\n    await access(targetPath);\n  }\n\n  await chmod(targetPath, 0o755);\n  return changed;\n}\n\nexport async function readOptionalUtf8(filePath: string, fallback: string): Promise<string> {\n  try {\n    return await readFile(filePath, 'utf-8');\n  } catch (error) {\n    if (isMissingFileError(error)) {\n      return fallback;\n    }\n    throw error;\n  }\n}\n\nexport async function writeBackupIfPresent(filePath: string, raw: string): Promise<string | undefined> {\n  if (!existsSync(filePath)) {\n    return undefined;\n  }\n\n  const backupPath = `${filePath}.dollhouse.bak`;\n  await writeFile(backupPath, raw, 'utf-8');\n  return backupPath;\n}\n\nexport async function writeHookMarker(\n  homeDir: string,\n  marker: PermissionHookMarker,\n): Promise<string> {\n  const markerPath = getPermissionHookMarkerPath(homeDir, marker.host);\n  await mkdir(dirname(markerPath), { recursive: true });\n  await writeFile(markerPath, JSON.stringify(marker, null, 2) + '\\n', 'utf-8');\n  return markerPath;\n}\n\nexport async function installHookAssetsForHost(\n  client: string,\n  homeDir: string,\n  sourceScriptPath?: string,\n): Promise<{ scriptPath: string }> {\n  const normalizedClient = normalizeHookHost(client);\n  const hooksDir = dirname(getPermissionHookScriptPath(homeDir));\n  const sharedTargetPath = getPermissionHookScriptPath(homeDir);\n  const sharedSourcePath = sourceScriptPath ?? join(repoRootFromModule(), 'scripts', 'pretooluse-dollhouse.sh');\n  const portHelperSourcePath = join(repoRootFromModule(), 'scripts', 'permission-port-discovery.sh');\n  const portHelperTargetPath = join(hooksDir, 'permission-port-discovery.sh');\n  const configHelperSourcePath = join(repoRootFromModule(), 'scripts', 'permission-hook-config.sh');\n  const configHelperTargetPath = join(hooksDir, 'permission-hook-config.sh');\n\n  const sharedStat = statSync(sharedSourcePath);\n  if (!sharedStat.isFile()) {\n    logger.warn(`[PermissionHooks] Shared hook bridge missing for ${normalizedClient}: ${sharedSourcePath}`);\n    throw new Error(`Permission hook source script not found: ${sharedSourcePath}`);\n  }\n  await copyHookAsset(sharedSourcePath, sharedTargetPath);\n\n  const portHelperStat = statSync(portHelperSourcePath);\n  if (!portHelperStat.isFile()) {\n    logger.warn(`[PermissionHooks] Port discovery helper missing for ${normalizedClient}: ${portHelperSourcePath}`);\n    throw new Error(`Permission hook helper script not found: ${portHelperSourcePath}`);\n  }\n  await copyHookAsset(portHelperSourcePath, portHelperTargetPath);\n\n  const configHelperStat = statSync(configHelperSourcePath);\n  if (!configHelperStat.isFile()) {\n    logger.warn(`[PermissionHooks] Config helper missing for ${normalizedClient}: ${configHelperSourcePath}`);\n    throw new Error(`Permission hook config helper not found: ${configHelperSourcePath}`);\n  }\n  await copyHookAsset(configHelperSourcePath, configHelperTargetPath);\n\n  const wrapperTargetPath = getHookWrapperPath(normalizedClient, homeDir);\n  if (!wrapperTargetPath) {\n    return { scriptPath: sharedTargetPath };\n  }\n\n  const wrapperSourcePath = getHookSourcePath(normalizedClient);\n  const wrapperStat = statSync(wrapperSourcePath);\n  if (!wrapperStat.isFile()) {\n    logger.warn(`[PermissionHooks] Wrapper hook script missing for ${normalizedClient}: ${wrapperSourcePath}`);\n    throw new Error(`Permission hook wrapper script not found: ${wrapperSourcePath}`);\n  }\n  await copyHookAsset(wrapperSourcePath, wrapperTargetPath);\n  return { scriptPath: wrapperTargetPath };\n}\n"]}
@@ -0,0 +1,10 @@
1
+ import { type PermissionHookAuditSummary, type PermissionHookHealthSummary, type PermissionHookStartupRepairSummary, type PermissionHookStatus, type ReconcilePermissionHookOptions } from './permissionHookShared.js';
2
+ export declare function getPermissionHookStatus(homeDir?: string, host?: string): PermissionHookStatus;
3
+ export declare function getPermissionHookStatusAsync(homeDir?: string, host?: string): Promise<PermissionHookStatus>;
4
+ export declare function getLastPermissionHookStartupRepairSummary(): PermissionHookStartupRepairSummary | null;
5
+ export declare function _resetPermissionHookStartupRepairSummaryForTests(): void;
6
+ export declare function reconcilePermissionHookStatus(host: string, options?: ReconcilePermissionHookOptions): Promise<PermissionHookStatus>;
7
+ export declare function getPermissionHookAuditSummary(homeDir?: string): Promise<PermissionHookAuditSummary>;
8
+ export declare function summarizePermissionHookHealth(summary: PermissionHookAuditSummary): PermissionHookHealthSummary;
9
+ export declare function repairPermissionHooksOnStartup(homeDir?: string, sourceScriptPath?: string): Promise<PermissionHookStartupRepairSummary>;
10
+ //# sourceMappingURL=permissionHookStatus.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"permissionHookStatus.d.ts","sourceRoot":"","sources":["../../src/utils/permissionHookStatus.ts"],"names":[],"mappings":"AAEA,OAAO,EACL,KAAK,0BAA0B,EAC/B,KAAK,2BAA2B,EAGhC,KAAK,kCAAkC,EACvC,KAAK,oBAAoB,EACzB,KAAK,8BAA8B,EAWpC,MAAM,2BAA2B,CAAC;AAInC,wBAAgB,uBAAuB,CAAC,OAAO,SAAY,EAAE,IAAI,CAAC,EAAE,MAAM,GAAG,oBAAoB,CAMhG;AAED,wBAAsB,4BAA4B,CAAC,OAAO,SAAY,EAAE,IAAI,CAAC,EAAE,MAAM,GAAG,OAAO,CAAC,oBAAoB,CAAC,CAMpH;AA2FD,wBAAgB,yCAAyC,IAAI,kCAAkC,GAAG,IAAI,CAErG;AAED,wBAAgB,gDAAgD,IAAI,IAAI,CAEvE;AAED,wBAAsB,6BAA6B,CACjD,IAAI,EAAE,MAAM,EACZ,OAAO,GAAE,8BAAmC,GAC3C,OAAO,CAAC,oBAAoB,CAAC,CAgC/B;AAED,wBAAsB,6BAA6B,CAAC,OAAO,SAAY,GAAG,OAAO,CAAC,0BAA0B,CAAC,CA4B5G;AA4CD,wBAAgB,6BAA6B,CAC3C,OAAO,EAAE,0BAA0B,GAClC,2BAA2B,CAmD7B;AAED,wBAAsB,8BAA8B,CAClD,OAAO,SAAY,EACnB,gBAAgB,CAAC,EAAE,MAAM,GACxB,OAAO,CAAC,kCAAkC,CAAC,CA0D7C"}