@jmfederico/pi-web 1.202605.1
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/LICENSE +21 -0
- package/README.md +321 -0
- package/dist/cli.js +256 -0
- package/dist/cli.js.map +1 -0
- package/dist/client/assets/CodeViewer-DsXI9VCn.js +4 -0
- package/dist/client/assets/TerminalPanel-CpzJEFv1.js +47 -0
- package/dist/client/assets/index-Cbr8EG8h.js +687 -0
- package/dist/client/assets/vendor-editor-core-hulUn3GY.js +12 -0
- package/dist/client/assets/vendor-editor-languages-Cjllm-a8.js +26 -0
- package/dist/client/assets/vendor-editor-legacy-B4QLsWF8.js +1 -0
- package/dist/client/assets/vendor-terminal-DDGTF8rc.css +1 -0
- package/dist/client/assets/vendor-terminal-DjQ08hXu.js +16 -0
- package/dist/client/index.html +16 -0
- package/dist/config.js +92 -0
- package/dist/config.js.map +1 -0
- package/dist/server/app.js +80 -0
- package/dist/server/app.js.map +1 -0
- package/dist/server/git/gitService.js +118 -0
- package/dist/server/git/gitService.js.map +1 -0
- package/dist/server/gitRoutes.js +23 -0
- package/dist/server/gitRoutes.js.map +1 -0
- package/dist/server/index.js +7 -0
- package/dist/server/index.js.map +1 -0
- package/dist/server/projects/directorySuggestions.js +37 -0
- package/dist/server/projects/directorySuggestions.js.map +1 -0
- package/dist/server/projects/projectService.js +31 -0
- package/dist/server/projects/projectService.js.map +1 -0
- package/dist/server/realtime/sessionEventHub.js +36 -0
- package/dist/server/realtime/sessionEventHub.js.map +1 -0
- package/dist/server/sessiond/config.js +9 -0
- package/dist/server/sessiond/config.js.map +1 -0
- package/dist/server/sessiond/sessionDaemonClient.js +65 -0
- package/dist/server/sessiond/sessionDaemonClient.js.map +1 -0
- package/dist/server/sessiond/sessionProxyRoutes.js +63 -0
- package/dist/server/sessiond/sessionProxyRoutes.js.map +1 -0
- package/dist/server/sessiond.js +45 -0
- package/dist/server/sessiond.js.map +1 -0
- package/dist/server/sessions/builtinCommands.js +27 -0
- package/dist/server/sessions/builtinCommands.js.map +1 -0
- package/dist/server/sessions/piSessionService.js +517 -0
- package/dist/server/sessions/piSessionService.js.map +1 -0
- package/dist/server/sessions/sessionArchiveStore.js +68 -0
- package/dist/server/sessions/sessionArchiveStore.js.map +1 -0
- package/dist/server/sessions/sessionCommandService.js +134 -0
- package/dist/server/sessions/sessionCommandService.js.map +1 -0
- package/dist/server/sessions/sessionNameGenerator.js +68 -0
- package/dist/server/sessions/sessionNameGenerator.js.map +1 -0
- package/dist/server/sessions/sessionRoutes.js +116 -0
- package/dist/server/sessions/sessionRoutes.js.map +1 -0
- package/dist/server/sessions/sessionRuntimeStore.js +2 -0
- package/dist/server/sessions/sessionRuntimeStore.js.map +1 -0
- package/dist/server/storage/projectStore.js +88 -0
- package/dist/server/storage/projectStore.js.map +1 -0
- package/dist/server/terminalProxyRoutes.js +70 -0
- package/dist/server/terminalProxyRoutes.js.map +1 -0
- package/dist/server/terminals/terminalRoutes.js +70 -0
- package/dist/server/terminals/terminalRoutes.js.map +1 -0
- package/dist/server/terminals/terminalService.js +115 -0
- package/dist/server/terminals/terminalService.js.map +1 -0
- package/dist/server/types.js +2 -0
- package/dist/server/types.js.map +1 -0
- package/dist/server/workspaceExplorerRoutes.js +24 -0
- package/dist/server/workspaceExplorerRoutes.js.map +1 -0
- package/dist/server/workspaces/fileContentService.js +50 -0
- package/dist/server/workspaces/fileContentService.js.map +1 -0
- package/dist/server/workspaces/fileSuggestions.js +84 -0
- package/dist/server/workspaces/fileSuggestions.js.map +1 -0
- package/dist/server/workspaces/fileTreeService.js +26 -0
- package/dist/server/workspaces/fileTreeService.js.map +1 -0
- package/dist/server/workspaces/gitWorktreeDiscovery.js +33 -0
- package/dist/server/workspaces/gitWorktreeDiscovery.js.map +1 -0
- package/dist/server/workspaces/pathSafety.js +38 -0
- package/dist/server/workspaces/pathSafety.js.map +1 -0
- package/dist/server/workspaces/workspaceContext.js +8 -0
- package/dist/server/workspaces/workspaceContext.js.map +1 -0
- package/dist/server/workspaces/workspaceService.js +39 -0
- package/dist/server/workspaces/workspaceService.js.map +1 -0
- package/dist/shared/apiTypes.js +2 -0
- package/dist/shared/apiTypes.js.map +1 -0
- package/extensions/pi-web.ts +144 -0
- package/install.sh +5 -0
- package/package.json +107 -0
|
@@ -0,0 +1,134 @@
|
|
|
1
|
+
import crypto from "node:crypto";
|
|
2
|
+
import { isBuiltinCommand } from "./builtinCommands.js";
|
|
3
|
+
export class SessionCommandService {
|
|
4
|
+
constructor(getActive, prompt, events, lifecycle = {}) {
|
|
5
|
+
this.getActive = getActive;
|
|
6
|
+
this.prompt = prompt;
|
|
7
|
+
this.events = events;
|
|
8
|
+
this.lifecycle = lifecycle;
|
|
9
|
+
this.pendingSelects = new Map();
|
|
10
|
+
}
|
|
11
|
+
async run(sessionId, text) {
|
|
12
|
+
const active = await this.getActive(sessionId);
|
|
13
|
+
const session = active.runtime.session;
|
|
14
|
+
const [name = "", ...args] = text.trim().replace(/^\//, "").split(/\s+/);
|
|
15
|
+
const rest = args.join(" ").trim();
|
|
16
|
+
if (!isBuiltinCommand(name)) {
|
|
17
|
+
if (this.isRuntimeCommand(session, name)) {
|
|
18
|
+
await this.prompt(sessionId, text);
|
|
19
|
+
return { type: "done", message: `Accepted ${text}` };
|
|
20
|
+
}
|
|
21
|
+
return { type: "unsupported", message: `Unknown command: /${name}` };
|
|
22
|
+
}
|
|
23
|
+
if (name === "session")
|
|
24
|
+
return { type: "done", message: formatSessionStats(session) };
|
|
25
|
+
if (name === "name")
|
|
26
|
+
return this.nameSession(active, rest);
|
|
27
|
+
if (name === "compact")
|
|
28
|
+
return this.compact(session, rest);
|
|
29
|
+
if (name === "clone")
|
|
30
|
+
return this.clone(active);
|
|
31
|
+
if (name === "fork")
|
|
32
|
+
return this.fork(active);
|
|
33
|
+
return { type: "unsupported", message: `/${name} is not implemented in the web UI yet` };
|
|
34
|
+
}
|
|
35
|
+
async respond(sessionId, requestId, value) {
|
|
36
|
+
const pending = this.pendingSelects.get(requestId);
|
|
37
|
+
if (pending?.sessionId !== sessionId)
|
|
38
|
+
return { type: "unsupported", message: "Command request expired" };
|
|
39
|
+
this.pendingSelects.delete(requestId);
|
|
40
|
+
const active = await this.getActive(sessionId);
|
|
41
|
+
const result = await active.runtime.fork(value);
|
|
42
|
+
if (result.cancelled)
|
|
43
|
+
return { type: "done", message: "Fork cancelled" };
|
|
44
|
+
return { type: "done", message: "Session forked", session: clientSessionFromRuntime(active.runtime) };
|
|
45
|
+
}
|
|
46
|
+
nameSession(active, name) {
|
|
47
|
+
if (name === "")
|
|
48
|
+
return { type: "unsupported", message: "Usage: /name <session name>" };
|
|
49
|
+
active.runtime.session.setSessionName(name);
|
|
50
|
+
return { type: "done", message: `Session named: ${name}`, session: clientSessionFromRuntime(active.runtime) };
|
|
51
|
+
}
|
|
52
|
+
compact(session, instructions) {
|
|
53
|
+
this.lifecycle.onCompactionStart?.(session);
|
|
54
|
+
void session.compact(instructions === "" ? undefined : instructions)
|
|
55
|
+
.then((result) => {
|
|
56
|
+
this.events.publish(session.sessionId, {
|
|
57
|
+
type: "command.output",
|
|
58
|
+
level: "success",
|
|
59
|
+
message: formatCompactionResult(result),
|
|
60
|
+
});
|
|
61
|
+
this.lifecycle.onCompactionEnd?.(session, "success");
|
|
62
|
+
})
|
|
63
|
+
.catch((error) => {
|
|
64
|
+
const message = error instanceof Error ? error.message : String(error);
|
|
65
|
+
this.events.publish(session.sessionId, { type: "command.output", level: "error", message: `Compaction failed: ${message}` });
|
|
66
|
+
this.events.publish(session.sessionId, { type: "session.error", message });
|
|
67
|
+
this.lifecycle.onCompactionEnd?.(session, "error", message);
|
|
68
|
+
});
|
|
69
|
+
return { type: "done", message: "Compaction started…" };
|
|
70
|
+
}
|
|
71
|
+
async clone(active) {
|
|
72
|
+
const leafId = active.runtime.session.sessionManager.getLeafId();
|
|
73
|
+
if (leafId === null || leafId === "")
|
|
74
|
+
return { type: "unsupported", message: "Cannot clone: no current session entry" };
|
|
75
|
+
const result = await active.runtime.fork(leafId, { position: "at" });
|
|
76
|
+
if (result.cancelled)
|
|
77
|
+
return { type: "done", message: "Clone cancelled" };
|
|
78
|
+
return { type: "done", message: "Session cloned", session: clientSessionFromRuntime(active.runtime) };
|
|
79
|
+
}
|
|
80
|
+
fork(active) {
|
|
81
|
+
const messages = active.runtime.session.getUserMessagesForForking();
|
|
82
|
+
if (!messages.length)
|
|
83
|
+
return { type: "unsupported", message: "No user messages to fork from" };
|
|
84
|
+
const requestId = crypto.randomUUID();
|
|
85
|
+
this.pendingSelects.set(requestId, { sessionId: active.runtime.session.sessionId, command: "fork" });
|
|
86
|
+
return {
|
|
87
|
+
type: "select",
|
|
88
|
+
requestId,
|
|
89
|
+
title: "Fork from message",
|
|
90
|
+
options: [...messages].reverse().map((message) => ({ value: message.entryId, label: truncate(message.text, 140) })),
|
|
91
|
+
};
|
|
92
|
+
}
|
|
93
|
+
isRuntimeCommand(session, name) {
|
|
94
|
+
return session.extensionRunner.getRegisteredCommands().some((command) => command.invocationName === name)
|
|
95
|
+
|| session.promptTemplates.some((template) => template.name === name)
|
|
96
|
+
|| session.resourceLoader.getSkills().skills.some((skill) => `skill:${skill.name}` === name);
|
|
97
|
+
}
|
|
98
|
+
}
|
|
99
|
+
function clientSessionFromRuntime(runtime) {
|
|
100
|
+
const session = runtime.session;
|
|
101
|
+
return {
|
|
102
|
+
id: session.sessionId,
|
|
103
|
+
path: session.sessionFile ?? "",
|
|
104
|
+
cwd: runtime.cwd,
|
|
105
|
+
...(session.sessionName === undefined ? {} : { name: session.sessionName }),
|
|
106
|
+
created: new Date().toISOString(),
|
|
107
|
+
modified: new Date().toISOString(),
|
|
108
|
+
messageCount: session.messages.length,
|
|
109
|
+
firstMessage: "",
|
|
110
|
+
};
|
|
111
|
+
}
|
|
112
|
+
function formatSessionStats(session) {
|
|
113
|
+
const stats = session.getSessionStats();
|
|
114
|
+
return [
|
|
115
|
+
`Session: ${stats.sessionId}`,
|
|
116
|
+
`Messages: ${String(stats.totalMessages)} (${String(stats.userMessages)} user, ${String(stats.assistantMessages)} assistant)`,
|
|
117
|
+
`Tool calls: ${String(stats.toolCalls)}`,
|
|
118
|
+
`Tokens: ↑${String(stats.tokens.input)} ↓${String(stats.tokens.output)} total ${String(stats.tokens.total)}`,
|
|
119
|
+
`Cost: $${stats.cost.toFixed(4)}`,
|
|
120
|
+
].join("\n");
|
|
121
|
+
}
|
|
122
|
+
function formatCompactionResult(result) {
|
|
123
|
+
return [
|
|
124
|
+
"Compaction complete.",
|
|
125
|
+
`Tokens before: ${String(result.tokensBefore)}`,
|
|
126
|
+
"",
|
|
127
|
+
result.summary,
|
|
128
|
+
].join("\n");
|
|
129
|
+
}
|
|
130
|
+
function truncate(text, maxLength) {
|
|
131
|
+
const singleLine = text.replace(/\s+/g, " ").trim();
|
|
132
|
+
return singleLine.length <= maxLength ? singleLine : `${singleLine.slice(0, maxLength - 1)}…`;
|
|
133
|
+
}
|
|
134
|
+
//# sourceMappingURL=sessionCommandService.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"sessionCommandService.js","sourceRoot":"","sources":["../../../src/server/sessions/sessionCommandService.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,68 @@
|
|
|
1
|
+
import { getApiProvider } from "@earendil-works/pi-ai";
|
|
2
|
+
const SESSION_NAME_TIMEOUT_MS = 10_000;
|
|
3
|
+
const SESSION_NAME_MAX_INPUT_CHARS = 4_000;
|
|
4
|
+
const SESSION_NAME_MAX_LENGTH = 60;
|
|
5
|
+
const FALLBACK_SESSION_NAME_MAX_WORDS = 6;
|
|
6
|
+
export async function generateShortSessionName(modelRegistry, model, firstMessage) {
|
|
7
|
+
const provider = getApiProvider(model.api);
|
|
8
|
+
if (provider === undefined)
|
|
9
|
+
return undefined;
|
|
10
|
+
const auth = await modelRegistry.getApiKeyAndHeaders(model);
|
|
11
|
+
if (!auth.ok)
|
|
12
|
+
return undefined;
|
|
13
|
+
const stream = provider.streamSimple(model, {
|
|
14
|
+
systemPrompt: "Generate a concise title for a coding-agent chat session. Return only the title, with no quotes or punctuation wrapper.",
|
|
15
|
+
messages: [{
|
|
16
|
+
role: "user",
|
|
17
|
+
content: `Create a 2-6 word title for this request:\n\n${truncateInput(firstMessage)}`,
|
|
18
|
+
timestamp: Date.now(),
|
|
19
|
+
}],
|
|
20
|
+
}, {
|
|
21
|
+
maxTokens: 24,
|
|
22
|
+
reasoning: "minimal",
|
|
23
|
+
signal: AbortSignal.timeout(SESSION_NAME_TIMEOUT_MS),
|
|
24
|
+
...(auth.apiKey === undefined ? {} : { apiKey: auth.apiKey }),
|
|
25
|
+
...(auth.headers === undefined ? {} : { headers: auth.headers }),
|
|
26
|
+
});
|
|
27
|
+
let streamedText = "";
|
|
28
|
+
let finalMessage;
|
|
29
|
+
for await (const event of stream) {
|
|
30
|
+
if (event.type === "text_delta")
|
|
31
|
+
streamedText += event.delta;
|
|
32
|
+
if (event.type === "done")
|
|
33
|
+
finalMessage = event.message;
|
|
34
|
+
if (event.type === "error")
|
|
35
|
+
return undefined;
|
|
36
|
+
}
|
|
37
|
+
return cleanSessionName(finalMessage === undefined ? streamedText : textFromAssistant(finalMessage));
|
|
38
|
+
}
|
|
39
|
+
export function fallbackSessionName(firstMessage) {
|
|
40
|
+
return cleanSessionName(firstMessage
|
|
41
|
+
.replace(/<skill name="[^"]+" location="[^"]+">[\s\S]*?<\/skill>/g, "")
|
|
42
|
+
.replace(/```[\s\S]*?```/g, " ")
|
|
43
|
+
.replace(/[`*_#[\](){}<>]/g, " ")
|
|
44
|
+
.split(/\s+/)
|
|
45
|
+
.filter(Boolean)
|
|
46
|
+
.slice(0, FALLBACK_SESSION_NAME_MAX_WORDS)
|
|
47
|
+
.join(" "));
|
|
48
|
+
}
|
|
49
|
+
export function cleanSessionName(value) {
|
|
50
|
+
const title = (value.split("\n", 1)[0] ?? "")
|
|
51
|
+
.replace(/^\s*(title|session title)\s*:\s*/i, "")
|
|
52
|
+
.replace(/^\s*["'`]+|["'`.]+\s*$/g, "")
|
|
53
|
+
.replace(/\s+/g, " ")
|
|
54
|
+
.trim()
|
|
55
|
+
.slice(0, SESSION_NAME_MAX_LENGTH)
|
|
56
|
+
.trim();
|
|
57
|
+
return title === "" ? undefined : title;
|
|
58
|
+
}
|
|
59
|
+
function textFromAssistant(message) {
|
|
60
|
+
return message.content
|
|
61
|
+
.filter((part) => part.type === "text")
|
|
62
|
+
.map((part) => part.text)
|
|
63
|
+
.join("");
|
|
64
|
+
}
|
|
65
|
+
function truncateInput(value) {
|
|
66
|
+
return value.length <= SESSION_NAME_MAX_INPUT_CHARS ? value : `${value.slice(0, SESSION_NAME_MAX_INPUT_CHARS)}…`;
|
|
67
|
+
}
|
|
68
|
+
//# sourceMappingURL=sessionNameGenerator.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"sessionNameGenerator.js","sourceRoot":"","sources":["../../../src/server/sessions/sessionNameGenerator.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,116 @@
|
|
|
1
|
+
export function registerSessionRoutes(app, sessions, eventHub, prefix = "") {
|
|
2
|
+
app.get(`${prefix}/sessions`, async (request, reply) => {
|
|
3
|
+
if (request.query.cwd === undefined || request.query.cwd === "")
|
|
4
|
+
return reply.code(400).send({ error: "cwd query parameter is required" });
|
|
5
|
+
return sessions.list(request.query.cwd);
|
|
6
|
+
});
|
|
7
|
+
app.post(`${prefix}/sessions`, async (request, reply) => {
|
|
8
|
+
try {
|
|
9
|
+
return await sessions.start(request.body.cwd);
|
|
10
|
+
}
|
|
11
|
+
catch (error) {
|
|
12
|
+
return reply.code(400).send({ error: error instanceof Error ? error.message : String(error) });
|
|
13
|
+
}
|
|
14
|
+
});
|
|
15
|
+
app.get(`${prefix}/sessions/:sessionId/messages`, async (request, reply) => {
|
|
16
|
+
try {
|
|
17
|
+
const page = { ...optionalField("before", optionalNumber(request.query.before)), ...optionalField("limit", optionalNumber(request.query.limit)) };
|
|
18
|
+
return await sessions.messages(request.params.sessionId, page);
|
|
19
|
+
}
|
|
20
|
+
catch (error) {
|
|
21
|
+
return reply.code(404).send({ error: error instanceof Error ? error.message : String(error) });
|
|
22
|
+
}
|
|
23
|
+
});
|
|
24
|
+
app.get(`${prefix}/sessions/:sessionId/status`, async (request, reply) => {
|
|
25
|
+
try {
|
|
26
|
+
return await sessions.status(request.params.sessionId);
|
|
27
|
+
}
|
|
28
|
+
catch (error) {
|
|
29
|
+
return reply.code(404).send({ error: error instanceof Error ? error.message : String(error) });
|
|
30
|
+
}
|
|
31
|
+
});
|
|
32
|
+
app.get(`${prefix}/sessions/:sessionId/commands`, async (request, reply) => {
|
|
33
|
+
try {
|
|
34
|
+
return await sessions.commands(request.params.sessionId);
|
|
35
|
+
}
|
|
36
|
+
catch (error) {
|
|
37
|
+
return reply.code(404).send({ error: error instanceof Error ? error.message : String(error) });
|
|
38
|
+
}
|
|
39
|
+
});
|
|
40
|
+
app.post(`${prefix}/sessions/:sessionId/prompt`, async (request, reply) => {
|
|
41
|
+
try {
|
|
42
|
+
await sessions.prompt(request.params.sessionId, request.body.text, request.body.streamingBehavior);
|
|
43
|
+
return { accepted: true };
|
|
44
|
+
}
|
|
45
|
+
catch (error) {
|
|
46
|
+
return reply.code(400).send({ error: error instanceof Error ? error.message : String(error) });
|
|
47
|
+
}
|
|
48
|
+
});
|
|
49
|
+
app.post(`${prefix}/sessions/:sessionId/shell`, async (request, reply) => {
|
|
50
|
+
try {
|
|
51
|
+
await sessions.shell(request.params.sessionId, request.body.text);
|
|
52
|
+
return { accepted: true };
|
|
53
|
+
}
|
|
54
|
+
catch (error) {
|
|
55
|
+
return reply.code(400).send({ error: error instanceof Error ? error.message : String(error) });
|
|
56
|
+
}
|
|
57
|
+
});
|
|
58
|
+
app.post(`${prefix}/sessions/:sessionId/commands/run`, async (request, reply) => {
|
|
59
|
+
try {
|
|
60
|
+
return await sessions.runCommand(request.params.sessionId, request.body.text);
|
|
61
|
+
}
|
|
62
|
+
catch (error) {
|
|
63
|
+
return reply.code(400).send({ error: error instanceof Error ? error.message : String(error) });
|
|
64
|
+
}
|
|
65
|
+
});
|
|
66
|
+
app.post(`${prefix}/sessions/:sessionId/commands/respond`, async (request, reply) => {
|
|
67
|
+
try {
|
|
68
|
+
return await sessions.respondToCommand(request.params.sessionId, request.body.requestId, request.body.value);
|
|
69
|
+
}
|
|
70
|
+
catch (error) {
|
|
71
|
+
return reply.code(400).send({ error: error instanceof Error ? error.message : String(error) });
|
|
72
|
+
}
|
|
73
|
+
});
|
|
74
|
+
app.post(`${prefix}/sessions/:sessionId/abort`, async (request) => {
|
|
75
|
+
await sessions.abort(request.params.sessionId);
|
|
76
|
+
return { aborted: true };
|
|
77
|
+
});
|
|
78
|
+
app.post(`${prefix}/sessions/:sessionId/stop`, (request) => {
|
|
79
|
+
sessions.stop(request.params.sessionId);
|
|
80
|
+
return { stopped: true };
|
|
81
|
+
});
|
|
82
|
+
app.post(`${prefix}/sessions/:sessionId/archive`, async (request, reply) => {
|
|
83
|
+
try {
|
|
84
|
+
await sessions.archive(request.params.sessionId);
|
|
85
|
+
return { archived: true };
|
|
86
|
+
}
|
|
87
|
+
catch (error) {
|
|
88
|
+
return reply.code(400).send({ error: error instanceof Error ? error.message : String(error) });
|
|
89
|
+
}
|
|
90
|
+
});
|
|
91
|
+
app.post(`${prefix}/sessions/:sessionId/restore`, async (request, reply) => {
|
|
92
|
+
try {
|
|
93
|
+
await sessions.restore(request.params.sessionId);
|
|
94
|
+
return { restored: true };
|
|
95
|
+
}
|
|
96
|
+
catch (error) {
|
|
97
|
+
return reply.code(400).send({ error: error instanceof Error ? error.message : String(error) });
|
|
98
|
+
}
|
|
99
|
+
});
|
|
100
|
+
app.get(`${prefix}/sessions/:sessionId/events`, { websocket: true }, (socket, request) => {
|
|
101
|
+
eventHub.add(request.params.sessionId, socket);
|
|
102
|
+
});
|
|
103
|
+
app.get(`${prefix}/sessions/events`, { websocket: true }, (socket) => {
|
|
104
|
+
eventHub.addGlobal(socket);
|
|
105
|
+
});
|
|
106
|
+
}
|
|
107
|
+
function optionalField(key, value) {
|
|
108
|
+
return value === undefined ? {} : { [key]: value };
|
|
109
|
+
}
|
|
110
|
+
function optionalNumber(value) {
|
|
111
|
+
if (value === undefined || value === "")
|
|
112
|
+
return undefined;
|
|
113
|
+
const parsed = Number(value);
|
|
114
|
+
return Number.isFinite(parsed) ? parsed : undefined;
|
|
115
|
+
}
|
|
116
|
+
//# sourceMappingURL=sessionRoutes.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"sessionRoutes.js","sourceRoot":"","sources":["../../../src/server/sessions/sessionRoutes.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"sessionRuntimeStore.js","sourceRoot":"","sources":["../../../src/server/sessions/sessionRuntimeStore.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,88 @@
|
|
|
1
|
+
import { mkdir, readFile, writeFile } from "node:fs/promises";
|
|
2
|
+
import { dirname, join, resolve } from "node:path";
|
|
3
|
+
import { piWebDataDir } from "../../config.js";
|
|
4
|
+
import { randomUUID } from "node:crypto";
|
|
5
|
+
function isNodeErrorWithCode(error, code) {
|
|
6
|
+
return error instanceof Error && "code" in error && error.code === code;
|
|
7
|
+
}
|
|
8
|
+
function parseProjectFile(value) {
|
|
9
|
+
if (!isRecord(value) || !Array.isArray(value["projects"]))
|
|
10
|
+
throw new Error("Invalid project file");
|
|
11
|
+
return { projects: value["projects"].map(parseProject) };
|
|
12
|
+
}
|
|
13
|
+
function parseProject(value) {
|
|
14
|
+
if (!isRecord(value))
|
|
15
|
+
throw new Error("Invalid project");
|
|
16
|
+
const id = value["id"];
|
|
17
|
+
const name = value["name"];
|
|
18
|
+
const path = value["path"];
|
|
19
|
+
const createdAt = value["createdAt"];
|
|
20
|
+
if (typeof id !== "string" || typeof name !== "string" || typeof path !== "string" || typeof createdAt !== "string")
|
|
21
|
+
throw new Error("Invalid project");
|
|
22
|
+
return { id, name, path, createdAt };
|
|
23
|
+
}
|
|
24
|
+
function isRecord(value) {
|
|
25
|
+
return typeof value === "object" && value !== null;
|
|
26
|
+
}
|
|
27
|
+
export function defaultProjectStorePath(env = process.env, cwd = process.cwd()) {
|
|
28
|
+
return join(piWebDataDir(env, cwd), "projects.json");
|
|
29
|
+
}
|
|
30
|
+
export function projectStorePath(env = process.env, cwd = process.cwd()) {
|
|
31
|
+
const configured = env["PI_WEB_PROJECTS_FILE"];
|
|
32
|
+
if (configured === undefined || configured === "")
|
|
33
|
+
return defaultProjectStorePath(env, cwd);
|
|
34
|
+
return resolve(cwd, configured);
|
|
35
|
+
}
|
|
36
|
+
export class ProjectStore {
|
|
37
|
+
constructor(filePath = projectStorePath()) {
|
|
38
|
+
this.filePath = filePath;
|
|
39
|
+
}
|
|
40
|
+
async list() {
|
|
41
|
+
return (await this.read()).projects;
|
|
42
|
+
}
|
|
43
|
+
async add(input) {
|
|
44
|
+
const data = await this.read();
|
|
45
|
+
const path = input.path;
|
|
46
|
+
const existing = data.projects.find((p) => p.path === path);
|
|
47
|
+
if (existing)
|
|
48
|
+
return existing;
|
|
49
|
+
const trimmedName = input.name?.trim();
|
|
50
|
+
const leafName = path.split("/").filter((part) => part !== "").at(-1);
|
|
51
|
+
const project = {
|
|
52
|
+
id: randomUUID(),
|
|
53
|
+
name: trimmedName !== undefined && trimmedName !== "" ? trimmedName : leafName ?? path,
|
|
54
|
+
path,
|
|
55
|
+
createdAt: new Date().toISOString(),
|
|
56
|
+
};
|
|
57
|
+
data.projects.push(project);
|
|
58
|
+
await this.write(data);
|
|
59
|
+
return project;
|
|
60
|
+
}
|
|
61
|
+
async get(id) {
|
|
62
|
+
return (await this.list()).find((p) => p.id === id);
|
|
63
|
+
}
|
|
64
|
+
async remove(id) {
|
|
65
|
+
const data = await this.read();
|
|
66
|
+
const projects = data.projects.filter((p) => p.id !== id);
|
|
67
|
+
if (projects.length === data.projects.length)
|
|
68
|
+
return false;
|
|
69
|
+
await this.write({ projects });
|
|
70
|
+
return true;
|
|
71
|
+
}
|
|
72
|
+
async read() {
|
|
73
|
+
try {
|
|
74
|
+
const value = JSON.parse(await readFile(this.filePath, "utf8"));
|
|
75
|
+
return parseProjectFile(value);
|
|
76
|
+
}
|
|
77
|
+
catch (error) {
|
|
78
|
+
if (isNodeErrorWithCode(error, "ENOENT"))
|
|
79
|
+
return { projects: [] };
|
|
80
|
+
throw error;
|
|
81
|
+
}
|
|
82
|
+
}
|
|
83
|
+
async write(data) {
|
|
84
|
+
await mkdir(dirname(this.filePath), { recursive: true });
|
|
85
|
+
await writeFile(this.filePath, `${JSON.stringify(data, null, 2)}\n`, "utf8");
|
|
86
|
+
}
|
|
87
|
+
}
|
|
88
|
+
//# sourceMappingURL=projectStore.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"projectStore.js","sourceRoot":"","sources":["../../../src/server/storage/projectStore.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,70 @@
|
|
|
1
|
+
import { WebSocket } from "ws";
|
|
2
|
+
import { SessionDaemonClient } from "./sessiond/sessionDaemonClient.js";
|
|
3
|
+
import { resolveWorkspaceContext } from "./workspaces/workspaceContext.js";
|
|
4
|
+
export function registerTerminalProxyRoutes(app, projects, workspaces, daemon = new SessionDaemonClient()) {
|
|
5
|
+
app.get("/api/projects/:projectId/workspaces/:workspaceId/terminals", async (request, reply) => {
|
|
6
|
+
try {
|
|
7
|
+
const context = await resolveWorkspaceContext(projects, workspaces, request.params.projectId, request.params.workspaceId);
|
|
8
|
+
return await proxyJson(daemon, "GET", `/terminals?cwd=${encodeURIComponent(context.root)}`, undefined, reply);
|
|
9
|
+
}
|
|
10
|
+
catch (error) {
|
|
11
|
+
requestFailed(reply, error);
|
|
12
|
+
return undefined;
|
|
13
|
+
}
|
|
14
|
+
});
|
|
15
|
+
app.post("/api/projects/:projectId/workspaces/:workspaceId/terminals", async (request, reply) => {
|
|
16
|
+
try {
|
|
17
|
+
const context = await resolveWorkspaceContext(projects, workspaces, request.params.projectId, request.params.workspaceId);
|
|
18
|
+
return await proxyJson(daemon, "POST", "/terminals", { ...request.body, cwd: context.root }, reply);
|
|
19
|
+
}
|
|
20
|
+
catch (error) {
|
|
21
|
+
requestFailed(reply, error);
|
|
22
|
+
return undefined;
|
|
23
|
+
}
|
|
24
|
+
});
|
|
25
|
+
app.delete("/api/projects/:projectId/workspaces/:workspaceId/terminals/:terminalId", async (request, reply) => {
|
|
26
|
+
try {
|
|
27
|
+
await resolveWorkspaceContext(projects, workspaces, request.params.projectId, request.params.workspaceId);
|
|
28
|
+
return await proxyJson(daemon, "DELETE", `/terminals/${encodeURIComponent(request.params.terminalId)}`, undefined, reply);
|
|
29
|
+
}
|
|
30
|
+
catch (error) {
|
|
31
|
+
requestFailed(reply, error);
|
|
32
|
+
return undefined;
|
|
33
|
+
}
|
|
34
|
+
});
|
|
35
|
+
app.get("/api/projects/:projectId/workspaces/:workspaceId/terminals/:terminalId/socket", { websocket: true }, async (socket, request) => {
|
|
36
|
+
try {
|
|
37
|
+
await resolveWorkspaceContext(projects, workspaces, request.params.projectId, request.params.workspaceId);
|
|
38
|
+
bridgeSockets(socket, daemon.connectWebSocket(`/terminals/${request.params.terminalId}/socket`));
|
|
39
|
+
}
|
|
40
|
+
catch (error) {
|
|
41
|
+
socket.send(JSON.stringify({ type: "error", message: error instanceof Error ? error.message : String(error) }));
|
|
42
|
+
socket.close();
|
|
43
|
+
}
|
|
44
|
+
});
|
|
45
|
+
}
|
|
46
|
+
async function proxyJson(daemon, method, path, body, reply) {
|
|
47
|
+
const upstream = await daemon.request(method, path, body);
|
|
48
|
+
reply.code(upstream.statusCode);
|
|
49
|
+
const contentType = upstream.headers["content-type"];
|
|
50
|
+
if (contentType !== undefined && contentType !== "")
|
|
51
|
+
reply.header("content-type", contentType);
|
|
52
|
+
const value = upstream.body !== "" ? JSON.parse(upstream.body) : undefined;
|
|
53
|
+
return value;
|
|
54
|
+
}
|
|
55
|
+
function requestFailed(reply, error) {
|
|
56
|
+
reply.code(400).send({ error: error instanceof Error ? error.message : String(error) });
|
|
57
|
+
}
|
|
58
|
+
function bridgeSockets(client, upstream) {
|
|
59
|
+
client.on("message", (data) => { sendIfOpen(upstream, data); });
|
|
60
|
+
upstream.on("message", (data) => { sendIfOpen(client, data); });
|
|
61
|
+
client.on("close", () => { upstream.close(); });
|
|
62
|
+
upstream.on("close", () => { client.close(); });
|
|
63
|
+
upstream.on("error", () => { client.close(); });
|
|
64
|
+
client.on("error", () => { upstream.close(); });
|
|
65
|
+
}
|
|
66
|
+
function sendIfOpen(socket, data) {
|
|
67
|
+
if (socket.readyState === WebSocket.OPEN)
|
|
68
|
+
socket.send(data);
|
|
69
|
+
}
|
|
70
|
+
//# sourceMappingURL=terminalProxyRoutes.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"terminalProxyRoutes.js","sourceRoot":"","sources":["../../src/server/terminalProxyRoutes.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,70 @@
|
|
|
1
|
+
export function registerTerminalRoutes(app, terminals, prefix = "") {
|
|
2
|
+
app.get(`${prefix}/terminals`, (request, reply) => {
|
|
3
|
+
if (request.query.cwd === undefined || request.query.cwd === "")
|
|
4
|
+
return reply.code(400).send({ error: "cwd query parameter is required" });
|
|
5
|
+
return terminals.list(request.query.cwd);
|
|
6
|
+
});
|
|
7
|
+
app.post(`${prefix}/terminals`, (request, reply) => {
|
|
8
|
+
try {
|
|
9
|
+
return terminals.create(request.body);
|
|
10
|
+
}
|
|
11
|
+
catch (error) {
|
|
12
|
+
return reply.code(400).send({ error: error instanceof Error ? error.message : String(error) });
|
|
13
|
+
}
|
|
14
|
+
});
|
|
15
|
+
app.delete(`${prefix}/terminals/:terminalId`, (request) => {
|
|
16
|
+
terminals.close(request.params.terminalId);
|
|
17
|
+
return { closed: true };
|
|
18
|
+
});
|
|
19
|
+
app.get(`${prefix}/terminals/:terminalId/socket`, { websocket: true }, (socket, request) => {
|
|
20
|
+
let detach;
|
|
21
|
+
try {
|
|
22
|
+
detach = terminals.attach(request.params.terminalId, {
|
|
23
|
+
output: (data) => { socket.send(JSON.stringify({ type: "output", data })); },
|
|
24
|
+
exit: (exitCode) => { socket.send(JSON.stringify({ type: "exit", exitCode })); },
|
|
25
|
+
});
|
|
26
|
+
}
|
|
27
|
+
catch (error) {
|
|
28
|
+
socket.send(JSON.stringify({ type: "error", message: error instanceof Error ? error.message : String(error) }));
|
|
29
|
+
socket.close();
|
|
30
|
+
return;
|
|
31
|
+
}
|
|
32
|
+
socket.on("message", (data) => {
|
|
33
|
+
try {
|
|
34
|
+
const message = parseClientMessage(data);
|
|
35
|
+
if (message.type === "input")
|
|
36
|
+
terminals.write(request.params.terminalId, message.data);
|
|
37
|
+
if (message.type === "resize")
|
|
38
|
+
terminals.resize(request.params.terminalId, message.cols, message.rows);
|
|
39
|
+
}
|
|
40
|
+
catch (error) {
|
|
41
|
+
socket.send(JSON.stringify({ type: "error", message: error instanceof Error ? error.message : String(error) }));
|
|
42
|
+
}
|
|
43
|
+
});
|
|
44
|
+
socket.on("close", () => { detach(); });
|
|
45
|
+
socket.on("error", () => { detach(); });
|
|
46
|
+
});
|
|
47
|
+
}
|
|
48
|
+
function parseClientMessage(data) {
|
|
49
|
+
const value = JSON.parse(rawDataToString(data));
|
|
50
|
+
if (!isRecord(value) || typeof value["type"] !== "string")
|
|
51
|
+
throw new Error("Invalid terminal message");
|
|
52
|
+
if (value["type"] === "input" && typeof value["data"] === "string")
|
|
53
|
+
return { type: "input", data: value["data"] };
|
|
54
|
+
if (value["type"] === "resize" && typeof value["cols"] === "number" && typeof value["rows"] === "number")
|
|
55
|
+
return { type: "resize", cols: value["cols"], rows: value["rows"] };
|
|
56
|
+
throw new Error("Invalid terminal message");
|
|
57
|
+
}
|
|
58
|
+
function rawDataToString(data) {
|
|
59
|
+
if (typeof data === "string")
|
|
60
|
+
return data;
|
|
61
|
+
if (data instanceof ArrayBuffer)
|
|
62
|
+
return Buffer.from(data).toString("utf8");
|
|
63
|
+
if (Array.isArray(data))
|
|
64
|
+
return Buffer.concat(data).toString("utf8");
|
|
65
|
+
return data.toString("utf8");
|
|
66
|
+
}
|
|
67
|
+
function isRecord(value) {
|
|
68
|
+
return typeof value === "object" && value !== null;
|
|
69
|
+
}
|
|
70
|
+
//# sourceMappingURL=terminalRoutes.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"terminalRoutes.js","sourceRoot":"","sources":["../../../src/server/terminals/terminalRoutes.ts"],"names":[],"mappings":"AAIA,MAAM,UAAU,sBAAsB,CAAC,GAAoB,EAAE,SAA0B,EAAE,MAAM,GAAG,EAAE;IAClG,GAAG,CAAC,GAAG,CAAoC,GAAG,MAAM,YAAY,EAAE,CAAC,OAAO,EAAE,KAAK,EAAE,EAAE;QACnF,IAAI,OAAO,CAAC,KAAK,CAAC,GAAG,KAAK,SAAS,IAAI,OAAO,CAAC,KAAK,CAAC,GAAG,KAAK,EAAE;YAAE,OAAO,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,iCAAiC,EAAE,CAAC,CAAC;QAC3I,OAAO,SAAS,CAAC,IAAI,CAAC,OAAO,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;IAC3C,CAAC,CAAC,CAAC;IAEH,GAAG,CAAC,IAAI,CAAyE,GAAG,MAAM,YAAY,EAAE,CAAC,OAAO,EAAE,KAAK,EAAE,EAAE;QACzH,IAAI,CAAC;YACH,OAAO,SAAS,CAAC,MAAM,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC;QACxC,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,OAAO,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,KAAK,YAAY,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC;QACjG,CAAC;IACH,CAAC,CAAC,CAAC;IAEH,GAAG,CAAC,MAAM,CAAqC,GAAG,MAAM,wBAAwB,EAAE,CAAC,OAAO,EAAE,EAAE;QAC5F,SAAS,CAAC,KAAK,CAAC,OAAO,CAAC,MAAM,CAAC,UAAU,CAAC,CAAC;QAC3C,OAAO,EAAE,MAAM,EAAE,IAAI,EAAE,CAAC;IAC1B,CAAC,CAAC,CAAC;IAEH,GAAG,CAAC,GAAG,CAAqC,GAAG,MAAM,+BAA+B,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,EAAE,CAAC,MAAM,EAAE,OAAO,EAAE,EAAE;QAC7H,IAAI,MAAgC,CAAC;QACrC,IAAI,CAAC;YACH,MAAM,GAAG,SAAS,CAAC,MAAM,CAAC,OAAO,CAAC,MAAM,CAAC,UAAU,EAAE;gBACnD,MAAM,EAAE,CAAC,IAAI,EAAE,EAAE,GAAG,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,EAAE,IAAI,EAAE,QAAQ,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC;gBAC5E,IAAI,EAAE,CAAC,QAAQ,EAAE,EAAE,GAAG,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,EAAE,IAAI,EAAE,MAAM,EAAE,QAAQ,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC;aACjF,CAAC,CAAC;QACL,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,EAAE,IAAI,EAAE,OAAO,EAAE,OAAO,EAAE,KAAK,YAAY,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC,CAAC;YAChH,MAAM,CAAC,KAAK,EAAE,CAAC;YACf,OAAO;QACT,CAAC;QAED,MAAM,CAAC,EAAE,CAAC,SAAS,EAAE,CAAC,IAAI,EAAE,EAAE;YAC5B,IAAI,CAAC;gBACH,MAAM,OAAO,GAAG,kBAAkB,CAAC,IAAI,CAAC,CAAC;gBACzC,IAAI,OAAO,CAAC,IAAI,KAAK,OAAO;oBAAE,SAAS,CAAC,KAAK,CAAC,OAAO,CAAC,MAAM,CAAC,UAAU,EAAE,OAAO,CAAC,IAAI,CAAC,CAAC;gBACvF,IAAI,OAAO,CAAC,IAAI,KAAK,QAAQ;oBAAE,SAAS,CAAC,MAAM,CAAC,OAAO,CAAC,MAAM,CAAC,UAAU,EAAE,OAAO,CAAC,IAAI,EAAE,OAAO,CAAC,IAAI,CAAC,CAAC;YACzG,CAAC;YAAC,OAAO,KAAK,EAAE,CAAC;gBACf,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,EAAE,IAAI,EAAE,OAAO,EAAE,OAAO,EAAE,KAAK,YAAY,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC,CAAC;YAClH,CAAC;QACH,CAAC,CAAC,CAAC;QACH,MAAM,CAAC,EAAE,CAAC,OAAO,EAAE,GAAG,EAAE,GAAG,MAAM,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC;QACxC,MAAM,CAAC,EAAE,CAAC,OAAO,EAAE,GAAG,EAAE,GAAG,MAAM,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC;IAC1C,CAAC,CAAC,CAAC;AACL,CAAC;AAMD,SAAS,kBAAkB,CAAC,IAAa;IACvC,MAAM,KAAK,GAAY,IAAI,CAAC,KAAK,CAAC,eAAe,CAAC,IAAI,CAAC,CAAC,CAAC;IACzD,IAAI,CAAC,QAAQ,CAAC,KAAK,CAAC,IAAI,OAAO,KAAK,CAAC,MAAM,CAAC,KAAK,QAAQ;QAAE,MAAM,IAAI,KAAK,CAAC,0BAA0B,CAAC,CAAC;IACvG,IAAI,KAAK,CAAC,MAAM,CAAC,KAAK,OAAO,IAAI,OAAO,KAAK,CAAC,MAAM,CAAC,KAAK,QAAQ;QAAE,OAAO,EAAE,IAAI,EAAE,OAAO,EAAE,IAAI,EAAE,KAAK,CAAC,MAAM,CAAC,EAAE,CAAC;IAClH,IAAI,KAAK,CAAC,MAAM,CAAC,KAAK,QAAQ,IAAI,OAAO,KAAK,CAAC,MAAM,CAAC,KAAK,QAAQ,IAAI,OAAO,KAAK,CAAC,MAAM,CAAC,KAAK,QAAQ;QAAE,OAAO,EAAE,IAAI,EAAE,QAAQ,EAAE,IAAI,EAAE,KAAK,CAAC,MAAM,CAAC,EAAE,IAAI,EAAE,KAAK,CAAC,MAAM,CAAC,EAAE,CAAC;IAC9K,MAAM,IAAI,KAAK,CAAC,0BAA0B,CAAC,CAAC;AAC9C,CAAC;AAED,SAAS,eAAe,CAAC,IAAa;IACpC,IAAI,OAAO,IAAI,KAAK,QAAQ;QAAE,OAAO,IAAI,CAAC;IAC1C,IAAI,IAAI,YAAY,WAAW;QAAE,OAAO,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC;IAC3E,IAAI,KAAK,CAAC,OAAO,CAAC,IAAI,CAAC;QAAE,OAAO,MAAM,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC;IACrE,OAAO,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC;AAC/B,CAAC;AAED,SAAS,QAAQ,CAAC,KAAc;IAC9B,OAAO,OAAO,KAAK,KAAK,QAAQ,IAAI,KAAK,KAAK,IAAI,CAAC;AACrD,CAAC"}
|