@juspay/shooter 1.15.0 → 1.16.0
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/build/client/_app/immutable/assets/{0.BZLcOr5z.css → 0.DEfoFaGR.css} +1 -1
- package/build/client/_app/immutable/assets/0.DEfoFaGR.css.br +0 -0
- package/build/client/_app/immutable/assets/0.DEfoFaGR.css.gz +0 -0
- package/build/client/_app/immutable/chunks/Bkqjn62J.js +1 -0
- package/build/client/_app/immutable/chunks/Bkqjn62J.js.br +1 -0
- package/build/client/_app/immutable/chunks/Bkqjn62J.js.gz +0 -0
- package/build/client/_app/immutable/chunks/{gxvWeAns.js → DOHhmtDH.js} +1 -1
- package/build/client/_app/immutable/chunks/DOHhmtDH.js.br +0 -0
- package/build/client/_app/immutable/chunks/DOHhmtDH.js.gz +0 -0
- package/build/client/_app/immutable/chunks/{pMo6RVvN.js → DlS3abGJ.js} +1 -1
- package/build/client/_app/immutable/chunks/DlS3abGJ.js.br +0 -0
- package/build/client/_app/immutable/chunks/DlS3abGJ.js.gz +0 -0
- package/build/client/_app/immutable/entry/{app.B0PrrcUG.js → app.CSJG7N9H.js} +2 -2
- package/build/client/_app/immutable/entry/app.CSJG7N9H.js.br +0 -0
- package/build/client/_app/immutable/entry/app.CSJG7N9H.js.gz +0 -0
- package/build/client/_app/immutable/entry/start.CTt1901T.js +1 -0
- package/build/client/_app/immutable/entry/start.CTt1901T.js.br +2 -0
- package/build/client/_app/immutable/entry/start.CTt1901T.js.gz +0 -0
- package/build/client/_app/immutable/nodes/{0.D4GLHqPM.js → 0.qOL7xtFn.js} +1 -1
- package/build/client/_app/immutable/nodes/0.qOL7xtFn.js.br +0 -0
- package/build/client/_app/immutable/nodes/0.qOL7xtFn.js.gz +0 -0
- package/build/client/_app/immutable/nodes/{1.nJde5z5O.js → 1.Di708Ago.js} +1 -1
- package/build/client/_app/immutable/nodes/1.Di708Ago.js.br +0 -0
- package/build/client/_app/immutable/nodes/1.Di708Ago.js.gz +0 -0
- package/build/client/_app/immutable/nodes/{2.CLtsjLeG.js → 2.DSM1znqa.js} +1 -1
- package/build/client/_app/immutable/nodes/2.DSM1znqa.js.br +0 -0
- package/build/client/_app/immutable/nodes/2.DSM1znqa.js.gz +0 -0
- package/build/client/_app/immutable/nodes/{3.CKTUHtnx.js → 3.BPa5fh75.js} +1 -1
- package/build/client/_app/immutable/nodes/3.BPa5fh75.js.br +0 -0
- package/build/client/_app/immutable/nodes/3.BPa5fh75.js.gz +0 -0
- package/build/client/_app/immutable/nodes/6.B1LwwEF-.js +1 -0
- package/build/client/_app/immutable/nodes/6.B1LwwEF-.js.br +0 -0
- package/build/client/_app/immutable/nodes/6.B1LwwEF-.js.gz +0 -0
- package/build/client/_app/immutable/nodes/7.B7UJd8GQ.js +4 -0
- package/build/client/_app/immutable/nodes/7.B7UJd8GQ.js.br +0 -0
- package/build/client/_app/immutable/nodes/7.B7UJd8GQ.js.gz +0 -0
- package/build/client/_app/immutable/nodes/8.CG0mrgBU.js +2 -0
- package/build/client/_app/immutable/nodes/8.CG0mrgBU.js.br +0 -0
- package/build/client/_app/immutable/nodes/8.CG0mrgBU.js.gz +0 -0
- package/build/client/_app/immutable/nodes/9.KwzWaMHj.js +2 -0
- package/build/client/_app/immutable/nodes/9.KwzWaMHj.js.br +0 -0
- package/build/client/_app/immutable/nodes/9.KwzWaMHj.js.gz +0 -0
- package/build/client/_app/version.json +1 -1
- package/build/client/_app/version.json.br +0 -0
- package/build/client/_app/version.json.gz +0 -0
- package/build/server/chunks/{0-BUSWGJr9.js → 0-D8uPamNd.js} +3 -3
- package/build/server/chunks/{0-BUSWGJr9.js.map → 0-D8uPamNd.js.map} +1 -1
- package/build/server/chunks/{1-DjiQE1K0.js → 1-DhtioHbs.js} +2 -2
- package/build/server/chunks/{1-DjiQE1K0.js.map → 1-DhtioHbs.js.map} +1 -1
- package/build/server/chunks/{2-ThgVrRKa.js → 2-Cgh7ZFgE.js} +2 -2
- package/build/server/chunks/{2-ThgVrRKa.js.map → 2-Cgh7ZFgE.js.map} +1 -1
- package/build/server/chunks/{3-G5LiDFQ9.js → 3-I6hnjssH.js} +2 -2
- package/build/server/chunks/{3-G5LiDFQ9.js.map → 3-I6hnjssH.js.map} +1 -1
- package/build/server/chunks/{6--I7fF3Bx.js → 6-bPDbH1_W.js} +2 -2
- package/build/server/chunks/{6--I7fF3Bx.js.map → 6-bPDbH1_W.js.map} +1 -1
- package/build/server/chunks/{7-BwPLVwOR.js → 7-CpBrOkxQ.js} +2 -2
- package/build/server/chunks/{7-BwPLVwOR.js.map → 7-CpBrOkxQ.js.map} +1 -1
- package/build/server/chunks/{8-BwOMHaoQ.js → 8-BRGAVfze.js} +2 -2
- package/build/server/chunks/{8-BwOMHaoQ.js.map → 8-BRGAVfze.js.map} +1 -1
- package/build/server/chunks/{9-DkO6aJIB.js → 9-C6xuAb_Y.js} +2 -2
- package/build/server/chunks/{9-DkO6aJIB.js.map → 9-C6xuAb_Y.js.map} +1 -1
- package/build/server/chunks/{_server.ts-ByPExYfO.js → _server.ts-BLNDdFWC.js} +3 -3
- package/build/server/chunks/_server.ts-BLNDdFWC.js.map +1 -0
- package/build/server/chunks/{_server.ts-40c_epk8.js → _server.ts-BrRZXr-8.js} +5 -4
- package/build/server/chunks/_server.ts-BrRZXr-8.js.map +1 -0
- package/build/server/chunks/{_server.ts-BuYyCrnF.js → _server.ts-C6xbNz6d.js} +4 -3
- package/build/server/chunks/_server.ts-C6xbNz6d.js.map +1 -0
- package/build/server/chunks/{_server.ts-CjpQ10xh.js → _server.ts-CjK0g9dO.js} +18 -2
- package/build/server/chunks/_server.ts-CjK0g9dO.js.map +1 -0
- package/build/server/chunks/{_server.ts-0Xr2fWaq.js → _server.ts-Cq9_scaV.js} +17 -7
- package/build/server/chunks/_server.ts-Cq9_scaV.js.map +1 -0
- package/build/server/chunks/{_server.ts-CilRds58.js → _server.ts-D--_NXt2.js} +25 -16
- package/build/server/chunks/_server.ts-D--_NXt2.js.map +1 -0
- package/build/server/chunks/{_server.ts-2ixC-X3K.js → _server.ts-Dekgb6Hx.js} +4 -3
- package/build/server/chunks/_server.ts-Dekgb6Hx.js.map +1 -0
- package/build/server/chunks/opencode-db-path-CRgzBK5U.js +402 -0
- package/build/server/chunks/opencode-db-path-CRgzBK5U.js.map +1 -0
- package/build/server/chunks/{pty-manager-TyMUpDA9.js → pty-manager-aFpChJah.js} +35 -2
- package/build/server/chunks/pty-manager-aFpChJah.js.map +1 -0
- package/build/server/index.js +1 -1
- package/build/server/index.js.map +1 -1
- package/build/server/manifest.js +16 -16
- package/build/server/manifest.js.map +1 -1
- package/package.json +2 -2
- package/server.ts +12 -0
- package/src/lib/modules/client/common/index.ts +1 -0
- package/src/lib/modules/client/common/provider.ts +30 -0
- package/src/lib/modules/client/terminal/LaunchSheet.svelte +2 -0
- package/src/lib/modules/server/sessions/codex-parser.ts +286 -0
- package/src/lib/modules/server/sessions/codex-reader.ts +293 -0
- package/src/lib/modules/server/sessions/process-detector.ts +23 -0
- package/src/lib/modules/server/terminal/codex-watcher.ts +179 -0
- package/src/lib/modules/server/terminal/pty-manager.ts +41 -0
- package/src/lib/theme.css +21 -1
- package/src/lib/types/codex.ts +21 -0
- package/src/lib/types/generated/Sessions.ts +5 -1
- package/src/lib/types/index.ts +1 -0
- package/src/lib/types/server.ts +18 -5
- package/src/lib/types/sessions.ts +1 -1
- package/src/routes/api/device-token/+server.ts +7 -3
- package/src/routes/api/sessions/+server.ts +27 -15
- package/src/routes/api/sessions/connect/+server.ts +19 -11
- package/src/routes/api/terminals/+server.ts +1 -1
- package/src/routes/project/+page.svelte +7 -23
- package/src/routes/session/[id]/+page.svelte +3 -3
- package/src/routes/terminals/+page.svelte +1 -1
- package/src/routes/terminals/[id]/+page.svelte +1 -1
- package/build/client/_app/immutable/assets/0.BZLcOr5z.css.br +0 -0
- package/build/client/_app/immutable/assets/0.BZLcOr5z.css.gz +0 -0
- package/build/client/_app/immutable/chunks/X-tVU_3P.js +0 -1
- package/build/client/_app/immutable/chunks/X-tVU_3P.js.br +0 -0
- package/build/client/_app/immutable/chunks/X-tVU_3P.js.gz +0 -0
- package/build/client/_app/immutable/chunks/gxvWeAns.js.br +0 -0
- package/build/client/_app/immutable/chunks/gxvWeAns.js.gz +0 -0
- package/build/client/_app/immutable/chunks/pMo6RVvN.js.br +0 -0
- package/build/client/_app/immutable/chunks/pMo6RVvN.js.gz +0 -0
- package/build/client/_app/immutable/entry/app.B0PrrcUG.js.br +0 -0
- package/build/client/_app/immutable/entry/app.B0PrrcUG.js.gz +0 -0
- package/build/client/_app/immutable/entry/start.B1obDjVk.js +0 -1
- package/build/client/_app/immutable/entry/start.B1obDjVk.js.br +0 -0
- package/build/client/_app/immutable/entry/start.B1obDjVk.js.gz +0 -0
- package/build/client/_app/immutable/nodes/0.D4GLHqPM.js.br +0 -0
- package/build/client/_app/immutable/nodes/0.D4GLHqPM.js.gz +0 -0
- package/build/client/_app/immutable/nodes/1.nJde5z5O.js.br +0 -0
- package/build/client/_app/immutable/nodes/1.nJde5z5O.js.gz +0 -0
- package/build/client/_app/immutable/nodes/2.CLtsjLeG.js.br +0 -0
- package/build/client/_app/immutable/nodes/2.CLtsjLeG.js.gz +0 -0
- package/build/client/_app/immutable/nodes/3.CKTUHtnx.js.br +0 -0
- package/build/client/_app/immutable/nodes/3.CKTUHtnx.js.gz +0 -0
- package/build/client/_app/immutable/nodes/6.Cgc5iTlM.js +0 -1
- package/build/client/_app/immutable/nodes/6.Cgc5iTlM.js.br +0 -0
- package/build/client/_app/immutable/nodes/6.Cgc5iTlM.js.gz +0 -0
- package/build/client/_app/immutable/nodes/7.BXKvUopV.js +0 -4
- package/build/client/_app/immutable/nodes/7.BXKvUopV.js.br +0 -0
- package/build/client/_app/immutable/nodes/7.BXKvUopV.js.gz +0 -0
- package/build/client/_app/immutable/nodes/8.Df0leW0d.js +0 -2
- package/build/client/_app/immutable/nodes/8.Df0leW0d.js.br +0 -0
- package/build/client/_app/immutable/nodes/8.Df0leW0d.js.gz +0 -0
- package/build/client/_app/immutable/nodes/9.C4-N3geF.js +0 -2
- package/build/client/_app/immutable/nodes/9.C4-N3geF.js.br +0 -0
- package/build/client/_app/immutable/nodes/9.C4-N3geF.js.gz +0 -0
- package/build/server/chunks/_server.ts-0Xr2fWaq.js.map +0 -1
- package/build/server/chunks/_server.ts-2ixC-X3K.js.map +0 -1
- package/build/server/chunks/_server.ts-40c_epk8.js.map +0 -1
- package/build/server/chunks/_server.ts-BuYyCrnF.js.map +0 -1
- package/build/server/chunks/_server.ts-ByPExYfO.js.map +0 -1
- package/build/server/chunks/_server.ts-CilRds58.js.map +0 -1
- package/build/server/chunks/_server.ts-CjpQ10xh.js.map +0 -1
- package/build/server/chunks/opencode-db-path-DcfhJtJy.js +0 -15
- package/build/server/chunks/opencode-db-path-DcfhJtJy.js.map +0 -1
- package/build/server/chunks/pty-manager-TyMUpDA9.js.map +0 -1
package/src/lib/types/server.ts
CHANGED
|
@@ -8,11 +8,22 @@
|
|
|
8
8
|
import type { FSWatcher } from 'chokidar';
|
|
9
9
|
import type WebSocket from 'ws';
|
|
10
10
|
|
|
11
|
+
import type { CodexStreamParser } from '../modules/server/sessions/codex-parser';
|
|
11
12
|
import type { HolderClient } from '../modules/server/terminal/holder-client';
|
|
12
13
|
import type { ConversationMessage } from './sessions';
|
|
13
14
|
|
|
14
15
|
// ── holder-client types ─────────────────────────────────────────────
|
|
15
16
|
|
|
17
|
+
export interface CodexWatchState {
|
|
18
|
+
callbacks: Set<(messages: ConversationMessage[]) => void>;
|
|
19
|
+
idleTimer: null | ReturnType<typeof setTimeout>;
|
|
20
|
+
/** Incomplete trailing line buffered between reads. */
|
|
21
|
+
lineBuffer: string;
|
|
22
|
+
offset: number;
|
|
23
|
+
parser: CodexStreamParser;
|
|
24
|
+
watcher: FSWatcher;
|
|
25
|
+
}
|
|
26
|
+
|
|
16
27
|
/** Messages received from the holder process (local ndjson protocol). */
|
|
17
28
|
export type HolderIncomingMessage =
|
|
18
29
|
| { active: boolean; type: 'activity' }
|
|
@@ -22,19 +33,21 @@ export type HolderIncomingMessage =
|
|
|
22
33
|
| { exitCode: null | number; exited: boolean; pid: number; type: 'info' }
|
|
23
34
|
| { path: string; type: 'cwd' };
|
|
24
35
|
|
|
36
|
+
// ── pty-manager types ───────────────────────────────────────────────
|
|
37
|
+
|
|
25
38
|
/** Messages sent to the holder process (local ndjson protocol). */
|
|
26
39
|
export type HolderOutgoingMessage =
|
|
27
40
|
| { cols: number; rows: number; type: 'resize' }
|
|
28
41
|
| { data: string; type: 'input' }
|
|
29
42
|
| { signal?: string; type: 'kill' };
|
|
30
43
|
|
|
31
|
-
// ── pty-manager types ───────────────────────────────────────────────
|
|
32
|
-
|
|
33
44
|
/**
|
|
34
45
|
* Callback invoked when new JSONL entries are parsed from a watched file.
|
|
35
46
|
*/
|
|
36
47
|
export type OnNewEntries = (entries: ConversationMessage[]) => void;
|
|
37
48
|
|
|
49
|
+
// ── session-watcher types ───────────────────────────────────────────
|
|
50
|
+
|
|
38
51
|
export interface OpenCodeWatchState {
|
|
39
52
|
callbacks: Set<(messages: ConversationMessage[]) => void>;
|
|
40
53
|
/** Set of message IDs we have already emitted, to avoid duplicates. */
|
|
@@ -49,8 +62,6 @@ export interface OpenCodeWatchState {
|
|
|
49
62
|
sessionId: string;
|
|
50
63
|
}
|
|
51
64
|
|
|
52
|
-
// ── session-watcher types ───────────────────────────────────────────
|
|
53
|
-
|
|
54
65
|
export interface PtyManagedTerminal {
|
|
55
66
|
args: string[];
|
|
56
67
|
clients: Set<WebSocket>;
|
|
@@ -78,12 +89,14 @@ export interface PtyManagedTerminal {
|
|
|
78
89
|
watcherOffset: number;
|
|
79
90
|
}
|
|
80
91
|
|
|
92
|
+
// ── opencode-watcher types ──────────────────────────────────────────
|
|
93
|
+
|
|
81
94
|
export interface PtyOutputBuffer {
|
|
82
95
|
data: string[];
|
|
83
96
|
size: number;
|
|
84
97
|
}
|
|
85
98
|
|
|
86
|
-
// ──
|
|
99
|
+
// ── codex-watcher types ─────────────────────────────────────────────
|
|
87
100
|
|
|
88
101
|
export interface SessionWatchedFile {
|
|
89
102
|
callbacks: Set<OnNewEntries>;
|
|
@@ -1,4 +1,3 @@
|
|
|
1
|
-
import { env } from '$env/dynamic/private';
|
|
2
1
|
import { validateAuth } from '$lib/modules/server/auth';
|
|
3
2
|
import { json } from '@sveltejs/kit';
|
|
4
3
|
import { existsSync, mkdirSync, readFileSync, writeFileSync } from 'fs';
|
|
@@ -70,9 +69,14 @@ export const POST: RequestHandler = async ({ request }) => {
|
|
|
70
69
|
tokens[platform] = token;
|
|
71
70
|
writeTokens(tokens);
|
|
72
71
|
|
|
73
|
-
// Update in-memory env so APNs can use it immediately (iOS is the primary APNs target)
|
|
72
|
+
// Update in-memory env so APNs can use it immediately (iOS is the primary APNs target).
|
|
73
|
+
// SvelteKit's $env/dynamic/private exposes a Proxy whose getter reads process.env at
|
|
74
|
+
// access time but whose setter does NOT propagate to process.env. Assigning via the
|
|
75
|
+
// Proxy is a silent no-op, so subsequent /api/notify calls still read the stale value
|
|
76
|
+
// from .env. Write straight to process.env so env.DEVICE_TOKEN picks up the new token
|
|
77
|
+
// on the next read.
|
|
74
78
|
if (platform === 'ios') {
|
|
75
|
-
|
|
79
|
+
process.env.DEVICE_TOKEN = token;
|
|
76
80
|
}
|
|
77
81
|
|
|
78
82
|
console.log(`[device-token] Registered ${platform} token (length: ${token.length})`);
|
|
@@ -1,6 +1,7 @@
|
|
|
1
1
|
import type { ProjectGroup } from '$lib/types';
|
|
2
2
|
|
|
3
3
|
import { validateAuth } from '$lib/modules/server/auth';
|
|
4
|
+
import { getCodexConversation, listCodexProjects } from '$lib/modules/server/sessions/codex-reader';
|
|
4
5
|
import {
|
|
5
6
|
getSessionConversation,
|
|
6
7
|
listProjectsWithSessions,
|
|
@@ -26,27 +27,35 @@ function getMergedProjects(): ProjectGroup[] {
|
|
|
26
27
|
|
|
27
28
|
const claudeProjects = listProjectsWithSessions();
|
|
28
29
|
const openCodeProjects = listOpenCodeProjects();
|
|
30
|
+
const codexProjects = listCodexProjects();
|
|
29
31
|
const projectsByPath = new Map<string, ProjectGroup>();
|
|
30
32
|
|
|
31
33
|
for (const p of claudeProjects) {
|
|
32
34
|
projectsByPath.set(p.fullPath, { ...p });
|
|
33
35
|
}
|
|
34
36
|
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
)
|
|
42
|
-
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
|
|
37
|
+
// Merge a provider's projects into the map, deduplicating by absolute path so
|
|
38
|
+
// sessions from different agents in the same directory group under one project.
|
|
39
|
+
const mergeProvider = (incoming: ProjectGroup[], stripSuffix?: string): void => {
|
|
40
|
+
for (const group of incoming) {
|
|
41
|
+
const cleanName = stripSuffix ? group.name.replace(stripSuffix, '') : group.name;
|
|
42
|
+
const existing = projectsByPath.get(group.fullPath);
|
|
43
|
+
if (existing) {
|
|
44
|
+
existing.sessions.push(...group.sessions);
|
|
45
|
+
existing.sessions.sort(
|
|
46
|
+
(a, b) => new Date(b.modified).getTime() - new Date(a.modified).getTime()
|
|
47
|
+
);
|
|
48
|
+
existing.sessionCount = existing.sessions.length;
|
|
49
|
+
existing.lastModified = existing.sessions[0]?.modified || existing.lastModified;
|
|
50
|
+
existing.name = existing.name.replace(stripSuffix ?? '', '');
|
|
51
|
+
} else {
|
|
52
|
+
projectsByPath.set(group.fullPath, { ...group, name: cleanName });
|
|
53
|
+
}
|
|
48
54
|
}
|
|
49
|
-
}
|
|
55
|
+
};
|
|
56
|
+
|
|
57
|
+
mergeProvider(openCodeProjects, ' (OpenCode)');
|
|
58
|
+
mergeProvider(codexProjects);
|
|
50
59
|
|
|
51
60
|
cachedProjects = [...projectsByPath.values()].sort(
|
|
52
61
|
(a, b) => new Date(b.lastModified).getTime() - new Date(a.lastModified).getTime()
|
|
@@ -88,10 +97,13 @@ export const GET: RequestHandler = ({ request, url }) => {
|
|
|
88
97
|
: undefined;
|
|
89
98
|
let messages = getSessionConversation(sessionId, offset, limit, claudeProjectDir);
|
|
90
99
|
|
|
91
|
-
// If Claude Code reader found nothing, try OpenCode
|
|
100
|
+
// If Claude Code reader found nothing, try OpenCode, then Codex.
|
|
92
101
|
if (messages.length === 0) {
|
|
93
102
|
messages = getOpenCodeConversation(sessionId, offset, limit);
|
|
94
103
|
}
|
|
104
|
+
if (messages.length === 0) {
|
|
105
|
+
messages = getCodexConversation(sessionId, offset, limit);
|
|
106
|
+
}
|
|
95
107
|
|
|
96
108
|
// Find session info — short-circuit when project is already resolved
|
|
97
109
|
let sessionInfo = matchedProject?.sessions.find((s) => s.id === sessionId) ?? null;
|
|
@@ -38,8 +38,9 @@ export const POST: RequestHandler = async ({ request }) => {
|
|
|
38
38
|
return json({ error: 'cwd is required (string)' }, { status: 400 });
|
|
39
39
|
}
|
|
40
40
|
|
|
41
|
-
|
|
42
|
-
|
|
41
|
+
const VALID_COMMANDS = ['claude', 'opencode', 'codex', 'gemini'];
|
|
42
|
+
if (!command || !VALID_COMMANDS.includes(command)) {
|
|
43
|
+
return json({ error: `command must be one of: ${VALID_COMMANDS.join(', ')}` }, { status: 400 });
|
|
43
44
|
}
|
|
44
45
|
|
|
45
46
|
// --- Validate cwd (same checks as POST /api/terminals) ---
|
|
@@ -63,13 +64,14 @@ export const POST: RequestHandler = async ({ request }) => {
|
|
|
63
64
|
|
|
64
65
|
// --- Reuse existing terminal if one is already running for this session ---
|
|
65
66
|
|
|
66
|
-
const existing = ptyManager
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
|
|
72
|
-
|
|
67
|
+
const existing = ptyManager.list().find(
|
|
68
|
+
(t) =>
|
|
69
|
+
t.status === 'running' &&
|
|
70
|
+
// Claude: <id>.jsonl ; Codex rollout: rollout-<ts>-<id>.jsonl ; OpenCode: session id
|
|
71
|
+
(t.sessionFile?.endsWith(`/${sessionId}.jsonl`) ||
|
|
72
|
+
t.sessionFile?.endsWith(`-${sessionId}.jsonl`) ||
|
|
73
|
+
t.openCodeSessionId === sessionId)
|
|
74
|
+
);
|
|
73
75
|
|
|
74
76
|
if (existing) {
|
|
75
77
|
console.log(
|
|
@@ -94,9 +96,15 @@ export const POST: RequestHandler = async ({ request }) => {
|
|
|
94
96
|
return json({ error: 'No existing terminal for this session' }, { status: 404 });
|
|
95
97
|
}
|
|
96
98
|
|
|
97
|
-
// --- Build args based on command ---
|
|
99
|
+
// --- Build args based on command (resume convention differs per agent CLI) ---
|
|
98
100
|
|
|
99
|
-
const
|
|
101
|
+
const resumeArgs: Record<string, string[]> = {
|
|
102
|
+
claude: ['--resume', sessionId],
|
|
103
|
+
codex: ['resume', sessionId],
|
|
104
|
+
gemini: [], // Gemini CLI has no session-resume flag; launch fresh in the cwd.
|
|
105
|
+
opencode: ['--session', sessionId],
|
|
106
|
+
};
|
|
107
|
+
const args: string[] = resumeArgs[command] ?? [];
|
|
100
108
|
|
|
101
109
|
try {
|
|
102
110
|
const terminal = await ptyManager.create(command, args, realCwd, 120, 40);
|
|
@@ -7,7 +7,7 @@ import { basename, isAbsolute, relative } from 'path';
|
|
|
7
7
|
|
|
8
8
|
import type { RequestHandler } from './$types';
|
|
9
9
|
|
|
10
|
-
const ALLOWED_COMMANDS = ['zsh', 'bash', 'sh', 'fish', 'claude', 'opencode'];
|
|
10
|
+
const ALLOWED_COMMANDS = ['zsh', 'bash', 'sh', 'fish', 'claude', 'opencode', 'codex', 'gemini'];
|
|
11
11
|
|
|
12
12
|
/** Extract the last non-empty line from a scrollback string. */
|
|
13
13
|
function lastScrollbackLine(scrollback: string): null | string {
|
|
@@ -12,6 +12,8 @@
|
|
|
12
12
|
getCached,
|
|
13
13
|
isShooterConfig,
|
|
14
14
|
setCache,
|
|
15
|
+
sourceLabel,
|
|
16
|
+
sourceToCommand,
|
|
15
17
|
} from '$lib/modules/client/common';
|
|
16
18
|
import { Banner, Button, EmptyState, Icon, Pill, Shimmer } from '@juspay/svelte-ui-components';
|
|
17
19
|
import { onDestroy, onMount } from 'svelte';
|
|
@@ -246,10 +248,7 @@
|
|
|
246
248
|
Back to Projects
|
|
247
249
|
</a>
|
|
248
250
|
</div>
|
|
249
|
-
<EmptyState
|
|
250
|
-
title="Project Not Found"
|
|
251
|
-
description="The requested project could not be found."
|
|
252
|
-
>
|
|
251
|
+
<EmptyState title="Project Not Found" description="The requested project could not be found.">
|
|
253
252
|
{#snippet icon()}<Icon svg={AlertTriangleSvg} classes="icon-24" />{/snippet}
|
|
254
253
|
</EmptyState>
|
|
255
254
|
{:else}
|
|
@@ -269,10 +268,7 @@
|
|
|
269
268
|
</div>
|
|
270
269
|
|
|
271
270
|
{#if project.sessions.length === 0}
|
|
272
|
-
<EmptyState
|
|
273
|
-
title="No sessions yet"
|
|
274
|
-
description="Sessions for this project will appear here"
|
|
275
|
-
>
|
|
271
|
+
<EmptyState title="No sessions yet" description="Sessions for this project will appear here">
|
|
276
272
|
{#snippet icon()}<Icon svg={BellSvg} classes="icon-24" />{/snippet}
|
|
277
273
|
</EmptyState>
|
|
278
274
|
{:else}
|
|
@@ -292,11 +288,7 @@
|
|
|
292
288
|
<Button
|
|
293
289
|
classes="btn-connect btn-xs"
|
|
294
290
|
onclick={(e: MouseEvent): void =>
|
|
295
|
-
void connectToSession(
|
|
296
|
-
e,
|
|
297
|
-
session.id,
|
|
298
|
-
session.source === 'opencode' ? 'opencode' : 'claude'
|
|
299
|
-
)}
|
|
291
|
+
void connectToSession(e, session.id, sourceToCommand(session.source))}
|
|
300
292
|
disabled={connectingSessionId === session.id}
|
|
301
293
|
showLoader={connectingSessionId === session.id}
|
|
302
294
|
>
|
|
@@ -307,11 +299,7 @@
|
|
|
307
299
|
<Button
|
|
308
300
|
classes="btn-resume btn-xs"
|
|
309
301
|
onclick={(e: MouseEvent): void =>
|
|
310
|
-
void connectToSession(
|
|
311
|
-
e,
|
|
312
|
-
session.id,
|
|
313
|
-
session.source === 'opencode' ? 'opencode' : 'claude'
|
|
314
|
-
)}
|
|
302
|
+
void connectToSession(e, session.id, sourceToCommand(session.source))}
|
|
315
303
|
disabled={connectingSessionId === session.id}
|
|
316
304
|
showLoader={connectingSessionId === session.id}
|
|
317
305
|
text="Resume"
|
|
@@ -324,11 +312,7 @@
|
|
|
324
312
|
{#if session.gitBranch}
|
|
325
313
|
<Pill text="🌿 {session.gitBranch}" classes="pill-git-branch" />
|
|
326
314
|
{/if}
|
|
327
|
-
{
|
|
328
|
-
<Pill text="OpenCode" classes="pill-source-opencode" />
|
|
329
|
-
{:else}
|
|
330
|
-
<Pill text="Claude Code" classes="pill-source-claude" />
|
|
331
|
-
{/if}
|
|
315
|
+
<Pill text={sourceLabel(session.source)} classes="pill-source-{session.source}" />
|
|
332
316
|
</div>
|
|
333
317
|
<div class="session-meta-row">
|
|
334
318
|
<span class="session-modified">Last updated {formatDate(session.modified)}</span>
|
|
@@ -10,7 +10,7 @@
|
|
|
10
10
|
|
|
11
11
|
import { browser } from '$app/environment';
|
|
12
12
|
import { page } from '$app/state';
|
|
13
|
-
import { getCached, setCache } from '$lib/modules/client/common';
|
|
13
|
+
import { getCached, setCache, sourceToCommand } from '$lib/modules/client/common';
|
|
14
14
|
import ChatView from '$lib/modules/client/terminal/ChatView.svelte';
|
|
15
15
|
import { Button } from '@juspay/svelte-ui-components';
|
|
16
16
|
import { onMount } from 'svelte';
|
|
@@ -315,7 +315,7 @@
|
|
|
315
315
|
sendStatus = 'connecting';
|
|
316
316
|
|
|
317
317
|
try {
|
|
318
|
-
const command = session.source
|
|
318
|
+
const command = sourceToCommand(session.source);
|
|
319
319
|
const res = await fetch('/api/sessions/connect', {
|
|
320
320
|
body: JSON.stringify({ command, cwd: session.projectPath, sessionId }),
|
|
321
321
|
headers: {
|
|
@@ -510,7 +510,7 @@
|
|
|
510
510
|
}
|
|
511
511
|
|
|
512
512
|
try {
|
|
513
|
-
const command = session.source
|
|
513
|
+
const command = sourceToCommand(session.source);
|
|
514
514
|
const res = await fetch('/api/sessions/connect', {
|
|
515
515
|
body: JSON.stringify({
|
|
516
516
|
command,
|
|
@@ -21,7 +21,7 @@
|
|
|
21
21
|
|
|
22
22
|
const POLL_INTERVAL_MS = 10_000;
|
|
23
23
|
const CACHE_KEY = 'shooter_terminals';
|
|
24
|
-
const AI_COMMANDS = ['claude', 'opencode'];
|
|
24
|
+
const AI_COMMANDS = ['claude', 'opencode', 'codex', 'gemini'];
|
|
25
25
|
const SHELL_COMMANDS = ['zsh', 'bash', 'sh', 'fish'];
|
|
26
26
|
|
|
27
27
|
let terminals = $state<TerminalListItem[]>([]);
|
|
Binary file
|
|
Binary file
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
import{s as r,p as e}from"./gxvWeAns.js";const a={get params(){return e.params},get url(){return e.url}};r.updated.check;const s=a;export{s as p};
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
import{l as o,a as r}from"../chunks/gxvWeAns.js";export{o as load_css,r as start};
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
import{i as j,a as f,f as m,s as S}from"../chunks/Cg3dlX05.js";import{p as We,s as k,q as Xe,v as Ye,y as et,w as tt,z as st,g as e,b as c,$ as at,c as n,d as l,r as o,x as p,A as L,B as Q,t as W}from"../chunks/DYuMZGL5.js";import{b as rt,e as ve,i as ot,B as N,c as X,P as D,d as nt}from"../chunks/DZvnhU_8.js";import{h as it}from"../chunks/DZQMsHM5.js";import{g as ct}from"../chunks/gxvWeAns.js";import{p as lt}from"../chunks/X-tVU_3P.js";import{A as dt}from"../chunks/DzuS5Nbr.js";import{B as vt}from"../chunks/Bn-6X1BO.js";import{S as ue,R as ut}from"../chunks/BRqaaL5D.js";import{g as ft,s as pt,c as mt}from"../chunks/xs1Xl3_e.js";import{i as ht}from"../chunks/gQJcRhou.js";import"../chunks/BDcFu3l7.js";import{f as _t}from"../chunks/BLszSzTT.js";import{E as fe}from"../chunks/C5VOyQCG.js";var gt=m('<meta name="description" content="Project sessions sorted by latest update"/>'),yt=m('<div class="loading-container"><!> <!></div>'),bt=m('<div class="project-back-row svelte-urn6fi"><a href="/" class="back-link"><span class="back-arrow">←</span> Back to Projects</a></div> <!>',1),jt=m("<!> Refresh",1),St=m('<div class="session-card-subtitle"> </div>'),$t=m('<span class="connect-dot svelte-urn6fi"></span> Connect',1),xt=m('<a class="session-card"><div class="session-card-header"><div><h3 class="session-card-title"> </h3> <!></div> <div class="session-card-actions svelte-urn6fi"><!> <!></div></div> <div class="session-stats"><span><strong> </strong> messages</span> <!> <!></div> <div class="session-meta-row svelte-urn6fi"><span class="session-modified svelte-urn6fi"> </span> <span class="session-duration svelte-urn6fi"> </span></div></a>'),Pt=m('<div style="text-align: center; padding: 1rem;"><!></div>'),wt=m('<div class="sessions-container svelte-urn6fi"></div> <!>',1),kt=m('<div class="chat-session-header"><div class="chat-session-header-top svelte-urn6fi"><a href="/" class="back-link">← Back to Projects</a> <!></div> <h1 class="chat-session-title"> </h1> <div class="chat-session-meta"><span class="session-card-subtitle"> </span> <span> </span></div></div> <!>',1),Ct=m('<main class="main"><!> <!></main>');function Jt(pe,me){We(me,!0);const he=15e3,M=20;let i=k(null),g=k(!0),h=k(null),R=k(null),E=null,C=k(M),Y=k(Xe([])),$=k(null);const x=p(()=>lt.url.searchParams.get("id")||""),_e=p(()=>e(i)?e(i).sessions.slice(0,e(C)):[]),ge=p(()=>e(i)?e(C)<e(i).sessions.length:!1),ye=p(()=>new Set(e(Y).map(t=>t.sessionId).filter(t=>typeof t=="string")));Ye(()=>{be();const t=ft(`shooter_project_${e(x)}`);t&&(c(i,t,!0),c(g,!1)),setTimeout(()=>{O(),z()},50),E=setInterval(()=>{e(h)?.apiKey&&e(x)&&(O(),z())},he)}),et(()=>{E&&(clearInterval(E),E=null)});function be(){try{const t=localStorage.getItem("shooter_config");if(t){const s=JSON.parse(t);ht(s)?c(h,s,!0):(localStorage.removeItem("shooter_config"),c(h,null))}}catch{}}async function O(t=!1){if(!e(h)?.apiKey||!e(x)){c(g,!1);return}e(i)||c(g,!0);try{const d=await fetch(t?"/api/sessions?refresh=true":"/api/sessions",{headers:{Authorization:`Bearer ${e(h).apiKey}`}});if(!d.ok){c(R,`Failed to load project (HTTP ${d.status})`),c(g,!1);return}c(R,null);const u=(await d.json()).projects.find(B=>B.id===e(x))||null;c(i,u,!0),u&&pt(`shooter_project_${e(x)}`,u)}catch(s){console.error("Failed to fetch project:",s)}finally{c(g,!1)}}async function z(){if(e(h)?.apiKey)try{const t=await fetch("/api/sessions/detect",{headers:{Authorization:`Bearer ${e(h).apiKey}`}});if(t.ok){const s=await t.json();c(Y,s.processes,!0)}}catch{}}async function ee(t,s,d){if(t.preventDefault(),t.stopPropagation(),!(!e(h)?.apiKey||!e(i)||e($))){c($,s,!0);try{const v=await fetch("/api/sessions/connect",{body:JSON.stringify({command:d,cwd:e(i).fullPath,sessionId:s}),headers:{Authorization:`Bearer ${e(h).apiKey}`,"Content-Type":"application/json"},method:"POST"});if(v.ok){const u=await v.json();ct(`/terminals/${u.terminalId}`)}}catch(v){console.error("Failed to connect to session:",v)}finally{c($,null)}}}async function je(){c(g,!0),c(i,null),c(C,M),mt(`shooter_project_${e(x)}`),await O(!0),await z()}function Se(){c(C,e(C)+M)}function te(t){if(!t)return"";const s=new Date(t);return isNaN(s.getTime())?"":s.toLocaleDateString("en-US",{day:"numeric",hour:"2-digit",minute:"2-digit",month:"short"})}function $e(t,s){return t.length<=s?t:`${t.slice(0,s).trimEnd()}...`}var F=Ct();it("urn6fi",t=>{var s=gt();st(()=>{at.title=`${(e(i)?.name||"Project")??""} - Shooter`}),f(t,s)});var se=n(F);{var xe=t=>{rt(t,{get text(){return e(R)},classes:"banner-error"})};j(se,t=>{e(R)&&t(xe)})}var Pe=l(se,2);{var we=t=>{var s=yt(),d=n(s);ue(d,{classes:"shimmer-header"});var v=l(d,2);ve(v,16,()=>Array(4),ot,(u,B)=>{ue(u,{classes:"shimmer-card"})}),o(s),f(t,s)},ke=t=>{var s=bt(),d=l(L(s),2);fe(d,{title:"Project Not Found",description:"The requested project could not be found.",icon:u=>{X(u,{get svg(){return dt},classes:"icon-24"})}}),f(t,s)},Ce=t=>{var s=kt(),d=L(s),v=n(d),u=l(n(v),2);N(u,{classes:"btn-secondary",onclick:je,get disabled(){return e(g)},children:(y,K)=>{var b=jt(),J=L(b);X(J,{get svg(){return ut},classes:"icon-14"}),Q(),f(y,b)},$$slots:{default:!0}}),o(v);var B=l(v,2),Be=n(B,!0);o(B);var ae=l(B,2),q=n(ae),Te=n(q,!0);o(q);var re=l(q,2),Ie=n(re);o(re),o(ae),o(d);var Ae=l(d,2);{var Re=y=>{fe(y,{title:"No sessions yet",description:"Sessions for this project will appear here",icon:b=>{X(b,{get svg(){return vt},classes:"icon-24"})}})},Ee=y=>{var K=wt(),b=L(K);ve(b,21,()=>e(_e),P=>P.id,(P,r)=>{var T=xt(),I=n(T),G=n(I),H=n(G),Le=n(H,!0);o(H);var Ne=l(H,2);{var De=a=>{var _=St(),A=n(_,!0);o(_),W(w=>S(A,w),[()=>$e(e(r).summary,80)]),f(a,_)};j(Ne,a=>{e(r).summary&&a(De)})}o(G);var oe=l(G,2),ne=n(oe);{let a=p(()=>_t(e(r).modified));D(ne,{get text(){return e(a)},classes:"pill-session-time"})}var Me=l(ne,2);{var Oe=a=>{{let _=p(()=>e($)===e(r).id),A=p(()=>e($)===e(r).id);N(a,{classes:"btn-connect btn-xs",onclick:w=>{ee(w,e(r).id,e(r).source==="opencode"?"opencode":"claude")},get disabled(){return e(_)},get showLoader(){return e(A)},children:(w,Bt)=>{var Qe=$t();Q(),f(w,Qe)},$$slots:{default:!0}})}},ze=p(()=>e(ye).has(e(r).id)),Fe=a=>{{let _=p(()=>e($)===e(r).id),A=p(()=>e($)===e(r).id);N(a,{classes:"btn-resume btn-xs",onclick:w=>{ee(w,e(r).id,e(r).source==="opencode"?"opencode":"claude")},get disabled(){return e(_)},get showLoader(){return e(A)},text:"Resume"})}};j(Me,a=>{e(ze)?a(Oe):a(Fe,-1)})}o(oe),o(I);var U=l(I,2),V=n(U),ie=n(V),qe=n(ie,!0);o(ie),Q(),o(V);var ce=l(V,2);{var Je=a=>{D(a,{get text(){return`🌿 ${e(r).gitBranch??""}`},classes:"pill-git-branch"})};j(ce,a=>{e(r).gitBranch&&a(Je)})}var Ge=l(ce,2);{var He=a=>{D(a,{text:"OpenCode",classes:"pill-source-opencode"})},Ue=a=>{D(a,{text:"Claude Code",classes:"pill-source-claude"})};j(Ge,a=>{e(r).source==="opencode"?a(He):a(Ue,-1)})}o(U);var le=l(U,2),Z=n(le),Ve=n(Z);o(Z);var de=l(Z,2),Ze=n(de);o(de),o(le),o(T),W((a,_)=>{nt(T,"href",`/session/${e(r).id??""}?project=${e(x)??""}`),S(Le,e(r).title),S(qe,e(r).messageCount),S(Ve,`Last updated ${a??""}`),S(Ze,`Created ${_??""}`)},[()=>te(e(r).modified),()=>te(e(r).created)]),f(P,T)}),o(b);var J=l(b,2);{var Ke=P=>{var r=Pt(),T=n(r);{let I=p(()=>`Load More (${e(i).sessions.length-e(C)} remaining)`);N(T,{classes:"btn-secondary",onclick:Se,get text(){return e(I)}})}o(r),f(P,r)};j(J,P=>{e(ge)&&P(Ke)})}f(y,K)};j(Ae,y=>{e(i).sessions.length===0?y(Re):y(Ee,-1)})}W(()=>{S(Be,e(i).name),S(Te,e(i).fullPath),S(Ie,`${e(i).sessionCount??""} sessions`)}),f(t,s)};j(Pe,t=>{e(g)&&!e(i)?t(we):e(i)?t(Ce,-1):t(ke,1)})}o(F),f(pe,F),tt()}export{Jt as component};
|
|
Binary file
|
|
Binary file
|
|
@@ -1,4 +0,0 @@
|
|
|
1
|
-
import{i as K,a as h,f as g,s as V}from"../chunks/Cg3dlX05.js";import{p as ve,s as v,q as ge,v as we,w as ye,z as be,b as s,g as e,$ as Se,c as _,r as $,x as T,A as H,d as D,t as Q}from"../chunks/DYuMZGL5.js";import{h as _e}from"../chunks/DZQMsHM5.js";import{d as X,B as $e}from"../chunks/DZvnhU_8.js";import{p as x}from"../chunks/X-tVU_3P.js";import{g as ke,s as Y}from"../chunks/xs1Xl3_e.js";import"../chunks/BDcFu3l7.js";import{C as Ie}from"../chunks/Pw0jDB7M.js";var je=g('<meta name="description" content="Session conversation view"/>'),Ce=g('<div class="loading-container" aria-live="polite" aria-busy="true"><div class="shimmer shimmer-header" aria-hidden="true"></div> <div class="chat-container"><div class="shimmer shimmer-bubble shimmer-bubble-user" aria-hidden="true"></div> <div class="shimmer shimmer-bubble shimmer-bubble-assistant" aria-hidden="true"></div> <div class="shimmer shimmer-bubble shimmer-bubble-user-short" aria-hidden="true"></div> <div class="shimmer shimmer-bubble shimmer-bubble-assistant-wide" aria-hidden="true"></div></div></div>'),Oe=g('<div class="session-back-row svelte-1a2wdfb"><a class="back-link">← Back</a></div> <p class="error-text svelte-1a2wdfb"> </p>',1),Me=g('<span class="resume-status svelte-1a2wdfb">Connecting...</span>'),Ne=g('<span class="resume-status svelte-1a2wdfb">Resuming session...</span>'),Ae=g('<span class="connection-dot connected svelte-1a2wdfb"></span>'),Pe=g('<span class="connection-dot reconnecting svelte-1a2wdfb"></span>'),Te=g('<div class="load-earlier-row svelte-1a2wdfb"><!></div>'),De=g('<div class="session-header svelte-1a2wdfb"><div class="session-header-row svelte-1a2wdfb"><a class="back-link">← Back</a> <!></div> <h1 class="session-title svelte-1a2wdfb"> </h1></div> <div class="session-chat-container svelte-1a2wdfb"><!> <!></div>',1),Re=g('<main class="main session-page-main svelte-1a2wdfb"><!></main>');function Le(Z,ee){ve(ee,!0);let f=v(null),o=v(ge([])),w=v(!0),I=v(null),R=v(!1),N=v(!1),A=v(200),j=v(null),k=v("idle"),c=v("idle"),y=null,b=null,C=null,O=null,d=!1;const W=T(()=>x.params.id),M=T(()=>x.url.searchParams.get("project")||""),te=T(()=>e(c)==="connecting"||e(c)==="resuming");function P(){try{const t=localStorage.getItem("shooter_config");return t?JSON.parse(t):null}catch{return null}}async function L(){const t=P();if(!t)return null;try{const n=await fetch("/api/ws-ticket",{headers:{Authorization:`Bearer ${t.apiKey}`},method:"POST"});return n.ok?(await n.json()).ticket:null}catch{return null}}async function z(t,n){if(d||C&&(C.readyState===WebSocket.OPEN||C.readyState===WebSocket.CONNECTING))return;const a=await L();if(!a||d)return;s(k,"connecting");const i=window.location.protocol==="https:"?"wss:":"ws:",r=new WebSocket(`${i}//${window.location.host}${t}?ticket=${a}`);r.onopen=()=>{if(d){r.close();return}s(k,"connected"),r.send(JSON.stringify({sessionId:n,type:"subscribe"}))},r.onmessage=u=>{if(!d)try{const p=JSON.parse(u.data);se(p)}catch{}},r.onclose=()=>{d||(s(k,"disconnected"),C=null,e(j)&&(O&&clearTimeout(O),O=setTimeout(()=>{O=null,!d&&e(j)&&z(`/ws/session/${e(j)}`,e(j))},2e3)))},r.onerror=()=>{d||s(k,"disconnected")},C=r}async function q(t){if(d)return;const n=await L();if(!n||d)return;const a=window.location.protocol==="https:"?"wss:":"ws:",i=new WebSocket(`${a}//${window.location.host}${t}?ticket=${n}`);i.onopen=()=>{if(d){i.close();return}b=i,e(c)==="ready"&&y&&(i.send(JSON.stringify({data:`${y}
|
|
2
|
-
`,type:"input"})),y=null)},i.onclose=()=>{b===i&&(b=null)}}function se(t){if(t.type==="history"){const n=t.messages||[];s(o,n.map(a=>({id:a.id,parts:a.content,role:a.role,timestamp:a.timestamp})),!0),s(R,!1),e(c)==="resuming"&&(s(c,"ready"),y&&b?.readyState===WebSocket.OPEN&&(b.send(JSON.stringify({data:`${y}
|
|
3
|
-
`,type:"input"})),y=null))}else if(t.type==="message")s(o,[...e(o),{id:`msg-${Date.now()}-${Math.random().toString(36).slice(2,8)}`,parts:t.content||[],role:t.role||"assistant",timestamp:t.timestamp||new Date().toISOString()}],!0);else if(t.type==="tool-use"){const n={id:t.id||`tool-${Date.now()}`,input:t.input||{},toolName:t.name,type:"tool_use"},a=e(o).length>0?e(o)[e(o).length-1]:null;a?.role==="assistant"?s(o,[...e(o).slice(0,-1),{...a,parts:[...a.parts,n]}],!0):s(o,[...e(o),{id:`tool-${Date.now()}-${Math.random().toString(36).slice(2,8)}`,parts:[n],role:"assistant",timestamp:new Date().toISOString()}],!0)}else if(t.type==="tool-result"){const n={isError:t.isError||!1,output:t.output||"",toolUseId:t.id,type:"tool_result"},a=e(o).length>0?e(o)[e(o).length-1]:null;a?.role==="system"?s(o,[...e(o).slice(0,-1),{...a,parts:[...a.parts,n]}],!0):s(o,[...e(o),{id:`result-${Date.now()}-${Math.random().toString(36).slice(2,8)}`,parts:[n],role:"system",timestamp:new Date().toISOString()}],!0)}else if(t.type==="thinking"){const n={content:t.text||"",type:"thinking"},a=e(o).length>0?e(o)[e(o).length-1]:null;a?.role==="assistant"?s(o,[...e(o).slice(0,-1),{...a,parts:[...a.parts,n]}],!0):s(o,[...e(o),{id:`think-${Date.now()}-${Math.random().toString(36).slice(2,8)}`,parts:[n],role:"assistant",timestamp:new Date().toISOString()}],!0)}else t.type==="session-end"&&s(c,"idle")}async function ne(){if(!e(f)||d){s(c,"idle");return}const t=P();if(!t){s(c,"idle");return}s(c,"connecting");try{const n=e(f).source==="opencode"?"opencode":"claude",a=await fetch("/api/sessions/connect",{body:JSON.stringify({command:n,cwd:e(f).projectPath,sessionId:e(W)}),headers:{Authorization:`Bearer ${t.apiKey}`,"Content-Type":"application/json"},method:"POST"});if(!a.ok||d){s(c,"idle");return}const i=await a.json();s(j,i.terminalId,!0),s(c,"resuming"),z(i.sessionWs,i.terminalId),q(i.ws)}catch{s(c,"idle")}}function ae(t){if(t.trim()){if(e(c)==="ready"&&b?.readyState===WebSocket.OPEN){b.send(JSON.stringify({data:`${t}
|
|
4
|
-
`,type:"input"}));return}if(e(c)==="connecting"||e(c)==="resuming"){y=t;return}y=t,ne()}}async function oe(){e(o).length===0&&s(w,!0);try{const t=P();if(!t){s(I,"No configuration found. Please configure settings first."),s(w,!1);return}const n=e(W)??"",a=e(M);if(!n){s(I,"No session ID"),s(w,!1);return}const i=a?`id=${encodeURIComponent(n)}&project=${encodeURIComponent(a)}&limit=${e(A)}`:`id=${encodeURIComponent(n)}&limit=${e(A)}`;let r=await fetch(`/api/sessions?${i}`,{headers:{Authorization:`Bearer ${t.apiKey}`}});if(r.status===404&&!a){const S=await fetch("/api/sessions",{headers:{Authorization:`Bearer ${t.apiKey}`}});if(S.ok){const J=await S.json();for(const B of J.projects||[])if((B.sessions||[]).find(U=>U.id===n)){r=await fetch(`/api/sessions?id=${encodeURIComponent(n)}&project=${encodeURIComponent(B.id)}`,{headers:{Authorization:`Bearer ${t.apiKey}`}});break}}}if(!r.ok){s(I,"Session not found"),s(w,!1);return}const u=await r.json(),p=Array.isArray(u.messages)?u.messages:[];Y(`shooter_session_${n}`,{messages:p,session:u.session}),s(f,u.session,!0),s(o,p,!0),s(R,p.length>=e(A)),s(w,!1);return}catch{s(I,"Failed to load session")}s(w,!1)}async function ie(){if(e(N))return;const t=P();if(t){s(N,!0);try{const n=e(A)+200,a=e(W)??"",i=e(M);if(!a)return;const r=[`id=${encodeURIComponent(a)}`,`limit=${n}`];i&&r.push(`project=${encodeURIComponent(i)}`);const u=await fetch(`/api/sessions?${r.join("&")}`,{headers:{Authorization:`Bearer ${t.apiKey}`}});if(!u.ok)return;const p=await u.json(),S=Array.isArray(p.messages)?p.messages:[];s(o,S,!0),s(A,n),s(R,S.length>=n),Y(`shooter_session_${a}`,{messages:S,session:p.session})}catch{}finally{s(N,!1)}}}we(()=>{const t=ke(`shooter_session_${x.params.id}`);return t&&(s(f,t.session,!0),s(o,Array.isArray(t.messages)?t.messages:[],!0),s(w,!1)),oe().then(async()=>{if(!e(f)||d)return;const n=P();if(n)try{const a=e(f).source==="opencode"?"opencode":"claude",i=await fetch("/api/sessions/connect",{body:JSON.stringify({command:a,cwd:e(f).projectPath,noCreate:!0,sessionId:e(W)}),headers:{Authorization:`Bearer ${n.apiKey}`,"Content-Type":"application/json"},method:"POST"});if(i.ok&&!d){const r=await i.json();s(j,r.terminalId,!0),s(c,"resuming"),z(r.sessionWs,r.terminalId),q(r.ws)}}catch{}}),()=>{d=!0,O&&clearTimeout(O),C?.close(),b?.close()}});var E=Re();_e("1a2wdfb",t=>{var n=je();be(()=>{Se.title=`${(e(f)?.title||"Session")??""} - Shooter`}),h(t,n)});var re=_(E);{var ce=t=>{var n=Ce();h(t,n)},le=t=>{var n=Oe(),a=H(n),i=_(a);$(a);var r=D(a,2),u=_(r,!0);$(r),Q(()=>{X(i,"href",e(M)?`/project?id=${e(M)}`:"/"),V(u,e(I))}),h(t,n)},de=t=>{var n=De(),a=H(n),i=_(a),r=_(i),u=D(r,2);{var p=l=>{var m=Me();h(l,m)},S=l=>{var m=Ne();h(l,m)},J=l=>{var m=Ae();h(l,m)},B=l=>{var m=Pe();h(l,m)};K(u,l=>{e(c)==="connecting"?l(p):e(c)==="resuming"?l(S,1):e(k)==="connected"?l(J,2):e(k)==="reconnecting"&&l(B,3)})}$(i);var U=D(i,2),ue=_(U,!0);$(U),$(a);var F=D(a,2),G=_(F);{var fe=l=>{var m=Te(),me=_(m);{let he=T(()=>e(N)?"Loading...":"Load earlier messages");$e(me,{get text(){return e(he)},classes:"btn-ghost btn-sm",get disabled(){return e(N)},onclick:ie})}$(m),h(l,m)};K(G,l=>{e(R)&&l(fe)})}var pe=D(G,2);{let l=T(()=>[...e(o)].reverse());Ie(pe,{get messages(){return e(l)},newestFirst:!0,get connectionState(){return e(k)},showInput:!0,get sendDisabled(){return e(te)},onSendInput:ae,sessionEnded:!1})}$(F),Q(()=>{X(r,"href",e(M)?`/project?id=${e(M)}`:"/"),V(ue,e(f).title)}),h(t,n)};K(re,t=>{e(w)&&e(o).length===0?t(ce):e(I)?t(le,1):e(f)&&t(de,2)})}$(E),h(Z,E),ye()}export{Le as component};
|
|
Binary file
|
|
Binary file
|
|
@@ -1,2 +0,0 @@
|
|
|
1
|
-
import{p as fe,j as Ye,i as H,a as h,s as D,f as w,t as Ge}from"../chunks/Cg3dlX05.js";import{p as ke,v as $e,A as _e,w as Me,b as o,c as n,r as s,t as K,g as e,s as k,x as y,q as Ke,d as c,B as he,y as Ue,N as Xe,$ as et}from"../chunks/DYuMZGL5.js";import{T as Ae,s as Le,d as ne,S as tt,e as ge,C as at,f as st,I as rt,B as re,b as Be,c as Ie,i as nt,P as Pe,g as it}from"../chunks/DZvnhU_8.js";import{h as lt}from"../chunks/DZQMsHM5.js";import{g as De}from"../chunks/gxvWeAns.js";import{R as ot,S as ct}from"../chunks/BRqaaL5D.js";import{S as dt}from"../chunks/CG453M9q.js";import{T as ut}from"../chunks/CSjf4fBs.js";import{c as vt,s as Ee,g as mt}from"../chunks/xs1Xl3_e.js";import{i as pt}from"../chunks/gQJcRhou.js";import"../chunks/BDcFu3l7.js";import{E as Ne}from"../chunks/C5VOyQCG.js";var ft=w("<time> </time>"),ht=w("<time> </time>");function je(ie,d){ke(d,!0);let U=fe(d,"format",3,"long"),$=fe(d,"updateInterval",3,6e4),B=fe(d,"tooltip",3,!1),S=y(()=>d.date instanceof Date?d.date:new Date(d.date)),p=y(()=>e(S).toISOString()),A=y(()=>e(S).toLocaleString(d.locale,{dateStyle:"full",timeStyle:"short"})),T=k("");function C(){const _=Date.now(),g=e(S).getTime()-_,i=Math.abs(g),f=Math.round(i/1e3),q=Math.round(i/6e4),R=Math.round(i/36e5),J=Math.round(i/864e5),z=Math.round(i/6048e5),Z=Math.round(i/2592e6),X=Math.round(i/31536e6),ee=new Intl.RelativeTimeFormat(d.locale,{style:U()});let I,u;f<60?(I="second",u=f):q<60?(I="minute",u=q):R<24?(I="hour",u=R):J<7?(I="day",u=J):z<4?(I="week",u=z):Z<12?(I="month",u=Z):(I="year",u=X);const F=g<0?-1:1;return ee.format(F*u,I)}function x(){o(T,C(),!0)}$e(()=>{if(x(),$()>0){const _=setInterval(x,$());return()=>clearInterval(_)}});var E=Ye(),le=_e(E);{var oe=_=>{Ae(_,{get text(){return e(A)},children:(v,g)=>{var i=ft(),f=n(i,!0);s(i),K(()=>{Le(i,1,`relative-time ${d.classes??""??""}`,"svelte-o98vko"),ne(i,"datetime",e(p)),ne(i,"data-pw",typeof d.testId=="string"?d.testId:null),D(f,e(T))}),h(v,i)},$$slots:{default:!0}})},ce=_=>{var v=ht(),g=n(v,!0);s(v),K(()=>{Le(v,1,`relative-time ${d.classes??""??""}`,"svelte-o98vko"),ne(v,"datetime",e(p)),ne(v,"data-pw",typeof d.testId=="string"?d.testId:null),D(g,e(T))}),h(_,v)};H(le,_=>{B()?_(oe):_(ce,-1)})}h(ie,E),Me()}var gt=w('<div class="section svelte-plslui"><span class="section-label svelte-plslui">Quick Launch</span> <div class="preset-grid svelte-plslui"></div></div> <div class="section svelte-plslui"><span class="section-label svelte-plslui">Working Directory</span> <!> <div class="custom-cwd-group svelte-plslui"><!></div></div> <!> <!>',1);function _t(ie,d){ke(d,!0);let U=fe(d,"open",15,!1);const $=[{args:[],command:"claude",label:"Claude Code"},{args:[],command:"opencode",label:"OpenCode"},{args:[],command:"zsh",label:"Shell / zsh"},{args:[],command:"bash",label:"Bash"}];let B=k(0),S=k(Ke([])),p=k(""),A=k(""),T=k(!1),C=k("");$e(()=>{x()});async function x(){try{const v=await fetch("/api/sessions?limit=50&offset=0",{headers:{Authorization:`Bearer ${d.apiKey}`}});if(!v.ok)return;const i=(await v.json()).projects.map(f=>f.fullPath).filter(f=>typeof f=="string"&&f.length>0);o(S,[...new Set(i)],!0),e(S).length>0&&!e(p)&&o(p,e(S)[0],!0)}catch{}}function E(v){o(B,v,!0)}function le(){return $[e(B)].command}function oe(){return $[e(B)].args}function ce(){const v=e(A).trim();return v||(e(p)?e(p):"/tmp")}async function _(){const v=le();if(v){o(T,!0),o(C,"");try{const g=await fetch("/api/terminals",{body:JSON.stringify({args:oe(),command:v,cwd:ce()||void 0}),headers:{Authorization:`Bearer ${d.apiKey}`,"Content-Type":"application/json"},method:"POST"});if(!g.ok){const f=await g.json().catch(()=>({}));o(C,(typeof f.error=="string"?f.error:"")||`Failed to launch (${g.status})`,!0);return}const i=await g.json();d.onLaunch(i)}catch{o(C,"Network error — is the server running?")}finally{o(T,!1)}}}tt(ie,{side:"bottom",title:"New Terminal",get onclose(){return d.onClose},classes:"launch-sheet",get open(){return U()},set open(g){U(g)},content:g=>{var i=gt(),f=_e(i),q=c(n(f),2);ge(q,23,()=>$,u=>u.label,(u,F,M)=>{{let de=y(()=>e(B)===e(M));at(u,{mode:"radio",get selected(){return e(de)},onclick:()=>{E(e(M))},classes:"preset-choice",children:(ve,we)=>{he();var ue=Ge();K(()=>D(ue,e(F).label)),h(ve,ue)},$$slots:{default:!0}})}}),s(q),s(f);var R=c(f,2),J=c(n(R),2);{let u=y(()=>e(S).length>0?e(S).map(M=>({id:M,label:M})):[{id:"",label:"No recent projects"}]),F=y(()=>e(p)?[e(p)]:e(S).length>0?[e(S)[0]]:[""]);st(J,{get items(){return e(u)},get value(){return e(F)},placeholder:"Select a project",onchange:M=>{o(p,M[0]||"",!0)},classes:"launch-select"})}var z=c(J,2),Z=n(z);rt(Z,{label:"Or enter a custom path",dataType:"text",placeholder:"/path/to/project",classes:"input-mono launch-input",get value(){return e(A)},set value(u){o(A,u,!0)}}),s(z),s(R);var X=c(R,2);{let u=y(()=>e(T)?"Launching...":"Launch Terminal");re(X,{classes:"btn-launch",get disabled(){return e(T)},onclick:_,get showLoader(){return e(T)},get text(){return e(u)}})}var ee=c(X,2);{var I=u=>{Be(u,{get text(){return e(C)},classes:"banner-error launch-error-banner"})};H(ee,u=>{e(C)&&u(I)})}h(g,i)},$$slots:{content:!0}}),Me()}var wt=w('<meta name="description" content="Active terminal sessions on this machine"/>'),xt=w("<!> Refresh",1),bt=w('<span class="plus-icon svelte-1ewpmn2">+</span> New Terminal',1),yt=w('<div class="loading-container"></div>'),St=w('<span class="plus-icon svelte-1ewpmn2">+</span> New Terminal',1),Tt=w('<span class="terminal-cwd svelte-1ewpmn2"> </span>'),Ct=w('<div class="terminal-preview svelte-1ewpmn2"><span class="terminal-preview-text svelte-1ewpmn2"> </span></div>'),It=w('<a class="terminal-card svelte-1ewpmn2"><div class="terminal-card-header svelte-1ewpmn2"><div class="terminal-card-left svelte-1ewpmn2"><span class="status-indicator status-running svelte-1ewpmn2"><span></span></span> <span class="terminal-command svelte-1ewpmn2"> </span> <!></div> <!></div> <div class="terminal-card-meta svelte-1ewpmn2"><!> <span class="terminal-pid svelte-1ewpmn2"> </span></div> <!></a>'),Pt=w('<span class="terminal-cwd svelte-1ewpmn2"> </span>'),At=w('<div class="terminal-preview svelte-1ewpmn2"><span class="terminal-preview-text svelte-1ewpmn2"> </span></div>'),Lt=w('<a class="terminal-card terminal-card-exited svelte-1ewpmn2"><div class="terminal-card-header svelte-1ewpmn2"><div class="terminal-card-left svelte-1ewpmn2"><span class="status-indicator status-exited svelte-1ewpmn2"><span class="status-dot-static"></span></span> <span class="terminal-command svelte-1ewpmn2"> </span> <!> <!></div> <div class="terminal-card-right svelte-1ewpmn2"><!> <!></div></div> <div class="terminal-card-meta svelte-1ewpmn2"><!></div> <!></a>'),kt=w('<div class="terminals-container svelte-1ewpmn2"><!> <!></div>'),$t=w('<main class="main"><div class="page-header"><div class="page-header-content"><div><h1 class="page-title">Terminals</h1> <p class="page-description">Active terminal sessions on this machine</p></div> <div class="page-actions"><!> <!></div></div></div> <!> <!></main> <!>',1);function qt(ie,d){ke(d,!0);const U=1e4,$="shooter_terminals",B=["claude","opencode"],S=["zsh","bash","sh","fish"];let p=k(Ke([])),A=k(!1),T=!1,C=k(null),x=k(null),E=null;const le=y(()=>e(p).filter(t=>t.status==="running")),oe=y(()=>e(p).filter(t=>t.status==="exited"));$e(()=>{ce();const t=mt($,1e4);t&&(o(p,t,!0),o(A,!1)),_(),E=setInterval(()=>{e(x)?.apiKey&&_()},U)}),Ue(()=>{E&&(clearInterval(E),E=null)});function ce(){try{const t=localStorage.getItem("shooter_config");if(t){const a=JSON.parse(t);pt(a)?o(x,a,!0):(localStorage.removeItem("shooter_config"),o(x,null))}}catch{}}async function _(){if(!(!e(x)?.apiKey||T)){T=!0,e(p).length===0&&o(A,!0);try{const a=await fetch("/api/terminals",{headers:{Authorization:`Bearer ${e(x).apiKey}`}});if(!a.ok){o(C,`Failed to load terminals (HTTP ${a.status})`);return}o(C,null);const r=await a.json();o(p,r.terminals.map(b=>({...b,lastOutput:b.lastOutput??null})),!0),Ee($,e(p))}catch(t){o(C,"Failed to load terminals"),console.error("Failed to fetch terminals:",t)}finally{o(A,!1),T=!1}}}async function v(){o(A,!0),vt($),await _()}function g(){De("/config")}let i=k(!1);function f(){o(i,!0)}function q(){o(i,!1)}function R(t){o(i,!1),De(`/terminals/${t.id}`)}function J(t){const a=t.split("/").pop()||t;return B.includes(a)?"ai":(S.includes(a),"shell")}function z(t){return t.status==="exited"?{class:"pill-badge-ended",label:"ENDED"}:J(t.command)==="ai"?{class:"pill-badge-ai",label:"AI"}:{class:"pill-badge-shell",label:"SHELL"}}function Z(t,a=40){if(t.length<=a)return t;const r=t.split("/");return r.length<=3?t:`${r[0]}/.../${r.slice(-2).join("/")}`}function X(t){return t.replace(/\x1b\[[0-9;]*[a-zA-Z]|\x1b\][^\x07]*\x07|\x1b\[[\?]?[0-9;]*[a-zA-Z]|\x1b/g,"").replace(/[\x00-\x1f]/g,"").trim()}function ee(t,a=80){if(!t)return"";const r=t.split(`
|
|
2
|
-
`).map(P=>X(P)).filter(P=>P.trim()),b=r[r.length-1]||"";return b.length<=a?b:`${b.slice(0,a)}...`}function I(t){return t.split("/").pop()||t}async function u(t,a){if(t.preventDefault(),t.stopPropagation(),!!e(x)?.apiKey)try{(await fetch(`/api/terminals/${a}`,{headers:{Authorization:`Bearer ${e(x).apiKey}`},method:"DELETE"})).ok&&(o(p,e(p).filter(b=>b.id!==a),!0),Ee($,e(p)))}catch(r){console.error("Failed to remove terminal:",r)}}var F=$t();lt("1ewpmn2",t=>{var a=wt();Xe(()=>{et.title="Terminals - Shooter"}),h(t,a)});var M=_e(F),de=n(M),ve=n(de),we=c(n(ve),2),ue=n(we);re(ue,{classes:"btn-secondary",onclick:v,get disabled(){return e(A)},children:(t,a)=>{var r=xt(),b=_e(r);Ie(b,{get svg(){return ot},classes:"icon-14"}),he(),h(t,r)},$$slots:{default:!0}});var Re=c(ue,2);re(Re,{classes:"btn-primary",onclick:f,children:(t,a)=>{var r=bt();he(),h(t,r)},$$slots:{default:!0}}),s(we),s(ve),s(de);var Oe=c(de,2);{var ze=t=>{Be(t,{get text(){return e(C)},classes:"banner-error"})};H(Oe,t=>{e(C)&&t(ze)})}var Fe=c(Oe,2);{var He=t=>{var a=yt();ge(a,20,()=>Array(4),nt,(r,b)=>{ct(r,{classes:"shimmer-card"})}),s(a),h(t,a)},qe=t=>{Ne(t,{title:"Configuration Required",description:"Set up your API credentials to view terminal sessions",icon:r=>{Ie(r,{get svg(){return dt},classes:"icon-24"})},children:(r,b)=>{re(r,{classes:"btn-primary",onclick:g,text:"Configure Settings"})}})},Je=t=>{Ne(t,{title:"No terminals",description:"Launch a new terminal session to get started. Terminal sessions will appear here once created.",icon:r=>{Ie(r,{get svg(){return ut},classes:"icon-24"})},children:(r,b)=>{re(r,{classes:"btn-primary",onclick:f,children:(P,l)=>{var Q=St();he(),h(P,Q)},$$slots:{default:!0}})}})},Ze=t=>{var a=kt(),r=n(a);ge(r,17,()=>e(le),P=>P.id,(P,l)=>{const Q=y(()=>z(e(l)));var N=It(),V=n(N),W=n(V),Y=n(W),xe=n(Y);s(Y);var te=c(Y,2),be=n(te,!0);s(te);var ye=c(te,2);Pe(ye,{get text(){return e(Q).label},get classes(){return e(Q).class}}),s(W);var me=c(W,2);je(me,{get date(){return e(l).createdAt},format:"narrow",classes:"terminal-time"}),s(V);var ae=c(V,2),pe=n(ae);{let m=y(()=>e(l).currentCwd||e(l).cwd);Ae(pe,{get text(){return e(m)},position:"bottom",children:(L,O)=>{var j=Tt(),G=n(j,!0);s(j),K(We=>D(G,We),[()=>Z(e(l).currentCwd||e(l).cwd)]),h(L,j)},$$slots:{default:!0}})}var se=c(pe,2),Se=n(se);s(se),s(ae);var Te=c(ae,2);{var Ce=m=>{var L=Ct(),O=n(L),j=n(O,!0);s(O),s(L),K(G=>D(j,G),[()=>ee(e(l).lastOutput)]),h(m,L)};H(Te,m=>{e(l).lastOutput&&m(Ce)})}s(N),K(m=>{ne(N,"href",`/terminals/${e(l).id??""}`),Le(xe,1,it(e(l).isActive?"status-dot-active":"status-dot-idle")),D(be,m),D(Se,`PID ${e(l).pid??""}`)},[()=>I(e(l).command)]),h(P,N)});var b=c(r,2);ge(b,17,()=>e(oe),P=>P.id,(P,l)=>{const Q=y(()=>z(e(l)));var N=Lt(),V=n(N),W=n(V),Y=c(n(W),2),xe=n(Y,!0);s(Y);var te=c(Y,2);Pe(te,{get text(){return e(Q).label},get classes(){return e(Q).class}});var be=c(te,2);{var ye=m=>{{let L=y(()=>e(l).exitCode!==0?"pill-exit-error":"pill-exit-ok");Pe(m,{get text(){return`exit ${e(l).exitCode??""}`},get classes(){return e(L)}})}};H(be,m=>{e(l).exitCode!==null&&m(ye)})}s(W);var me=c(W,2),ae=n(me);{let m=y(()=>e(l).exitedAt||e(l).createdAt);je(ae,{get date(){return e(m)},format:"narrow",classes:"terminal-time"})}var pe=c(ae,2);re(pe,{classes:"btn-ghost btn-sm btn-remove",onclick:m=>{u(m,e(l).id)},text:"×"}),s(me),s(V);var se=c(V,2),Se=n(se);Ae(Se,{get text(){return e(l).cwd},position:"bottom",children:(m,L)=>{var O=Pt(),j=n(O,!0);s(O),K(G=>D(j,G),[()=>Z(e(l).cwd)]),h(m,O)},$$slots:{default:!0}}),s(se);var Te=c(se,2);{var Ce=m=>{var L=At(),O=n(L),j=n(O,!0);s(O),s(L),K(G=>D(j,G),[()=>ee(e(l).lastOutput)]),h(m,L)};H(Te,m=>{e(l).lastOutput&&m(Ce)})}s(N),K(m=>{ne(N,"href",`/terminals/${e(l).id??""}`),D(xe,m)},[()=>I(e(l).command)]),h(P,N)}),s(a),h(t,a)};H(Fe,t=>{e(A)&&e(p).length===0?t(He):e(x)?.apiKey?e(p).length===0?t(Je,2):t(Ze,-1):t(qe,1)})}s(M);var Qe=c(M,2);{var Ve=t=>{_t(t,{get apiKey(){return e(x).apiKey},onClose:q,onLaunch:R,get open(){return e(i)},set open(a){o(i,a,!0)}})};H(Qe,t=>{e(x)?.apiKey&&t(Ve)})}h(ie,F),Me()}export{qt as component};
|
|
Binary file
|
|
Binary file
|
|
@@ -1,2 +0,0 @@
|
|
|
1
|
-
const __vite__mapDeps=(i,m=__vite__mapDeps,d=(m.f||(m.f=["../chunks/CR6bkGJW.js","../chunks/PPVm8Dsz.js"])))=>i.map(i=>d[i]);
|
|
2
|
-
import{_ as ke}from"../chunks/PPVm8Dsz.js";import{p as Se,i as q,s as rt,a as S,f as C,j as xe,c as Ce,e as Ie,g as Nt}from"../chunks/Cg3dlX05.js";import{p as ot,w as it,g as t,x as h,c as k,d as c,r as g,t as H,A as Ft,s as m,q as Te,v as Ee,y as je,z as Ke,b as n,$ as Me}from"../chunks/DYuMZGL5.js";import{h as qe}from"../chunks/DZQMsHM5.js";import{m as Ae,s as ft,d as zt,B as U,e as Bt,M as De,K as Pe,P as Ot,k as Rt,c as $e,T as Lt,n as Ne,I as Oe}from"../chunks/DZvnhU_8.js";import{g as tt}from"../chunks/gxvWeAns.js";import{p as Re}from"../chunks/pMo6RVvN.js";import{A as Le}from"../chunks/DzuS5Nbr.js";import{C as We}from"../chunks/Pw0jDB7M.js";import{E as Fe}from"../chunks/C5VOyQCG.js";function ze(y,o){ot(o,!0);let b=Se(o,"open",15,!1);const u=h(()=>o.commands.map((w,a)=>({label:w.label,value:String(a)})));function T(w){const a=parseInt(w.value,10);o.commands[a]&&o.commands[a].action()}Ae(y,{get items(){return t(u)},placeholder:"Type a command...",emptyText:"No matching commands",onselect:T,get onclose(){return o.onClose},classes:"command-palette",get open(){return b()},set open(w){b(w)}}),it()}var Be=C('<div><span></span> <span class="status-label svelte-1cg9pai"> </span> <!></div>');function He(y,o){const b=h(()=>o.status==="connected"?"Connected":o.status==="reconnecting"?"Reconnecting...":"Disconnected");var u=Be();let T;var w=k(u),a=c(w,2),O=k(a,!0);g(a);var A=c(a,2);{var E=x=>{U(x,{classes:"btn-ghost btn-sm btn-retry",get onclick(){return o.onretry},text:"Retry"})};q(A,x=>{o.status==="disconnected"&&o.onretry&&x(E)})}g(u),H(()=>{T=ft(u,1,"connection-status svelte-1cg9pai",null,T,{reconnecting:o.status==="reconnecting",disconnected:o.status==="disconnected",connected:o.status==="connected"}),zt(u,"aria-label",`Connection: ${o.status??""}`),ft(w,1,`conn-dot ${o.status??""}`,"svelte-1cg9pai"),rt(O,t(b))}),S(y,u)}const Ht=typeof navigator<"u"&&/Mac|iPhone|iPad/.test(navigator.userAgent),Wt=Ht?"⌘":"Ctrl";function Ue(y){function o(b){if((Ht?b.metaKey:b.ctrlKey)&&!Ge()&&b.key==="/"){b.preventDefault(),b.stopPropagation(),y.onHelp();return}}return window.addEventListener("keydown",o,!0),{destroy(){window.removeEventListener("keydown",o,!0)}}}function Je(){return[{description:"Command palette",keys:`${Wt}+K`},{description:"Keyboard shortcuts",keys:`${Wt}+/`}]}function Ge(){const y=document.activeElement;if(!y)return!1;const o=y.tagName;return!!(o==="INPUT"||o==="TEXTAREA"||y.isContentEditable||y.closest(".xterm"))}var Ve=C('<div class="quick-keys svelte-64qat5" role="toolbar" aria-label="Quick terminal keys"></div>');function Qe(y,o){ot(o,!0);const b=[{escape:"",label:"Ctrl+C"},{escape:" ",label:"Tab"},{escape:"\x1B[A",label:"↑"},{escape:"\x1B[B",label:"↓"},{escape:"\x1B",label:"Esc"},{escape:"",label:"Ctrl+D"},{escape:"",label:"Ctrl+Z"}];var u=Ve();Bt(u,21,()=>b,T=>T.label,(T,w)=>{U(T,{classes:"btn-quick-key",onclick:()=>{o.onKey(t(w).escape)},get text(){return t(w).label}})}),g(u),S(y,u),it()}var Xe=C('<div class="shortcut-row svelte-1u7lstk"><span class="shortcut-desc svelte-1u7lstk"> </span> <!></div>'),Ze=C('<div class="shortcuts-list svelte-1u7lstk"></div>');function Ye(y,o){ot(o,!0);const b=Je();var u=xe(),T=Ft(u);{var w=a=>{De(a,{header:{text:"Keyboard Shortcuts"},get onoverlayClick(){return o.onClose},classes:"shortcuts-modal",content:A=>{var E=Ze();Bt(E,21,()=>b,x=>x.keys,(x,F)=>{var R=Xe(),d=k(R),j=k(d,!0);g(d);var P=c(d,2);Pe(P,{get keys(){return t(F).keys},classes:"shortcut-kbd"}),g(R),H(()=>rt(j,t(F).description)),S(x,R)}),g(E),S(A,E)},$$slots:{content:!0}})};q(T,a=>{o.open&&a(w)})}S(y,u),it()}var tn=C('<meta name="description" content="Interactive terminal session"/>'),en=C('<div class="term-page svelte-1tubujq"><div class="term-topbar svelte-1tubujq"><div class="skeleton" style="width: 120px; height: 20px;"></div></div> <div class="term-body-loading svelte-1tubujq"><div class="skeleton" style="width: 100%; height: 100%;"></div></div></div>'),nn=C('<main class="main"><div class="session-back-row"><a href="/terminals" class="back-link"><span class="back-arrow">←</span> Terminals</a></div> <!></main>'),sn=C("<span></span>"),an=C('<span class="term-cwd svelte-1tubujq"> </span>'),rn=C('<a class="term-session-link svelte-1tubujq" title="View session history">Session</a>'),on=C('<div class="term-input-area svelte-1tubujq"><!> <div class="term-input-bar svelte-1tubujq"><!> <!></div></div>'),ln=C('<div class="term-exited-bar svelte-1tubujq"><span>Process exited</span> <!></div>'),cn=C('<div class="term-page svelte-1tubujq"><div class="term-topbar svelte-1tubujq"><div class="term-topbar-left svelte-1tubujq"><a href="/terminals" class="term-back svelte-1tubujq" aria-label="Back to terminals">←</a> <span class="term-command-name svelte-1tubujq"> </span> <!> <!> <!> <!> <!></div> <div class="term-topbar-right svelte-1tubujq"><!> <!> <!></div></div> <div class="term-body svelte-1tubujq"></div> <!> <div class="term-chat-body svelte-1tubujq"><!></div> <!></div>'),un=C("<!> <!> <!>",1);function _n(y,o){ot(o,!0);const b=()=>Ie(Re,"$page",u),[u,T]=Ce(),w=["claude","opencode"];let a=m(null),O=m(!0),A=m(null),E=m(!1),x=m(!1),F=m(!1),R=m(null),d=m("raw"),j=m("disconnected"),P=m("disconnected"),z=m(""),D=m(Te([])),vt=m(!1),J=m(null),lt=m(null),$=null,pt=null,L=m(!1),ct=m(!1),f=null,W=null,v=!1;const G=h(()=>b().params.id),V=h(()=>t(a)?w.includes((t(a).command.split("/").pop()||"").toLowerCase()):!1),B=h(()=>t(a)?.status==="running"),mt=h(()=>t(a)?.command.split("/").pop()||"terminal"),Ut=h(()=>t(V)?"AI":"SHELL"),Jt=h(()=>t(V)?"pill-badge-ai":"pill-badge-shell"),Gt=["Raw","Chat"],ht=h(()=>t(d)==="raw"?t(j):t(P)),Vt=h(()=>t(d)==="raw"?0:1),gt=h(()=>Xt(t(R)||t(a)?.cwd||"")),Qt=h(()=>{const e=[{action:()=>{tt("/")},label:"Go to Home"},{action:()=>{tt("/terminals")},label:"Go to Terminals"},{action:()=>{tt("/config")},label:"Go to Settings"},{action:()=>{n(L,!0)},label:"Show keyboard shortcuts"}];return t(B)&&e.push({action:()=>{bt()},label:"Kill terminal"}),e});function Xt(e){if(!e)return"";const s="";let r=e;if(typeof navigator<"u"){const l=e.split("/");l.length>=3&&l[1]==="Users"?r=`~/${l.slice(3).join("/")}`:l.length>=3&&l[1]==="home"&&(r=`~/${l.slice(3).join("/")}`)}if(r.length>30){const l=r.split("/");if(l.length>2)return`.../${l.slice(-2).join("/")}`}return r||s}function Q(){try{const e=localStorage.getItem("shooter_config");if(!e)return null;const s=JSON.parse(e);return typeof s?.apiKey=="string"&&s.apiKey?s:null}catch{return null}}async function Zt(){const e=Q();if(!e){n(A,"No configuration found. Please configure settings first."),n(O,!1);return}try{const s=await fetch(`/api/terminals/${t(G)}`,{headers:{Authorization:`Bearer ${e.apiKey}`}});if(!s.ok){n(A,s.status===404?"Terminal not found":"Failed to load terminal",!0),n(O,!1);return}n(a,await s.json(),!0)}catch{n(A,"Failed to connect to server")}n(O,!1)}async function yt(){const e=Q();if(!e)return null;try{const s=await fetch("/api/ws-ticket",{headers:{Authorization:`Bearer ${e.apiKey}`},method:"POST"});return s.ok?(await s.json()).ticket:null}catch{return null}}async function bt(){if(!t(a)||t(E))return;const e=Q();if(e){n(E,!0);try{(await fetch(`/api/terminals/${t(G)}`,{headers:{Authorization:`Bearer ${e.apiKey}`},method:"DELETE"})).ok?tt("/terminals"):n(E,!1)}catch{n(E,!1)}}}async function Yt(){if(!t(a)||t(x))return;const e=Q();if(e){n(x,!0);try{(await fetch(`/api/terminals/${t(G)}`,{headers:{Authorization:`Bearer ${e.apiKey}`},method:"DELETE"})).ok?tt("/terminals"):n(x,!1)}catch{n(x,!1)}}}async function ut(){if(!t(J)||!t(a)||v)return;const s=`${window.location.protocol==="https:"?"wss:":"ws:"}//${window.location.host}${t(a).ws}`,r=async()=>{const l=await yt();if(!l)throw new Error("Failed to obtain WebSocket ticket");return l};try{const{createTerminal:l}=await ke(async()=>{const{createTerminal:K}=await import("../chunks/CR6bkGJW.js");return{createTerminal:K}},__vite__mapDeps([0,1]),import.meta.url);if(v||!t(J))return;const p=await l({apiKey:Q()?.apiKey,container:t(J),fontSize:window.innerWidth<768?12:14,getTicket:r,onActivity:K=>{v||n(F,K,!0)},onCwd:K=>{v||n(R,K,!0)},onDisconnect:()=>{v||n(j,"reconnecting")},onExit:K=>{!v&&t(a)&&(n(a,{...t(a),exitCode:K,exitedAt:new Date().toISOString(),status:"exited"},!0),n(j,"disconnected"))},onReconnect:()=>{v||n(j,"connected")},terminalId:t(G),wsUrl:s});if(v){p.dispose();return}$=p,n(j,"connected")}catch(l){console.error("Failed to initialize terminal:",l),n(j,"disconnected")}}function wt(){$&&($.dispose(),$=null)}async function et(){if(!t(a)||v||f&&(f.readyState===WebSocket.OPEN||f.readyState===WebSocket.CONNECTING))return;const e=await yt();if(!e||v||!t(a)){n(P,"disconnected");return}const r=`${window.location.protocol==="https:"?"wss:":"ws:"}//${window.location.host}${t(a).sessionWs}?ticket=${e}`;f=new WebSocket(r),f.onopen=()=>{v||(n(P,"connected"),f?.send(JSON.stringify({sessionId:t(G),type:"subscribe"})))},f.onmessage=l=>{if(!v)try{const p=JSON.parse(l.data);te(p)}catch{}},f.onclose=()=>{!v&&t(a)?.status==="running"&&(n(P,"reconnecting"),W&&clearTimeout(W),W=setTimeout(()=>{W=null,!v&&t(a)?.status==="running"&&et()},2e3))},f.onerror=()=>{v||n(P,"disconnected")}}function X(){return t(D)}function te(e){if(e.type==="history"){const s=e.messages||[];n(D,s.map(r=>({id:r.id,parts:r.content,role:r.role,timestamp:r.timestamp})),!0)}else if(e.type==="message"){const s=typeof e.role=="string"?e.role:"assistant",r=s==="user"?"user":s==="system"?"system":"assistant",l={id:`msg-${Date.now()}-${Math.random().toString(36).slice(2,8)}`,parts:e.content||[],role:r,timestamp:(typeof e.timestamp=="string"?e.timestamp:"")||new Date().toISOString()};n(D,X().concat(l),!0)}else if(e.type==="tool-use"){const s={id:e.id||`tool-${Date.now()}`,input:e.input||{},toolName:e.name,type:"tool_use"},r={id:`tool-${Date.now()}-${Math.random().toString(36).slice(2,8)}`,parts:[s],role:"assistant",timestamp:new Date().toISOString()};n(D,X().concat(r),!0)}else if(e.type==="tool-result"){const s={isError:e.isError||!1,output:e.output||"",toolUseId:e.id,type:"tool_result"},r={id:`result-${Date.now()}-${Math.random().toString(36).slice(2,8)}`,parts:[s],role:"system",timestamp:new Date().toISOString()};n(D,X().concat(r),!0)}else if(e.type==="thinking"){const s={content:e.text||"",type:"thinking"},r=X(),l=r.length-1;if(l>=0&&r[l].role==="assistant"){const p=r[l],K={id:p.id,parts:p.parts.concat(s),role:p.role,timestamp:p.timestamp};n(D,[...r.slice(0,l),K],!0)}else{const p={id:`think-${Date.now()}-${Math.random().toString(36).slice(2,8)}`,parts:[s],role:"assistant",timestamp:new Date().toISOString()};n(D,r.concat(p),!0)}}else if(e.type==="error"){const s={content:e.message||"Unknown error",type:"text"},r={id:`err-${Date.now()}-${Math.random().toString(36).slice(2,8)}`,parts:[s],role:"system",timestamp:new Date().toISOString()};n(D,X().concat(r),!0)}else e.type==="session-end"&&(n(vt,!0),t(a)&&n(a,{...t(a),status:"exited"},!0))}function _t(){W&&(clearTimeout(W),W=null),f&&(f.onclose=null,f.close(),f=null)}function kt(){t(z).trim()&&($&&t(j)==="connected"&&$.sendInput(`${t(z)}\r`),n(z,""),t(lt)?.focus())}function ee(e){!e.trim()||f?.readyState!==WebSocket.OPEN||f.send(JSON.stringify({text:e,type:"send-input"}))}function St(){f?.readyState===WebSocket.OPEN&&f.send(JSON.stringify({type:"cancel"}))}function ne(e){t(d)==="raw"?$&&t(j)==="connected"&&$.sendInput(e):t(d)==="chat"&&e===""&&St()}function se(e){e.key==="Enter"&&!e.shiftKey&&(e.preventDefault(),kt())}let nt=!1,st=!1;function ae(e){e!==t(d)&&(n(d,e,!0),e==="raw"&&!nt?(requestAnimationFrame(()=>{ut()}),nt=!0):e==="chat"&&!st&&(et(),st=!0))}function re(){t(d)==="raw"?(wt(),n(j,"reconnecting"),ut(),nt=!0):(_t(),n(P,"reconnecting"),et(),st=!0)}Ee(async()=>{await Zt(),!v&&(pt=Ue({onHelp:()=>{n(L,!t(L))}}),!(!t(a)||t(A))&&(t(V)&&window.innerWidth<768?n(d,"chat"):n(d,"raw"),t(d)==="raw"?(requestAnimationFrame(()=>{ut()}),nt=!0):(et(),st=!0)))}),je(()=>{v=!0,wt(),_t(),pt?.destroy()});var xt=un();qe("1tubujq",e=>{var s=tn();Ke(()=>{Me.title=`${t(mt)??""} - Terminal - Shooter`}),S(e,s)});var Ct=Ft(xt);{var oe=e=>{var s=en();S(e,s)},ie=e=>{var s=nn(),r=c(k(s),2);Fe(r,{title:"Error",get description(){return t(A)},icon:p=>{$e(p,{get svg(){return Le},classes:"icon-24"})}}),g(s),S(e,s)},le=e=>{var s=cn(),r=k(s),l=k(r),p=c(k(l),2),K=k(p,!0);g(p);var Tt=c(p,2);Ot(Tt,{get text(){return t(Ut)},get classes(){return t(Jt)}});var Et=c(Tt,2);{var ue=i=>{{let _=h(()=>t(F)?"Active":"Idle");Lt(i,{get text(){return t(_)},position:"bottom",children:(M,N)=>{var I=sn();H(()=>ft(I,1,`activity-dot ${t(F)?"activity-active":"activity-idle"}`,"svelte-1tubujq")),S(M,I)},$$slots:{default:!0}})}};q(Et,i=>{t(B)&&i(ue)})}var jt=c(Et,2);{var de=i=>{{let _=h(()=>t(R)||t(a)?.cwd||"");Lt(i,{get text(){return t(_)},position:"bottom",children:(M,N)=>{var I=an(),Z=k(I,!0);g(I),H(()=>rt(Z,t(gt))),S(M,I)},$$slots:{default:!0}})}};q(jt,i=>{t(gt)&&i(de)})}var Kt=c(jt,2);He(Kt,{get status(){return t(ht)},onretry:re});var fe=c(Kt,2);{var ve=i=>{const _=h(()=>t(a).sessionFile??"");var M=rn();H(N=>zt(M,"href",`/session/${N??""}`),[()=>t(_).split(/[\\/]/).pop()?.replace(".jsonl","")||""]),S(i,M)};q(fe,i=>{t(V)&&t(a)?.sessionFile&&i(ve)})}g(l);var Mt=c(l,2),qt=k(Mt);{var pe=i=>{Ne(i,{get items(){return Gt},get activeIndex(){return t(Vt)},onchange:_=>{ae(_===0?"raw":"chat")},classes:"term-tabs"})};q(qt,i=>{t(V)&&i(pe)})}var At=c(qt,2);U(At,{classes:"term-shortcuts-btn",onclick:()=>{n(L,!t(L))},text:"?",ariaLabel:"Keyboard shortcuts"});var me=c(At,2);{var he=i=>{U(i,{classes:"btn-danger btn-sm",onclick:bt,get disabled(){return t(E)},get showLoader(){return t(E)},text:"Kill"})},ge=i=>{U(i,{classes:"btn-secondary btn-sm",onclick:Yt,get disabled(){return t(x)},get showLoader(){return t(x)},text:"Remove"})};q(me,i=>{t(B)?i(he):i(ge,-1)})}g(Mt),g(r);var dt=c(r,2);let Dt;Nt(dt,i=>n(J,i),()=>t(J));var Pt=c(dt,2);{var ye=i=>{var _=on(),M=k(_);Qe(M,{onKey:ne});var N=c(M,2),I=k(N);Nt(Oe(I,{dataType:"text",useTextArea:!0,placeholder:"Type command... (Shift+Enter for new line)",classes:"input-mono term-input-field",onKeyDown:se,get value(){return t(z)},set value(Y){n(z,Y,!0)}}),Y=>n(lt,Y,!0),()=>t(lt));var Z=c(I,2);{let Y=h(()=>!t(z).trim());U(Z,{classes:"btn-primary btn-send",onclick:kt,get disabled(){return t(Y)},text:"↵"})}g(N),g(_),S(i,_)};q(Pt,i=>{t(B)&&t(d)==="raw"&&i(ye)})}var at=c(Pt,2);let $t;var be=k(at);We(be,{get messages(){return t(D)},get connectionState(){return t(ht)},get sessionEnded(){return t(vt)},get showInput(){return t(B)},onSendInput:ee,onCancel:St}),g(at);var we=c(at,2);{var _e=i=>{var _=ln(),M=c(k(_),2);{var N=I=>{{let Z=h(()=>t(a).exitCode!==0?"pill-exit-error":"pill-exit-ok");Ot(I,{get text(){return`code ${t(a).exitCode??""}`},get classes(){return t(Z)}})}};q(M,I=>{t(a).exitCode!==null&&I(N)})}g(_),S(i,_)};q(we,i=>{t(B)||i(_e)})}g(s),H(()=>{rt(K,t(mt)),Dt=Rt(dt,"",Dt,{display:t(d)==="raw"?"flex":"none"}),$t=Rt(at,"",$t,{display:t(d)==="chat"?"flex":"none"})}),S(e,s)};q(Ct,e=>{t(O)?e(oe):t(A)?e(ie,1):t(a)&&e(le,2)})}var It=c(Ct,2);Ye(It,{get open(){return t(L)},onClose:()=>{n(L,!1)}});var ce=c(It,2);ze(ce,{get commands(){return t(Qt)},onClose:()=>{n(ct,!1)},get open(){return t(ct)},set open(e){n(ct,e,!0)}}),S(y,xt),it(),T()}export{_n as component};
|
|
Binary file
|
|
Binary file
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"_server.ts-0Xr2fWaq.js","sources":["../../../.svelte-kit/adapter-node/entries/endpoints/api/sessions/connect/_server.ts.js"],"sourcesContent":["import { v as validateAuth } from \"../../../../../chunks/auth.js\";\nimport { p as ptyManager } from \"../../../../../chunks/pty-manager.js\";\nimport { t as toErrorMessage } from \"../../../../../chunks/error.js\";\nimport { json } from \"@sveltejs/kit\";\nimport { realpathSync, statSync } from \"fs\";\nimport { relative, isAbsolute } from \"path\";\nconst POST = async ({ request }) => {\n const authError = validateAuth(request);\n if (authError) {\n return authError;\n }\n let body;\n try {\n body = await request.json();\n } catch {\n return json({ error: \"Invalid JSON in request body\" }, { status: 400 });\n }\n const { command, cwd, noCreate, sessionId } = body;\n if (!sessionId || typeof sessionId !== \"string\") {\n return json({ error: \"sessionId is required (string)\" }, { status: 400 });\n }\n if (!cwd || typeof cwd !== \"string\") {\n return json({ error: \"cwd is required (string)\" }, { status: 400 });\n }\n if (!command || command !== \"claude\" && command !== \"opencode\") {\n return json({ error: 'command must be \"claude\" or \"opencode\"' }, { status: 400 });\n }\n let realCwd;\n try {\n realCwd = realpathSync(cwd);\n if (!statSync(realCwd).isDirectory()) {\n return json({ error: \"Invalid working directory\" }, { status: 400 });\n }\n } catch {\n return json({ error: \"Invalid working directory\" }, { status: 400 });\n }\n const home = process.env.HOME || \"\";\n if (home) {\n const rel = relative(home, realCwd);\n if (rel.startsWith(\"..\") || isAbsolute(rel)) {\n return json({ error: \"Working directory must be under home directory\" }, { status: 400 });\n }\n }\n const existing = ptyManager.list().find(\n (t) => t.status === \"running\" && (t.sessionFile?.endsWith(`/${sessionId}.jsonl`) || t.openCodeSessionId === sessionId)\n );\n if (existing) {\n console.log(\n `[sessions/connect] Reusing terminal ${existing.id} for ${command} session ${sessionId}`\n );\n return json({\n command: existing.command,\n createdAt: existing.createdAt.toISOString(),\n cwd: existing.cwd,\n id: existing.id,\n pid: existing.pid,\n reused: true,\n sessionId,\n sessionWs: `/ws/session/${existing.id}`,\n terminalId: existing.id,\n ws: `/ws/terminal/${existing.id}`\n });\n }\n if (noCreate) {\n return json({ error: \"No existing terminal for this session\" }, { status: 404 });\n }\n const args = command === \"claude\" ? [\"--resume\", sessionId] : [\"--session\", sessionId];\n try {\n const terminal = await ptyManager.create(command, args, realCwd, 120, 40);\n console.log(\n `[sessions/connect] Created terminal ${terminal.id} for ${command} session ${sessionId} (pid=${terminal.pid})`\n );\n return json(\n {\n command: terminal.command,\n createdAt: terminal.createdAt instanceof Date ? terminal.createdAt.toISOString() : terminal.createdAt,\n cwd: terminal.cwd,\n id: terminal.id,\n pid: terminal.pid,\n sessionId,\n sessionWs: `/ws/session/${terminal.id}`,\n terminalId: terminal.id,\n ws: `/ws/terminal/${terminal.id}`\n },\n { status: 201 }\n );\n } catch (error) {\n console.error(\"[sessions/connect] Failed to create terminal:\", toErrorMessage(error));\n return json({ error: \"Failed to create terminal for session\" }, { status: 500 });\n }\n};\nexport {\n POST\n};\n"],"names":[],"mappings":"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"}
|