@code-yeongyu/senpi 2026.5.24 → 2026.5.29-2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +27 -0
- package/dist/cli/args.d.ts +0 -6
- package/dist/cli/args.d.ts.map +1 -1
- package/dist/cli/args.js +1 -2
- package/dist/cli/args.js.map +1 -1
- package/dist/config.d.ts.map +1 -1
- package/dist/config.js +15 -2
- package/dist/config.js.map +1 -1
- package/dist/core/agent-session.d.ts +4 -0
- package/dist/core/agent-session.d.ts.map +1 -1
- package/dist/core/agent-session.js +116 -80
- package/dist/core/agent-session.js.map +1 -1
- package/dist/core/compaction/compaction.d.ts.map +1 -1
- package/dist/core/compaction/compaction.js +18 -24
- package/dist/core/compaction/compaction.js.map +1 -1
- package/dist/core/extensions/builtin/gpt-apply-patch/streaming-render.d.ts.map +1 -1
- package/dist/core/extensions/builtin/gpt-apply-patch/streaming-render.js +4 -2
- package/dist/core/extensions/builtin/gpt-apply-patch/streaming-render.js.map +1 -1
- package/dist/core/extensions/builtin/session-observer/overlay.d.ts.map +1 -1
- package/dist/core/extensions/builtin/session-observer/overlay.js +0 -5
- package/dist/core/extensions/builtin/session-observer/overlay.js.map +1 -1
- package/dist/core/extensions/builtin/session-observer/scanner.d.ts.map +1 -1
- package/dist/core/extensions/builtin/session-observer/scanner.js +2 -0
- package/dist/core/extensions/builtin/session-observer/scanner.js.map +1 -1
- package/dist/core/extensions/loader.d.ts.map +1 -1
- package/dist/core/extensions/loader.js +21 -9
- package/dist/core/extensions/loader.js.map +1 -1
- package/dist/core/output-guard.d.ts +1 -0
- package/dist/core/output-guard.d.ts.map +1 -1
- package/dist/core/output-guard.js +52 -22
- package/dist/core/output-guard.js.map +1 -1
- package/dist/core/session-work-barrier.d.ts +9 -0
- package/dist/core/session-work-barrier.d.ts.map +1 -0
- package/dist/core/session-work-barrier.js +50 -0
- package/dist/core/session-work-barrier.js.map +1 -0
- package/dist/main.d.ts.map +1 -1
- package/dist/main.js +0 -15
- package/dist/main.js.map +1 -1
- package/dist/modes/interactive/components/user-message.d.ts.map +1 -1
- package/dist/modes/interactive/components/user-message.js +1 -1
- package/dist/modes/interactive/components/user-message.js.map +1 -1
- package/dist/modes/interactive/interactive-mode.d.ts.map +1 -1
- package/dist/modes/interactive/interactive-mode.js +6 -0
- package/dist/modes/interactive/interactive-mode.js.map +1 -1
- package/dist/modes/rpc/rpc-client.d.ts +3 -0
- package/dist/modes/rpc/rpc-client.d.ts.map +1 -1
- package/dist/modes/rpc/rpc-client.js +64 -7
- package/dist/modes/rpc/rpc-client.js.map +1 -1
- package/dist/modes/rpc/rpc-mode.d.ts.map +1 -1
- package/dist/modes/rpc/rpc-mode.js +15 -3
- package/dist/modes/rpc/rpc-mode.js.map +1 -1
- package/docs/settings.md +3 -1
- package/node_modules/@earendil-works/pi-agent-core/dist/harness/compaction/compaction.d.ts.map +1 -1
- package/node_modules/@earendil-works/pi-agent-core/dist/harness/compaction/compaction.js +18 -24
- package/node_modules/@earendil-works/pi-agent-core/dist/harness/compaction/compaction.js.map +1 -1
- package/node_modules/@earendil-works/pi-agent-core/package.json +2 -2
- package/node_modules/@earendil-works/pi-ai/dist/models.generated.d.ts +249 -39
- package/node_modules/@earendil-works/pi-ai/dist/models.generated.d.ts.map +1 -1
- package/node_modules/@earendil-works/pi-ai/dist/models.generated.js +349 -144
- package/node_modules/@earendil-works/pi-ai/dist/models.generated.js.map +1 -1
- package/node_modules/@earendil-works/pi-ai/dist/providers/anthropic.d.ts.map +1 -1
- package/node_modules/@earendil-works/pi-ai/dist/providers/anthropic.js +1 -1
- package/node_modules/@earendil-works/pi-ai/dist/providers/anthropic.js.map +1 -1
- package/node_modules/@earendil-works/pi-ai/dist/providers/azure-openai-responses.d.ts.map +1 -1
- package/node_modules/@earendil-works/pi-ai/dist/providers/azure-openai-responses.js +1 -1
- package/node_modules/@earendil-works/pi-ai/dist/providers/azure-openai-responses.js.map +1 -1
- package/node_modules/@earendil-works/pi-ai/dist/providers/images/openrouter.d.ts.map +1 -1
- package/node_modules/@earendil-works/pi-ai/dist/providers/images/openrouter.js +1 -1
- package/node_modules/@earendil-works/pi-ai/dist/providers/images/openrouter.js.map +1 -1
- package/node_modules/@earendil-works/pi-ai/dist/providers/openai-codex-responses.d.ts.map +1 -1
- package/node_modules/@earendil-works/pi-ai/dist/providers/openai-codex-responses.js +46 -29
- package/node_modules/@earendil-works/pi-ai/dist/providers/openai-codex-responses.js.map +1 -1
- package/node_modules/@earendil-works/pi-ai/dist/providers/openai-completions.d.ts.map +1 -1
- package/node_modules/@earendil-works/pi-ai/dist/providers/openai-completions.js +1 -1
- package/node_modules/@earendil-works/pi-ai/dist/providers/openai-completions.js.map +1 -1
- package/node_modules/@earendil-works/pi-ai/dist/providers/openai-responses.d.ts.map +1 -1
- package/node_modules/@earendil-works/pi-ai/dist/providers/openai-responses.js +1 -1
- package/node_modules/@earendil-works/pi-ai/dist/providers/openai-responses.js.map +1 -1
- package/node_modules/@earendil-works/pi-ai/dist/utils/overflow.d.ts +2 -1
- package/node_modules/@earendil-works/pi-ai/dist/utils/overflow.d.ts.map +1 -1
- package/node_modules/@earendil-works/pi-ai/dist/utils/overflow.js +5 -2
- package/node_modules/@earendil-works/pi-ai/dist/utils/overflow.js.map +1 -1
- package/node_modules/@earendil-works/pi-ai/package.json +1 -1
- package/node_modules/@earendil-works/pi-tui/dist/autocomplete.d.ts.map +1 -1
- package/node_modules/@earendil-works/pi-tui/dist/autocomplete.js +2 -17
- package/node_modules/@earendil-works/pi-tui/dist/autocomplete.js.map +1 -1
- package/node_modules/@earendil-works/pi-tui/dist/components/editor.d.ts.map +1 -1
- package/node_modules/@earendil-works/pi-tui/dist/components/editor.js +40 -55
- package/node_modules/@earendil-works/pi-tui/dist/components/editor.js.map +1 -1
- package/node_modules/@earendil-works/pi-tui/dist/components/input.d.ts.map +1 -1
- package/node_modules/@earendil-works/pi-tui/dist/components/input.js +2 -2
- package/node_modules/@earendil-works/pi-tui/dist/components/input.js.map +1 -1
- package/node_modules/@earendil-works/pi-tui/dist/components/markdown.d.ts +7 -1
- package/node_modules/@earendil-works/pi-tui/dist/components/markdown.d.ts.map +1 -1
- package/node_modules/@earendil-works/pi-tui/dist/components/markdown.js +12 -2
- package/node_modules/@earendil-works/pi-tui/dist/components/markdown.js.map +1 -1
- package/node_modules/@earendil-works/pi-tui/dist/index.d.ts +1 -1
- package/node_modules/@earendil-works/pi-tui/dist/index.d.ts.map +1 -1
- package/node_modules/@earendil-works/pi-tui/dist/index.js.map +1 -1
- package/node_modules/@earendil-works/pi-tui/dist/slash-command-autocomplete.d.ts +3 -0
- package/node_modules/@earendil-works/pi-tui/dist/slash-command-autocomplete.d.ts.map +1 -0
- package/node_modules/@earendil-works/pi-tui/dist/slash-command-autocomplete.js +38 -0
- package/node_modules/@earendil-works/pi-tui/dist/slash-command-autocomplete.js.map +1 -0
- package/node_modules/@earendil-works/pi-tui/dist/terminal-image.d.ts.map +1 -1
- package/node_modules/@earendil-works/pi-tui/dist/terminal-image.js +4 -1
- package/node_modules/@earendil-works/pi-tui/dist/terminal-image.js.map +1 -1
- package/node_modules/@earendil-works/pi-tui/dist/utils.d.ts +5 -1
- package/node_modules/@earendil-works/pi-tui/dist/utils.d.ts.map +1 -1
- package/node_modules/@earendil-works/pi-tui/dist/utils.js +66 -14
- package/node_modules/@earendil-works/pi-tui/dist/utils.js.map +1 -1
- package/node_modules/@earendil-works/pi-tui/package.json +1 -1
- package/npm-shrinkwrap.json +13 -13
- package/package.json +6 -7
- package/dist/modes/neo-mode.d.ts +0 -43
- package/dist/modes/neo-mode.d.ts.map +0 -1
- package/dist/modes/neo-mode.js +0 -142
- package/dist/modes/neo-mode.js.map +0 -1
- package/dist/neo-tui-bin/senpi-neo-tui-linux-x64 +0 -0
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"overlay.d.ts","sourceRoot":"","sources":["../../../../../src/core/extensions/builtin/session-observer/overlay.ts"],"names":[],"mappings":"AAAA,OAAO,EACN,SAAS,EACT,KAAK,SAAS,EAOd,MAAM,wBAAwB,CAAC;AAUhC,OAAO,KAAK,EAAE,eAAe,EAAwC,MAAM,YAAY,CAAC;AAIxF,KAAK,IAAI,GAAG,QAAQ,GAAG,QAAQ,CAAC;AAUhC,UAAU,wBAAwB;IACjC,QAAQ,CAAC,QAAQ,EAAE,SAAS,eAAe,EAAE,CAAC;IAC9C,QAAQ,CAAC,IAAI,EAAE,MAAM,IAAI,CAAC;IAC1B,QAAQ,CAAC,aAAa,EAAE,MAAM,IAAI,CAAC;CACnC;AAED,qBAAa,iBAAkB,SAAQ,SAAU,YAAW,SAAS;IACpE,OAAO,CAAC,QAAQ,CAAC,OAAO,CAA2B;IACnD,OAAO,CAAC,QAAQ,CAAC,eAAe,CAAsC;IACtE,OAAO,CAAC,QAAQ,CAAC,SAAS,CAAyD;IACnF,OAAO,CAAC,QAAQ,CAAC,YAAY,CAAyD;IACtF,OAAO,CAAC,QAAQ,CAAC,YAAY,CAAyD;IACtF,OAAO,CAAC,IAAI,CAAyB;IACrC,OAAO,CAAC,IAAI,CAAkB;IAC9B,OAAO,CAAC,eAAe,CAA8B;IACrD,OAAO,CAAC,QAAQ,CAAiC;IACjD,OAAO,CAAC,aAAa,CAAyB;IAC9C,OAAO,CAAC,MAAM,CAAmC;IACjD,OAAO,CAAC,kBAAkB,CAAM;IAChC,OAAO,CAAC,sBAAsB,CAAS;IACvC,OAAO,CAAC,eAAe,CAAqB;IAC5C,OAAO,CAAC,YAAY,CAAK;IACzB,OAAO,CAAC,cAAc,CAAM;IAC5B,OAAO,CAAC,WAAW,CAAqB;IACxC,OAAO,CAAC,QAAQ,CAAS;IAEzB,YAAY,OAAO,EAAE,wBAAwB,EAI5C;IAED,IAAI,OAAO,IAAI,OAAO,CAErB;IAED,IAAI,OAAO,CAAC,KAAK,EAAE,OAAO,EAEzB;IAED,WAAW,CAAC,KAAK,EAAE,MAAM,GAAG,IAAI,CAM/B;IAEQ,MAAM,CAAC,KAAK,EAAE,MAAM,GAAG,MAAM,EAAE,CAGvC;IAED,OAAO,IAAI,IAAI,CAEd;IAED,qBAAqB,IAAI,MAAM,CAE9B;IAED,qBAAqB,IAAI,MAAM,CAE9B;IAED,OAAO,CAAC,iBAAiB;IAOzB,OAAO,CAAC,aAAa;IAqCrB,OAAO,CAAC,YAAY;YAMN,WAAW;IAiBzB,OAAO,CAAC,gBAAgB;IASxB,OAAO,CAAC,iBAAiB;IAkBzB,OAAO,CAAC,YAAY;IA4BpB,OAAO,CAAC,iBAAiB;IAgBzB,OAAO,CAAC,YAAY;IAKpB,OAAO,CAAC,aAAa;IAMrB,OAAO,CAAC,MAAM;IAMd,OAAO,CAAC,cAAc;IAMtB,OAAO,CAAC,gBAAgB;CAQxB","sourcesContent":["import {\n\tContainer,\n\ttype Focusable,\n\tgetKeybindings,\n\ttype SelectItem,\n\tSelectList,\n\tSpacer,\n\tText,\n\tTruncatedText,\n} from \"@earendil-works/pi-tui\";\nimport { DynamicBorder } from \"../../../../modes/interactive/components/dynamic-border.ts\";\nimport { keyHint } from \"../../../../modes/interactive/components/keybinding-hints.ts\";\nimport { getMarkdownTheme, theme } from \"../../../../modes/interactive/theme/theme.ts\";\nimport { shortenPath } from \"../../../../utils/paths.ts\";\nimport type {} from \"../../../keybindings.ts\";\nimport { loadTranscriptSnapshot } from \"./loader.ts\";\nimport { describeSession, pickerLabel, renderLine, sessionAge, viewerFooter } from \"./overlay-format.ts\";\nimport { sanitizeLine } from \"./text.ts\";\nimport { renderTranscript } from \"./transcript.ts\";\nimport type { SessionHudEntry, TranscriptSnapshot, ViewerEntryRange } from \"./types.ts\";\n\nconst MAX_VISIBLE_SESSIONS = 12;\n\ntype Mode = \"picker\" | \"viewer\";\ntype PickerAction = \"tui.select.up\" | \"tui.select.down\" | \"tui.select.confirm\" | \"tui.select.cancel\";\n\nconst PICKER_ACTIONS: readonly PickerAction[] = [\n\t\"tui.select.up\",\n\t\"tui.select.down\",\n\t\"tui.select.confirm\",\n\t\"tui.select.cancel\",\n];\n\ninterface SessionHudOverlayOptions {\n\treadonly sessions: readonly SessionHudEntry[];\n\treadonly done: () => void;\n\treadonly requestRender: () => void;\n}\n\nexport class SessionHudOverlay extends Container implements Focusable {\n\tprivate readonly options: SessionHudOverlayOptions;\n\tprivate readonly sessionsByValue = new Map<string, SessionHudEntry>();\n\tprivate readonly topBorder = new DynamicBorder((text) => theme.fg(\"accent\", text));\n\tprivate readonly middleBorder = new DynamicBorder((text) => theme.fg(\"accent\", text));\n\tprivate readonly bottomBorder = new DynamicBorder((text) => theme.fg(\"accent\", text));\n\tprivate list: SelectList | undefined;\n\tprivate mode: Mode = \"picker\";\n\tprivate selectedSession: SessionHudEntry | undefined;\n\tprivate snapshot: TranscriptSnapshot | undefined;\n\tprivate renderedLines: readonly string[] = [];\n\tprivate ranges: readonly ViewerEntryRange[] = [];\n\tprivate selectedEntryIndex = -1;\n\tprivate shouldSelectLastOnLoad = false;\n\tprivate expandedEntries = new Set<number>();\n\tprivate scrollOffset = 0;\n\tprivate viewportHeight = 12;\n\tprivate loadingText: string | undefined;\n\tprivate _focused = false;\n\n\tconstructor(options: SessionHudOverlayOptions) {\n\t\tsuper();\n\t\tthis.options = options;\n\t\tthis.rebuildPicker();\n\t}\n\n\tget focused(): boolean {\n\t\treturn this._focused;\n\t}\n\n\tset focused(value: boolean) {\n\t\tthis._focused = value;\n\t}\n\n\thandleInput(input: string): void {\n\t\tif (this.mode === \"picker\") {\n\t\t\tthis.handlePickerInput(input);\n\t\t\treturn;\n\t\t}\n\t\tthis.handleViewerInput(input);\n\t}\n\n\toverride render(width: number): string[] {\n\t\tif (this.mode === \"picker\") return super.render(width);\n\t\treturn this.renderViewer(width);\n\t}\n\n\tgetMode(): Mode {\n\t\treturn this.mode;\n\t}\n\n\tgetSelectedEntryIndex(): number {\n\t\treturn this.selectedEntryIndex;\n\t}\n\n\tgetExpandedEntryCount(): number {\n\t\treturn this.expandedEntries.size;\n\t}\n\n\tprivate handlePickerInput(input: string): void {\n\t\tconst keybindings = getKeybindings();\n\t\tif (PICKER_ACTIONS.some((action) => keybindings.matches(input, action))) {\n\t\t\tthis.list?.handleInput(input);\n\t\t}\n\t}\n\n\tprivate rebuildPicker(): void {\n\t\tthis.sessionsByValue.clear();\n\t\tconst items = this.options.sessions.map((session, index) => this.toPickerItem(session, index));\n\t\tconst list = new SelectList(items, Math.min(MAX_VISIBLE_SESSIONS, Math.max(1, items.length)), {\n\t\t\tselectedPrefix: (text) => theme.fg(\"accent\", text),\n\t\t\tselectedText: (text) => text,\n\t\t\tdescription: (text) => theme.fg(\"muted\", text),\n\t\t\tscrollInfo: (text) => theme.fg(\"dim\", text),\n\t\t\tnoMatch: (text) => theme.fg(\"warning\", text.replace(\"commands\", \"sessions\")),\n\t\t});\n\t\tlist.onSelect = (item) => {\n\t\t\tconst session = this.sessionsByValue.get(item.value);\n\t\t\tif (session) void this.openSession(session);\n\t\t};\n\t\tlist.onCancel = () => this.options.done();\n\t\tthis.list = list;\n\t\tthis.clear();\n\t\tthis.addChild(new Spacer(1));\n\t\tthis.addChild(this.topBorder);\n\t\tthis.addChild(new Spacer(1));\n\t\tthis.addChild(\n\t\t\tnew Text(\n\t\t\t\t`${theme.bold(theme.fg(\"accent\", \" Sessions\"))}${theme.fg(\"dim\", ` ${this.options.sessions.length} sessions`)}`,\n\t\t\t\t0,\n\t\t\t\t0,\n\t\t\t),\n\t\t);\n\t\tthis.addChild(new Spacer(1));\n\t\tthis.addChild(list);\n\t\tthis.addChild(new Spacer(1));\n\t\tthis.addChild(\n\t\t\tnew TruncatedText(`${keyHint(\"tui.select.confirm\", \"view\")} ${keyHint(\"tui.select.cancel\", \"close\")}`, 0, 0),\n\t\t);\n\t\tthis.addChild(new Spacer(1));\n\t\tthis.addChild(this.bottomBorder);\n\t}\n\n\tprivate toPickerItem(session: SessionHudEntry, index: number): SelectItem {\n\t\tconst value = String(index);\n\t\tthis.sessionsByValue.set(value, session);\n\t\treturn { value, label: pickerLabel(session), description: describeSession(session) };\n\t}\n\n\tprivate async openSession(session: SessionHudEntry): Promise<void> {\n\t\tthis.mode = \"viewer\";\n\t\tthis.selectedSession = session;\n\t\tthis.snapshot = undefined;\n\t\tthis.loadingText = \"Loading session transcript...\";\n\t\tthis.resetViewerState();\n\t\tthis.options.requestRender();\n\t\ttry {\n\t\t\tthis.snapshot = await loadTranscriptSnapshot(session.path);\n\t\t\tthis.loadingText = undefined;\n\t\t} catch (error) {\n\t\t\tthis.loadingText = `Failed to read session: ${error instanceof Error ? error.message : String(error)}`;\n\t\t}\n\t\tthis.rebuildTranscript(process.stdout.columns || 80);\n\t\tthis.options.requestRender();\n\t}\n\n\tprivate resetViewerState(): void {\n\t\tthis.expandedEntries = new Set<number>();\n\t\tthis.scrollOffset = 0;\n\t\tthis.selectedEntryIndex = -1;\n\t\tthis.shouldSelectLastOnLoad = true;\n\t\tthis.renderedLines = [];\n\t\tthis.ranges = [];\n\t}\n\n\tprivate rebuildTranscript(width: number): void {\n\t\tif (!this.snapshot) return;\n\t\tconst rendered = renderTranscript(this.snapshot.entries, {\n\t\t\twidth,\n\t\t\tselectedIndex: this.selectedEntryIndex,\n\t\t\texpandedEntries: this.expandedEntries,\n\t\t\tmarkdownTheme: getMarkdownTheme(),\n\t\t});\n\t\tthis.renderedLines = rendered.lines;\n\t\tthis.ranges = rendered.ranges;\n\t\tif (this.ranges.length > 0 && this.shouldSelectLastOnLoad) {\n\t\t\tthis.shouldSelectLastOnLoad = false;\n\t\t\tthis.selectedEntryIndex = this.ranges.length - 1;\n\t\t\tthis.rebuildTranscript(width);\n\t\t}\n\t\tthis.scrollToSelected();\n\t}\n\n\tprivate renderViewer(width: number): string[] {\n\t\tthis.viewportHeight = Math.max(5, (process.stdout.rows || 32) - 8);\n\t\tthis.rebuildTranscript(width);\n\t\tconst session = this.selectedSession;\n\t\tconst title = session ? `Sessions > ${shortenPath(session.cwd) || \"unknown\"} · ${session.shortId}` : \"Sessions\";\n\t\tconst status = session\n\t\t\t? `${session.messageCount} messages · ${sessionAge(session)}${this.snapshot?.model ? ` · ${this.snapshot.model}` : \"\"}`\n\t\t\t: \"\";\n\t\tconst maxScroll = Math.max(0, this.renderedLines.length - this.viewportHeight);\n\t\tthis.scrollOffset = Math.max(0, Math.min(this.scrollOffset, maxScroll));\n\t\tconst content = this.loadingText ? [theme.fg(\"dim\", this.loadingText)] : this.renderedLines;\n\t\tconst visible = content.slice(this.scrollOffset, this.scrollOffset + this.viewportHeight);\n\t\tconst lines: string[] = [];\n\t\tlines.push(...this.topBorder.render(width));\n\t\tlines.push(renderLine(` ${theme.bold(theme.fg(\"accent\", title))}`, width));\n\t\tif (status) lines.push(renderLine(` ${theme.fg(\"dim\", status)}`, width));\n\t\tlines.push(...this.middleBorder.render(width));\n\t\tfor (const line of visible) lines.push(` ${sanitizeLine(line, width - 2)}`);\n\t\tfor (let index = visible.length; index < this.viewportHeight; index += 1) lines.push(\"\");\n\t\tconst scroll =\n\t\t\tcontent.length > this.viewportHeight\n\t\t\t\t? ` [${this.scrollOffset + 1}-${Math.min(this.scrollOffset + this.viewportHeight, content.length)}/${content.length}]`\n\t\t\t\t: \"\";\n\t\tlines.push(renderLine(` ${viewerFooter(scroll)}`, width));\n\t\tlines.push(...this.bottomBorder.render(width));\n\t\treturn lines;\n\t}\n\n\tprivate handleViewerInput(input: string): void {\n\t\tconst keybindings = getKeybindings();\n\t\tif (keybindings.matches(input, \"app.sessions.observe\")) {\n\t\t\tthis.options.done();\n\t\t\treturn;\n\t\t} else if (keybindings.matches(input, \"tui.select.cancel\")) this.backToPicker();\n\t\telse if (input === \"j\" || keybindings.matches(input, \"tui.select.down\")) this.moveSelection(1);\n\t\telse if (input === \"k\" || keybindings.matches(input, \"tui.select.up\")) this.moveSelection(-1);\n\t\telse if (keybindings.matches(input, \"tui.select.pageDown\")) this.moveSelection(5);\n\t\telse if (keybindings.matches(input, \"tui.select.pageUp\")) this.moveSelection(-5);\n\t\telse if (input === \"g\") this.jumpTo(0);\n\t\telse if (input === \"G\") this.jumpTo(this.ranges.length - 1);\n\t\telse if (keybindings.matches(input, \"tui.select.confirm\")) this.toggleExpanded();\n\t\tthis.options.requestRender();\n\t}\n\n\tprivate backToPicker(): void {\n\t\tthis.mode = \"picker\";\n\t\tthis.rebuildPicker();\n\t}\n\n\tprivate moveSelection(delta: number): void {\n\t\tif (this.ranges.length === 0) return;\n\t\tthis.selectedEntryIndex = Math.max(0, Math.min(this.selectedEntryIndex + delta, this.ranges.length - 1));\n\t\tthis.scrollToSelected();\n\t}\n\n\tprivate jumpTo(index: number): void {\n\t\tif (this.ranges.length === 0) return;\n\t\tthis.selectedEntryIndex = Math.max(0, Math.min(index, this.ranges.length - 1));\n\t\tthis.scrollToSelected();\n\t}\n\n\tprivate toggleExpanded(): void {\n\t\tif (this.ranges.length === 0) return;\n\t\tif (this.expandedEntries.has(this.selectedEntryIndex)) this.expandedEntries.delete(this.selectedEntryIndex);\n\t\telse this.expandedEntries.add(this.selectedEntryIndex);\n\t}\n\n\tprivate scrollToSelected(): void {\n\t\tconst selected = this.ranges[this.selectedEntryIndex];\n\t\tif (!selected) return;\n\t\tconst bottom = selected.lineStart + selected.lineCount;\n\t\tif (selected.lineStart < this.scrollOffset) this.scrollOffset = Math.max(0, selected.lineStart - 1);\n\t\tif (bottom > this.scrollOffset + this.viewportHeight)\n\t\t\tthis.scrollOffset = Math.max(0, bottom - this.viewportHeight + 1);\n\t}\n}\n"]}
|
|
1
|
+
{"version":3,"file":"overlay.d.ts","sourceRoot":"","sources":["../../../../../src/core/extensions/builtin/session-observer/overlay.ts"],"names":[],"mappings":"AAAA,OAAO,EACN,SAAS,EACT,KAAK,SAAS,EAOd,MAAM,wBAAwB,CAAC;AAUhC,OAAO,KAAK,EAAE,eAAe,EAAwC,MAAM,YAAY,CAAC;AAIxF,KAAK,IAAI,GAAG,QAAQ,GAAG,QAAQ,CAAC;AAUhC,UAAU,wBAAwB;IACjC,QAAQ,CAAC,QAAQ,EAAE,SAAS,eAAe,EAAE,CAAC;IAC9C,QAAQ,CAAC,IAAI,EAAE,MAAM,IAAI,CAAC;IAC1B,QAAQ,CAAC,aAAa,EAAE,MAAM,IAAI,CAAC;CACnC;AAED,qBAAa,iBAAkB,SAAQ,SAAU,YAAW,SAAS;IACpE,OAAO,CAAC,QAAQ,CAAC,OAAO,CAA2B;IACnD,OAAO,CAAC,QAAQ,CAAC,eAAe,CAAsC;IACtE,OAAO,CAAC,QAAQ,CAAC,SAAS,CAAyD;IACnF,OAAO,CAAC,QAAQ,CAAC,YAAY,CAAyD;IACtF,OAAO,CAAC,QAAQ,CAAC,YAAY,CAAyD;IACtF,OAAO,CAAC,IAAI,CAAyB;IACrC,OAAO,CAAC,IAAI,CAAkB;IAC9B,OAAO,CAAC,eAAe,CAA8B;IACrD,OAAO,CAAC,QAAQ,CAAiC;IACjD,OAAO,CAAC,aAAa,CAAyB;IAC9C,OAAO,CAAC,MAAM,CAAmC;IACjD,OAAO,CAAC,kBAAkB,CAAM;IAChC,OAAO,CAAC,sBAAsB,CAAS;IACvC,OAAO,CAAC,eAAe,CAAqB;IAC5C,OAAO,CAAC,YAAY,CAAK;IACzB,OAAO,CAAC,cAAc,CAAM;IAC5B,OAAO,CAAC,WAAW,CAAqB;IACxC,OAAO,CAAC,QAAQ,CAAS;IAEzB,YAAY,OAAO,EAAE,wBAAwB,EAI5C;IAED,IAAI,OAAO,IAAI,OAAO,CAErB;IAED,IAAI,OAAO,CAAC,KAAK,EAAE,OAAO,EAEzB;IAED,WAAW,CAAC,KAAK,EAAE,MAAM,GAAG,IAAI,CAM/B;IAEQ,MAAM,CAAC,KAAK,EAAE,MAAM,GAAG,MAAM,EAAE,CAGvC;IAED,OAAO,IAAI,IAAI,CAEd;IAED,qBAAqB,IAAI,MAAM,CAE9B;IAED,qBAAqB,IAAI,MAAM,CAE9B;IAED,OAAO,CAAC,iBAAiB;IAOzB,OAAO,CAAC,aAAa;IAgCrB,OAAO,CAAC,YAAY;YAMN,WAAW;IAiBzB,OAAO,CAAC,gBAAgB;IASxB,OAAO,CAAC,iBAAiB;IAkBzB,OAAO,CAAC,YAAY;IA4BpB,OAAO,CAAC,iBAAiB;IAgBzB,OAAO,CAAC,YAAY;IAKpB,OAAO,CAAC,aAAa;IAMrB,OAAO,CAAC,MAAM;IAMd,OAAO,CAAC,cAAc;IAMtB,OAAO,CAAC,gBAAgB;CAQxB","sourcesContent":["import {\n\tContainer,\n\ttype Focusable,\n\tgetKeybindings,\n\ttype SelectItem,\n\tSelectList,\n\tSpacer,\n\tText,\n\tTruncatedText,\n} from \"@earendil-works/pi-tui\";\nimport { DynamicBorder } from \"../../../../modes/interactive/components/dynamic-border.ts\";\nimport { keyHint } from \"../../../../modes/interactive/components/keybinding-hints.ts\";\nimport { getMarkdownTheme, theme } from \"../../../../modes/interactive/theme/theme.ts\";\nimport { shortenPath } from \"../../../../utils/paths.ts\";\nimport type {} from \"../../../keybindings.ts\";\nimport { loadTranscriptSnapshot } from \"./loader.ts\";\nimport { describeSession, pickerLabel, renderLine, sessionAge, viewerFooter } from \"./overlay-format.ts\";\nimport { sanitizeLine } from \"./text.ts\";\nimport { renderTranscript } from \"./transcript.ts\";\nimport type { SessionHudEntry, TranscriptSnapshot, ViewerEntryRange } from \"./types.ts\";\n\nconst MAX_VISIBLE_SESSIONS = 12;\n\ntype Mode = \"picker\" | \"viewer\";\ntype PickerAction = \"tui.select.up\" | \"tui.select.down\" | \"tui.select.confirm\" | \"tui.select.cancel\";\n\nconst PICKER_ACTIONS: readonly PickerAction[] = [\n\t\"tui.select.up\",\n\t\"tui.select.down\",\n\t\"tui.select.confirm\",\n\t\"tui.select.cancel\",\n];\n\ninterface SessionHudOverlayOptions {\n\treadonly sessions: readonly SessionHudEntry[];\n\treadonly done: () => void;\n\treadonly requestRender: () => void;\n}\n\nexport class SessionHudOverlay extends Container implements Focusable {\n\tprivate readonly options: SessionHudOverlayOptions;\n\tprivate readonly sessionsByValue = new Map<string, SessionHudEntry>();\n\tprivate readonly topBorder = new DynamicBorder((text) => theme.fg(\"accent\", text));\n\tprivate readonly middleBorder = new DynamicBorder((text) => theme.fg(\"accent\", text));\n\tprivate readonly bottomBorder = new DynamicBorder((text) => theme.fg(\"accent\", text));\n\tprivate list: SelectList | undefined;\n\tprivate mode: Mode = \"picker\";\n\tprivate selectedSession: SessionHudEntry | undefined;\n\tprivate snapshot: TranscriptSnapshot | undefined;\n\tprivate renderedLines: readonly string[] = [];\n\tprivate ranges: readonly ViewerEntryRange[] = [];\n\tprivate selectedEntryIndex = -1;\n\tprivate shouldSelectLastOnLoad = false;\n\tprivate expandedEntries = new Set<number>();\n\tprivate scrollOffset = 0;\n\tprivate viewportHeight = 12;\n\tprivate loadingText: string | undefined;\n\tprivate _focused = false;\n\n\tconstructor(options: SessionHudOverlayOptions) {\n\t\tsuper();\n\t\tthis.options = options;\n\t\tthis.rebuildPicker();\n\t}\n\n\tget focused(): boolean {\n\t\treturn this._focused;\n\t}\n\n\tset focused(value: boolean) {\n\t\tthis._focused = value;\n\t}\n\n\thandleInput(input: string): void {\n\t\tif (this.mode === \"picker\") {\n\t\t\tthis.handlePickerInput(input);\n\t\t\treturn;\n\t\t}\n\t\tthis.handleViewerInput(input);\n\t}\n\n\toverride render(width: number): string[] {\n\t\tif (this.mode === \"picker\") return super.render(width);\n\t\treturn this.renderViewer(width);\n\t}\n\n\tgetMode(): Mode {\n\t\treturn this.mode;\n\t}\n\n\tgetSelectedEntryIndex(): number {\n\t\treturn this.selectedEntryIndex;\n\t}\n\n\tgetExpandedEntryCount(): number {\n\t\treturn this.expandedEntries.size;\n\t}\n\n\tprivate handlePickerInput(input: string): void {\n\t\tconst keybindings = getKeybindings();\n\t\tif (PICKER_ACTIONS.some((action) => keybindings.matches(input, action))) {\n\t\t\tthis.list?.handleInput(input);\n\t\t}\n\t}\n\n\tprivate rebuildPicker(): void {\n\t\tthis.sessionsByValue.clear();\n\t\tconst items = this.options.sessions.map((session, index) => this.toPickerItem(session, index));\n\t\tconst list = new SelectList(items, Math.min(MAX_VISIBLE_SESSIONS, Math.max(1, items.length)), {\n\t\t\tselectedPrefix: (text) => theme.fg(\"accent\", text),\n\t\t\tselectedText: (text) => text,\n\t\t\tdescription: (text) => theme.fg(\"muted\", text),\n\t\t\tscrollInfo: (text) => theme.fg(\"dim\", text),\n\t\t\tnoMatch: (text) => theme.fg(\"warning\", text.replace(\"commands\", \"sessions\")),\n\t\t});\n\t\tlist.onSelect = (item) => {\n\t\t\tconst session = this.sessionsByValue.get(item.value);\n\t\t\tif (session) void this.openSession(session);\n\t\t};\n\t\tlist.onCancel = () => this.options.done();\n\t\tthis.list = list;\n\t\tthis.clear();\n\t\tthis.addChild(\n\t\t\tnew Text(\n\t\t\t\t`${theme.bold(theme.fg(\"accent\", \" Sessions\"))}${theme.fg(\"dim\", ` ${this.options.sessions.length} sessions`)}`,\n\t\t\t\t0,\n\t\t\t\t0,\n\t\t\t),\n\t\t);\n\t\tthis.addChild(new Spacer(1));\n\t\tthis.addChild(list);\n\t\tthis.addChild(new Spacer(1));\n\t\tthis.addChild(\n\t\t\tnew TruncatedText(`${keyHint(\"tui.select.confirm\", \"view\")} ${keyHint(\"tui.select.cancel\", \"close\")}`, 0, 0),\n\t\t);\n\t}\n\n\tprivate toPickerItem(session: SessionHudEntry, index: number): SelectItem {\n\t\tconst value = String(index);\n\t\tthis.sessionsByValue.set(value, session);\n\t\treturn { value, label: pickerLabel(session), description: describeSession(session) };\n\t}\n\n\tprivate async openSession(session: SessionHudEntry): Promise<void> {\n\t\tthis.mode = \"viewer\";\n\t\tthis.selectedSession = session;\n\t\tthis.snapshot = undefined;\n\t\tthis.loadingText = \"Loading session transcript...\";\n\t\tthis.resetViewerState();\n\t\tthis.options.requestRender();\n\t\ttry {\n\t\t\tthis.snapshot = await loadTranscriptSnapshot(session.path);\n\t\t\tthis.loadingText = undefined;\n\t\t} catch (error) {\n\t\t\tthis.loadingText = `Failed to read session: ${error instanceof Error ? error.message : String(error)}`;\n\t\t}\n\t\tthis.rebuildTranscript(process.stdout.columns || 80);\n\t\tthis.options.requestRender();\n\t}\n\n\tprivate resetViewerState(): void {\n\t\tthis.expandedEntries = new Set<number>();\n\t\tthis.scrollOffset = 0;\n\t\tthis.selectedEntryIndex = -1;\n\t\tthis.shouldSelectLastOnLoad = true;\n\t\tthis.renderedLines = [];\n\t\tthis.ranges = [];\n\t}\n\n\tprivate rebuildTranscript(width: number): void {\n\t\tif (!this.snapshot) return;\n\t\tconst rendered = renderTranscript(this.snapshot.entries, {\n\t\t\twidth,\n\t\t\tselectedIndex: this.selectedEntryIndex,\n\t\t\texpandedEntries: this.expandedEntries,\n\t\t\tmarkdownTheme: getMarkdownTheme(),\n\t\t});\n\t\tthis.renderedLines = rendered.lines;\n\t\tthis.ranges = rendered.ranges;\n\t\tif (this.ranges.length > 0 && this.shouldSelectLastOnLoad) {\n\t\t\tthis.shouldSelectLastOnLoad = false;\n\t\t\tthis.selectedEntryIndex = this.ranges.length - 1;\n\t\t\tthis.rebuildTranscript(width);\n\t\t}\n\t\tthis.scrollToSelected();\n\t}\n\n\tprivate renderViewer(width: number): string[] {\n\t\tthis.viewportHeight = Math.max(5, (process.stdout.rows || 32) - 8);\n\t\tthis.rebuildTranscript(width);\n\t\tconst session = this.selectedSession;\n\t\tconst title = session ? `Sessions > ${shortenPath(session.cwd) || \"unknown\"} · ${session.shortId}` : \"Sessions\";\n\t\tconst status = session\n\t\t\t? `${session.messageCount} messages · ${sessionAge(session)}${this.snapshot?.model ? ` · ${this.snapshot.model}` : \"\"}`\n\t\t\t: \"\";\n\t\tconst maxScroll = Math.max(0, this.renderedLines.length - this.viewportHeight);\n\t\tthis.scrollOffset = Math.max(0, Math.min(this.scrollOffset, maxScroll));\n\t\tconst content = this.loadingText ? [theme.fg(\"dim\", this.loadingText)] : this.renderedLines;\n\t\tconst visible = content.slice(this.scrollOffset, this.scrollOffset + this.viewportHeight);\n\t\tconst lines: string[] = [];\n\t\tlines.push(...this.topBorder.render(width));\n\t\tlines.push(renderLine(` ${theme.bold(theme.fg(\"accent\", title))}`, width));\n\t\tif (status) lines.push(renderLine(` ${theme.fg(\"dim\", status)}`, width));\n\t\tlines.push(...this.middleBorder.render(width));\n\t\tfor (const line of visible) lines.push(` ${sanitizeLine(line, width - 2)}`);\n\t\tfor (let index = visible.length; index < this.viewportHeight; index += 1) lines.push(\"\");\n\t\tconst scroll =\n\t\t\tcontent.length > this.viewportHeight\n\t\t\t\t? ` [${this.scrollOffset + 1}-${Math.min(this.scrollOffset + this.viewportHeight, content.length)}/${content.length}]`\n\t\t\t\t: \"\";\n\t\tlines.push(renderLine(` ${viewerFooter(scroll)}`, width));\n\t\tlines.push(...this.bottomBorder.render(width));\n\t\treturn lines;\n\t}\n\n\tprivate handleViewerInput(input: string): void {\n\t\tconst keybindings = getKeybindings();\n\t\tif (keybindings.matches(input, \"app.sessions.observe\")) {\n\t\t\tthis.options.done();\n\t\t\treturn;\n\t\t} else if (keybindings.matches(input, \"tui.select.cancel\")) this.backToPicker();\n\t\telse if (input === \"j\" || keybindings.matches(input, \"tui.select.down\")) this.moveSelection(1);\n\t\telse if (input === \"k\" || keybindings.matches(input, \"tui.select.up\")) this.moveSelection(-1);\n\t\telse if (keybindings.matches(input, \"tui.select.pageDown\")) this.moveSelection(5);\n\t\telse if (keybindings.matches(input, \"tui.select.pageUp\")) this.moveSelection(-5);\n\t\telse if (input === \"g\") this.jumpTo(0);\n\t\telse if (input === \"G\") this.jumpTo(this.ranges.length - 1);\n\t\telse if (keybindings.matches(input, \"tui.select.confirm\")) this.toggleExpanded();\n\t\tthis.options.requestRender();\n\t}\n\n\tprivate backToPicker(): void {\n\t\tthis.mode = \"picker\";\n\t\tthis.rebuildPicker();\n\t}\n\n\tprivate moveSelection(delta: number): void {\n\t\tif (this.ranges.length === 0) return;\n\t\tthis.selectedEntryIndex = Math.max(0, Math.min(this.selectedEntryIndex + delta, this.ranges.length - 1));\n\t\tthis.scrollToSelected();\n\t}\n\n\tprivate jumpTo(index: number): void {\n\t\tif (this.ranges.length === 0) return;\n\t\tthis.selectedEntryIndex = Math.max(0, Math.min(index, this.ranges.length - 1));\n\t\tthis.scrollToSelected();\n\t}\n\n\tprivate toggleExpanded(): void {\n\t\tif (this.ranges.length === 0) return;\n\t\tif (this.expandedEntries.has(this.selectedEntryIndex)) this.expandedEntries.delete(this.selectedEntryIndex);\n\t\telse this.expandedEntries.add(this.selectedEntryIndex);\n\t}\n\n\tprivate scrollToSelected(): void {\n\t\tconst selected = this.ranges[this.selectedEntryIndex];\n\t\tif (!selected) return;\n\t\tconst bottom = selected.lineStart + selected.lineCount;\n\t\tif (selected.lineStart < this.scrollOffset) this.scrollOffset = Math.max(0, selected.lineStart - 1);\n\t\tif (bottom > this.scrollOffset + this.viewportHeight)\n\t\t\tthis.scrollOffset = Math.max(0, bottom - this.viewportHeight + 1);\n\t}\n}\n"]}
|
|
@@ -89,16 +89,11 @@ export class SessionHudOverlay extends Container {
|
|
|
89
89
|
list.onCancel = () => this.options.done();
|
|
90
90
|
this.list = list;
|
|
91
91
|
this.clear();
|
|
92
|
-
this.addChild(new Spacer(1));
|
|
93
|
-
this.addChild(this.topBorder);
|
|
94
|
-
this.addChild(new Spacer(1));
|
|
95
92
|
this.addChild(new Text(`${theme.bold(theme.fg("accent", " Sessions"))}${theme.fg("dim", ` ${this.options.sessions.length} sessions`)}`, 0, 0));
|
|
96
93
|
this.addChild(new Spacer(1));
|
|
97
94
|
this.addChild(list);
|
|
98
95
|
this.addChild(new Spacer(1));
|
|
99
96
|
this.addChild(new TruncatedText(`${keyHint("tui.select.confirm", "view")} ${keyHint("tui.select.cancel", "close")}`, 0, 0));
|
|
100
|
-
this.addChild(new Spacer(1));
|
|
101
|
-
this.addChild(this.bottomBorder);
|
|
102
97
|
}
|
|
103
98
|
toPickerItem(session, index) {
|
|
104
99
|
const value = String(index);
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"overlay.js","sourceRoot":"","sources":["../../../../../src/core/extensions/builtin/session-observer/overlay.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n\tContainer,\n\ttype Focusable,\n\tgetKeybindings,\n\ttype SelectItem,\n\tSelectList,\n\tSpacer,\n\tText,\n\tTruncatedText,\n} from \"@earendil-works/pi-tui\";\nimport { DynamicBorder } from \"../../../../modes/interactive/components/dynamic-border.ts\";\nimport { keyHint } from \"../../../../modes/interactive/components/keybinding-hints.ts\";\nimport { getMarkdownTheme, theme } from \"../../../../modes/interactive/theme/theme.ts\";\nimport { shortenPath } from \"../../../../utils/paths.ts\";\nimport type {} from \"../../../keybindings.ts\";\nimport { loadTranscriptSnapshot } from \"./loader.ts\";\nimport { describeSession, pickerLabel, renderLine, sessionAge, viewerFooter } from \"./overlay-format.ts\";\nimport { sanitizeLine } from \"./text.ts\";\nimport { renderTranscript } from \"./transcript.ts\";\nimport type { SessionHudEntry, TranscriptSnapshot, ViewerEntryRange } from \"./types.ts\";\n\nconst MAX_VISIBLE_SESSIONS = 12;\n\ntype Mode = \"picker\" | \"viewer\";\ntype PickerAction = \"tui.select.up\" | \"tui.select.down\" | \"tui.select.confirm\" | \"tui.select.cancel\";\n\nconst PICKER_ACTIONS: readonly PickerAction[] = [\n\t\"tui.select.up\",\n\t\"tui.select.down\",\n\t\"tui.select.confirm\",\n\t\"tui.select.cancel\",\n];\n\ninterface SessionHudOverlayOptions {\n\treadonly sessions: readonly SessionHudEntry[];\n\treadonly done: () => void;\n\treadonly requestRender: () => void;\n}\n\nexport class SessionHudOverlay extends Container implements Focusable {\n\tprivate readonly options: SessionHudOverlayOptions;\n\tprivate readonly sessionsByValue = new Map<string, SessionHudEntry>();\n\tprivate readonly topBorder = new DynamicBorder((text) => theme.fg(\"accent\", text));\n\tprivate readonly middleBorder = new DynamicBorder((text) => theme.fg(\"accent\", text));\n\tprivate readonly bottomBorder = new DynamicBorder((text) => theme.fg(\"accent\", text));\n\tprivate list: SelectList | undefined;\n\tprivate mode: Mode = \"picker\";\n\tprivate selectedSession: SessionHudEntry | undefined;\n\tprivate snapshot: TranscriptSnapshot | undefined;\n\tprivate renderedLines: readonly string[] = [];\n\tprivate ranges: readonly ViewerEntryRange[] = [];\n\tprivate selectedEntryIndex = -1;\n\tprivate shouldSelectLastOnLoad = false;\n\tprivate expandedEntries = new Set<number>();\n\tprivate scrollOffset = 0;\n\tprivate viewportHeight = 12;\n\tprivate loadingText: string | undefined;\n\tprivate _focused = false;\n\n\tconstructor(options: SessionHudOverlayOptions) {\n\t\tsuper();\n\t\tthis.options = options;\n\t\tthis.rebuildPicker();\n\t}\n\n\tget focused(): boolean {\n\t\treturn this._focused;\n\t}\n\n\tset focused(value: boolean) {\n\t\tthis._focused = value;\n\t}\n\n\thandleInput(input: string): void {\n\t\tif (this.mode === \"picker\") {\n\t\t\tthis.handlePickerInput(input);\n\t\t\treturn;\n\t\t}\n\t\tthis.handleViewerInput(input);\n\t}\n\n\toverride render(width: number): string[] {\n\t\tif (this.mode === \"picker\") return super.render(width);\n\t\treturn this.renderViewer(width);\n\t}\n\n\tgetMode(): Mode {\n\t\treturn this.mode;\n\t}\n\n\tgetSelectedEntryIndex(): number {\n\t\treturn this.selectedEntryIndex;\n\t}\n\n\tgetExpandedEntryCount(): number {\n\t\treturn this.expandedEntries.size;\n\t}\n\n\tprivate handlePickerInput(input: string): void {\n\t\tconst keybindings = getKeybindings();\n\t\tif (PICKER_ACTIONS.some((action) => keybindings.matches(input, action))) {\n\t\t\tthis.list?.handleInput(input);\n\t\t}\n\t}\n\n\tprivate rebuildPicker(): void {\n\t\tthis.sessionsByValue.clear();\n\t\tconst items = this.options.sessions.map((session, index) => this.toPickerItem(session, index));\n\t\tconst list = new SelectList(items, Math.min(MAX_VISIBLE_SESSIONS, Math.max(1, items.length)), {\n\t\t\tselectedPrefix: (text) => theme.fg(\"accent\", text),\n\t\t\tselectedText: (text) => text,\n\t\t\tdescription: (text) => theme.fg(\"muted\", text),\n\t\t\tscrollInfo: (text) => theme.fg(\"dim\", text),\n\t\t\tnoMatch: (text) => theme.fg(\"warning\", text.replace(\"commands\", \"sessions\")),\n\t\t});\n\t\tlist.onSelect = (item) => {\n\t\t\tconst session = this.sessionsByValue.get(item.value);\n\t\t\tif (session) void this.openSession(session);\n\t\t};\n\t\tlist.onCancel = () => this.options.done();\n\t\tthis.list = list;\n\t\tthis.clear();\n\t\tthis.addChild(new Spacer(1));\n\t\tthis.addChild(this.topBorder);\n\t\tthis.addChild(new Spacer(1));\n\t\tthis.addChild(\n\t\t\tnew Text(\n\t\t\t\t`${theme.bold(theme.fg(\"accent\", \" Sessions\"))}${theme.fg(\"dim\", ` ${this.options.sessions.length} sessions`)}`,\n\t\t\t\t0,\n\t\t\t\t0,\n\t\t\t),\n\t\t);\n\t\tthis.addChild(new Spacer(1));\n\t\tthis.addChild(list);\n\t\tthis.addChild(new Spacer(1));\n\t\tthis.addChild(\n\t\t\tnew TruncatedText(`${keyHint(\"tui.select.confirm\", \"view\")} ${keyHint(\"tui.select.cancel\", \"close\")}`, 0, 0),\n\t\t);\n\t\tthis.addChild(new Spacer(1));\n\t\tthis.addChild(this.bottomBorder);\n\t}\n\n\tprivate toPickerItem(session: SessionHudEntry, index: number): SelectItem {\n\t\tconst value = String(index);\n\t\tthis.sessionsByValue.set(value, session);\n\t\treturn { value, label: pickerLabel(session), description: describeSession(session) };\n\t}\n\n\tprivate async openSession(session: SessionHudEntry): Promise<void> {\n\t\tthis.mode = \"viewer\";\n\t\tthis.selectedSession = session;\n\t\tthis.snapshot = undefined;\n\t\tthis.loadingText = \"Loading session transcript...\";\n\t\tthis.resetViewerState();\n\t\tthis.options.requestRender();\n\t\ttry {\n\t\t\tthis.snapshot = await loadTranscriptSnapshot(session.path);\n\t\t\tthis.loadingText = undefined;\n\t\t} catch (error) {\n\t\t\tthis.loadingText = `Failed to read session: ${error instanceof Error ? error.message : String(error)}`;\n\t\t}\n\t\tthis.rebuildTranscript(process.stdout.columns || 80);\n\t\tthis.options.requestRender();\n\t}\n\n\tprivate resetViewerState(): void {\n\t\tthis.expandedEntries = new Set<number>();\n\t\tthis.scrollOffset = 0;\n\t\tthis.selectedEntryIndex = -1;\n\t\tthis.shouldSelectLastOnLoad = true;\n\t\tthis.renderedLines = [];\n\t\tthis.ranges = [];\n\t}\n\n\tprivate rebuildTranscript(width: number): void {\n\t\tif (!this.snapshot) return;\n\t\tconst rendered = renderTranscript(this.snapshot.entries, {\n\t\t\twidth,\n\t\t\tselectedIndex: this.selectedEntryIndex,\n\t\t\texpandedEntries: this.expandedEntries,\n\t\t\tmarkdownTheme: getMarkdownTheme(),\n\t\t});\n\t\tthis.renderedLines = rendered.lines;\n\t\tthis.ranges = rendered.ranges;\n\t\tif (this.ranges.length > 0 && this.shouldSelectLastOnLoad) {\n\t\t\tthis.shouldSelectLastOnLoad = false;\n\t\t\tthis.selectedEntryIndex = this.ranges.length - 1;\n\t\t\tthis.rebuildTranscript(width);\n\t\t}\n\t\tthis.scrollToSelected();\n\t}\n\n\tprivate renderViewer(width: number): string[] {\n\t\tthis.viewportHeight = Math.max(5, (process.stdout.rows || 32) - 8);\n\t\tthis.rebuildTranscript(width);\n\t\tconst session = this.selectedSession;\n\t\tconst title = session ? `Sessions > ${shortenPath(session.cwd) || \"unknown\"} · ${session.shortId}` : \"Sessions\";\n\t\tconst status = session\n\t\t\t? `${session.messageCount} messages · ${sessionAge(session)}${this.snapshot?.model ? ` · ${this.snapshot.model}` : \"\"}`\n\t\t\t: \"\";\n\t\tconst maxScroll = Math.max(0, this.renderedLines.length - this.viewportHeight);\n\t\tthis.scrollOffset = Math.max(0, Math.min(this.scrollOffset, maxScroll));\n\t\tconst content = this.loadingText ? [theme.fg(\"dim\", this.loadingText)] : this.renderedLines;\n\t\tconst visible = content.slice(this.scrollOffset, this.scrollOffset + this.viewportHeight);\n\t\tconst lines: string[] = [];\n\t\tlines.push(...this.topBorder.render(width));\n\t\tlines.push(renderLine(` ${theme.bold(theme.fg(\"accent\", title))}`, width));\n\t\tif (status) lines.push(renderLine(` ${theme.fg(\"dim\", status)}`, width));\n\t\tlines.push(...this.middleBorder.render(width));\n\t\tfor (const line of visible) lines.push(` ${sanitizeLine(line, width - 2)}`);\n\t\tfor (let index = visible.length; index < this.viewportHeight; index += 1) lines.push(\"\");\n\t\tconst scroll =\n\t\t\tcontent.length > this.viewportHeight\n\t\t\t\t? ` [${this.scrollOffset + 1}-${Math.min(this.scrollOffset + this.viewportHeight, content.length)}/${content.length}]`\n\t\t\t\t: \"\";\n\t\tlines.push(renderLine(` ${viewerFooter(scroll)}`, width));\n\t\tlines.push(...this.bottomBorder.render(width));\n\t\treturn lines;\n\t}\n\n\tprivate handleViewerInput(input: string): void {\n\t\tconst keybindings = getKeybindings();\n\t\tif (keybindings.matches(input, \"app.sessions.observe\")) {\n\t\t\tthis.options.done();\n\t\t\treturn;\n\t\t} else if (keybindings.matches(input, \"tui.select.cancel\")) this.backToPicker();\n\t\telse if (input === \"j\" || keybindings.matches(input, \"tui.select.down\")) this.moveSelection(1);\n\t\telse if (input === \"k\" || keybindings.matches(input, \"tui.select.up\")) this.moveSelection(-1);\n\t\telse if (keybindings.matches(input, \"tui.select.pageDown\")) this.moveSelection(5);\n\t\telse if (keybindings.matches(input, \"tui.select.pageUp\")) this.moveSelection(-5);\n\t\telse if (input === \"g\") this.jumpTo(0);\n\t\telse if (input === \"G\") this.jumpTo(this.ranges.length - 1);\n\t\telse if (keybindings.matches(input, \"tui.select.confirm\")) this.toggleExpanded();\n\t\tthis.options.requestRender();\n\t}\n\n\tprivate backToPicker(): void {\n\t\tthis.mode = \"picker\";\n\t\tthis.rebuildPicker();\n\t}\n\n\tprivate moveSelection(delta: number): void {\n\t\tif (this.ranges.length === 0) return;\n\t\tthis.selectedEntryIndex = Math.max(0, Math.min(this.selectedEntryIndex + delta, this.ranges.length - 1));\n\t\tthis.scrollToSelected();\n\t}\n\n\tprivate jumpTo(index: number): void {\n\t\tif (this.ranges.length === 0) return;\n\t\tthis.selectedEntryIndex = Math.max(0, Math.min(index, this.ranges.length - 1));\n\t\tthis.scrollToSelected();\n\t}\n\n\tprivate toggleExpanded(): void {\n\t\tif (this.ranges.length === 0) return;\n\t\tif (this.expandedEntries.has(this.selectedEntryIndex)) this.expandedEntries.delete(this.selectedEntryIndex);\n\t\telse this.expandedEntries.add(this.selectedEntryIndex);\n\t}\n\n\tprivate scrollToSelected(): void {\n\t\tconst selected = this.ranges[this.selectedEntryIndex];\n\t\tif (!selected) return;\n\t\tconst bottom = selected.lineStart + selected.lineCount;\n\t\tif (selected.lineStart < this.scrollOffset) this.scrollOffset = Math.max(0, selected.lineStart - 1);\n\t\tif (bottom > this.scrollOffset + this.viewportHeight)\n\t\t\tthis.scrollOffset = Math.max(0, bottom - this.viewportHeight + 1);\n\t}\n}\n"]}
|
|
1
|
+
{"version":3,"file":"overlay.js","sourceRoot":"","sources":["../../../../../src/core/extensions/builtin/session-observer/overlay.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n\tContainer,\n\ttype Focusable,\n\tgetKeybindings,\n\ttype SelectItem,\n\tSelectList,\n\tSpacer,\n\tText,\n\tTruncatedText,\n} from \"@earendil-works/pi-tui\";\nimport { DynamicBorder } from \"../../../../modes/interactive/components/dynamic-border.ts\";\nimport { keyHint } from \"../../../../modes/interactive/components/keybinding-hints.ts\";\nimport { getMarkdownTheme, theme } from \"../../../../modes/interactive/theme/theme.ts\";\nimport { shortenPath } from \"../../../../utils/paths.ts\";\nimport type {} from \"../../../keybindings.ts\";\nimport { loadTranscriptSnapshot } from \"./loader.ts\";\nimport { describeSession, pickerLabel, renderLine, sessionAge, viewerFooter } from \"./overlay-format.ts\";\nimport { sanitizeLine } from \"./text.ts\";\nimport { renderTranscript } from \"./transcript.ts\";\nimport type { SessionHudEntry, TranscriptSnapshot, ViewerEntryRange } from \"./types.ts\";\n\nconst MAX_VISIBLE_SESSIONS = 12;\n\ntype Mode = \"picker\" | \"viewer\";\ntype PickerAction = \"tui.select.up\" | \"tui.select.down\" | \"tui.select.confirm\" | \"tui.select.cancel\";\n\nconst PICKER_ACTIONS: readonly PickerAction[] = [\n\t\"tui.select.up\",\n\t\"tui.select.down\",\n\t\"tui.select.confirm\",\n\t\"tui.select.cancel\",\n];\n\ninterface SessionHudOverlayOptions {\n\treadonly sessions: readonly SessionHudEntry[];\n\treadonly done: () => void;\n\treadonly requestRender: () => void;\n}\n\nexport class SessionHudOverlay extends Container implements Focusable {\n\tprivate readonly options: SessionHudOverlayOptions;\n\tprivate readonly sessionsByValue = new Map<string, SessionHudEntry>();\n\tprivate readonly topBorder = new DynamicBorder((text) => theme.fg(\"accent\", text));\n\tprivate readonly middleBorder = new DynamicBorder((text) => theme.fg(\"accent\", text));\n\tprivate readonly bottomBorder = new DynamicBorder((text) => theme.fg(\"accent\", text));\n\tprivate list: SelectList | undefined;\n\tprivate mode: Mode = \"picker\";\n\tprivate selectedSession: SessionHudEntry | undefined;\n\tprivate snapshot: TranscriptSnapshot | undefined;\n\tprivate renderedLines: readonly string[] = [];\n\tprivate ranges: readonly ViewerEntryRange[] = [];\n\tprivate selectedEntryIndex = -1;\n\tprivate shouldSelectLastOnLoad = false;\n\tprivate expandedEntries = new Set<number>();\n\tprivate scrollOffset = 0;\n\tprivate viewportHeight = 12;\n\tprivate loadingText: string | undefined;\n\tprivate _focused = false;\n\n\tconstructor(options: SessionHudOverlayOptions) {\n\t\tsuper();\n\t\tthis.options = options;\n\t\tthis.rebuildPicker();\n\t}\n\n\tget focused(): boolean {\n\t\treturn this._focused;\n\t}\n\n\tset focused(value: boolean) {\n\t\tthis._focused = value;\n\t}\n\n\thandleInput(input: string): void {\n\t\tif (this.mode === \"picker\") {\n\t\t\tthis.handlePickerInput(input);\n\t\t\treturn;\n\t\t}\n\t\tthis.handleViewerInput(input);\n\t}\n\n\toverride render(width: number): string[] {\n\t\tif (this.mode === \"picker\") return super.render(width);\n\t\treturn this.renderViewer(width);\n\t}\n\n\tgetMode(): Mode {\n\t\treturn this.mode;\n\t}\n\n\tgetSelectedEntryIndex(): number {\n\t\treturn this.selectedEntryIndex;\n\t}\n\n\tgetExpandedEntryCount(): number {\n\t\treturn this.expandedEntries.size;\n\t}\n\n\tprivate handlePickerInput(input: string): void {\n\t\tconst keybindings = getKeybindings();\n\t\tif (PICKER_ACTIONS.some((action) => keybindings.matches(input, action))) {\n\t\t\tthis.list?.handleInput(input);\n\t\t}\n\t}\n\n\tprivate rebuildPicker(): void {\n\t\tthis.sessionsByValue.clear();\n\t\tconst items = this.options.sessions.map((session, index) => this.toPickerItem(session, index));\n\t\tconst list = new SelectList(items, Math.min(MAX_VISIBLE_SESSIONS, Math.max(1, items.length)), {\n\t\t\tselectedPrefix: (text) => theme.fg(\"accent\", text),\n\t\t\tselectedText: (text) => text,\n\t\t\tdescription: (text) => theme.fg(\"muted\", text),\n\t\t\tscrollInfo: (text) => theme.fg(\"dim\", text),\n\t\t\tnoMatch: (text) => theme.fg(\"warning\", text.replace(\"commands\", \"sessions\")),\n\t\t});\n\t\tlist.onSelect = (item) => {\n\t\t\tconst session = this.sessionsByValue.get(item.value);\n\t\t\tif (session) void this.openSession(session);\n\t\t};\n\t\tlist.onCancel = () => this.options.done();\n\t\tthis.list = list;\n\t\tthis.clear();\n\t\tthis.addChild(\n\t\t\tnew Text(\n\t\t\t\t`${theme.bold(theme.fg(\"accent\", \" Sessions\"))}${theme.fg(\"dim\", ` ${this.options.sessions.length} sessions`)}`,\n\t\t\t\t0,\n\t\t\t\t0,\n\t\t\t),\n\t\t);\n\t\tthis.addChild(new Spacer(1));\n\t\tthis.addChild(list);\n\t\tthis.addChild(new Spacer(1));\n\t\tthis.addChild(\n\t\t\tnew TruncatedText(`${keyHint(\"tui.select.confirm\", \"view\")} ${keyHint(\"tui.select.cancel\", \"close\")}`, 0, 0),\n\t\t);\n\t}\n\n\tprivate toPickerItem(session: SessionHudEntry, index: number): SelectItem {\n\t\tconst value = String(index);\n\t\tthis.sessionsByValue.set(value, session);\n\t\treturn { value, label: pickerLabel(session), description: describeSession(session) };\n\t}\n\n\tprivate async openSession(session: SessionHudEntry): Promise<void> {\n\t\tthis.mode = \"viewer\";\n\t\tthis.selectedSession = session;\n\t\tthis.snapshot = undefined;\n\t\tthis.loadingText = \"Loading session transcript...\";\n\t\tthis.resetViewerState();\n\t\tthis.options.requestRender();\n\t\ttry {\n\t\t\tthis.snapshot = await loadTranscriptSnapshot(session.path);\n\t\t\tthis.loadingText = undefined;\n\t\t} catch (error) {\n\t\t\tthis.loadingText = `Failed to read session: ${error instanceof Error ? error.message : String(error)}`;\n\t\t}\n\t\tthis.rebuildTranscript(process.stdout.columns || 80);\n\t\tthis.options.requestRender();\n\t}\n\n\tprivate resetViewerState(): void {\n\t\tthis.expandedEntries = new Set<number>();\n\t\tthis.scrollOffset = 0;\n\t\tthis.selectedEntryIndex = -1;\n\t\tthis.shouldSelectLastOnLoad = true;\n\t\tthis.renderedLines = [];\n\t\tthis.ranges = [];\n\t}\n\n\tprivate rebuildTranscript(width: number): void {\n\t\tif (!this.snapshot) return;\n\t\tconst rendered = renderTranscript(this.snapshot.entries, {\n\t\t\twidth,\n\t\t\tselectedIndex: this.selectedEntryIndex,\n\t\t\texpandedEntries: this.expandedEntries,\n\t\t\tmarkdownTheme: getMarkdownTheme(),\n\t\t});\n\t\tthis.renderedLines = rendered.lines;\n\t\tthis.ranges = rendered.ranges;\n\t\tif (this.ranges.length > 0 && this.shouldSelectLastOnLoad) {\n\t\t\tthis.shouldSelectLastOnLoad = false;\n\t\t\tthis.selectedEntryIndex = this.ranges.length - 1;\n\t\t\tthis.rebuildTranscript(width);\n\t\t}\n\t\tthis.scrollToSelected();\n\t}\n\n\tprivate renderViewer(width: number): string[] {\n\t\tthis.viewportHeight = Math.max(5, (process.stdout.rows || 32) - 8);\n\t\tthis.rebuildTranscript(width);\n\t\tconst session = this.selectedSession;\n\t\tconst title = session ? `Sessions > ${shortenPath(session.cwd) || \"unknown\"} · ${session.shortId}` : \"Sessions\";\n\t\tconst status = session\n\t\t\t? `${session.messageCount} messages · ${sessionAge(session)}${this.snapshot?.model ? ` · ${this.snapshot.model}` : \"\"}`\n\t\t\t: \"\";\n\t\tconst maxScroll = Math.max(0, this.renderedLines.length - this.viewportHeight);\n\t\tthis.scrollOffset = Math.max(0, Math.min(this.scrollOffset, maxScroll));\n\t\tconst content = this.loadingText ? [theme.fg(\"dim\", this.loadingText)] : this.renderedLines;\n\t\tconst visible = content.slice(this.scrollOffset, this.scrollOffset + this.viewportHeight);\n\t\tconst lines: string[] = [];\n\t\tlines.push(...this.topBorder.render(width));\n\t\tlines.push(renderLine(` ${theme.bold(theme.fg(\"accent\", title))}`, width));\n\t\tif (status) lines.push(renderLine(` ${theme.fg(\"dim\", status)}`, width));\n\t\tlines.push(...this.middleBorder.render(width));\n\t\tfor (const line of visible) lines.push(` ${sanitizeLine(line, width - 2)}`);\n\t\tfor (let index = visible.length; index < this.viewportHeight; index += 1) lines.push(\"\");\n\t\tconst scroll =\n\t\t\tcontent.length > this.viewportHeight\n\t\t\t\t? ` [${this.scrollOffset + 1}-${Math.min(this.scrollOffset + this.viewportHeight, content.length)}/${content.length}]`\n\t\t\t\t: \"\";\n\t\tlines.push(renderLine(` ${viewerFooter(scroll)}`, width));\n\t\tlines.push(...this.bottomBorder.render(width));\n\t\treturn lines;\n\t}\n\n\tprivate handleViewerInput(input: string): void {\n\t\tconst keybindings = getKeybindings();\n\t\tif (keybindings.matches(input, \"app.sessions.observe\")) {\n\t\t\tthis.options.done();\n\t\t\treturn;\n\t\t} else if (keybindings.matches(input, \"tui.select.cancel\")) this.backToPicker();\n\t\telse if (input === \"j\" || keybindings.matches(input, \"tui.select.down\")) this.moveSelection(1);\n\t\telse if (input === \"k\" || keybindings.matches(input, \"tui.select.up\")) this.moveSelection(-1);\n\t\telse if (keybindings.matches(input, \"tui.select.pageDown\")) this.moveSelection(5);\n\t\telse if (keybindings.matches(input, \"tui.select.pageUp\")) this.moveSelection(-5);\n\t\telse if (input === \"g\") this.jumpTo(0);\n\t\telse if (input === \"G\") this.jumpTo(this.ranges.length - 1);\n\t\telse if (keybindings.matches(input, \"tui.select.confirm\")) this.toggleExpanded();\n\t\tthis.options.requestRender();\n\t}\n\n\tprivate backToPicker(): void {\n\t\tthis.mode = \"picker\";\n\t\tthis.rebuildPicker();\n\t}\n\n\tprivate moveSelection(delta: number): void {\n\t\tif (this.ranges.length === 0) return;\n\t\tthis.selectedEntryIndex = Math.max(0, Math.min(this.selectedEntryIndex + delta, this.ranges.length - 1));\n\t\tthis.scrollToSelected();\n\t}\n\n\tprivate jumpTo(index: number): void {\n\t\tif (this.ranges.length === 0) return;\n\t\tthis.selectedEntryIndex = Math.max(0, Math.min(index, this.ranges.length - 1));\n\t\tthis.scrollToSelected();\n\t}\n\n\tprivate toggleExpanded(): void {\n\t\tif (this.ranges.length === 0) return;\n\t\tif (this.expandedEntries.has(this.selectedEntryIndex)) this.expandedEntries.delete(this.selectedEntryIndex);\n\t\telse this.expandedEntries.add(this.selectedEntryIndex);\n\t}\n\n\tprivate scrollToSelected(): void {\n\t\tconst selected = this.ranges[this.selectedEntryIndex];\n\t\tif (!selected) return;\n\t\tconst bottom = selected.lineStart + selected.lineCount;\n\t\tif (selected.lineStart < this.scrollOffset) this.scrollOffset = Math.max(0, selected.lineStart - 1);\n\t\tif (bottom > this.scrollOffset + this.viewportHeight)\n\t\t\tthis.scrollOffset = Math.max(0, bottom - this.viewportHeight + 1);\n\t}\n}\n"]}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"scanner.d.ts","sourceRoot":"","sources":["../../../../../src/core/extensions/builtin/session-observer/scanner.ts"],"names":[],"mappings":"AAOA,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,YAAY,CAAC;AAElD,UAAU,QAAQ;IACjB,OAAO,CAAC,IAAI,EAAE,MAAM,GAAG,MAAM,CAAC;IAC9B,QAAQ,CAAC,IAAI,EAAE,MAAM,EAAE,EAAE,EAAE,MAAM,GAAG,MAAM,CAAC;IAC3C,UAAU,CAAC,IAAI,EAAE,MAAM,GAAG,OAAO,CAAC;CAClC;AAED,wBAAgB,qBAAqB,CACpC,iBAAiB,EAAE,MAAM,EACzB,mBAAmB,GAAE,MAAyB,EAC9C,QAAQ,GAAE,QAA4C,GACpD,MAAM,CAQR;
|
|
1
|
+
{"version":3,"file":"scanner.d.ts","sourceRoot":"","sources":["../../../../../src/core/extensions/builtin/session-observer/scanner.ts"],"names":[],"mappings":"AAOA,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,YAAY,CAAC;AAElD,UAAU,QAAQ;IACjB,OAAO,CAAC,IAAI,EAAE,MAAM,GAAG,MAAM,CAAC;IAC9B,QAAQ,CAAC,IAAI,EAAE,MAAM,EAAE,EAAE,EAAE,MAAM,GAAG,MAAM,CAAC;IAC3C,UAAU,CAAC,IAAI,EAAE,MAAM,GAAG,OAAO,CAAC;CAClC;AAED,wBAAgB,qBAAqB,CACpC,iBAAiB,EAAE,MAAM,EACzB,mBAAmB,GAAE,MAAyB,EAC9C,QAAQ,GAAE,QAA4C,GACpD,MAAM,CAQR;AAuGD,wBAAsB,qBAAqB,CAC1C,IAAI,EAAE,MAAM,EACZ,kBAAkB,CAAC,EAAE,MAAM,GACzB,OAAO,CAAC,SAAS,eAAe,EAAE,CAAC,CAarC","sourcesContent":["import type { Stats } from \"node:fs\";\nimport { readdir, readFile, stat } from \"node:fs/promises\";\nimport { basename, isAbsolute, join, relative, resolve } from \"node:path\";\nimport { getSessionsDir } from \"../../../../config.ts\";\nimport type { FileEntry, SessionHeader } from \"../../../session-manager.ts\";\nimport { parseSessionEntries } from \"../../../session-manager.ts\";\nimport { compactWhitespace, getTextContent } from \"./text.ts\";\nimport type { SessionHudEntry } from \"./types.ts\";\n\ninterface PathLike {\n\tresolve(path: string): string;\n\trelative(from: string, to: string): string;\n\tisAbsolute(path: string): boolean;\n}\n\nexport function resolveSessionHudRoot(\n\tcurrentSessionDir: string,\n\tdefaultSessionsRoot: string = getSessionsDir(),\n\tpathImpl: PathLike = { resolve, relative, isAbsolute },\n): string {\n\tconst defaultRoot = pathImpl.resolve(defaultSessionsRoot);\n\tif (!currentSessionDir) return defaultRoot;\n\tconst current = pathImpl.resolve(currentSessionDir);\n\tif (current === defaultRoot) return defaultRoot;\n\tconst rel = pathImpl.relative(defaultRoot, current);\n\tif (rel && !rel.startsWith(\"..\") && !pathImpl.isAbsolute(rel)) return defaultRoot;\n\treturn current;\n}\n\nfunction hasErrorCode(error: unknown, code: string): boolean {\n\treturn error instanceof Error && \"code\" in error && error.code === code;\n}\n\nasync function readDirIfExists(path: string): Promise<readonly string[]> {\n\ttry {\n\t\treturn await readdir(path);\n\t} catch (error) {\n\t\tif (hasErrorCode(error, \"ENOENT\")) return [];\n\t\tthrow error;\n\t}\n}\n\nasync function statIfFile(path: string): Promise<Stats | undefined> {\n\ttry {\n\t\tconst fileStat = await stat(path);\n\t\treturn fileStat.isFile() ? fileStat : undefined;\n\t} catch (error) {\n\t\tif (hasErrorCode(error, \"ENOENT\")) return undefined;\n\t\tthrow error;\n\t}\n}\n\nasync function collectFilesInDir(dir: string): Promise<readonly string[]> {\n\tconst names = await readDirIfExists(dir);\n\tconst files: string[] = [];\n\tfor (const name of names) {\n\t\tif (!name.endsWith(\".jsonl\")) continue;\n\t\tconst file = join(dir, name);\n\t\tif (await statIfFile(file)) files.push(file);\n\t}\n\treturn files;\n}\n\nasync function discoverSessionFiles(root: string): Promise<readonly string[]> {\n\tconst names = await readDirIfExists(root);\n\tconst files: string[] = [...(await collectFilesInDir(root))];\n\tfor (const name of names) {\n\t\tif (name.endsWith(\".jsonl\")) continue;\n\t\tconst dir = join(root, name);\n\t\ttry {\n\t\t\tconst dirStat = await stat(dir);\n\t\t\tif (dirStat.isDirectory()) files.push(...(await collectFilesInDir(dir)));\n\t\t} catch (error) {\n\t\t\tif (!hasErrorCode(error, \"ENOENT\")) throw error;\n\t\t}\n\t}\n\treturn files;\n}\n\nfunction firstHeader(entries: readonly FileEntry[], filePath: string): SessionHeader | undefined {\n\tconst header = entries[0];\n\tif (header?.type === \"session\") return header;\n\tif (entries.length === 0) return undefined;\n\treturn { type: \"session\", id: basename(filePath, \".jsonl\"), timestamp: new Date(0).toISOString(), cwd: \"\" };\n}\n\nfunction lastUserText(entries: readonly FileEntry[]): string {\n\tfor (let index = entries.length - 1; index >= 0; index -= 1) {\n\t\tconst entry = entries[index];\n\t\tif (entry?.type !== \"message\") continue;\n\t\tconst message = entry.message;\n\t\tif (message.role !== \"user\") continue;\n\t\tconst text = compactWhitespace(getTextContent(message.content));\n\t\tif (text) return text;\n\t}\n\treturn \"(no user prompt)\";\n}\n\nfunction latestMessageTimestamp(entries: readonly FileEntry[], fallback: number): number {\n\tfor (let index = entries.length - 1; index >= 0; index -= 1) {\n\t\tconst entry = entries[index];\n\t\tif (!entry || entry.type === \"session\") continue;\n\t\tconst timestamp = Date.parse(entry.timestamp);\n\t\tif (Number.isFinite(timestamp)) return timestamp;\n\t}\n\treturn fallback;\n}\n\nasync function summarizeSession(\n\tfilePath: string,\n\tcurrentSessionFile: string | undefined,\n): Promise<SessionHudEntry | undefined> {\n\tconst [content, fileStat] = await Promise.all([readFile(filePath, \"utf-8\"), stat(filePath)]);\n\tconst entries = parseSessionEntries(content);\n\tconst header = firstHeader(entries, filePath);\n\tif (!header) return undefined;\n\tconst messageCount = entries.filter((entry) => entry.type === \"message\").length;\n\treturn {\n\t\tid: header.id,\n\t\tshortId: header.id.length <= 8 ? header.id : header.id.slice(0, 8),\n\t\tpath: filePath,\n\t\tcwd: header.cwd,\n\t\tcreatedAt: Date.parse(header.timestamp),\n\t\tmodifiedAt: latestMessageTimestamp(entries, fileStat.mtime.getTime()),\n\t\tmessageCount,\n\t\tlastUserText: lastUserText(entries),\n\t\tisCurrent: currentSessionFile === filePath,\n\t};\n}\n\nexport async function scanSessionHudEntries(\n\troot: string,\n\tcurrentSessionFile?: string,\n): Promise<readonly SessionHudEntry[]> {\n\tconst files = await discoverSessionFiles(root);\n\tconst sessions: SessionHudEntry[] = [];\n\tfor (const file of files) {\n\t\ttry {\n\t\t\tconst session = await summarizeSession(file, currentSessionFile);\n\t\t\tif (session) sessions.push(session);\n\t\t} catch (error) {\n\t\t\tif (!hasErrorCode(error, \"ENOENT\")) throw error;\n\t\t}\n\t}\n\tsessions.sort((left, right) => right.modifiedAt - left.modifiedAt);\n\treturn sessions;\n}\n"]}
|
|
@@ -74,6 +74,8 @@ function firstHeader(entries, filePath) {
|
|
|
74
74
|
const header = entries[0];
|
|
75
75
|
if (header?.type === "session")
|
|
76
76
|
return header;
|
|
77
|
+
if (entries.length === 0)
|
|
78
|
+
return undefined;
|
|
77
79
|
return { type: "session", id: basename(filePath, ".jsonl"), timestamp: new Date(0).toISOString(), cwd: "" };
|
|
78
80
|
}
|
|
79
81
|
function lastUserText(entries) {
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"scanner.js","sourceRoot":"","sources":["../../../../../src/core/extensions/builtin/session-observer/scanner.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,OAAO,EAAE,QAAQ,EAAE,IAAI,EAAE,MAAM,kBAAkB,CAAC;AAC3D,OAAO,EAAE,QAAQ,EAAE,UAAU,EAAE,IAAI,EAAE,QAAQ,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AAC1E,OAAO,EAAE,cAAc,EAAE,MAAM,uBAAuB,CAAC;AAEvD,OAAO,EAAE,mBAAmB,EAAE,MAAM,6BAA6B,CAAC;AAClE,OAAO,EAAE,iBAAiB,EAAE,cAAc,EAAE,MAAM,WAAW,CAAC;AAS9D,MAAM,UAAU,qBAAqB,CACpC,iBAAyB,EACzB,mBAAmB,GAAW,cAAc,EAAE,EAC9C,QAAQ,GAAa,EAAE,OAAO,EAAE,QAAQ,EAAE,UAAU,EAAE,EAC7C;IACT,MAAM,WAAW,GAAG,QAAQ,CAAC,OAAO,CAAC,mBAAmB,CAAC,CAAC;IAC1D,IAAI,CAAC,iBAAiB;QAAE,OAAO,WAAW,CAAC;IAC3C,MAAM,OAAO,GAAG,QAAQ,CAAC,OAAO,CAAC,iBAAiB,CAAC,CAAC;IACpD,IAAI,OAAO,KAAK,WAAW;QAAE,OAAO,WAAW,CAAC;IAChD,MAAM,GAAG,GAAG,QAAQ,CAAC,QAAQ,CAAC,WAAW,EAAE,OAAO,CAAC,CAAC;IACpD,IAAI,GAAG,IAAI,CAAC,GAAG,CAAC,UAAU,CAAC,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,UAAU,CAAC,GAAG,CAAC;QAAE,OAAO,WAAW,CAAC;IAClF,OAAO,OAAO,CAAC;AAAA,CACf;AAED,SAAS,YAAY,CAAC,KAAc,EAAE,IAAY,EAAW;IAC5D,OAAO,KAAK,YAAY,KAAK,IAAI,MAAM,IAAI,KAAK,IAAI,KAAK,CAAC,IAAI,KAAK,IAAI,CAAC;AAAA,CACxE;AAED,KAAK,UAAU,eAAe,CAAC,IAAY,EAA8B;IACxE,IAAI,CAAC;QACJ,OAAO,MAAM,OAAO,CAAC,IAAI,CAAC,CAAC;IAC5B,CAAC;IAAC,OAAO,KAAK,EAAE,CAAC;QAChB,IAAI,YAAY,CAAC,KAAK,EAAE,QAAQ,CAAC;YAAE,OAAO,EAAE,CAAC;QAC7C,MAAM,KAAK,CAAC;IACb,CAAC;AAAA,CACD;AAED,KAAK,UAAU,UAAU,CAAC,IAAY,EAA8B;IACnE,IAAI,CAAC;QACJ,MAAM,QAAQ,GAAG,MAAM,IAAI,CAAC,IAAI,CAAC,CAAC;QAClC,OAAO,QAAQ,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC,CAAC,SAAS,CAAC;IACjD,CAAC;IAAC,OAAO,KAAK,EAAE,CAAC;QAChB,IAAI,YAAY,CAAC,KAAK,EAAE,QAAQ,CAAC;YAAE,OAAO,SAAS,CAAC;QACpD,MAAM,KAAK,CAAC;IACb,CAAC;AAAA,CACD;AAED,KAAK,UAAU,iBAAiB,CAAC,GAAW,EAA8B;IACzE,MAAM,KAAK,GAAG,MAAM,eAAe,CAAC,GAAG,CAAC,CAAC;IACzC,MAAM,KAAK,GAAa,EAAE,CAAC;IAC3B,KAAK,MAAM,IAAI,IAAI,KAAK,EAAE,CAAC;QAC1B,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC;YAAE,SAAS;QACvC,MAAM,IAAI,GAAG,IAAI,CAAC,GAAG,EAAE,IAAI,CAAC,CAAC;QAC7B,IAAI,MAAM,UAAU,CAAC,IAAI,CAAC;YAAE,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;IAC9C,CAAC;IACD,OAAO,KAAK,CAAC;AAAA,CACb;AAED,KAAK,UAAU,oBAAoB,CAAC,IAAY,EAA8B;IAC7E,MAAM,KAAK,GAAG,MAAM,eAAe,CAAC,IAAI,CAAC,CAAC;IAC1C,MAAM,KAAK,GAAa,CAAC,GAAG,CAAC,MAAM,iBAAiB,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IAC7D,KAAK,MAAM,IAAI,IAAI,KAAK,EAAE,CAAC;QAC1B,IAAI,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC;YAAE,SAAS;QACtC,MAAM,GAAG,GAAG,IAAI,CAAC,IAAI,EAAE,IAAI,CAAC,CAAC;QAC7B,IAAI,CAAC;YACJ,MAAM,OAAO,GAAG,MAAM,IAAI,CAAC,GAAG,CAAC,CAAC;YAChC,IAAI,OAAO,CAAC,WAAW,EAAE;gBAAE,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,MAAM,iBAAiB,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC;QAC1E,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YAChB,IAAI,CAAC,YAAY,CAAC,KAAK,EAAE,QAAQ,CAAC;gBAAE,MAAM,KAAK,CAAC;QACjD,CAAC;IACF,CAAC;IACD,OAAO,KAAK,CAAC;AAAA,CACb;AAED,SAAS,WAAW,CAAC,OAA6B,EAAE,QAAgB,EAA6B;IAChG,MAAM,MAAM,GAAG,OAAO,CAAC,CAAC,CAAC,CAAC;IAC1B,IAAI,MAAM,EAAE,IAAI,KAAK,SAAS;QAAE,OAAO,MAAM,CAAC;IAC9C,OAAO,EAAE,IAAI,EAAE,SAAS,EAAE,EAAE,EAAE,QAAQ,CAAC,QAAQ,EAAE,QAAQ,CAAC,EAAE,SAAS,EAAE,IAAI,IAAI,CAAC,CAAC,CAAC,CAAC,WAAW,EAAE,EAAE,GAAG,EAAE,EAAE,EAAE,CAAC;AAAA,CAC5G;AAED,SAAS,YAAY,CAAC,OAA6B,EAAU;IAC5D,KAAK,IAAI,KAAK,GAAG,OAAO,CAAC,MAAM,GAAG,CAAC,EAAE,KAAK,IAAI,CAAC,EAAE,KAAK,IAAI,CAAC,EAAE,CAAC;QAC7D,MAAM,KAAK,GAAG,OAAO,CAAC,KAAK,CAAC,CAAC;QAC7B,IAAI,KAAK,EAAE,IAAI,KAAK,SAAS;YAAE,SAAS;QACxC,MAAM,OAAO,GAAG,KAAK,CAAC,OAAO,CAAC;QAC9B,IAAI,OAAO,CAAC,IAAI,KAAK,MAAM;YAAE,SAAS;QACtC,MAAM,IAAI,GAAG,iBAAiB,CAAC,cAAc,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC,CAAC;QAChE,IAAI,IAAI;YAAE,OAAO,IAAI,CAAC;IACvB,CAAC;IACD,OAAO,kBAAkB,CAAC;AAAA,CAC1B;AAED,SAAS,sBAAsB,CAAC,OAA6B,EAAE,QAAgB,EAAU;IACxF,KAAK,IAAI,KAAK,GAAG,OAAO,CAAC,MAAM,GAAG,CAAC,EAAE,KAAK,IAAI,CAAC,EAAE,KAAK,IAAI,CAAC,EAAE,CAAC;QAC7D,MAAM,KAAK,GAAG,OAAO,CAAC,KAAK,CAAC,CAAC;QAC7B,IAAI,CAAC,KAAK,IAAI,KAAK,CAAC,IAAI,KAAK,SAAS;YAAE,SAAS;QACjD,MAAM,SAAS,GAAG,IAAI,CAAC,KAAK,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC;QAC9C,IAAI,MAAM,CAAC,QAAQ,CAAC,SAAS,CAAC;YAAE,OAAO,SAAS,CAAC;IAClD,CAAC;IACD,OAAO,QAAQ,CAAC;AAAA,CAChB;AAED,KAAK,UAAU,gBAAgB,CAC9B,QAAgB,EAChB,kBAAsC,EACC;IACvC,MAAM,CAAC,OAAO,EAAE,QAAQ,CAAC,GAAG,MAAM,OAAO,CAAC,GAAG,CAAC,CAAC,QAAQ,CAAC,QAAQ,EAAE,OAAO,CAAC,EAAE,IAAI,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAC;IAC7F,MAAM,OAAO,GAAG,mBAAmB,CAAC,OAAO,CAAC,CAAC;IAC7C,MAAM,MAAM,GAAG,WAAW,CAAC,OAAO,EAAE,QAAQ,CAAC,CAAC;IAC9C,IAAI,CAAC,MAAM;QAAE,OAAO,SAAS,CAAC;IAC9B,MAAM,YAAY,GAAG,OAAO,CAAC,MAAM,CAAC,CAAC,KAAK,EAAE,EAAE,CAAC,KAAK,CAAC,IAAI,KAAK,SAAS,CAAC,CAAC,MAAM,CAAC;IAChF,OAAO;QACN,EAAE,EAAE,MAAM,CAAC,EAAE;QACb,OAAO,EAAE,MAAM,CAAC,EAAE,CAAC,MAAM,IAAI,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC,CAAC,MAAM,CAAC,EAAE,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC,CAAC;QAClE,IAAI,EAAE,QAAQ;QACd,GAAG,EAAE,MAAM,CAAC,GAAG;QACf,SAAS,EAAE,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,SAAS,CAAC;QACvC,UAAU,EAAE,sBAAsB,CAAC,OAAO,EAAE,QAAQ,CAAC,KAAK,CAAC,OAAO,EAAE,CAAC;QACrE,YAAY;QACZ,YAAY,EAAE,YAAY,CAAC,OAAO,CAAC;QACnC,SAAS,EAAE,kBAAkB,KAAK,QAAQ;KAC1C,CAAC;AAAA,CACF;AAED,MAAM,CAAC,KAAK,UAAU,qBAAqB,CAC1C,IAAY,EACZ,kBAA2B,EACW;IACtC,MAAM,KAAK,GAAG,MAAM,oBAAoB,CAAC,IAAI,CAAC,CAAC;IAC/C,MAAM,QAAQ,GAAsB,EAAE,CAAC;IACvC,KAAK,MAAM,IAAI,IAAI,KAAK,EAAE,CAAC;QAC1B,IAAI,CAAC;YACJ,MAAM,OAAO,GAAG,MAAM,gBAAgB,CAAC,IAAI,EAAE,kBAAkB,CAAC,CAAC;YACjE,IAAI,OAAO;gBAAE,QAAQ,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;QACrC,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YAChB,IAAI,CAAC,YAAY,CAAC,KAAK,EAAE,QAAQ,CAAC;gBAAE,MAAM,KAAK,CAAC;QACjD,CAAC;IACF,CAAC;IACD,QAAQ,CAAC,IAAI,CAAC,CAAC,IAAI,EAAE,KAAK,EAAE,EAAE,CAAC,KAAK,CAAC,UAAU,GAAG,IAAI,CAAC,UAAU,CAAC,CAAC;IACnE,OAAO,QAAQ,CAAC;AAAA,CAChB","sourcesContent":["import type { Stats } from \"node:fs\";\nimport { readdir, readFile, stat } from \"node:fs/promises\";\nimport { basename, isAbsolute, join, relative, resolve } from \"node:path\";\nimport { getSessionsDir } from \"../../../../config.ts\";\nimport type { FileEntry, SessionHeader } from \"../../../session-manager.ts\";\nimport { parseSessionEntries } from \"../../../session-manager.ts\";\nimport { compactWhitespace, getTextContent } from \"./text.ts\";\nimport type { SessionHudEntry } from \"./types.ts\";\n\ninterface PathLike {\n\tresolve(path: string): string;\n\trelative(from: string, to: string): string;\n\tisAbsolute(path: string): boolean;\n}\n\nexport function resolveSessionHudRoot(\n\tcurrentSessionDir: string,\n\tdefaultSessionsRoot: string = getSessionsDir(),\n\tpathImpl: PathLike = { resolve, relative, isAbsolute },\n): string {\n\tconst defaultRoot = pathImpl.resolve(defaultSessionsRoot);\n\tif (!currentSessionDir) return defaultRoot;\n\tconst current = pathImpl.resolve(currentSessionDir);\n\tif (current === defaultRoot) return defaultRoot;\n\tconst rel = pathImpl.relative(defaultRoot, current);\n\tif (rel && !rel.startsWith(\"..\") && !pathImpl.isAbsolute(rel)) return defaultRoot;\n\treturn current;\n}\n\nfunction hasErrorCode(error: unknown, code: string): boolean {\n\treturn error instanceof Error && \"code\" in error && error.code === code;\n}\n\nasync function readDirIfExists(path: string): Promise<readonly string[]> {\n\ttry {\n\t\treturn await readdir(path);\n\t} catch (error) {\n\t\tif (hasErrorCode(error, \"ENOENT\")) return [];\n\t\tthrow error;\n\t}\n}\n\nasync function statIfFile(path: string): Promise<Stats | undefined> {\n\ttry {\n\t\tconst fileStat = await stat(path);\n\t\treturn fileStat.isFile() ? fileStat : undefined;\n\t} catch (error) {\n\t\tif (hasErrorCode(error, \"ENOENT\")) return undefined;\n\t\tthrow error;\n\t}\n}\n\nasync function collectFilesInDir(dir: string): Promise<readonly string[]> {\n\tconst names = await readDirIfExists(dir);\n\tconst files: string[] = [];\n\tfor (const name of names) {\n\t\tif (!name.endsWith(\".jsonl\")) continue;\n\t\tconst file = join(dir, name);\n\t\tif (await statIfFile(file)) files.push(file);\n\t}\n\treturn files;\n}\n\nasync function discoverSessionFiles(root: string): Promise<readonly string[]> {\n\tconst names = await readDirIfExists(root);\n\tconst files: string[] = [...(await collectFilesInDir(root))];\n\tfor (const name of names) {\n\t\tif (name.endsWith(\".jsonl\")) continue;\n\t\tconst dir = join(root, name);\n\t\ttry {\n\t\t\tconst dirStat = await stat(dir);\n\t\t\tif (dirStat.isDirectory()) files.push(...(await collectFilesInDir(dir)));\n\t\t} catch (error) {\n\t\t\tif (!hasErrorCode(error, \"ENOENT\")) throw error;\n\t\t}\n\t}\n\treturn files;\n}\n\nfunction firstHeader(entries: readonly FileEntry[], filePath: string): SessionHeader | undefined {\n\tconst header = entries[0];\n\tif (header?.type === \"session\") return header;\n\treturn { type: \"session\", id: basename(filePath, \".jsonl\"), timestamp: new Date(0).toISOString(), cwd: \"\" };\n}\n\nfunction lastUserText(entries: readonly FileEntry[]): string {\n\tfor (let index = entries.length - 1; index >= 0; index -= 1) {\n\t\tconst entry = entries[index];\n\t\tif (entry?.type !== \"message\") continue;\n\t\tconst message = entry.message;\n\t\tif (message.role !== \"user\") continue;\n\t\tconst text = compactWhitespace(getTextContent(message.content));\n\t\tif (text) return text;\n\t}\n\treturn \"(no user prompt)\";\n}\n\nfunction latestMessageTimestamp(entries: readonly FileEntry[], fallback: number): number {\n\tfor (let index = entries.length - 1; index >= 0; index -= 1) {\n\t\tconst entry = entries[index];\n\t\tif (!entry || entry.type === \"session\") continue;\n\t\tconst timestamp = Date.parse(entry.timestamp);\n\t\tif (Number.isFinite(timestamp)) return timestamp;\n\t}\n\treturn fallback;\n}\n\nasync function summarizeSession(\n\tfilePath: string,\n\tcurrentSessionFile: string | undefined,\n): Promise<SessionHudEntry | undefined> {\n\tconst [content, fileStat] = await Promise.all([readFile(filePath, \"utf-8\"), stat(filePath)]);\n\tconst entries = parseSessionEntries(content);\n\tconst header = firstHeader(entries, filePath);\n\tif (!header) return undefined;\n\tconst messageCount = entries.filter((entry) => entry.type === \"message\").length;\n\treturn {\n\t\tid: header.id,\n\t\tshortId: header.id.length <= 8 ? header.id : header.id.slice(0, 8),\n\t\tpath: filePath,\n\t\tcwd: header.cwd,\n\t\tcreatedAt: Date.parse(header.timestamp),\n\t\tmodifiedAt: latestMessageTimestamp(entries, fileStat.mtime.getTime()),\n\t\tmessageCount,\n\t\tlastUserText: lastUserText(entries),\n\t\tisCurrent: currentSessionFile === filePath,\n\t};\n}\n\nexport async function scanSessionHudEntries(\n\troot: string,\n\tcurrentSessionFile?: string,\n): Promise<readonly SessionHudEntry[]> {\n\tconst files = await discoverSessionFiles(root);\n\tconst sessions: SessionHudEntry[] = [];\n\tfor (const file of files) {\n\t\ttry {\n\t\t\tconst session = await summarizeSession(file, currentSessionFile);\n\t\t\tif (session) sessions.push(session);\n\t\t} catch (error) {\n\t\t\tif (!hasErrorCode(error, \"ENOENT\")) throw error;\n\t\t}\n\t}\n\tsessions.sort((left, right) => right.modifiedAt - left.modifiedAt);\n\treturn sessions;\n}\n"]}
|
|
1
|
+
{"version":3,"file":"scanner.js","sourceRoot":"","sources":["../../../../../src/core/extensions/builtin/session-observer/scanner.ts"],"names":[],"mappings":"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","sourcesContent":["import type { Stats } from \"node:fs\";\nimport { readdir, readFile, stat } from \"node:fs/promises\";\nimport { basename, isAbsolute, join, relative, resolve } from \"node:path\";\nimport { getSessionsDir } from \"../../../../config.ts\";\nimport type { FileEntry, SessionHeader } from \"../../../session-manager.ts\";\nimport { parseSessionEntries } from \"../../../session-manager.ts\";\nimport { compactWhitespace, getTextContent } from \"./text.ts\";\nimport type { SessionHudEntry } from \"./types.ts\";\n\ninterface PathLike {\n\tresolve(path: string): string;\n\trelative(from: string, to: string): string;\n\tisAbsolute(path: string): boolean;\n}\n\nexport function resolveSessionHudRoot(\n\tcurrentSessionDir: string,\n\tdefaultSessionsRoot: string = getSessionsDir(),\n\tpathImpl: PathLike = { resolve, relative, isAbsolute },\n): string {\n\tconst defaultRoot = pathImpl.resolve(defaultSessionsRoot);\n\tif (!currentSessionDir) return defaultRoot;\n\tconst current = pathImpl.resolve(currentSessionDir);\n\tif (current === defaultRoot) return defaultRoot;\n\tconst rel = pathImpl.relative(defaultRoot, current);\n\tif (rel && !rel.startsWith(\"..\") && !pathImpl.isAbsolute(rel)) return defaultRoot;\n\treturn current;\n}\n\nfunction hasErrorCode(error: unknown, code: string): boolean {\n\treturn error instanceof Error && \"code\" in error && error.code === code;\n}\n\nasync function readDirIfExists(path: string): Promise<readonly string[]> {\n\ttry {\n\t\treturn await readdir(path);\n\t} catch (error) {\n\t\tif (hasErrorCode(error, \"ENOENT\")) return [];\n\t\tthrow error;\n\t}\n}\n\nasync function statIfFile(path: string): Promise<Stats | undefined> {\n\ttry {\n\t\tconst fileStat = await stat(path);\n\t\treturn fileStat.isFile() ? fileStat : undefined;\n\t} catch (error) {\n\t\tif (hasErrorCode(error, \"ENOENT\")) return undefined;\n\t\tthrow error;\n\t}\n}\n\nasync function collectFilesInDir(dir: string): Promise<readonly string[]> {\n\tconst names = await readDirIfExists(dir);\n\tconst files: string[] = [];\n\tfor (const name of names) {\n\t\tif (!name.endsWith(\".jsonl\")) continue;\n\t\tconst file = join(dir, name);\n\t\tif (await statIfFile(file)) files.push(file);\n\t}\n\treturn files;\n}\n\nasync function discoverSessionFiles(root: string): Promise<readonly string[]> {\n\tconst names = await readDirIfExists(root);\n\tconst files: string[] = [...(await collectFilesInDir(root))];\n\tfor (const name of names) {\n\t\tif (name.endsWith(\".jsonl\")) continue;\n\t\tconst dir = join(root, name);\n\t\ttry {\n\t\t\tconst dirStat = await stat(dir);\n\t\t\tif (dirStat.isDirectory()) files.push(...(await collectFilesInDir(dir)));\n\t\t} catch (error) {\n\t\t\tif (!hasErrorCode(error, \"ENOENT\")) throw error;\n\t\t}\n\t}\n\treturn files;\n}\n\nfunction firstHeader(entries: readonly FileEntry[], filePath: string): SessionHeader | undefined {\n\tconst header = entries[0];\n\tif (header?.type === \"session\") return header;\n\tif (entries.length === 0) return undefined;\n\treturn { type: \"session\", id: basename(filePath, \".jsonl\"), timestamp: new Date(0).toISOString(), cwd: \"\" };\n}\n\nfunction lastUserText(entries: readonly FileEntry[]): string {\n\tfor (let index = entries.length - 1; index >= 0; index -= 1) {\n\t\tconst entry = entries[index];\n\t\tif (entry?.type !== \"message\") continue;\n\t\tconst message = entry.message;\n\t\tif (message.role !== \"user\") continue;\n\t\tconst text = compactWhitespace(getTextContent(message.content));\n\t\tif (text) return text;\n\t}\n\treturn \"(no user prompt)\";\n}\n\nfunction latestMessageTimestamp(entries: readonly FileEntry[], fallback: number): number {\n\tfor (let index = entries.length - 1; index >= 0; index -= 1) {\n\t\tconst entry = entries[index];\n\t\tif (!entry || entry.type === \"session\") continue;\n\t\tconst timestamp = Date.parse(entry.timestamp);\n\t\tif (Number.isFinite(timestamp)) return timestamp;\n\t}\n\treturn fallback;\n}\n\nasync function summarizeSession(\n\tfilePath: string,\n\tcurrentSessionFile: string | undefined,\n): Promise<SessionHudEntry | undefined> {\n\tconst [content, fileStat] = await Promise.all([readFile(filePath, \"utf-8\"), stat(filePath)]);\n\tconst entries = parseSessionEntries(content);\n\tconst header = firstHeader(entries, filePath);\n\tif (!header) return undefined;\n\tconst messageCount = entries.filter((entry) => entry.type === \"message\").length;\n\treturn {\n\t\tid: header.id,\n\t\tshortId: header.id.length <= 8 ? header.id : header.id.slice(0, 8),\n\t\tpath: filePath,\n\t\tcwd: header.cwd,\n\t\tcreatedAt: Date.parse(header.timestamp),\n\t\tmodifiedAt: latestMessageTimestamp(entries, fileStat.mtime.getTime()),\n\t\tmessageCount,\n\t\tlastUserText: lastUserText(entries),\n\t\tisCurrent: currentSessionFile === filePath,\n\t};\n}\n\nexport async function scanSessionHudEntries(\n\troot: string,\n\tcurrentSessionFile?: string,\n): Promise<readonly SessionHudEntry[]> {\n\tconst files = await discoverSessionFiles(root);\n\tconst sessions: SessionHudEntry[] = [];\n\tfor (const file of files) {\n\t\ttry {\n\t\t\tconst session = await summarizeSession(file, currentSessionFile);\n\t\t\tif (session) sessions.push(session);\n\t\t} catch (error) {\n\t\t\tif (!hasErrorCode(error, \"ENOENT\")) throw error;\n\t\t}\n\t}\n\tsessions.sort((left, right) => right.modifiedAt - left.modifiedAt);\n\treturn sessions;\n}\n"]}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"loader.d.ts","sourceRoot":"","sources":["../../../src/core/extensions/loader.ts"],"names":[],"mappings":"AAAA;;;GAGG;AAuBH,OAAO,EAAkB,KAAK,QAAQ,EAAE,MAAM,iBAAiB,CAAC;AAIhE,OAAO,KAAK,EACX,SAAS,EAET,gBAAgB,EAChB,gBAAgB,EAChB,oBAAoB,EAKpB,MAAM,YAAY,CAAC;AAiFpB,MAAM,MAAM,wBAAwB,GAAG,CAAC,aAAa,EAAE,MAAM,EAAE,YAAY,EAAE,MAAM,KAAK,gBAAgB,GAAG,SAAS,CAAC;AAErH;;;GAGG;AACH,wBAAgB,sBAAsB,IAAI,gBAAgB,CA8CzD;AAoOD;;GAEG;AACH,wBAAsB,wBAAwB,CAC7C,OAAO,EAAE,gBAAgB,EACzB,GAAG,EAAE,MAAM,EACX,QAAQ,EAAE,QAAQ,EAClB,OAAO,EAAE,gBAAgB,EACzB,aAAa,SAAa,GACxB,OAAO,CAAC,SAAS,CAAC,CAMpB;AAED;;GAEG;AACH,wBAAsB,cAAc,CACnC,KAAK,EAAE,MAAM,EAAE,EACf,GAAG,EAAE,MAAM,EACX,QAAQ,CAAC,EAAE,QAAQ,EACnB,OAAO,CAAC,EAAE;IAAE,eAAe,CAAC,EAAE,wBAAwB,CAAA;CAAE,GACtD,OAAO,CAAC,oBAAoB,CAAC,CAqC/B;AA+GD;;GAEG;AACH,wBAAsB,yBAAyB,CAC9C,eAAe,EAAE,MAAM,EAAE,EACzB,GAAG,EAAE,MAAM,EACX,QAAQ,GAAE,MAAsB,EAChC,QAAQ,CAAC,EAAE,QAAQ,GACjB,OAAO,CAAC,oBAAoB,CAAC,CA2C/B","sourcesContent":["/**\n * Extension loader - loads TypeScript extension modules using jiti.\n *\n */\n\nimport * as fs from \"node:fs\";\nimport { createRequire } from \"node:module\";\nimport * as path from \"node:path\";\nimport { fileURLToPath } from \"node:url\";\nimport * as _bundledPiAgentCore from \"@earendil-works/pi-agent-core\";\nimport * as _bundledPiAi from \"@earendil-works/pi-ai\";\nimport * as _bundledPiAiOauth from \"@earendil-works/pi-ai/oauth\";\nimport type { KeyId } from \"@earendil-works/pi-tui\";\nimport * as _bundledPiTui from \"@earendil-works/pi-tui\";\nimport { createJiti } from \"jiti/static\";\n// Static imports of packages that extensions may use.\n// These MUST be static so Bun bundles them into the compiled binary.\n// The virtualModules option then makes them available to extensions.\nimport * as _bundledTypebox from \"typebox\";\nimport * as _bundledTypeboxCompile from \"typebox/compile\";\nimport * as _bundledTypeboxValue from \"typebox/value\";\nimport { CONFIG_DIR_NAME, getAgentDir, isBunBinary } from \"../../config.ts\";\n// NOTE: This import works because loader.ts exports are NOT re-exported from index.ts,\n// avoiding a circular dependency. Extensions can import from @code-yeongyu/senpi.\nimport * as _bundledPiCodingAgent from \"../../index.ts\";\nimport { resolvePath } from \"../../utils/paths.ts\";\nimport { createEventBus, type EventBus } from \"../event-bus.ts\";\nimport type { ExecOptions } from \"../exec.ts\";\nimport { execCommand } from \"../exec.ts\";\nimport { createSyntheticSourceInfo } from \"../source-info.ts\";\nimport type {\n\tExtension,\n\tExtensionAPI,\n\tExtensionFactory,\n\tExtensionRuntime,\n\tLoadExtensionsResult,\n\tMessageRenderer,\n\tProviderConfig,\n\tRegisteredCommand,\n\tToolDefinition,\n} from \"./types.ts\";\n\n/** Modules available to extensions via virtualModules (for compiled Bun binary) */\nconst VIRTUAL_MODULES: Record<string, unknown> = {\n\ttypebox: _bundledTypebox,\n\t\"typebox/compile\": _bundledTypeboxCompile,\n\t\"typebox/value\": _bundledTypeboxValue,\n\t\"@sinclair/typebox\": _bundledTypebox,\n\t\"@sinclair/typebox/compile\": _bundledTypeboxCompile,\n\t\"@sinclair/typebox/value\": _bundledTypeboxValue,\n\t\"@mariozechner/pi-agent-core\": _bundledPiAgentCore,\n\t\"@mariozechner/pi-tui\": _bundledPiTui,\n\t\"@mariozechner/pi-ai\": _bundledPiAi,\n\t\"@mariozechner/pi-ai/oauth\": _bundledPiAiOauth,\n\t\"@mariozechner/pi-coding-agent\": _bundledPiCodingAgent,\n\t\"@earendil-works/pi-agent-core\": _bundledPiAgentCore,\n\t\"@earendil-works/pi-tui\": _bundledPiTui,\n\t\"@earendil-works/pi-ai\": _bundledPiAi,\n\t\"@earendil-works/pi-ai/oauth\": _bundledPiAiOauth,\n\t\"@earendil-works/pi-coding-agent\": _bundledPiCodingAgent,\n\t\"@code-yeongyu/senpi\": _bundledPiCodingAgent,\n};\n\nconst require = createRequire(import.meta.url);\n\n/**\n * Get aliases for jiti (used in Node.js/development mode).\n * In Bun binary mode, virtualModules is used instead.\n */\nlet _aliases: Record<string, string> | null = null;\n\nfunction getAliases(): Record<string, string> {\n\tif (_aliases) return _aliases;\n\n\tconst __dirname = path.dirname(fileURLToPath(import.meta.url));\n\tconst packageIndex = path.resolve(__dirname, \"../..\", \"index.js\");\n\n\tconst typeboxEntry = require.resolve(\"typebox\");\n\tconst typeboxCompileEntry = require.resolve(\"typebox/compile\");\n\tconst typeboxValueEntry = require.resolve(\"typebox/value\");\n\n\tconst packagesRoot = path.resolve(__dirname, \"../../../../\");\n\tconst resolveWorkspaceOrImport = (workspaceRelativePath: string, specifier: string): string => {\n\t\tconst workspacePath = path.join(packagesRoot, workspaceRelativePath);\n\t\tif (fs.existsSync(workspacePath)) {\n\t\t\treturn workspacePath;\n\t\t}\n\t\treturn fileURLToPath(import.meta.resolve(specifier));\n\t};\n\n\tconst piCodingAgentEntry = packageIndex;\n\tconst piAgentCoreEntry = resolveWorkspaceOrImport(\"agent/dist/index.js\", \"@earendil-works/pi-agent-core\");\n\tconst piTuiEntry = resolveWorkspaceOrImport(\"tui/dist/index.js\", \"@earendil-works/pi-tui\");\n\tconst piAiEntry = resolveWorkspaceOrImport(\"ai/dist/index.js\", \"@earendil-works/pi-ai\");\n\tconst piAiOauthEntry = resolveWorkspaceOrImport(\"ai/dist/oauth.js\", \"@earendil-works/pi-ai/oauth\");\n\n\t_aliases = {\n\t\t\"@code-yeongyu/senpi\": piCodingAgentEntry,\n\t\t\"@mariozechner/pi-coding-agent\": piCodingAgentEntry,\n\t\t\"@mariozechner/pi-agent-core\": piAgentCoreEntry,\n\t\t\"@mariozechner/pi-tui\": piTuiEntry,\n\t\t\"@mariozechner/pi-ai\": piAiEntry,\n\t\t\"@mariozechner/pi-ai/oauth\": piAiOauthEntry,\n\t\t\"@earendil-works/pi-agent-core\": piAgentCoreEntry,\n\t\t\"@earendil-works/pi-tui\": piTuiEntry,\n\t\t\"@earendil-works/pi-ai\": piAiEntry,\n\t\t\"@earendil-works/pi-ai/oauth\": piAiOauthEntry,\n\t\t\"@earendil-works/pi-coding-agent\": piCodingAgentEntry,\n\t\ttypebox: typeboxEntry,\n\t\t\"typebox/compile\": typeboxCompileEntry,\n\t\t\"typebox/value\": typeboxValueEntry,\n\t\t\"@sinclair/typebox\": typeboxEntry,\n\t\t\"@sinclair/typebox/compile\": typeboxCompileEntry,\n\t\t\"@sinclair/typebox/value\": typeboxValueEntry,\n\t};\n\n\treturn _aliases;\n}\n\ntype HandlerFn = (...args: unknown[]) => Promise<unknown>;\ntype ExtensionModuleImporter = ReturnType<typeof createJiti>;\nexport type ExtensionFactoryResolver = (extensionPath: string, resolvedPath: string) => ExtensionFactory | undefined;\n\n/**\n * Create a runtime with throwing stubs for action methods.\n * Runner.bindCore() replaces these with real implementations.\n */\nexport function createExtensionRuntime(): ExtensionRuntime {\n\tconst notInitialized = () => {\n\t\tthrow new Error(\"Extension runtime not initialized. Action methods cannot be called during extension loading.\");\n\t};\n\tconst state: { staleMessage?: string } = {};\n\tconst assertActive = () => {\n\t\tif (state.staleMessage) {\n\t\t\tthrow new Error(state.staleMessage);\n\t\t}\n\t};\n\n\tconst runtime: ExtensionRuntime = {\n\t\tsendMessage: notInitialized,\n\t\tsendUserMessage: notInitialized,\n\t\tappendEntry: notInitialized,\n\t\tsetSessionName: notInitialized,\n\t\tgetSessionName: notInitialized,\n\t\tsetLabel: notInitialized,\n\t\tgetActiveTools: notInitialized,\n\t\tgetAllTools: notInitialized,\n\t\tsetActiveTools: notInitialized,\n\t\t// registerTool() is valid during extension load; refresh is only needed post-bind.\n\t\trefreshTools: () => {},\n\t\tgetCommands: notInitialized,\n\t\tsetModel: () => Promise.reject(new Error(\"Extension runtime not initialized\")),\n\t\tgetThinkingLevel: notInitialized,\n\t\tsetThinkingLevel: notInitialized,\n\t\tflagValues: new Map(),\n\t\tpendingProviderRegistrations: [],\n\t\tassertActive,\n\t\tinvalidate: (message) => {\n\t\t\tstate.staleMessage ??=\n\t\t\t\tmessage ??\n\t\t\t\t\"This extension ctx is stale after session replacement or reload. Do not use a captured pi or command ctx after ctx.newSession(), ctx.fork(), ctx.switchSession(), or ctx.reload(). For newSession, fork, and switchSession, move post-replacement work into withSession and use the ctx passed to withSession. For reload, do not use the old ctx after await ctx.reload().\";\n\t\t},\n\t\t// Pre-bind: queue registrations so bindCore() can flush them once the\n\t\t// model registry is available. bindCore() replaces both with direct calls.\n\t\tregisterProvider: (name, config, extensionPath = \"<unknown>\") => {\n\t\t\truntime.pendingProviderRegistrations.push({ name, config, extensionPath });\n\t\t},\n\t\tunregisterProvider: (name) => {\n\t\t\truntime.pendingProviderRegistrations = runtime.pendingProviderRegistrations.filter((r) => r.name !== name);\n\t\t},\n\t};\n\n\treturn runtime;\n}\n\n/**\n * Create the ExtensionAPI for an extension.\n * Registration methods write to the extension object.\n * Action methods delegate to the shared runtime.\n */\nfunction createExtensionAPI(\n\textension: Extension,\n\truntime: ExtensionRuntime,\n\tcwd: string,\n\teventBus: EventBus,\n): ExtensionAPI {\n\tconst api = {\n\t\t// Registration methods - write to extension\n\t\ton(event: string, handler: HandlerFn): void {\n\t\t\truntime.assertActive();\n\t\t\tconst list = extension.handlers.get(event) ?? [];\n\t\t\tlist.push(handler);\n\t\t\textension.handlers.set(event, list);\n\t\t},\n\n\t\tregisterTool(tool: ToolDefinition): void {\n\t\t\truntime.assertActive();\n\t\t\textension.tools.set(tool.name, {\n\t\t\t\tdefinition: tool,\n\t\t\t\tsourceInfo: extension.sourceInfo,\n\t\t\t});\n\t\t\truntime.refreshTools();\n\t\t},\n\n\t\tregisterCommand(name: string, options: Omit<RegisteredCommand, \"name\" | \"sourceInfo\">): void {\n\t\t\truntime.assertActive();\n\t\t\textension.commands.set(name, {\n\t\t\t\tname,\n\t\t\t\tsourceInfo: extension.sourceInfo,\n\t\t\t\t...options,\n\t\t\t});\n\t\t},\n\n\t\tregisterShortcut(\n\t\t\tshortcut: KeyId,\n\t\t\toptions: {\n\t\t\t\tdescription?: string;\n\t\t\t\thandler: (ctx: import(\"./types.ts\").ExtensionContext) => Promise<void> | void;\n\t\t\t},\n\t\t): void {\n\t\t\truntime.assertActive();\n\t\t\textension.shortcuts.set(shortcut, { shortcut, extensionPath: extension.path, ...options });\n\t\t},\n\n\t\tregisterFlag(\n\t\t\tname: string,\n\t\t\toptions: { description?: string; type: \"boolean\" | \"string\"; default?: boolean | string },\n\t\t): void {\n\t\t\truntime.assertActive();\n\t\t\textension.flags.set(name, { name, extensionPath: extension.path, ...options });\n\t\t\tif (options.default !== undefined && !runtime.flagValues.has(name)) {\n\t\t\t\truntime.flagValues.set(name, options.default);\n\t\t\t}\n\t\t},\n\n\t\tregisterMessageRenderer<T>(customType: string, renderer: MessageRenderer<T>): void {\n\t\t\truntime.assertActive();\n\t\t\textension.messageRenderers.set(customType, renderer as MessageRenderer);\n\t\t},\n\n\t\t// Flag access - checks extension registered it, reads from runtime\n\t\tgetFlag(name: string): boolean | string | undefined {\n\t\t\truntime.assertActive();\n\t\t\tif (!extension.flags.has(name)) return undefined;\n\t\t\treturn runtime.flagValues.get(name);\n\t\t},\n\n\t\t// Action methods - delegate to shared runtime\n\t\tsendMessage(message, options): void {\n\t\t\truntime.assertActive();\n\t\t\truntime.sendMessage(message, options);\n\t\t},\n\n\t\tsendUserMessage(content, options): void {\n\t\t\truntime.assertActive();\n\t\t\truntime.sendUserMessage(content, options);\n\t\t},\n\n\t\tappendEntry(customType: string, data?: unknown): void {\n\t\t\truntime.assertActive();\n\t\t\truntime.appendEntry(customType, data);\n\t\t},\n\n\t\tsetSessionName(name: string): void {\n\t\t\truntime.assertActive();\n\t\t\truntime.setSessionName(name);\n\t\t},\n\n\t\tgetSessionName(): string | undefined {\n\t\t\truntime.assertActive();\n\t\t\treturn runtime.getSessionName();\n\t\t},\n\n\t\tsetLabel(entryId: string, label: string | undefined): void {\n\t\t\truntime.assertActive();\n\t\t\truntime.setLabel(entryId, label);\n\t\t},\n\n\t\texec(command: string, args: string[], options?: ExecOptions) {\n\t\t\truntime.assertActive();\n\t\t\treturn execCommand(command, args, options?.cwd ?? cwd, options);\n\t\t},\n\n\t\tgetActiveTools(): string[] {\n\t\t\truntime.assertActive();\n\t\t\treturn runtime.getActiveTools();\n\t\t},\n\n\t\tgetAllTools() {\n\t\t\truntime.assertActive();\n\t\t\treturn runtime.getAllTools();\n\t\t},\n\n\t\tsetActiveTools(toolNames: string[]): void {\n\t\t\truntime.assertActive();\n\t\t\truntime.setActiveTools(toolNames);\n\t\t},\n\n\t\tgetCommands() {\n\t\t\truntime.assertActive();\n\t\t\treturn runtime.getCommands();\n\t\t},\n\n\t\tsetModel(model) {\n\t\t\truntime.assertActive();\n\t\t\treturn runtime.setModel(model);\n\t\t},\n\n\t\tgetThinkingLevel() {\n\t\t\truntime.assertActive();\n\t\t\treturn runtime.getThinkingLevel();\n\t\t},\n\n\t\tsetThinkingLevel(level) {\n\t\t\truntime.assertActive();\n\t\t\truntime.setThinkingLevel(level);\n\t\t},\n\n\t\tregisterProvider(name: string, config: ProviderConfig) {\n\t\t\truntime.assertActive();\n\t\t\truntime.registerProvider(name, config, extension.path);\n\t\t},\n\n\t\tunregisterProvider(name: string) {\n\t\t\truntime.assertActive();\n\t\t\truntime.unregisterProvider(name, extension.path);\n\t\t},\n\n\t\tevents: eventBus,\n\t} as ExtensionAPI;\n\n\treturn api;\n}\n\nfunction createExtensionModuleImporter(): ExtensionModuleImporter {\n\treturn createJiti(import.meta.url, {\n\t\tmoduleCache: false,\n\t\t// In Bun binary: use virtualModules for bundled packages (no filesystem resolution)\n\t\t// Also disable tryNative so jiti handles ALL imports (not just the entry point)\n\t\t// In Node.js/dev: use aliases to resolve to node_modules paths\n\t\t...(isBunBinary ? { virtualModules: VIRTUAL_MODULES, tryNative: false } : { alias: getAliases() }),\n\t});\n}\n\nasync function loadExtensionModule(extensionPath: string, importer: ExtensionModuleImporter) {\n\tconst module = await importer.import(extensionPath, { default: true });\n\tconst factory = module as ExtensionFactory;\n\treturn typeof factory !== \"function\" ? undefined : factory;\n}\n\n/**\n * Create an Extension object with empty collections.\n */\nfunction createExtension(extensionPath: string, resolvedPath: string): Extension {\n\tconst source =\n\t\textensionPath.startsWith(\"<\") && extensionPath.endsWith(\">\")\n\t\t\t? extensionPath.slice(1, -1).split(\":\")[0] || \"temporary\"\n\t\t\t: \"local\";\n\tconst baseDir = extensionPath.startsWith(\"<\") ? undefined : path.dirname(resolvedPath);\n\n\treturn {\n\t\tpath: extensionPath,\n\t\tresolvedPath,\n\t\tsourceInfo: createSyntheticSourceInfo(extensionPath, { source, baseDir }),\n\t\thandlers: new Map(),\n\t\ttools: new Map(),\n\t\tmessageRenderers: new Map(),\n\t\tcommands: new Map(),\n\t\tflags: new Map(),\n\t\tshortcuts: new Map(),\n\t};\n}\n\nasync function loadExtension(\n\textensionPath: string,\n\tcwd: string,\n\teventBus: EventBus,\n\truntime: ExtensionRuntime,\n\tgetImporter: () => ExtensionModuleImporter,\n\tfactoryResolver?: ExtensionFactoryResolver,\n): Promise<{ extension: Extension | null; error: string | null }> {\n\tconst resolvedPath = resolvePath(extensionPath, cwd, { normalizeUnicodeSpaces: true });\n\n\ttry {\n\t\tconst factory =\n\t\t\tfactoryResolver?.(extensionPath, resolvedPath) ?? (await loadExtensionModule(resolvedPath, getImporter()));\n\t\tif (!factory) {\n\t\t\treturn { extension: null, error: `Extension does not export a valid factory function: ${extensionPath}` };\n\t\t}\n\n\t\tconst extension = createExtension(extensionPath, resolvedPath);\n\t\tconst api = createExtensionAPI(extension, runtime, cwd, eventBus);\n\t\tawait factory(api);\n\n\t\treturn { extension, error: null };\n\t} catch (err) {\n\t\tconst message = err instanceof Error ? err.message : String(err);\n\t\treturn { extension: null, error: `Failed to load extension: ${message}` };\n\t}\n}\n\n/**\n * Create an Extension from an inline factory function.\n */\nexport async function loadExtensionFromFactory(\n\tfactory: ExtensionFactory,\n\tcwd: string,\n\teventBus: EventBus,\n\truntime: ExtensionRuntime,\n\textensionPath = \"<inline>\",\n): Promise<Extension> {\n\tconst extension = createExtension(extensionPath, extensionPath);\n\tconst resolvedCwd = resolvePath(cwd);\n\tconst api = createExtensionAPI(extension, runtime, resolvedCwd, eventBus);\n\tawait factory(api);\n\treturn extension;\n}\n\n/**\n * Load extensions from paths.\n */\nexport async function loadExtensions(\n\tpaths: string[],\n\tcwd: string,\n\teventBus?: EventBus,\n\toptions?: { factoryResolver?: ExtensionFactoryResolver },\n): Promise<LoadExtensionsResult> {\n\tconst extensions: Extension[] = [];\n\tconst errors: Array<{ path: string; error: string }> = [];\n\tconst resolvedCwd = resolvePath(cwd);\n\tconst resolvedEventBus = eventBus ?? createEventBus();\n\tconst runtime = createExtensionRuntime();\n\tlet importer: ExtensionModuleImporter | undefined;\n\tconst getImporter = () => {\n\t\timporter ??= createExtensionModuleImporter();\n\t\treturn importer;\n\t};\n\n\tfor (const extPath of paths) {\n\t\tconst { extension, error } = await loadExtension(\n\t\t\textPath,\n\t\t\tresolvedCwd,\n\t\t\tresolvedEventBus,\n\t\t\truntime,\n\t\t\tgetImporter,\n\t\t\toptions?.factoryResolver,\n\t\t);\n\n\t\tif (error) {\n\t\t\terrors.push({ path: extPath, error });\n\t\t\tcontinue;\n\t\t}\n\n\t\tif (extension) {\n\t\t\textensions.push(extension);\n\t\t}\n\t}\n\n\treturn {\n\t\textensions,\n\t\terrors,\n\t\truntime,\n\t};\n}\n\ninterface PiManifest {\n\textensions?: string[];\n\tthemes?: string[];\n\tskills?: string[];\n\tprompts?: string[];\n}\n\nfunction readPiManifest(packageJsonPath: string): PiManifest | null {\n\ttry {\n\t\tconst content = fs.readFileSync(packageJsonPath, \"utf-8\");\n\t\tconst pkg = JSON.parse(content);\n\t\tif (pkg.pi && typeof pkg.pi === \"object\") {\n\t\t\treturn pkg.pi as PiManifest;\n\t\t}\n\t\treturn null;\n\t} catch {\n\t\treturn null;\n\t}\n}\n\nfunction isExtensionFile(name: string): boolean {\n\treturn name.endsWith(\".ts\") || name.endsWith(\".js\");\n}\n\n/**\n * Resolve extension entry points from a directory.\n *\n * Checks for:\n * 1. package.json with \"pi.extensions\" field -> returns declared paths\n * 2. index.ts or index.js -> returns the index file\n *\n * Returns resolved paths or null if no entry points found.\n */\nfunction resolveExtensionEntries(dir: string): string[] | null {\n\t// Check for package.json with \"pi\" field first\n\tconst packageJsonPath = path.join(dir, \"package.json\");\n\tif (fs.existsSync(packageJsonPath)) {\n\t\tconst manifest = readPiManifest(packageJsonPath);\n\t\tif (manifest?.extensions?.length) {\n\t\t\tconst entries: string[] = [];\n\t\t\tfor (const extPath of manifest.extensions) {\n\t\t\t\tconst resolvedExtPath = path.resolve(dir, extPath);\n\t\t\t\tif (fs.existsSync(resolvedExtPath)) {\n\t\t\t\t\tentries.push(resolvedExtPath);\n\t\t\t\t}\n\t\t\t}\n\t\t\tif (entries.length > 0) {\n\t\t\t\treturn entries;\n\t\t\t}\n\t\t}\n\t}\n\n\t// Check for index.ts or index.js\n\tconst indexTs = path.join(dir, \"index.ts\");\n\tconst indexJs = path.join(dir, \"index.js\");\n\tif (fs.existsSync(indexTs)) {\n\t\treturn [indexTs];\n\t}\n\tif (fs.existsSync(indexJs)) {\n\t\treturn [indexJs];\n\t}\n\n\treturn null;\n}\n\n/**\n * Discover extensions in a directory.\n *\n * Discovery rules:\n * 1. Direct files: `extensions/*.ts` or `*.js` → load\n * 2. Subdirectory with index: `extensions/* /index.ts` or `index.js` → load\n * 3. Subdirectory with package.json: `extensions/* /package.json` with \"pi\" field → load what it declares\n *\n * No recursion beyond one level. Complex packages must use package.json manifest.\n */\nfunction discoverExtensionsInDir(dir: string): string[] {\n\tif (!fs.existsSync(dir)) {\n\t\treturn [];\n\t}\n\n\tconst discovered: string[] = [];\n\n\ttry {\n\t\tconst entries = fs.readdirSync(dir, { withFileTypes: true });\n\n\t\tfor (const entry of entries) {\n\t\t\tconst entryPath = path.join(dir, entry.name);\n\n\t\t\t// 1. Direct files: *.ts or *.js\n\t\t\tif ((entry.isFile() || entry.isSymbolicLink()) && isExtensionFile(entry.name)) {\n\t\t\t\tdiscovered.push(entryPath);\n\t\t\t\tcontinue;\n\t\t\t}\n\n\t\t\t// 2 & 3. Subdirectories\n\t\t\tif (entry.isDirectory() || entry.isSymbolicLink()) {\n\t\t\t\tconst entries = resolveExtensionEntries(entryPath);\n\t\t\t\tif (entries) {\n\t\t\t\t\tdiscovered.push(...entries);\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\t} catch {\n\t\treturn [];\n\t}\n\n\treturn discovered;\n}\n\n/**\n * Discover and load extensions from standard locations.\n */\nexport async function discoverAndLoadExtensions(\n\tconfiguredPaths: string[],\n\tcwd: string,\n\tagentDir: string = getAgentDir(),\n\teventBus?: EventBus,\n): Promise<LoadExtensionsResult> {\n\tconst resolvedCwd = resolvePath(cwd);\n\tconst resolvedAgentDir = resolvePath(agentDir);\n\tconst allPaths: string[] = [];\n\tconst seen = new Set<string>();\n\n\tconst addPaths = (paths: string[]) => {\n\t\tfor (const p of paths) {\n\t\t\tconst resolved = path.resolve(p);\n\t\t\tif (!seen.has(resolved)) {\n\t\t\t\tseen.add(resolved);\n\t\t\t\tallPaths.push(p);\n\t\t\t}\n\t\t}\n\t};\n\n\t// 1. Project-local extensions: cwd/${CONFIG_DIR_NAME}/extensions/\n\tconst localExtDir = path.join(resolvedCwd, CONFIG_DIR_NAME, \"extensions\");\n\taddPaths(discoverExtensionsInDir(localExtDir));\n\n\t// 2. Global extensions: agentDir/extensions/\n\tconst globalExtDir = path.join(resolvedAgentDir, \"extensions\");\n\taddPaths(discoverExtensionsInDir(globalExtDir));\n\n\t// 3. Explicitly configured paths\n\tfor (const p of configuredPaths) {\n\t\tconst resolved = resolvePath(p, resolvedCwd, { normalizeUnicodeSpaces: true });\n\t\tif (fs.existsSync(resolved) && fs.statSync(resolved).isDirectory()) {\n\t\t\t// Check for package.json with pi manifest or index.ts\n\t\t\tconst entries = resolveExtensionEntries(resolved);\n\t\t\tif (entries) {\n\t\t\t\taddPaths(entries);\n\t\t\t\tcontinue;\n\t\t\t}\n\t\t\t// No explicit entries - discover individual files in directory\n\t\t\taddPaths(discoverExtensionsInDir(resolved));\n\t\t\tcontinue;\n\t\t}\n\n\t\taddPaths([resolved]);\n\t}\n\n\treturn loadExtensions(allPaths, resolvedCwd, eventBus);\n}\n"]}
|
|
1
|
+
{"version":3,"file":"loader.d.ts","sourceRoot":"","sources":["../../../src/core/extensions/loader.ts"],"names":[],"mappings":"AAAA;;;GAGG;AAuBH,OAAO,EAAkB,KAAK,QAAQ,EAAE,MAAM,iBAAiB,CAAC;AAIhE,OAAO,KAAK,EACX,SAAS,EAET,gBAAgB,EAChB,gBAAgB,EAChB,oBAAoB,EAKpB,MAAM,YAAY,CAAC;AA0HpB,MAAM,MAAM,wBAAwB,GAAG,CAAC,aAAa,EAAE,MAAM,EAAE,YAAY,EAAE,MAAM,KAAK,gBAAgB,GAAG,SAAS,CAAC;AAErH;;;GAGG;AACH,wBAAgB,sBAAsB,IAAI,gBAAgB,CA8CzD;AAoOD;;GAEG;AACH,wBAAsB,wBAAwB,CAC7C,OAAO,EAAE,gBAAgB,EACzB,GAAG,EAAE,MAAM,EACX,QAAQ,EAAE,QAAQ,EAClB,OAAO,EAAE,gBAAgB,EACzB,aAAa,SAAa,GACxB,OAAO,CAAC,SAAS,CAAC,CAMpB;AAED;;GAEG;AACH,wBAAsB,cAAc,CACnC,KAAK,EAAE,MAAM,EAAE,EACf,GAAG,EAAE,MAAM,EACX,QAAQ,CAAC,EAAE,QAAQ,EACnB,OAAO,CAAC,EAAE;IAAE,eAAe,CAAC,EAAE,wBAAwB,CAAA;CAAE,GACtD,OAAO,CAAC,oBAAoB,CAAC,CAqC/B;AA+GD;;GAEG;AACH,wBAAsB,yBAAyB,CAC9C,eAAe,EAAE,MAAM,EAAE,EACzB,GAAG,EAAE,MAAM,EACX,QAAQ,GAAE,MAAsB,EAChC,QAAQ,CAAC,EAAE,QAAQ,GACjB,OAAO,CAAC,oBAAoB,CAAC,CA2C/B","sourcesContent":["/**\n * Extension loader - loads TypeScript extension modules using jiti.\n *\n */\n\nimport * as fs from \"node:fs\";\nimport { createRequire } from \"node:module\";\nimport * as path from \"node:path\";\nimport { fileURLToPath } from \"node:url\";\nimport * as _bundledPiAgentCore from \"@earendil-works/pi-agent-core\";\nimport * as _bundledPiAi from \"@earendil-works/pi-ai\";\nimport * as _bundledPiAiOauth from \"@earendil-works/pi-ai/oauth\";\nimport type { KeyId } from \"@earendil-works/pi-tui\";\nimport * as _bundledPiTui from \"@earendil-works/pi-tui\";\nimport { createJiti } from \"jiti/static\";\n// Static imports of packages that extensions may use.\n// These MUST be static so Bun bundles them into the compiled binary.\n// The virtualModules option then makes them available to extensions.\nimport * as _bundledTypebox from \"typebox\";\nimport * as _bundledTypeboxCompile from \"typebox/compile\";\nimport * as _bundledTypeboxValue from \"typebox/value\";\nimport { CONFIG_DIR_NAME, getAgentDir, isBunBinary } from \"../../config.ts\";\n// NOTE: This import works because loader.ts exports are NOT re-exported from index.ts,\n// avoiding a circular dependency. Extensions can import from @code-yeongyu/senpi.\nimport * as _bundledPiCodingAgent from \"../../index.ts\";\nimport { resolvePath } from \"../../utils/paths.ts\";\nimport { createEventBus, type EventBus } from \"../event-bus.ts\";\nimport type { ExecOptions } from \"../exec.ts\";\nimport { execCommand } from \"../exec.ts\";\nimport { createSyntheticSourceInfo } from \"../source-info.ts\";\nimport type {\n\tExtension,\n\tExtensionAPI,\n\tExtensionFactory,\n\tExtensionRuntime,\n\tLoadExtensionsResult,\n\tMessageRenderer,\n\tProviderConfig,\n\tRegisteredCommand,\n\tToolDefinition,\n} from \"./types.ts\";\n\n/** Modules available to extensions via virtualModules (for compiled Bun binary) */\nconst VIRTUAL_MODULES: Record<string, unknown> = {\n\ttypebox: _bundledTypebox,\n\t\"typebox/compile\": _bundledTypeboxCompile,\n\t\"typebox/value\": _bundledTypeboxValue,\n\t\"@sinclair/typebox\": _bundledTypebox,\n\t\"@sinclair/typebox/compile\": _bundledTypeboxCompile,\n\t\"@sinclair/typebox/value\": _bundledTypeboxValue,\n\t\"@mariozechner/pi-agent-core\": _bundledPiAgentCore,\n\t\"@mariozechner/pi-tui\": _bundledPiTui,\n\t\"@mariozechner/pi-ai\": _bundledPiAi,\n\t\"@mariozechner/pi-ai/oauth\": _bundledPiAiOauth,\n\t\"@mariozechner/pi-coding-agent\": _bundledPiCodingAgent,\n\t\"@earendil-works/pi-agent-core\": _bundledPiAgentCore,\n\t\"@earendil-works/pi-tui\": _bundledPiTui,\n\t\"@earendil-works/pi-ai\": _bundledPiAi,\n\t\"@earendil-works/pi-ai/oauth\": _bundledPiAiOauth,\n\t\"@earendil-works/pi-coding-agent\": _bundledPiCodingAgent,\n\t\"@code-yeongyu/senpi\": _bundledPiCodingAgent,\n};\n\nconst require = createRequire(import.meta.url);\n\n/**\n * Get aliases for jiti (used in Node.js/development mode).\n * In Bun binary mode, virtualModules is used instead.\n */\nlet _aliases: Record<string, string> | null = null;\n\nfunction getAliases(): Record<string, string> {\n\tif (_aliases) return _aliases;\n\n\tconst __dirname = path.dirname(fileURLToPath(import.meta.url));\n\tconst typeboxEntry = require.resolve(\"typebox\");\n\tconst typeboxCompileEntry = require.resolve(\"typebox/compile\");\n\tconst typeboxValueEntry = require.resolve(\"typebox/value\");\n\n\tconst codingAgentPackageRoot = path.resolve(__dirname, \"../../..\");\n\tconst packagesRoot = path.dirname(codingAgentPackageRoot);\n\tconst resolveWorkspaceOrBundled = (\n\t\tpackageName: string,\n\t\tpackageEntryRelativePath: string,\n\t\tworkspaceRelativePath: string,\n\t\tsourceRelativePath: string,\n\t\tspecifier: string,\n\t): string => {\n\t\tconst bundledPath = path.join(codingAgentPackageRoot, \"node_modules\", packageName, packageEntryRelativePath);\n\t\tif (fs.existsSync(bundledPath)) {\n\t\t\treturn bundledPath;\n\t\t}\n\t\tconst workspacePath = path.join(packagesRoot, workspaceRelativePath);\n\t\tif (fs.existsSync(workspacePath)) {\n\t\t\treturn workspacePath;\n\t\t}\n\t\tconst sourcePath = path.join(packagesRoot, sourceRelativePath);\n\t\tif (fs.existsSync(sourcePath)) {\n\t\t\treturn sourcePath;\n\t\t}\n\t\treturn typeof import.meta.resolve === \"function\"\n\t\t\t? fileURLToPath(import.meta.resolve(specifier))\n\t\t\t: require.resolve(specifier);\n\t};\n\n\tconst piCodingAgentDistEntry = path.join(codingAgentPackageRoot, \"dist/index.js\");\n\tconst piCodingAgentSourceEntry = path.join(codingAgentPackageRoot, \"src/index.ts\");\n\tconst piCodingAgentEntry = fs.existsSync(piCodingAgentDistEntry) ? piCodingAgentDistEntry : piCodingAgentSourceEntry;\n\tconst piAgentCoreEntry = resolveWorkspaceOrBundled(\n\t\t\"@earendil-works/pi-agent-core\",\n\t\t\"dist/index.js\",\n\t\t\"agent/dist/index.js\",\n\t\t\"agent/src/index.ts\",\n\t\t\"@earendil-works/pi-agent-core\",\n\t);\n\tconst piTuiEntry = resolveWorkspaceOrBundled(\n\t\t\"@earendil-works/pi-tui\",\n\t\t\"dist/index.js\",\n\t\t\"tui/dist/index.js\",\n\t\t\"tui/src/index.ts\",\n\t\t\"@earendil-works/pi-tui\",\n\t);\n\tconst piAiEntry = resolveWorkspaceOrBundled(\n\t\t\"@earendil-works/pi-ai\",\n\t\t\"dist/index.js\",\n\t\t\"ai/dist/index.js\",\n\t\t\"ai/src/index.ts\",\n\t\t\"@earendil-works/pi-ai\",\n\t);\n\tconst piAiOauthEntry = resolveWorkspaceOrBundled(\n\t\t\"@earendil-works/pi-ai\",\n\t\t\"dist/oauth.js\",\n\t\t\"ai/dist/oauth.js\",\n\t\t\"ai/src/oauth.ts\",\n\t\t\"@earendil-works/pi-ai/oauth\",\n\t);\n\n\t_aliases = {\n\t\t\"@code-yeongyu/senpi\": piCodingAgentEntry,\n\t\t\"@mariozechner/pi-coding-agent\": piCodingAgentEntry,\n\t\t\"@mariozechner/pi-agent-core\": piAgentCoreEntry,\n\t\t\"@mariozechner/pi-tui\": piTuiEntry,\n\t\t\"@mariozechner/pi-ai\": piAiEntry,\n\t\t\"@mariozechner/pi-ai/oauth\": piAiOauthEntry,\n\t\t\"@earendil-works/pi-agent-core\": piAgentCoreEntry,\n\t\t\"@earendil-works/pi-tui\": piTuiEntry,\n\t\t\"@earendil-works/pi-ai\": piAiEntry,\n\t\t\"@earendil-works/pi-ai/oauth\": piAiOauthEntry,\n\t\t\"@earendil-works/pi-coding-agent\": piCodingAgentEntry,\n\t\ttypebox: typeboxEntry,\n\t\t\"typebox/compile\": typeboxCompileEntry,\n\t\t\"typebox/value\": typeboxValueEntry,\n\t\t\"@sinclair/typebox\": typeboxEntry,\n\t\t\"@sinclair/typebox/compile\": typeboxCompileEntry,\n\t\t\"@sinclair/typebox/value\": typeboxValueEntry,\n\t};\n\n\treturn _aliases;\n}\n\ntype HandlerFn = (...args: unknown[]) => Promise<unknown>;\ntype ExtensionModuleImporter = ReturnType<typeof createJiti>;\nexport type ExtensionFactoryResolver = (extensionPath: string, resolvedPath: string) => ExtensionFactory | undefined;\n\n/**\n * Create a runtime with throwing stubs for action methods.\n * Runner.bindCore() replaces these with real implementations.\n */\nexport function createExtensionRuntime(): ExtensionRuntime {\n\tconst notInitialized = () => {\n\t\tthrow new Error(\"Extension runtime not initialized. Action methods cannot be called during extension loading.\");\n\t};\n\tconst state: { staleMessage?: string } = {};\n\tconst assertActive = () => {\n\t\tif (state.staleMessage) {\n\t\t\tthrow new Error(state.staleMessage);\n\t\t}\n\t};\n\n\tconst runtime: ExtensionRuntime = {\n\t\tsendMessage: notInitialized,\n\t\tsendUserMessage: notInitialized,\n\t\tappendEntry: notInitialized,\n\t\tsetSessionName: notInitialized,\n\t\tgetSessionName: notInitialized,\n\t\tsetLabel: notInitialized,\n\t\tgetActiveTools: notInitialized,\n\t\tgetAllTools: notInitialized,\n\t\tsetActiveTools: notInitialized,\n\t\t// registerTool() is valid during extension load; refresh is only needed post-bind.\n\t\trefreshTools: () => {},\n\t\tgetCommands: notInitialized,\n\t\tsetModel: () => Promise.reject(new Error(\"Extension runtime not initialized\")),\n\t\tgetThinkingLevel: notInitialized,\n\t\tsetThinkingLevel: notInitialized,\n\t\tflagValues: new Map(),\n\t\tpendingProviderRegistrations: [],\n\t\tassertActive,\n\t\tinvalidate: (message) => {\n\t\t\tstate.staleMessage ??=\n\t\t\t\tmessage ??\n\t\t\t\t\"This extension ctx is stale after session replacement or reload. Do not use a captured pi or command ctx after ctx.newSession(), ctx.fork(), ctx.switchSession(), or ctx.reload(). For newSession, fork, and switchSession, move post-replacement work into withSession and use the ctx passed to withSession. For reload, do not use the old ctx after await ctx.reload().\";\n\t\t},\n\t\t// Pre-bind: queue registrations so bindCore() can flush them once the\n\t\t// model registry is available. bindCore() replaces both with direct calls.\n\t\tregisterProvider: (name, config, extensionPath = \"<unknown>\") => {\n\t\t\truntime.pendingProviderRegistrations.push({ name, config, extensionPath });\n\t\t},\n\t\tunregisterProvider: (name) => {\n\t\t\truntime.pendingProviderRegistrations = runtime.pendingProviderRegistrations.filter((r) => r.name !== name);\n\t\t},\n\t};\n\n\treturn runtime;\n}\n\n/**\n * Create the ExtensionAPI for an extension.\n * Registration methods write to the extension object.\n * Action methods delegate to the shared runtime.\n */\nfunction createExtensionAPI(\n\textension: Extension,\n\truntime: ExtensionRuntime,\n\tcwd: string,\n\teventBus: EventBus,\n): ExtensionAPI {\n\tconst api = {\n\t\t// Registration methods - write to extension\n\t\ton(event: string, handler: HandlerFn): void {\n\t\t\truntime.assertActive();\n\t\t\tconst list = extension.handlers.get(event) ?? [];\n\t\t\tlist.push(handler);\n\t\t\textension.handlers.set(event, list);\n\t\t},\n\n\t\tregisterTool(tool: ToolDefinition): void {\n\t\t\truntime.assertActive();\n\t\t\textension.tools.set(tool.name, {\n\t\t\t\tdefinition: tool,\n\t\t\t\tsourceInfo: extension.sourceInfo,\n\t\t\t});\n\t\t\truntime.refreshTools();\n\t\t},\n\n\t\tregisterCommand(name: string, options: Omit<RegisteredCommand, \"name\" | \"sourceInfo\">): void {\n\t\t\truntime.assertActive();\n\t\t\textension.commands.set(name, {\n\t\t\t\tname,\n\t\t\t\tsourceInfo: extension.sourceInfo,\n\t\t\t\t...options,\n\t\t\t});\n\t\t},\n\n\t\tregisterShortcut(\n\t\t\tshortcut: KeyId,\n\t\t\toptions: {\n\t\t\t\tdescription?: string;\n\t\t\t\thandler: (ctx: import(\"./types.ts\").ExtensionContext) => Promise<void> | void;\n\t\t\t},\n\t\t): void {\n\t\t\truntime.assertActive();\n\t\t\textension.shortcuts.set(shortcut, { shortcut, extensionPath: extension.path, ...options });\n\t\t},\n\n\t\tregisterFlag(\n\t\t\tname: string,\n\t\t\toptions: { description?: string; type: \"boolean\" | \"string\"; default?: boolean | string },\n\t\t): void {\n\t\t\truntime.assertActive();\n\t\t\textension.flags.set(name, { name, extensionPath: extension.path, ...options });\n\t\t\tif (options.default !== undefined && !runtime.flagValues.has(name)) {\n\t\t\t\truntime.flagValues.set(name, options.default);\n\t\t\t}\n\t\t},\n\n\t\tregisterMessageRenderer<T>(customType: string, renderer: MessageRenderer<T>): void {\n\t\t\truntime.assertActive();\n\t\t\textension.messageRenderers.set(customType, renderer as MessageRenderer);\n\t\t},\n\n\t\t// Flag access - checks extension registered it, reads from runtime\n\t\tgetFlag(name: string): boolean | string | undefined {\n\t\t\truntime.assertActive();\n\t\t\tif (!extension.flags.has(name)) return undefined;\n\t\t\treturn runtime.flagValues.get(name);\n\t\t},\n\n\t\t// Action methods - delegate to shared runtime\n\t\tsendMessage(message, options): void {\n\t\t\truntime.assertActive();\n\t\t\truntime.sendMessage(message, options);\n\t\t},\n\n\t\tsendUserMessage(content, options): void {\n\t\t\truntime.assertActive();\n\t\t\truntime.sendUserMessage(content, options);\n\t\t},\n\n\t\tappendEntry(customType: string, data?: unknown): void {\n\t\t\truntime.assertActive();\n\t\t\truntime.appendEntry(customType, data);\n\t\t},\n\n\t\tsetSessionName(name: string): void {\n\t\t\truntime.assertActive();\n\t\t\truntime.setSessionName(name);\n\t\t},\n\n\t\tgetSessionName(): string | undefined {\n\t\t\truntime.assertActive();\n\t\t\treturn runtime.getSessionName();\n\t\t},\n\n\t\tsetLabel(entryId: string, label: string | undefined): void {\n\t\t\truntime.assertActive();\n\t\t\truntime.setLabel(entryId, label);\n\t\t},\n\n\t\texec(command: string, args: string[], options?: ExecOptions) {\n\t\t\truntime.assertActive();\n\t\t\treturn execCommand(command, args, options?.cwd ?? cwd, options);\n\t\t},\n\n\t\tgetActiveTools(): string[] {\n\t\t\truntime.assertActive();\n\t\t\treturn runtime.getActiveTools();\n\t\t},\n\n\t\tgetAllTools() {\n\t\t\truntime.assertActive();\n\t\t\treturn runtime.getAllTools();\n\t\t},\n\n\t\tsetActiveTools(toolNames: string[]): void {\n\t\t\truntime.assertActive();\n\t\t\truntime.setActiveTools(toolNames);\n\t\t},\n\n\t\tgetCommands() {\n\t\t\truntime.assertActive();\n\t\t\treturn runtime.getCommands();\n\t\t},\n\n\t\tsetModel(model) {\n\t\t\truntime.assertActive();\n\t\t\treturn runtime.setModel(model);\n\t\t},\n\n\t\tgetThinkingLevel() {\n\t\t\truntime.assertActive();\n\t\t\treturn runtime.getThinkingLevel();\n\t\t},\n\n\t\tsetThinkingLevel(level) {\n\t\t\truntime.assertActive();\n\t\t\truntime.setThinkingLevel(level);\n\t\t},\n\n\t\tregisterProvider(name: string, config: ProviderConfig) {\n\t\t\truntime.assertActive();\n\t\t\truntime.registerProvider(name, config, extension.path);\n\t\t},\n\n\t\tunregisterProvider(name: string) {\n\t\t\truntime.assertActive();\n\t\t\truntime.unregisterProvider(name, extension.path);\n\t\t},\n\n\t\tevents: eventBus,\n\t} as ExtensionAPI;\n\n\treturn api;\n}\n\nfunction createExtensionModuleImporter(): ExtensionModuleImporter {\n\treturn createJiti(import.meta.url, {\n\t\tmoduleCache: false,\n\t\t// In Bun binary: use virtualModules for bundled packages (no filesystem resolution)\n\t\t// Also disable tryNative so jiti handles ALL imports (not just the entry point)\n\t\t// In Node.js/dev: use aliases to resolve to node_modules paths\n\t\t...(isBunBinary ? { virtualModules: VIRTUAL_MODULES, tryNative: false } : { alias: getAliases() }),\n\t});\n}\n\nasync function loadExtensionModule(extensionPath: string, importer: ExtensionModuleImporter) {\n\tconst module = await importer.import(extensionPath, { default: true });\n\tconst factory = module as ExtensionFactory;\n\treturn typeof factory !== \"function\" ? undefined : factory;\n}\n\n/**\n * Create an Extension object with empty collections.\n */\nfunction createExtension(extensionPath: string, resolvedPath: string): Extension {\n\tconst source =\n\t\textensionPath.startsWith(\"<\") && extensionPath.endsWith(\">\")\n\t\t\t? extensionPath.slice(1, -1).split(\":\")[0] || \"temporary\"\n\t\t\t: \"local\";\n\tconst baseDir = extensionPath.startsWith(\"<\") ? undefined : path.dirname(resolvedPath);\n\n\treturn {\n\t\tpath: extensionPath,\n\t\tresolvedPath,\n\t\tsourceInfo: createSyntheticSourceInfo(extensionPath, { source, baseDir }),\n\t\thandlers: new Map(),\n\t\ttools: new Map(),\n\t\tmessageRenderers: new Map(),\n\t\tcommands: new Map(),\n\t\tflags: new Map(),\n\t\tshortcuts: new Map(),\n\t};\n}\n\nasync function loadExtension(\n\textensionPath: string,\n\tcwd: string,\n\teventBus: EventBus,\n\truntime: ExtensionRuntime,\n\tgetImporter: () => ExtensionModuleImporter,\n\tfactoryResolver?: ExtensionFactoryResolver,\n): Promise<{ extension: Extension | null; error: string | null }> {\n\tconst resolvedPath = resolvePath(extensionPath, cwd, { normalizeUnicodeSpaces: true });\n\n\ttry {\n\t\tconst factory =\n\t\t\tfactoryResolver?.(extensionPath, resolvedPath) ?? (await loadExtensionModule(resolvedPath, getImporter()));\n\t\tif (!factory) {\n\t\t\treturn { extension: null, error: `Extension does not export a valid factory function: ${extensionPath}` };\n\t\t}\n\n\t\tconst extension = createExtension(extensionPath, resolvedPath);\n\t\tconst api = createExtensionAPI(extension, runtime, cwd, eventBus);\n\t\tawait factory(api);\n\n\t\treturn { extension, error: null };\n\t} catch (err) {\n\t\tconst message = err instanceof Error ? err.message : String(err);\n\t\treturn { extension: null, error: `Failed to load extension: ${message}` };\n\t}\n}\n\n/**\n * Create an Extension from an inline factory function.\n */\nexport async function loadExtensionFromFactory(\n\tfactory: ExtensionFactory,\n\tcwd: string,\n\teventBus: EventBus,\n\truntime: ExtensionRuntime,\n\textensionPath = \"<inline>\",\n): Promise<Extension> {\n\tconst extension = createExtension(extensionPath, extensionPath);\n\tconst resolvedCwd = resolvePath(cwd);\n\tconst api = createExtensionAPI(extension, runtime, resolvedCwd, eventBus);\n\tawait factory(api);\n\treturn extension;\n}\n\n/**\n * Load extensions from paths.\n */\nexport async function loadExtensions(\n\tpaths: string[],\n\tcwd: string,\n\teventBus?: EventBus,\n\toptions?: { factoryResolver?: ExtensionFactoryResolver },\n): Promise<LoadExtensionsResult> {\n\tconst extensions: Extension[] = [];\n\tconst errors: Array<{ path: string; error: string }> = [];\n\tconst resolvedCwd = resolvePath(cwd);\n\tconst resolvedEventBus = eventBus ?? createEventBus();\n\tconst runtime = createExtensionRuntime();\n\tlet importer: ExtensionModuleImporter | undefined;\n\tconst getImporter = () => {\n\t\timporter ??= createExtensionModuleImporter();\n\t\treturn importer;\n\t};\n\n\tfor (const extPath of paths) {\n\t\tconst { extension, error } = await loadExtension(\n\t\t\textPath,\n\t\t\tresolvedCwd,\n\t\t\tresolvedEventBus,\n\t\t\truntime,\n\t\t\tgetImporter,\n\t\t\toptions?.factoryResolver,\n\t\t);\n\n\t\tif (error) {\n\t\t\terrors.push({ path: extPath, error });\n\t\t\tcontinue;\n\t\t}\n\n\t\tif (extension) {\n\t\t\textensions.push(extension);\n\t\t}\n\t}\n\n\treturn {\n\t\textensions,\n\t\terrors,\n\t\truntime,\n\t};\n}\n\ninterface PiManifest {\n\textensions?: string[];\n\tthemes?: string[];\n\tskills?: string[];\n\tprompts?: string[];\n}\n\nfunction readPiManifest(packageJsonPath: string): PiManifest | null {\n\ttry {\n\t\tconst content = fs.readFileSync(packageJsonPath, \"utf-8\");\n\t\tconst pkg = JSON.parse(content);\n\t\tif (pkg.pi && typeof pkg.pi === \"object\") {\n\t\t\treturn pkg.pi as PiManifest;\n\t\t}\n\t\treturn null;\n\t} catch {\n\t\treturn null;\n\t}\n}\n\nfunction isExtensionFile(name: string): boolean {\n\treturn name.endsWith(\".ts\") || name.endsWith(\".js\");\n}\n\n/**\n * Resolve extension entry points from a directory.\n *\n * Checks for:\n * 1. package.json with \"pi.extensions\" field -> returns declared paths\n * 2. index.ts or index.js -> returns the index file\n *\n * Returns resolved paths or null if no entry points found.\n */\nfunction resolveExtensionEntries(dir: string): string[] | null {\n\t// Check for package.json with \"pi\" field first\n\tconst packageJsonPath = path.join(dir, \"package.json\");\n\tif (fs.existsSync(packageJsonPath)) {\n\t\tconst manifest = readPiManifest(packageJsonPath);\n\t\tif (manifest?.extensions?.length) {\n\t\t\tconst entries: string[] = [];\n\t\t\tfor (const extPath of manifest.extensions) {\n\t\t\t\tconst resolvedExtPath = path.resolve(dir, extPath);\n\t\t\t\tif (fs.existsSync(resolvedExtPath)) {\n\t\t\t\t\tentries.push(resolvedExtPath);\n\t\t\t\t}\n\t\t\t}\n\t\t\tif (entries.length > 0) {\n\t\t\t\treturn entries;\n\t\t\t}\n\t\t}\n\t}\n\n\t// Check for index.ts or index.js\n\tconst indexTs = path.join(dir, \"index.ts\");\n\tconst indexJs = path.join(dir, \"index.js\");\n\tif (fs.existsSync(indexTs)) {\n\t\treturn [indexTs];\n\t}\n\tif (fs.existsSync(indexJs)) {\n\t\treturn [indexJs];\n\t}\n\n\treturn null;\n}\n\n/**\n * Discover extensions in a directory.\n *\n * Discovery rules:\n * 1. Direct files: `extensions/*.ts` or `*.js` → load\n * 2. Subdirectory with index: `extensions/* /index.ts` or `index.js` → load\n * 3. Subdirectory with package.json: `extensions/* /package.json` with \"pi\" field → load what it declares\n *\n * No recursion beyond one level. Complex packages must use package.json manifest.\n */\nfunction discoverExtensionsInDir(dir: string): string[] {\n\tif (!fs.existsSync(dir)) {\n\t\treturn [];\n\t}\n\n\tconst discovered: string[] = [];\n\n\ttry {\n\t\tconst entries = fs.readdirSync(dir, { withFileTypes: true });\n\n\t\tfor (const entry of entries) {\n\t\t\tconst entryPath = path.join(dir, entry.name);\n\n\t\t\t// 1. Direct files: *.ts or *.js\n\t\t\tif ((entry.isFile() || entry.isSymbolicLink()) && isExtensionFile(entry.name)) {\n\t\t\t\tdiscovered.push(entryPath);\n\t\t\t\tcontinue;\n\t\t\t}\n\n\t\t\t// 2 & 3. Subdirectories\n\t\t\tif (entry.isDirectory() || entry.isSymbolicLink()) {\n\t\t\t\tconst entries = resolveExtensionEntries(entryPath);\n\t\t\t\tif (entries) {\n\t\t\t\t\tdiscovered.push(...entries);\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\t} catch {\n\t\treturn [];\n\t}\n\n\treturn discovered;\n}\n\n/**\n * Discover and load extensions from standard locations.\n */\nexport async function discoverAndLoadExtensions(\n\tconfiguredPaths: string[],\n\tcwd: string,\n\tagentDir: string = getAgentDir(),\n\teventBus?: EventBus,\n): Promise<LoadExtensionsResult> {\n\tconst resolvedCwd = resolvePath(cwd);\n\tconst resolvedAgentDir = resolvePath(agentDir);\n\tconst allPaths: string[] = [];\n\tconst seen = new Set<string>();\n\n\tconst addPaths = (paths: string[]) => {\n\t\tfor (const p of paths) {\n\t\t\tconst resolved = path.resolve(p);\n\t\t\tif (!seen.has(resolved)) {\n\t\t\t\tseen.add(resolved);\n\t\t\t\tallPaths.push(p);\n\t\t\t}\n\t\t}\n\t};\n\n\t// 1. Project-local extensions: cwd/${CONFIG_DIR_NAME}/extensions/\n\tconst localExtDir = path.join(resolvedCwd, CONFIG_DIR_NAME, \"extensions\");\n\taddPaths(discoverExtensionsInDir(localExtDir));\n\n\t// 2. Global extensions: agentDir/extensions/\n\tconst globalExtDir = path.join(resolvedAgentDir, \"extensions\");\n\taddPaths(discoverExtensionsInDir(globalExtDir));\n\n\t// 3. Explicitly configured paths\n\tfor (const p of configuredPaths) {\n\t\tconst resolved = resolvePath(p, resolvedCwd, { normalizeUnicodeSpaces: true });\n\t\tif (fs.existsSync(resolved) && fs.statSync(resolved).isDirectory()) {\n\t\t\t// Check for package.json with pi manifest or index.ts\n\t\t\tconst entries = resolveExtensionEntries(resolved);\n\t\t\tif (entries) {\n\t\t\t\taddPaths(entries);\n\t\t\t\tcontinue;\n\t\t\t}\n\t\t\t// No explicit entries - discover individual files in directory\n\t\t\taddPaths(discoverExtensionsInDir(resolved));\n\t\t\tcontinue;\n\t\t}\n\n\t\taddPaths([resolved]);\n\t}\n\n\treturn loadExtensions(allPaths, resolvedCwd, eventBus);\n}\n"]}
|
|
@@ -55,23 +55,35 @@ function getAliases() {
|
|
|
55
55
|
if (_aliases)
|
|
56
56
|
return _aliases;
|
|
57
57
|
const __dirname = path.dirname(fileURLToPath(import.meta.url));
|
|
58
|
-
const packageIndex = path.resolve(__dirname, "../..", "index.js");
|
|
59
58
|
const typeboxEntry = require.resolve("typebox");
|
|
60
59
|
const typeboxCompileEntry = require.resolve("typebox/compile");
|
|
61
60
|
const typeboxValueEntry = require.resolve("typebox/value");
|
|
62
|
-
const
|
|
63
|
-
const
|
|
61
|
+
const codingAgentPackageRoot = path.resolve(__dirname, "../../..");
|
|
62
|
+
const packagesRoot = path.dirname(codingAgentPackageRoot);
|
|
63
|
+
const resolveWorkspaceOrBundled = (packageName, packageEntryRelativePath, workspaceRelativePath, sourceRelativePath, specifier) => {
|
|
64
|
+
const bundledPath = path.join(codingAgentPackageRoot, "node_modules", packageName, packageEntryRelativePath);
|
|
65
|
+
if (fs.existsSync(bundledPath)) {
|
|
66
|
+
return bundledPath;
|
|
67
|
+
}
|
|
64
68
|
const workspacePath = path.join(packagesRoot, workspaceRelativePath);
|
|
65
69
|
if (fs.existsSync(workspacePath)) {
|
|
66
70
|
return workspacePath;
|
|
67
71
|
}
|
|
68
|
-
|
|
72
|
+
const sourcePath = path.join(packagesRoot, sourceRelativePath);
|
|
73
|
+
if (fs.existsSync(sourcePath)) {
|
|
74
|
+
return sourcePath;
|
|
75
|
+
}
|
|
76
|
+
return typeof import.meta.resolve === "function"
|
|
77
|
+
? fileURLToPath(import.meta.resolve(specifier))
|
|
78
|
+
: require.resolve(specifier);
|
|
69
79
|
};
|
|
70
|
-
const
|
|
71
|
-
const
|
|
72
|
-
const
|
|
73
|
-
const
|
|
74
|
-
const
|
|
80
|
+
const piCodingAgentDistEntry = path.join(codingAgentPackageRoot, "dist/index.js");
|
|
81
|
+
const piCodingAgentSourceEntry = path.join(codingAgentPackageRoot, "src/index.ts");
|
|
82
|
+
const piCodingAgentEntry = fs.existsSync(piCodingAgentDistEntry) ? piCodingAgentDistEntry : piCodingAgentSourceEntry;
|
|
83
|
+
const piAgentCoreEntry = resolveWorkspaceOrBundled("@earendil-works/pi-agent-core", "dist/index.js", "agent/dist/index.js", "agent/src/index.ts", "@earendil-works/pi-agent-core");
|
|
84
|
+
const piTuiEntry = resolveWorkspaceOrBundled("@earendil-works/pi-tui", "dist/index.js", "tui/dist/index.js", "tui/src/index.ts", "@earendil-works/pi-tui");
|
|
85
|
+
const piAiEntry = resolveWorkspaceOrBundled("@earendil-works/pi-ai", "dist/index.js", "ai/dist/index.js", "ai/src/index.ts", "@earendil-works/pi-ai");
|
|
86
|
+
const piAiOauthEntry = resolveWorkspaceOrBundled("@earendil-works/pi-ai", "dist/oauth.js", "ai/dist/oauth.js", "ai/src/oauth.ts", "@earendil-works/pi-ai/oauth");
|
|
75
87
|
_aliases = {
|
|
76
88
|
"@code-yeongyu/senpi": piCodingAgentEntry,
|
|
77
89
|
"@mariozechner/pi-coding-agent": piCodingAgentEntry,
|