@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.
Files changed (82) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +321 -0
  3. package/dist/cli.js +256 -0
  4. package/dist/cli.js.map +1 -0
  5. package/dist/client/assets/CodeViewer-DsXI9VCn.js +4 -0
  6. package/dist/client/assets/TerminalPanel-CpzJEFv1.js +47 -0
  7. package/dist/client/assets/index-Cbr8EG8h.js +687 -0
  8. package/dist/client/assets/vendor-editor-core-hulUn3GY.js +12 -0
  9. package/dist/client/assets/vendor-editor-languages-Cjllm-a8.js +26 -0
  10. package/dist/client/assets/vendor-editor-legacy-B4QLsWF8.js +1 -0
  11. package/dist/client/assets/vendor-terminal-DDGTF8rc.css +1 -0
  12. package/dist/client/assets/vendor-terminal-DjQ08hXu.js +16 -0
  13. package/dist/client/index.html +16 -0
  14. package/dist/config.js +92 -0
  15. package/dist/config.js.map +1 -0
  16. package/dist/server/app.js +80 -0
  17. package/dist/server/app.js.map +1 -0
  18. package/dist/server/git/gitService.js +118 -0
  19. package/dist/server/git/gitService.js.map +1 -0
  20. package/dist/server/gitRoutes.js +23 -0
  21. package/dist/server/gitRoutes.js.map +1 -0
  22. package/dist/server/index.js +7 -0
  23. package/dist/server/index.js.map +1 -0
  24. package/dist/server/projects/directorySuggestions.js +37 -0
  25. package/dist/server/projects/directorySuggestions.js.map +1 -0
  26. package/dist/server/projects/projectService.js +31 -0
  27. package/dist/server/projects/projectService.js.map +1 -0
  28. package/dist/server/realtime/sessionEventHub.js +36 -0
  29. package/dist/server/realtime/sessionEventHub.js.map +1 -0
  30. package/dist/server/sessiond/config.js +9 -0
  31. package/dist/server/sessiond/config.js.map +1 -0
  32. package/dist/server/sessiond/sessionDaemonClient.js +65 -0
  33. package/dist/server/sessiond/sessionDaemonClient.js.map +1 -0
  34. package/dist/server/sessiond/sessionProxyRoutes.js +63 -0
  35. package/dist/server/sessiond/sessionProxyRoutes.js.map +1 -0
  36. package/dist/server/sessiond.js +45 -0
  37. package/dist/server/sessiond.js.map +1 -0
  38. package/dist/server/sessions/builtinCommands.js +27 -0
  39. package/dist/server/sessions/builtinCommands.js.map +1 -0
  40. package/dist/server/sessions/piSessionService.js +517 -0
  41. package/dist/server/sessions/piSessionService.js.map +1 -0
  42. package/dist/server/sessions/sessionArchiveStore.js +68 -0
  43. package/dist/server/sessions/sessionArchiveStore.js.map +1 -0
  44. package/dist/server/sessions/sessionCommandService.js +134 -0
  45. package/dist/server/sessions/sessionCommandService.js.map +1 -0
  46. package/dist/server/sessions/sessionNameGenerator.js +68 -0
  47. package/dist/server/sessions/sessionNameGenerator.js.map +1 -0
  48. package/dist/server/sessions/sessionRoutes.js +116 -0
  49. package/dist/server/sessions/sessionRoutes.js.map +1 -0
  50. package/dist/server/sessions/sessionRuntimeStore.js +2 -0
  51. package/dist/server/sessions/sessionRuntimeStore.js.map +1 -0
  52. package/dist/server/storage/projectStore.js +88 -0
  53. package/dist/server/storage/projectStore.js.map +1 -0
  54. package/dist/server/terminalProxyRoutes.js +70 -0
  55. package/dist/server/terminalProxyRoutes.js.map +1 -0
  56. package/dist/server/terminals/terminalRoutes.js +70 -0
  57. package/dist/server/terminals/terminalRoutes.js.map +1 -0
  58. package/dist/server/terminals/terminalService.js +115 -0
  59. package/dist/server/terminals/terminalService.js.map +1 -0
  60. package/dist/server/types.js +2 -0
  61. package/dist/server/types.js.map +1 -0
  62. package/dist/server/workspaceExplorerRoutes.js +24 -0
  63. package/dist/server/workspaceExplorerRoutes.js.map +1 -0
  64. package/dist/server/workspaces/fileContentService.js +50 -0
  65. package/dist/server/workspaces/fileContentService.js.map +1 -0
  66. package/dist/server/workspaces/fileSuggestions.js +84 -0
  67. package/dist/server/workspaces/fileSuggestions.js.map +1 -0
  68. package/dist/server/workspaces/fileTreeService.js +26 -0
  69. package/dist/server/workspaces/fileTreeService.js.map +1 -0
  70. package/dist/server/workspaces/gitWorktreeDiscovery.js +33 -0
  71. package/dist/server/workspaces/gitWorktreeDiscovery.js.map +1 -0
  72. package/dist/server/workspaces/pathSafety.js +38 -0
  73. package/dist/server/workspaces/pathSafety.js.map +1 -0
  74. package/dist/server/workspaces/workspaceContext.js +8 -0
  75. package/dist/server/workspaces/workspaceContext.js.map +1 -0
  76. package/dist/server/workspaces/workspaceService.js +39 -0
  77. package/dist/server/workspaces/workspaceService.js.map +1 -0
  78. package/dist/shared/apiTypes.js +2 -0
  79. package/dist/shared/apiTypes.js.map +1 -0
  80. package/extensions/pi-web.ts +144 -0
  81. package/install.sh +5 -0
  82. 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":"AAAA,OAAO,MAAM,MAAM,aAAa,CAAC;AAIjC,OAAO,EAAE,gBAAgB,EAAE,MAAM,sBAAsB,CAAC;AAQxD,MAAM,OAAO,qBAAqB;IAGhC,YACmB,SAA2B,EAC3B,MAA0D,EAC1D,MAAuB,EACvB,YAGb,EAAE;QANW,cAAS,GAAT,SAAS,CAAkB;QAC3B,WAAM,GAAN,MAAM,CAAoD;QAC1D,WAAM,GAAN,MAAM,CAAiB;QACvB,cAAS,GAAT,SAAS,CAGpB;QATS,mBAAc,GAAG,IAAI,GAAG,EAAgC,CAAC;IAUvE,CAAC;IAEJ,KAAK,CAAC,GAAG,CAAC,SAAiB,EAAE,IAAY;QACvC,MAAM,MAAM,GAAG,MAAM,IAAI,CAAC,SAAS,CAAC,SAAS,CAAC,CAAC;QAC/C,MAAM,OAAO,GAAG,MAAM,CAAC,OAAO,CAAC,OAAO,CAAC;QACvC,MAAM,CAAC,IAAI,GAAG,EAAE,EAAE,GAAG,IAAI,CAAC,GAAG,IAAI,CAAC,IAAI,EAAE,CAAC,OAAO,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC;QACzE,MAAM,IAAI,GAAG,IAAI,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,IAAI,EAAE,CAAC;QAEnC,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,EAAE,CAAC;YAC5B,IAAI,IAAI,CAAC,gBAAgB,CAAC,OAAO,EAAE,IAAI,CAAC,EAAE,CAAC;gBACzC,MAAM,IAAI,CAAC,MAAM,CAAC,SAAS,EAAE,IAAI,CAAC,CAAC;gBACnC,OAAO,EAAE,IAAI,EAAE,MAAM,EAAE,OAAO,EAAE,YAAY,IAAI,EAAE,EAAE,CAAC;YACvD,CAAC;YACD,OAAO,EAAE,IAAI,EAAE,aAAa,EAAE,OAAO,EAAE,qBAAqB,IAAI,EAAE,EAAE,CAAC;QACvE,CAAC;QAED,IAAI,IAAI,KAAK,SAAS;YAAE,OAAO,EAAE,IAAI,EAAE,MAAM,EAAE,OAAO,EAAE,kBAAkB,CAAC,OAAO,CAAC,EAAE,CAAC;QACtF,IAAI,IAAI,KAAK,MAAM;YAAE,OAAO,IAAI,CAAC,WAAW,CAAC,MAAM,EAAE,IAAI,CAAC,CAAC;QAC3D,IAAI,IAAI,KAAK,SAAS;YAAE,OAAO,IAAI,CAAC,OAAO,CAAC,OAAO,EAAE,IAAI,CAAC,CAAC;QAC3D,IAAI,IAAI,KAAK,OAAO;YAAE,OAAO,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC;QAChD,IAAI,IAAI,KAAK,MAAM;YAAE,OAAO,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;QAE9C,OAAO,EAAE,IAAI,EAAE,aAAa,EAAE,OAAO,EAAE,IAAI,IAAI,uCAAuC,EAAE,CAAC;IAC3F,CAAC;IAED,KAAK,CAAC,OAAO,CAAC,SAAiB,EAAE,SAAiB,EAAE,KAAa;QAC/D,MAAM,OAAO,GAAG,IAAI,CAAC,cAAc,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC;QACnD,IAAI,OAAO,EAAE,SAAS,KAAK,SAAS;YAAE,OAAO,EAAE,IAAI,EAAE,aAAa,EAAE,OAAO,EAAE,yBAAyB,EAAE,CAAC;QACzG,IAAI,CAAC,cAAc,CAAC,MAAM,CAAC,SAAS,CAAC,CAAC;QAEtC,MAAM,MAAM,GAAG,MAAM,IAAI,CAAC,SAAS,CAAC,SAAS,CAAC,CAAC;QAC/C,MAAM,MAAM,GAAG,MAAM,MAAM,CAAC,OAAO,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QAChD,IAAI,MAAM,CAAC,SAAS;YAAE,OAAO,EAAE,IAAI,EAAE,MAAM,EAAE,OAAO,EAAE,gBAAgB,EAAE,CAAC;QACzE,OAAO,EAAE,IAAI,EAAE,MAAM,EAAE,OAAO,EAAE,gBAAgB,EAAE,OAAO,EAAE,wBAAwB,CAAC,MAAM,CAAC,OAAO,CAAC,EAAE,CAAC;IACxG,CAAC;IAEO,WAAW,CAAC,MAAqB,EAAE,IAAY;QACrD,IAAI,IAAI,KAAK,EAAE;YAAE,OAAO,EAAE,IAAI,EAAE,aAAa,EAAE,OAAO,EAAE,6BAA6B,EAAE,CAAC;QACxF,MAAM,CAAC,OAAO,CAAC,OAAO,CAAC,cAAc,CAAC,IAAI,CAAC,CAAC;QAC5C,OAAO,EAAE,IAAI,EAAE,MAAM,EAAE,OAAO,EAAE,kBAAkB,IAAI,EAAE,EAAE,OAAO,EAAE,wBAAwB,CAAC,MAAM,CAAC,OAAO,CAAC,EAAE,CAAC;IAChH,CAAC;IAEO,OAAO,CAAC,OAAqB,EAAE,YAAoB;QACzD,IAAI,CAAC,SAAS,CAAC,iBAAiB,EAAE,CAAC,OAAO,CAAC,CAAC;QAC5C,KAAK,OAAO,CAAC,OAAO,CAAC,YAAY,KAAK,EAAE,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,YAAY,CAAC;aACjE,IAAI,CAAC,CAAC,MAAM,EAAE,EAAE;YACf,IAAI,CAAC,MAAM,CAAC,OAAO,CAAC,OAAO,CAAC,SAAS,EAAE;gBACrC,IAAI,EAAE,gBAAgB;gBACtB,KAAK,EAAE,SAAS;gBAChB,OAAO,EAAE,sBAAsB,CAAC,MAAM,CAAC;aACxC,CAAC,CAAC;YACH,IAAI,CAAC,SAAS,CAAC,eAAe,EAAE,CAAC,OAAO,EAAE,SAAS,CAAC,CAAC;QACvD,CAAC,CAAC;aACD,KAAK,CAAC,CAAC,KAAc,EAAE,EAAE;YACxB,MAAM,OAAO,GAAG,KAAK,YAAY,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;YACvE,IAAI,CAAC,MAAM,CAAC,OAAO,CAAC,OAAO,CAAC,SAAS,EAAE,EAAE,IAAI,EAAE,gBAAgB,EAAE,KAAK,EAAE,OAAO,EAAE,OAAO,EAAE,sBAAsB,OAAO,EAAE,EAAE,CAAC,CAAC;YAC7H,IAAI,CAAC,MAAM,CAAC,OAAO,CAAC,OAAO,CAAC,SAAS,EAAE,EAAE,IAAI,EAAE,eAAe,EAAE,OAAO,EAAE,CAAC,CAAC;YAC3E,IAAI,CAAC,SAAS,CAAC,eAAe,EAAE,CAAC,OAAO,EAAE,OAAO,EAAE,OAAO,CAAC,CAAC;QAC9D,CAAC,CAAC,CAAC;QACL,OAAO,EAAE,IAAI,EAAE,MAAM,EAAE,OAAO,EAAE,qBAAqB,EAAE,CAAC;IAC1D,CAAC;IAEO,KAAK,CAAC,KAAK,CAAC,MAAqB;QACvC,MAAM,MAAM,GAAG,MAAM,CAAC,OAAO,CAAC,OAAO,CAAC,cAAc,CAAC,SAAS,EAAE,CAAC;QACjE,IAAI,MAAM,KAAK,IAAI,IAAI,MAAM,KAAK,EAAE;YAAE,OAAO,EAAE,IAAI,EAAE,aAAa,EAAE,OAAO,EAAE,wCAAwC,EAAE,CAAC;QACxH,MAAM,MAAM,GAAG,MAAM,MAAM,CAAC,OAAO,CAAC,IAAI,CAAC,MAAM,EAAE,EAAE,QAAQ,EAAE,IAAI,EAAE,CAAC,CAAC;QACrE,IAAI,MAAM,CAAC,SAAS;YAAE,OAAO,EAAE,IAAI,EAAE,MAAM,EAAE,OAAO,EAAE,iBAAiB,EAAE,CAAC;QAC1E,OAAO,EAAE,IAAI,EAAE,MAAM,EAAE,OAAO,EAAE,gBAAgB,EAAE,OAAO,EAAE,wBAAwB,CAAC,MAAM,CAAC,OAAO,CAAC,EAAE,CAAC;IACxG,CAAC;IAEO,IAAI,CAAC,MAAqB;QAChC,MAAM,QAAQ,GAAG,MAAM,CAAC,OAAO,CAAC,OAAO,CAAC,yBAAyB,EAAE,CAAC;QACpE,IAAI,CAAC,QAAQ,CAAC,MAAM;YAAE,OAAO,EAAE,IAAI,EAAE,aAAa,EAAE,OAAO,EAAE,+BAA+B,EAAE,CAAC;QAC/F,MAAM,SAAS,GAAG,MAAM,CAAC,UAAU,EAAE,CAAC;QACtC,IAAI,CAAC,cAAc,CAAC,GAAG,CAAC,SAAS,EAAE,EAAE,SAAS,EAAE,MAAM,CAAC,OAAO,CAAC,OAAO,CAAC,SAAS,EAAE,OAAO,EAAE,MAAM,EAAE,CAAC,CAAC;QACrG,OAAO;YACL,IAAI,EAAE,QAAQ;YACd,SAAS;YACT,KAAK,EAAE,mBAAmB;YAC1B,OAAO,EAAE,CAAC,GAAG,QAAQ,CAAC,CAAC,OAAO,EAAE,CAAC,GAAG,CAAC,CAAC,OAAO,EAAE,EAAE,CAAC,CAAC,EAAE,KAAK,EAAE,OAAO,CAAC,OAAO,EAAE,KAAK,EAAE,QAAQ,CAAC,OAAO,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,CAAC;SACpH,CAAC;IACJ,CAAC;IAEO,gBAAgB,CAAC,OAAqB,EAAE,IAAY;QAC1D,OAAO,OAAO,CAAC,eAAe,CAAC,qBAAqB,EAAE,CAAC,IAAI,CAAC,CAAC,OAAO,EAAE,EAAE,CAAC,OAAO,CAAC,cAAc,KAAK,IAAI,CAAC;eACpG,OAAO,CAAC,eAAe,CAAC,IAAI,CAAC,CAAC,QAAQ,EAAE,EAAE,CAAC,QAAQ,CAAC,IAAI,KAAK,IAAI,CAAC;eAClE,OAAO,CAAC,cAAc,CAAC,SAAS,EAAE,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,KAAK,EAAE,EAAE,CAAC,SAAS,KAAK,CAAC,IAAI,EAAE,KAAK,IAAI,CAAC,CAAC;IACjG,CAAC;CACF;AAED,SAAS,wBAAwB,CAAC,OAA4B;IAC5D,MAAM,OAAO,GAAG,OAAO,CAAC,OAAO,CAAC;IAChC,OAAO;QACL,EAAE,EAAE,OAAO,CAAC,SAAS;QACrB,IAAI,EAAE,OAAO,CAAC,WAAW,IAAI,EAAE;QAC/B,GAAG,EAAE,OAAO,CAAC,GAAG;QAChB,GAAG,CAAC,OAAO,CAAC,WAAW,KAAK,SAAS,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,EAAE,IAAI,EAAE,OAAO,CAAC,WAAW,EAAE,CAAC;QAC3E,OAAO,EAAE,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE;QACjC,QAAQ,EAAE,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE;QAClC,YAAY,EAAE,OAAO,CAAC,QAAQ,CAAC,MAAM;QACrC,YAAY,EAAE,EAAE;KACjB,CAAC;AACJ,CAAC;AAED,SAAS,kBAAkB,CAAC,OAAqB;IAC/C,MAAM,KAAK,GAAG,OAAO,CAAC,eAAe,EAAE,CAAC;IACxC,OAAO;QACL,YAAY,KAAK,CAAC,SAAS,EAAE;QAC7B,aAAa,MAAM,CAAC,KAAK,CAAC,aAAa,CAAC,KAAK,MAAM,CAAC,KAAK,CAAC,YAAY,CAAC,UAAU,MAAM,CAAC,KAAK,CAAC,iBAAiB,CAAC,aAAa;QAC7H,eAAe,MAAM,CAAC,KAAK,CAAC,SAAS,CAAC,EAAE;QACxC,YAAY,MAAM,CAAC,KAAK,CAAC,MAAM,CAAC,KAAK,CAAC,KAAK,MAAM,CAAC,KAAK,CAAC,MAAM,CAAC,MAAM,CAAC,UAAU,MAAM,CAAC,KAAK,CAAC,MAAM,CAAC,KAAK,CAAC,EAAE;QAC5G,UAAU,KAAK,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE;KAClC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;AACf,CAAC;AAED,SAAS,sBAAsB,CAAC,MAAiD;IAC/E,OAAO;QACL,sBAAsB;QACtB,kBAAkB,MAAM,CAAC,MAAM,CAAC,YAAY,CAAC,EAAE;QAC/C,EAAE;QACF,MAAM,CAAC,OAAO;KACf,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;AACf,CAAC;AAED,SAAS,QAAQ,CAAC,IAAY,EAAE,SAAiB;IAC/C,MAAM,UAAU,GAAG,IAAI,CAAC,OAAO,CAAC,MAAM,EAAE,GAAG,CAAC,CAAC,IAAI,EAAE,CAAC;IACpD,OAAO,UAAU,CAAC,MAAM,IAAI,SAAS,CAAC,CAAC,CAAC,UAAU,CAAC,CAAC,CAAC,GAAG,UAAU,CAAC,KAAK,CAAC,CAAC,EAAE,SAAS,GAAG,CAAC,CAAC,GAAG,CAAC;AAChG,CAAC"}
@@ -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,2 @@
1
+ export {};
2
+ //# sourceMappingURL=sessionRuntimeStore.js.map
@@ -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":"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"}