@renxqoo/renx-code 0.0.2 → 0.0.4
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/README.md +59 -223
- package/bin/renx.cjs +34 -0
- package/package.json +27 -83
- package/src/App.tsx +297 -0
- package/src/agent/runtime/event-format.ts +258 -0
- package/src/agent/runtime/model-types.ts +13 -0
- package/src/agent/runtime/runtime.context-usage.test.ts +193 -0
- package/src/agent/runtime/runtime.error-handling.test.ts +236 -0
- package/src/agent/runtime/runtime.simple.test.ts +16 -0
- package/src/agent/runtime/runtime.test.ts +293 -0
- package/src/agent/runtime/runtime.ts +881 -0
- package/src/agent/runtime/runtime.usage-forwarding.test.ts +229 -0
- package/src/agent/runtime/source-modules.test.ts +57 -0
- package/src/agent/runtime/source-modules.ts +353 -0
- package/src/agent/runtime/tool-call-buffer.test.ts +65 -0
- package/src/agent/runtime/tool-call-buffer.ts +60 -0
- package/src/agent/runtime/tool-confirmation.test.ts +56 -0
- package/src/agent/runtime/tool-confirmation.ts +15 -0
- package/src/agent/runtime/types.ts +99 -0
- package/src/commands/slash-commands.test.ts +216 -0
- package/src/commands/slash-commands.ts +64 -0
- package/src/components/chat/assistant-reply.test.tsx +47 -0
- package/src/components/chat/assistant-reply.tsx +136 -0
- package/src/components/chat/assistant-segment.test.ts +99 -0
- package/src/components/chat/assistant-segment.tsx +125 -0
- package/src/components/chat/assistant-tool-group.tsx +900 -0
- package/src/components/chat/code-block.test.tsx +206 -0
- package/src/components/chat/code-block.tsx +313 -0
- package/src/components/chat/prompt-card.tsx +81 -0
- package/src/components/chat/segment-groups.test.ts +52 -0
- package/src/components/chat/segment-groups.ts +106 -0
- package/src/components/chat/turn-item.tsx +39 -0
- package/src/components/conversation-panel.tsx +43 -0
- package/src/components/file-mention-menu.tsx +77 -0
- package/src/components/file-picker-dialog.tsx +206 -0
- package/src/components/footer-hints.tsx +75 -0
- package/src/components/model-picker-dialog.tsx +248 -0
- package/src/components/prompt.tsx +233 -0
- package/src/components/slash-command-menu.tsx +65 -0
- package/src/components/tool-confirm-dialog-content.test.ts +103 -0
- package/src/components/tool-confirm-dialog-content.ts +186 -0
- package/src/components/tool-confirm-dialog.tsx +187 -0
- package/src/components/tool-display-config.ts +119 -0
- package/src/context-usage-regressions.test.ts +26 -0
- package/src/files/attachment-capabilities.test.ts +30 -0
- package/src/files/attachment-capabilities.ts +50 -0
- package/src/files/attachment-content.ts +153 -0
- package/src/files/file-mention-query.test.ts +34 -0
- package/src/files/file-mention-query.ts +32 -0
- package/src/files/prompt-display.ts +13 -0
- package/src/files/types.ts +5 -0
- package/src/files/workspace-files.ts +63 -0
- package/src/hooks/agent-event-handlers.test.ts +207 -0
- package/src/hooks/agent-event-handlers.ts +196 -0
- package/src/hooks/chat-local-replies.fixed.test.ts +119 -0
- package/src/hooks/chat-local-replies.test.ts +153 -0
- package/src/hooks/chat-local-replies.ts +63 -0
- package/src/hooks/turn-updater.test.ts +70 -0
- package/src/hooks/turn-updater.ts +166 -0
- package/src/hooks/use-agent-chat.context.test.ts +10 -0
- package/src/hooks/use-agent-chat.status.test.ts +14 -0
- package/src/hooks/use-agent-chat.test.ts +80 -0
- package/src/hooks/use-agent-chat.ts +621 -0
- package/src/hooks/use-file-mention-menu.ts +196 -0
- package/src/hooks/use-file-picker.ts +185 -0
- package/src/hooks/use-model-picker.ts +196 -0
- package/src/hooks/use-slash-command-menu.ts +154 -0
- package/src/index.tsx +55 -0
- package/src/runtime/clipboard.test.ts +43 -0
- package/src/runtime/clipboard.ts +89 -0
- package/src/runtime/exit.test.ts +177 -0
- package/src/runtime/exit.ts +98 -0
- package/src/runtime/runtime-support.test.ts +31 -0
- package/src/runtime/terminal-theme.test.ts +55 -0
- package/src/runtime/terminal-theme.ts +196 -0
- package/src/types/chat.ts +32 -0
- package/src/types/message-content.ts +48 -0
- package/src/ui/open-code-theme.ts +176 -0
- package/src/ui/opencode-markdown.ts +211 -0
- package/src/ui/theme.simple.test.ts +52 -0
- package/src/ui/theme.test.ts +151 -0
- package/src/ui/theme.ts +152 -0
- package/src/utils/time.test.ts +144 -0
- package/src/utils/time.ts +7 -0
- package/tsconfig.json +30 -0
- package/LICENSE +0 -21
- package/dist/App.d.ts +0 -2
- package/dist/App.d.ts.map +0 -1
- package/dist/App.js +0 -170
- package/dist/App.js.map +0 -1
- package/dist/agent/prompts/system.d.ts +0 -24
- package/dist/agent/prompts/system.d.ts.map +0 -1
- package/dist/agent/prompts/system.js +0 -222
- package/dist/agent/prompts/system.js.map +0 -1
- package/dist/agent/runtime/event-format.d.ts +0 -17
- package/dist/agent/runtime/event-format.d.ts.map +0 -1
- package/dist/agent/runtime/event-format.js +0 -194
- package/dist/agent/runtime/event-format.js.map +0 -1
- package/dist/agent/runtime/model-types.d.ts +0 -13
- package/dist/agent/runtime/model-types.d.ts.map +0 -1
- package/dist/agent/runtime/model-types.js +0 -1
- package/dist/agent/runtime/model-types.js.map +0 -1
- package/dist/agent/runtime/runtime.d.ts +0 -16
- package/dist/agent/runtime/runtime.d.ts.map +0 -1
- package/dist/agent/runtime/runtime.js +0 -691
- package/dist/agent/runtime/runtime.js.map +0 -1
- package/dist/agent/runtime/source-modules.d.ts +0 -176
- package/dist/agent/runtime/source-modules.d.ts.map +0 -1
- package/dist/agent/runtime/source-modules.js +0 -110
- package/dist/agent/runtime/source-modules.js.map +0 -1
- package/dist/agent/runtime/tool-call-buffer.d.ts +0 -12
- package/dist/agent/runtime/tool-call-buffer.d.ts.map +0 -1
- package/dist/agent/runtime/tool-call-buffer.js +0 -48
- package/dist/agent/runtime/tool-call-buffer.js.map +0 -1
- package/dist/agent/runtime/tool-confirmation.d.ts +0 -3
- package/dist/agent/runtime/tool-confirmation.d.ts.map +0 -1
- package/dist/agent/runtime/tool-confirmation.js +0 -9
- package/dist/agent/runtime/tool-confirmation.js.map +0 -1
- package/dist/agent/runtime/types.d.ts +0 -86
- package/dist/agent/runtime/types.d.ts.map +0 -1
- package/dist/agent/runtime/types.js +0 -1
- package/dist/agent/runtime/types.js.map +0 -1
- package/dist/cli.d.ts +0 -3
- package/dist/cli.d.ts.map +0 -1
- package/dist/cli.js +0 -43
- package/dist/cli.js.map +0 -1
- package/dist/commands/slash-commands.d.ts +0 -11
- package/dist/commands/slash-commands.d.ts.map +0 -1
- package/dist/commands/slash-commands.js +0 -48
- package/dist/commands/slash-commands.js.map +0 -1
- package/dist/components/chat/assistant-reply.d.ts +0 -13
- package/dist/components/chat/assistant-reply.d.ts.map +0 -1
- package/dist/components/chat/assistant-reply.js +0 -78
- package/dist/components/chat/assistant-reply.js.map +0 -1
- package/dist/components/chat/assistant-segment.d.ts +0 -8
- package/dist/components/chat/assistant-segment.d.ts.map +0 -1
- package/dist/components/chat/assistant-segment.js +0 -54
- package/dist/components/chat/assistant-segment.js.map +0 -1
- package/dist/components/chat/assistant-tool-group.d.ts +0 -7
- package/dist/components/chat/assistant-tool-group.d.ts.map +0 -1
- package/dist/components/chat/assistant-tool-group.js +0 -695
- package/dist/components/chat/assistant-tool-group.js.map +0 -1
- package/dist/components/chat/code-block.d.ts +0 -16
- package/dist/components/chat/code-block.d.ts.map +0 -1
- package/dist/components/chat/code-block.js +0 -194
- package/dist/components/chat/code-block.js.map +0 -1
- package/dist/components/chat/prompt-card.d.ts +0 -9
- package/dist/components/chat/prompt-card.d.ts.map +0 -1
- package/dist/components/chat/prompt-card.js +0 -18
- package/dist/components/chat/prompt-card.js.map +0 -1
- package/dist/components/chat/segment-groups.d.ts +0 -24
- package/dist/components/chat/segment-groups.d.ts.map +0 -1
- package/dist/components/chat/segment-groups.js +0 -69
- package/dist/components/chat/segment-groups.js.map +0 -1
- package/dist/components/chat/turn-item.d.ts +0 -9
- package/dist/components/chat/turn-item.d.ts.map +0 -1
- package/dist/components/chat/turn-item.js +0 -11
- package/dist/components/chat/turn-item.js.map +0 -1
- package/dist/components/conversation-panel.d.ts +0 -8
- package/dist/components/conversation-panel.d.ts.map +0 -1
- package/dist/components/conversation-panel.js +0 -8
- package/dist/components/conversation-panel.js.map +0 -1
- package/dist/components/file-mention-menu.d.ts +0 -11
- package/dist/components/file-mention-menu.d.ts.map +0 -1
- package/dist/components/file-mention-menu.js +0 -15
- package/dist/components/file-mention-menu.js.map +0 -1
- package/dist/components/file-picker-dialog.d.ts +0 -21
- package/dist/components/file-picker-dialog.d.ts.map +0 -1
- package/dist/components/file-picker-dialog.js +0 -48
- package/dist/components/file-picker-dialog.js.map +0 -1
- package/dist/components/footer-hints.d.ts +0 -7
- package/dist/components/footer-hints.d.ts.map +0 -1
- package/dist/components/footer-hints.js +0 -29
- package/dist/components/footer-hints.js.map +0 -1
- package/dist/components/model-picker-dialog.d.ts +0 -20
- package/dist/components/model-picker-dialog.d.ts.map +0 -1
- package/dist/components/model-picker-dialog.js +0 -72
- package/dist/components/model-picker-dialog.js.map +0 -1
- package/dist/components/prompt.d.ts +0 -18
- package/dist/components/prompt.d.ts.map +0 -1
- package/dist/components/prompt.js +0 -96
- package/dist/components/prompt.js.map +0 -1
- package/dist/components/slash-command-menu.d.ts +0 -9
- package/dist/components/slash-command-menu.d.ts.map +0 -1
- package/dist/components/slash-command-menu.js +0 -20
- package/dist/components/slash-command-menu.js.map +0 -1
- package/dist/components/tool-confirm-dialog-content.d.ts +0 -15
- package/dist/components/tool-confirm-dialog-content.d.ts.map +0 -1
- package/dist/components/tool-confirm-dialog-content.js +0 -143
- package/dist/components/tool-confirm-dialog-content.js.map +0 -1
- package/dist/components/tool-confirm-dialog.d.ts +0 -12
- package/dist/components/tool-confirm-dialog.d.ts.map +0 -1
- package/dist/components/tool-confirm-dialog.js +0 -21
- package/dist/components/tool-confirm-dialog.js.map +0 -1
- package/dist/components/tool-display-config.d.ts +0 -11
- package/dist/components/tool-display-config.d.ts.map +0 -1
- package/dist/components/tool-display-config.js +0 -94
- package/dist/components/tool-display-config.js.map +0 -1
- package/dist/config/paths.d.ts +0 -7
- package/dist/config/paths.d.ts.map +0 -1
- package/dist/config/paths.js +0 -24
- package/dist/config/paths.js.map +0 -1
- package/dist/files/attachment-capabilities.d.ts +0 -19
- package/dist/files/attachment-capabilities.d.ts.map +0 -1
- package/dist/files/attachment-capabilities.js +0 -26
- package/dist/files/attachment-capabilities.js.map +0 -1
- package/dist/files/attachment-content.d.ts +0 -5
- package/dist/files/attachment-content.d.ts.map +0 -1
- package/dist/files/attachment-content.js +0 -117
- package/dist/files/attachment-content.js.map +0 -1
- package/dist/files/file-mention-query.d.ts +0 -9
- package/dist/files/file-mention-query.d.ts.map +0 -1
- package/dist/files/file-mention-query.js +0 -23
- package/dist/files/file-mention-query.js.map +0 -1
- package/dist/files/prompt-display.d.ts +0 -3
- package/dist/files/prompt-display.d.ts.map +0 -1
- package/dist/files/prompt-display.js +0 -11
- package/dist/files/prompt-display.js.map +0 -1
- package/dist/files/types.d.ts +0 -6
- package/dist/files/types.d.ts.map +0 -1
- package/dist/files/types.js +0 -1
- package/dist/files/types.js.map +0 -1
- package/dist/files/workspace-files.d.ts +0 -3
- package/dist/files/workspace-files.d.ts.map +0 -1
- package/dist/files/workspace-files.js +0 -50
- package/dist/files/workspace-files.js.map +0 -1
- package/dist/hooks/agent-event-handlers.d.ts +0 -11
- package/dist/hooks/agent-event-handlers.d.ts.map +0 -1
- package/dist/hooks/agent-event-handlers.js +0 -137
- package/dist/hooks/agent-event-handlers.js.map +0 -1
- package/dist/hooks/chat-local-replies.d.ts +0 -9
- package/dist/hooks/chat-local-replies.d.ts.map +0 -1
- package/dist/hooks/chat-local-replies.js +0 -54
- package/dist/hooks/chat-local-replies.js.map +0 -1
- package/dist/hooks/turn-updater.d.ts +0 -9
- package/dist/hooks/turn-updater.d.ts.map +0 -1
- package/dist/hooks/turn-updater.js +0 -103
- package/dist/hooks/turn-updater.js.map +0 -1
- package/dist/hooks/use-agent-chat.d.ts +0 -29
- package/dist/hooks/use-agent-chat.d.ts.map +0 -1
- package/dist/hooks/use-agent-chat.js +0 -455
- package/dist/hooks/use-agent-chat.js.map +0 -1
- package/dist/hooks/use-file-mention-menu.d.ts +0 -22
- package/dist/hooks/use-file-mention-menu.d.ts.map +0 -1
- package/dist/hooks/use-file-mention-menu.js +0 -137
- package/dist/hooks/use-file-mention-menu.js.map +0 -1
- package/dist/hooks/use-file-picker.d.ts +0 -21
- package/dist/hooks/use-file-picker.d.ts.map +0 -1
- package/dist/hooks/use-file-picker.js +0 -145
- package/dist/hooks/use-file-picker.js.map +0 -1
- package/dist/hooks/use-model-picker.d.ts +0 -23
- package/dist/hooks/use-model-picker.d.ts.map +0 -1
- package/dist/hooks/use-model-picker.js +0 -151
- package/dist/hooks/use-model-picker.js.map +0 -1
- package/dist/hooks/use-slash-command-menu.d.ts +0 -19
- package/dist/hooks/use-slash-command-menu.d.ts.map +0 -1
- package/dist/hooks/use-slash-command-menu.js +0 -101
- package/dist/hooks/use-slash-command-menu.js.map +0 -1
- package/dist/index.d.ts +0 -2
- package/dist/index.d.ts.map +0 -1
- package/dist/index.js +0 -39
- package/dist/index.js.map +0 -1
- package/dist/runtime/clipboard.d.ts +0 -10
- package/dist/runtime/clipboard.d.ts.map +0 -1
- package/dist/runtime/clipboard.js +0 -64
- package/dist/runtime/clipboard.js.map +0 -1
- package/dist/runtime/exit.d.ts +0 -7
- package/dist/runtime/exit.d.ts.map +0 -1
- package/dist/runtime/exit.js +0 -85
- package/dist/runtime/exit.js.map +0 -1
- package/dist/runtime/terminal-theme.d.ts +0 -25
- package/dist/runtime/terminal-theme.d.ts.map +0 -1
- package/dist/runtime/terminal-theme.js +0 -148
- package/dist/runtime/terminal-theme.js.map +0 -1
- package/dist/types/chat.d.ts +0 -29
- package/dist/types/chat.d.ts.map +0 -1
- package/dist/types/chat.js +0 -1
- package/dist/types/chat.js.map +0 -1
- package/dist/types/message-content.d.ts +0 -38
- package/dist/types/message-content.d.ts.map +0 -1
- package/dist/types/message-content.js +0 -1
- package/dist/types/message-content.js.map +0 -1
- package/dist/ui/open-code-theme.d.ts +0 -58
- package/dist/ui/open-code-theme.d.ts.map +0 -1
- package/dist/ui/open-code-theme.js +0 -113
- package/dist/ui/open-code-theme.js.map +0 -1
- package/dist/ui/opencode-markdown.d.ts +0 -7
- package/dist/ui/opencode-markdown.d.ts.map +0 -1
- package/dist/ui/opencode-markdown.js +0 -169
- package/dist/ui/opencode-markdown.js.map +0 -1
- package/dist/ui/theme.d.ts +0 -68
- package/dist/ui/theme.d.ts.map +0 -1
- package/dist/ui/theme.js +0 -80
- package/dist/ui/theme.js.map +0 -1
- package/dist/utils/time.d.ts +0 -2
- package/dist/utils/time.d.ts.map +0 -1
- package/dist/utils/time.js +0 -7
- package/dist/utils/time.js.map +0 -1
|
@@ -0,0 +1,206 @@
|
|
|
1
|
+
import { describe, expect, it } from 'vitest';
|
|
2
|
+
|
|
3
|
+
import {
|
|
4
|
+
CodeBlock,
|
|
5
|
+
extractDiffPath,
|
|
6
|
+
inferCodeFiletype,
|
|
7
|
+
inferFiletypeFromPath,
|
|
8
|
+
looksLikeDiff,
|
|
9
|
+
} from './code-block';
|
|
10
|
+
|
|
11
|
+
type ElementLike = {
|
|
12
|
+
type: unknown;
|
|
13
|
+
props?: {
|
|
14
|
+
children?: unknown;
|
|
15
|
+
[key: string]: unknown;
|
|
16
|
+
};
|
|
17
|
+
};
|
|
18
|
+
|
|
19
|
+
const isElementLike = (value: unknown): value is ElementLike => {
|
|
20
|
+
return (
|
|
21
|
+
Boolean(value) && typeof value === 'object' && 'type' in (value as Record<string, unknown>)
|
|
22
|
+
);
|
|
23
|
+
};
|
|
24
|
+
|
|
25
|
+
const findElementByType = (node: unknown, targetType: string): ElementLike | null => {
|
|
26
|
+
if (!node) {
|
|
27
|
+
return null;
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
if (Array.isArray(node)) {
|
|
31
|
+
for (const child of node) {
|
|
32
|
+
const match = findElementByType(child, targetType);
|
|
33
|
+
if (match) {
|
|
34
|
+
return match;
|
|
35
|
+
}
|
|
36
|
+
}
|
|
37
|
+
return null;
|
|
38
|
+
}
|
|
39
|
+
|
|
40
|
+
if (!isElementLike(node)) {
|
|
41
|
+
return null;
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
if (typeof node.type === 'function') {
|
|
45
|
+
return findElementByType(
|
|
46
|
+
(node.type as (props: object) => unknown)(node.props ?? {}),
|
|
47
|
+
targetType
|
|
48
|
+
);
|
|
49
|
+
}
|
|
50
|
+
|
|
51
|
+
if (node.type === targetType) {
|
|
52
|
+
return node;
|
|
53
|
+
}
|
|
54
|
+
|
|
55
|
+
return findElementByType(node.props?.children, targetType);
|
|
56
|
+
};
|
|
57
|
+
|
|
58
|
+
describe('CodeBlock', () => {
|
|
59
|
+
it('detects unified diff content and extracts the changed path', () => {
|
|
60
|
+
const diff = [
|
|
61
|
+
'diff --git a/src/App.tsx b/src/App.tsx',
|
|
62
|
+
'--- a/src/App.tsx',
|
|
63
|
+
'+++ b/src/App.tsx',
|
|
64
|
+
'@@ -1,2 +1,2 @@',
|
|
65
|
+
'-const before = true;',
|
|
66
|
+
'+const after = true;',
|
|
67
|
+
].join('\n');
|
|
68
|
+
|
|
69
|
+
expect(looksLikeDiff(diff)).toBe(true);
|
|
70
|
+
expect(extractDiffPath(diff)).toBe('src/App.tsx');
|
|
71
|
+
expect(inferCodeFiletype(diff)).toBe('diff');
|
|
72
|
+
expect(inferFiletypeFromPath('src/App.tsx')).toBe('tsx');
|
|
73
|
+
});
|
|
74
|
+
|
|
75
|
+
it('infers json and bash snippets without explicit metadata', () => {
|
|
76
|
+
expect(inferCodeFiletype('{\n "name": "demo"\n}')).toBe('json');
|
|
77
|
+
expect(inferCodeFiletype('$ pnpm test\n$ pnpm lint')).toBe('bash');
|
|
78
|
+
});
|
|
79
|
+
|
|
80
|
+
it('renders diff snippets with the OpenTUI diff component', () => {
|
|
81
|
+
const tree = CodeBlock({
|
|
82
|
+
label: 'output',
|
|
83
|
+
content: ['--- a/a.ts', '+++ b/a.ts', '@@ -1 +1 @@', '-a', '+b'].join('\n'),
|
|
84
|
+
});
|
|
85
|
+
|
|
86
|
+
const diffNode = findElementByType(tree, 'diff');
|
|
87
|
+
|
|
88
|
+
expect(diffNode).not.toBeNull();
|
|
89
|
+
expect(diffNode?.props?.view).toBe('unified');
|
|
90
|
+
expect(diffNode?.props?.showLineNumbers).toBe(true);
|
|
91
|
+
});
|
|
92
|
+
|
|
93
|
+
it('falls back to code preview when a diff block is collapsed', () => {
|
|
94
|
+
const content = [
|
|
95
|
+
'diff --git a/src/a.ts b/src/a.ts',
|
|
96
|
+
'--- a/src/a.ts',
|
|
97
|
+
'+++ b/src/a.ts',
|
|
98
|
+
'@@ -1,2 +1,22 @@',
|
|
99
|
+
'-const oldValue = 1;',
|
|
100
|
+
'+const newValue = 1;',
|
|
101
|
+
...Array.from({ length: 20 }, (_, index) => `+const line${index + 1} = ${index + 1};`),
|
|
102
|
+
].join('\n');
|
|
103
|
+
|
|
104
|
+
const tree = CodeBlock({
|
|
105
|
+
label: 'output',
|
|
106
|
+
content,
|
|
107
|
+
collapsible: true,
|
|
108
|
+
expanded: false,
|
|
109
|
+
});
|
|
110
|
+
|
|
111
|
+
expect(findElementByType(tree, 'diff')).toBeNull();
|
|
112
|
+
expect(findElementByType(tree, 'code')).not.toBeNull();
|
|
113
|
+
});
|
|
114
|
+
|
|
115
|
+
it('renders diff component when a collapsed diff block is expanded', () => {
|
|
116
|
+
const content = [
|
|
117
|
+
'diff --git a/src/a.ts b/src/a.ts',
|
|
118
|
+
'--- a/src/a.ts',
|
|
119
|
+
'+++ b/src/a.ts',
|
|
120
|
+
'@@ -1,2 +1,22 @@',
|
|
121
|
+
'-const oldValue = 1;',
|
|
122
|
+
'+const newValue = 1;',
|
|
123
|
+
...Array.from({ length: 20 }, (_, index) => `+const line${index + 1} = ${index + 1};`),
|
|
124
|
+
].join('\n');
|
|
125
|
+
|
|
126
|
+
const tree = CodeBlock({
|
|
127
|
+
label: 'output',
|
|
128
|
+
content,
|
|
129
|
+
collapsible: true,
|
|
130
|
+
expanded: true,
|
|
131
|
+
});
|
|
132
|
+
|
|
133
|
+
expect(findElementByType(tree, 'diff')).not.toBeNull();
|
|
134
|
+
});
|
|
135
|
+
|
|
136
|
+
it('renders regular snippets with the OpenTUI code component', () => {
|
|
137
|
+
const tree = CodeBlock({
|
|
138
|
+
label: 'arguments',
|
|
139
|
+
content: '{\n "timeout": 1000\n}',
|
|
140
|
+
languageHint: 'json',
|
|
141
|
+
});
|
|
142
|
+
|
|
143
|
+
const codeNode = findElementByType(tree, 'code');
|
|
144
|
+
|
|
145
|
+
expect(codeNode).not.toBeNull();
|
|
146
|
+
expect(codeNode?.props?.filetype).toBe('json');
|
|
147
|
+
});
|
|
148
|
+
|
|
149
|
+
it('collapses long output to 16 lines by default when enabled', () => {
|
|
150
|
+
const content = Array.from({ length: 20 }, (_, index) => `line ${index + 1}`).join('\n');
|
|
151
|
+
const expected = `${Array.from({ length: 16 }, (_, index) => `line ${index + 1}`).join('\n')}\n`;
|
|
152
|
+
|
|
153
|
+
const tree = CodeBlock({
|
|
154
|
+
label: 'output',
|
|
155
|
+
content,
|
|
156
|
+
collapsible: true,
|
|
157
|
+
expanded: false,
|
|
158
|
+
});
|
|
159
|
+
|
|
160
|
+
const codeNode = findElementByType(tree, 'code');
|
|
161
|
+
|
|
162
|
+
expect(codeNode).not.toBeNull();
|
|
163
|
+
expect(codeNode?.props?.content).toBe(expected);
|
|
164
|
+
});
|
|
165
|
+
|
|
166
|
+
it('shows full output when expanded is true', () => {
|
|
167
|
+
const content = Array.from({ length: 20 }, (_, index) => `line ${index + 1}`).join('\n');
|
|
168
|
+
const expected = content;
|
|
169
|
+
|
|
170
|
+
const tree = CodeBlock({
|
|
171
|
+
label: 'output',
|
|
172
|
+
content,
|
|
173
|
+
collapsible: true,
|
|
174
|
+
expanded: true,
|
|
175
|
+
});
|
|
176
|
+
|
|
177
|
+
const codeNode = findElementByType(tree, 'code');
|
|
178
|
+
|
|
179
|
+
expect(codeNode).not.toBeNull();
|
|
180
|
+
expect(codeNode?.props?.content).toBe(expected);
|
|
181
|
+
});
|
|
182
|
+
|
|
183
|
+
it('does not treat diagnostic text followed by a diff as a diff block', () => {
|
|
184
|
+
const content = [
|
|
185
|
+
"Error parsing diff: Hunk at line 5 contained invalid line Line 2');",
|
|
186
|
+
'Index: /tmp/task-errors.test.ts',
|
|
187
|
+
'===================================================================',
|
|
188
|
+
'--- /tmp/task-errors.test.ts original',
|
|
189
|
+
'+++ /tmp/task-errors.test.ts modified',
|
|
190
|
+
'@@ -1 +1 @@',
|
|
191
|
+
'-a',
|
|
192
|
+
'+b',
|
|
193
|
+
].join('\n');
|
|
194
|
+
|
|
195
|
+
expect(looksLikeDiff(content)).toBe(false);
|
|
196
|
+
expect(inferCodeFiletype(content)).toBeUndefined();
|
|
197
|
+
|
|
198
|
+
const tree = CodeBlock({
|
|
199
|
+
label: 'output',
|
|
200
|
+
content,
|
|
201
|
+
});
|
|
202
|
+
|
|
203
|
+
expect(findElementByType(tree, 'diff')).toBeNull();
|
|
204
|
+
expect(findElementByType(tree, 'code')).not.toBeNull();
|
|
205
|
+
});
|
|
206
|
+
});
|
|
@@ -0,0 +1,313 @@
|
|
|
1
|
+
import { opencodeMarkdownSyntax } from '../../ui/opencode-markdown';
|
|
2
|
+
import { uiTheme } from '../../ui/theme';
|
|
3
|
+
|
|
4
|
+
type CodeBlockProps = {
|
|
5
|
+
content: string;
|
|
6
|
+
label?: string;
|
|
7
|
+
languageHint?: string;
|
|
8
|
+
collapsible?: boolean;
|
|
9
|
+
collapsedLines?: number;
|
|
10
|
+
expanded?: boolean;
|
|
11
|
+
onToggleExpanded?: () => void;
|
|
12
|
+
};
|
|
13
|
+
|
|
14
|
+
const FILETYPE_BY_EXTENSION: Record<string, string> = {
|
|
15
|
+
bash: 'bash',
|
|
16
|
+
cjs: 'javascript',
|
|
17
|
+
css: 'css',
|
|
18
|
+
html: 'html',
|
|
19
|
+
java: 'java',
|
|
20
|
+
js: 'javascript',
|
|
21
|
+
json: 'json',
|
|
22
|
+
jsx: 'tsx',
|
|
23
|
+
md: 'markdown',
|
|
24
|
+
mjs: 'javascript',
|
|
25
|
+
py: 'python',
|
|
26
|
+
sh: 'bash',
|
|
27
|
+
sql: 'sql',
|
|
28
|
+
ts: 'typescript',
|
|
29
|
+
tsx: 'tsx',
|
|
30
|
+
txt: 'text',
|
|
31
|
+
xml: 'xml',
|
|
32
|
+
yaml: 'yaml',
|
|
33
|
+
yml: 'yaml',
|
|
34
|
+
zsh: 'bash',
|
|
35
|
+
};
|
|
36
|
+
|
|
37
|
+
const DIFF_HEADER_PATTERNS = [/^diff --git /m, /^Index:\s+/m, /^@@ /m];
|
|
38
|
+
|
|
39
|
+
const normalizeHint = (value?: string): string | undefined => {
|
|
40
|
+
const normalized = value?.trim().toLowerCase();
|
|
41
|
+
if (!normalized || normalized === 'auto') {
|
|
42
|
+
return undefined;
|
|
43
|
+
}
|
|
44
|
+
if (normalized === 'sh' || normalized === 'shell' || normalized === 'zsh') {
|
|
45
|
+
return 'bash';
|
|
46
|
+
}
|
|
47
|
+
if (normalized === 'js') {
|
|
48
|
+
return 'javascript';
|
|
49
|
+
}
|
|
50
|
+
if (normalized === 'ts') {
|
|
51
|
+
return 'typescript';
|
|
52
|
+
}
|
|
53
|
+
return normalized;
|
|
54
|
+
};
|
|
55
|
+
|
|
56
|
+
export const looksLikeDiff = (value: string): boolean => {
|
|
57
|
+
const normalized = value.trim();
|
|
58
|
+
if (!normalized) {
|
|
59
|
+
return false;
|
|
60
|
+
}
|
|
61
|
+
|
|
62
|
+
const firstLine = normalized.split('\n', 1)[0]?.trim();
|
|
63
|
+
if (
|
|
64
|
+
!firstLine ||
|
|
65
|
+
!(
|
|
66
|
+
firstLine.startsWith('diff --git ') ||
|
|
67
|
+
firstLine.startsWith('Index: ') ||
|
|
68
|
+
firstLine.startsWith('--- ') ||
|
|
69
|
+
firstLine.startsWith('@@ ')
|
|
70
|
+
)
|
|
71
|
+
) {
|
|
72
|
+
return false;
|
|
73
|
+
}
|
|
74
|
+
|
|
75
|
+
if (DIFF_HEADER_PATTERNS.some(pattern => pattern.test(normalized))) {
|
|
76
|
+
return (
|
|
77
|
+
/^@@ /m.test(normalized) ||
|
|
78
|
+
(/^--- /m.test(normalized) && /^\+\+\+ /m.test(normalized)) ||
|
|
79
|
+
/^diff --git /m.test(normalized)
|
|
80
|
+
);
|
|
81
|
+
}
|
|
82
|
+
|
|
83
|
+
return false;
|
|
84
|
+
};
|
|
85
|
+
|
|
86
|
+
export const inferFiletypeFromPath = (value?: string): string | undefined => {
|
|
87
|
+
if (!value) {
|
|
88
|
+
return undefined;
|
|
89
|
+
}
|
|
90
|
+
|
|
91
|
+
const fileName = value.split('/').pop();
|
|
92
|
+
if (!fileName || !fileName.includes('.')) {
|
|
93
|
+
return undefined;
|
|
94
|
+
}
|
|
95
|
+
|
|
96
|
+
const extension = fileName.split('.').pop()?.toLowerCase();
|
|
97
|
+
if (!extension) {
|
|
98
|
+
return undefined;
|
|
99
|
+
}
|
|
100
|
+
|
|
101
|
+
return FILETYPE_BY_EXTENSION[extension];
|
|
102
|
+
};
|
|
103
|
+
|
|
104
|
+
export const extractDiffPath = (value: string): string | undefined => {
|
|
105
|
+
const lines = value.split('\n');
|
|
106
|
+
for (const line of lines) {
|
|
107
|
+
if (line.startsWith('Index: ')) {
|
|
108
|
+
return line.slice('Index: '.length).trim();
|
|
109
|
+
}
|
|
110
|
+
if (line.startsWith('diff --git ')) {
|
|
111
|
+
const match = line.match(/^diff --git a\/(.+?) b\/(.+)$/);
|
|
112
|
+
if (match?.[2]) {
|
|
113
|
+
return match[2];
|
|
114
|
+
}
|
|
115
|
+
}
|
|
116
|
+
if (line.startsWith('+++ ')) {
|
|
117
|
+
const path = line.slice(4).trim();
|
|
118
|
+
if (path && path !== '/dev/null') {
|
|
119
|
+
return path.replace(/^b\//, '');
|
|
120
|
+
}
|
|
121
|
+
}
|
|
122
|
+
}
|
|
123
|
+
|
|
124
|
+
return undefined;
|
|
125
|
+
};
|
|
126
|
+
|
|
127
|
+
const looksLikeJson = (value: string): boolean => {
|
|
128
|
+
const normalized = value.trim();
|
|
129
|
+
if (!normalized || !['{', '['].includes(normalized[0] ?? '')) {
|
|
130
|
+
return false;
|
|
131
|
+
}
|
|
132
|
+
|
|
133
|
+
try {
|
|
134
|
+
JSON.parse(normalized);
|
|
135
|
+
return true;
|
|
136
|
+
} catch {
|
|
137
|
+
return false;
|
|
138
|
+
}
|
|
139
|
+
};
|
|
140
|
+
|
|
141
|
+
export const inferCodeFiletype = (value: string, languageHint?: string): string | undefined => {
|
|
142
|
+
const hint = normalizeHint(languageHint);
|
|
143
|
+
if (hint) {
|
|
144
|
+
return hint;
|
|
145
|
+
}
|
|
146
|
+
|
|
147
|
+
if (looksLikeDiff(value)) {
|
|
148
|
+
return 'diff';
|
|
149
|
+
}
|
|
150
|
+
|
|
151
|
+
const normalized = value.trim();
|
|
152
|
+
if (!normalized) {
|
|
153
|
+
return undefined;
|
|
154
|
+
}
|
|
155
|
+
|
|
156
|
+
if (looksLikeJson(normalized)) {
|
|
157
|
+
return 'json';
|
|
158
|
+
}
|
|
159
|
+
|
|
160
|
+
if (
|
|
161
|
+
normalized.startsWith('#!/bin/bash') ||
|
|
162
|
+
normalized.startsWith('#!/usr/bin/env bash') ||
|
|
163
|
+
normalized.startsWith('#!/bin/sh') ||
|
|
164
|
+
normalized.startsWith('#!/usr/bin/env zsh')
|
|
165
|
+
) {
|
|
166
|
+
return 'bash';
|
|
167
|
+
}
|
|
168
|
+
|
|
169
|
+
const meaningfulLines = normalized
|
|
170
|
+
.split('\n')
|
|
171
|
+
.map(line => line.trim())
|
|
172
|
+
.filter(Boolean);
|
|
173
|
+
if (meaningfulLines.length > 0 && meaningfulLines.every(line => line.startsWith('$ '))) {
|
|
174
|
+
return 'bash';
|
|
175
|
+
}
|
|
176
|
+
|
|
177
|
+
return undefined;
|
|
178
|
+
};
|
|
179
|
+
|
|
180
|
+
const buildHeaderMeta = (content: string, filetype?: string): string | undefined => {
|
|
181
|
+
if (filetype === 'diff') {
|
|
182
|
+
return extractDiffPath(content) ?? 'unified';
|
|
183
|
+
}
|
|
184
|
+
|
|
185
|
+
if (!filetype || filetype === 'text') {
|
|
186
|
+
return undefined;
|
|
187
|
+
}
|
|
188
|
+
|
|
189
|
+
return filetype;
|
|
190
|
+
};
|
|
191
|
+
|
|
192
|
+
const DEFAULT_COLLAPSED_LINES = 16;
|
|
193
|
+
|
|
194
|
+
const toContentLines = (value: string): string[] => {
|
|
195
|
+
if (!value) {
|
|
196
|
+
return [];
|
|
197
|
+
}
|
|
198
|
+
|
|
199
|
+
const trimmed = value.replace(/\n+$/, '');
|
|
200
|
+
if (!trimmed) {
|
|
201
|
+
return [];
|
|
202
|
+
}
|
|
203
|
+
|
|
204
|
+
return trimmed.split('\n');
|
|
205
|
+
};
|
|
206
|
+
|
|
207
|
+
const buildCollapsedContent = (
|
|
208
|
+
content: string,
|
|
209
|
+
lineLimit: number
|
|
210
|
+
): { content: string; hiddenLines: number } => {
|
|
211
|
+
const lines = toContentLines(content);
|
|
212
|
+
if (lineLimit <= 0 || lines.length <= lineLimit) {
|
|
213
|
+
return {
|
|
214
|
+
content,
|
|
215
|
+
hiddenLines: 0,
|
|
216
|
+
};
|
|
217
|
+
}
|
|
218
|
+
|
|
219
|
+
return {
|
|
220
|
+
content: `${lines.slice(0, lineLimit).join('\n')}\n`,
|
|
221
|
+
hiddenLines: lines.length - lineLimit,
|
|
222
|
+
};
|
|
223
|
+
};
|
|
224
|
+
|
|
225
|
+
export const CodeBlock = ({
|
|
226
|
+
content,
|
|
227
|
+
label,
|
|
228
|
+
languageHint,
|
|
229
|
+
collapsible = false,
|
|
230
|
+
collapsedLines = DEFAULT_COLLAPSED_LINES,
|
|
231
|
+
expanded = false,
|
|
232
|
+
onToggleExpanded,
|
|
233
|
+
}: CodeBlockProps) => {
|
|
234
|
+
const normalized = content.replace(/\n+$/, '\n');
|
|
235
|
+
const collapsed = buildCollapsedContent(normalized, collapsedLines);
|
|
236
|
+
const isTruncated = collapsible && collapsed.hiddenLines > 0;
|
|
237
|
+
const renderedContent = isTruncated && !expanded ? collapsed.content : normalized;
|
|
238
|
+
const filetype = inferCodeFiletype(normalized, languageHint);
|
|
239
|
+
const isDiff = filetype === 'diff';
|
|
240
|
+
const shouldRenderDiff = isDiff && (!isTruncated || expanded);
|
|
241
|
+
const codeFiletype = isDiff ? 'diff' : filetype;
|
|
242
|
+
const diffFiletype = isDiff ? inferFiletypeFromPath(extractDiffPath(normalized)) : undefined;
|
|
243
|
+
const headerLabel = label ?? 'code';
|
|
244
|
+
const headerMeta = buildHeaderMeta(normalized, filetype);
|
|
245
|
+
const toggleHint = expanded
|
|
246
|
+
? 'show less (click to collapse)'
|
|
247
|
+
: `... hidden ${collapsed.hiddenLines} lines (click to expand)`;
|
|
248
|
+
const clickHandler = isTruncated ? onToggleExpanded : undefined;
|
|
249
|
+
|
|
250
|
+
return (
|
|
251
|
+
<box
|
|
252
|
+
flexDirection="column"
|
|
253
|
+
// backgroundColor={uiTheme.codeBlock.bg}
|
|
254
|
+
paddingLeft={1}
|
|
255
|
+
paddingRight={1}
|
|
256
|
+
paddingTop={1}
|
|
257
|
+
paddingBottom={1}
|
|
258
|
+
>
|
|
259
|
+
<text fg={uiTheme.codeBlock.header} attributes={uiTheme.typography.note}>
|
|
260
|
+
{headerLabel}
|
|
261
|
+
{headerMeta ? <span fg={uiTheme.codeBlock.language}> · {headerMeta}</span> : null}
|
|
262
|
+
</text>
|
|
263
|
+
<box marginTop={1} onMouseUp={clickHandler}>
|
|
264
|
+
{shouldRenderDiff ? (
|
|
265
|
+
<diff
|
|
266
|
+
diff={renderedContent}
|
|
267
|
+
view="unified"
|
|
268
|
+
filetype={diffFiletype}
|
|
269
|
+
fg={uiTheme.codeBlock.text}
|
|
270
|
+
syntaxStyle={opencodeMarkdownSyntax}
|
|
271
|
+
wrapMode="char"
|
|
272
|
+
conceal={true}
|
|
273
|
+
showLineNumbers={true}
|
|
274
|
+
selectionBg={uiTheme.codeBlock.selectionBg}
|
|
275
|
+
selectionFg={uiTheme.codeBlock.selectionText}
|
|
276
|
+
lineNumberFg={uiTheme.diff.lineNumberFg}
|
|
277
|
+
lineNumberBg={uiTheme.diff.lineNumberBg}
|
|
278
|
+
addedBg={uiTheme.diff.addedBg}
|
|
279
|
+
removedBg={uiTheme.diff.removedBg}
|
|
280
|
+
contextBg={uiTheme.diff.contextBg}
|
|
281
|
+
addedContentBg={uiTheme.diff.addedContentBg}
|
|
282
|
+
removedContentBg={uiTheme.diff.removedContentBg}
|
|
283
|
+
contextContentBg={uiTheme.diff.contextContentBg}
|
|
284
|
+
addedSignColor={uiTheme.diff.addedSign}
|
|
285
|
+
removedSignColor={uiTheme.diff.removedSign}
|
|
286
|
+
addedLineNumberBg={uiTheme.diff.addedLineNumberBg}
|
|
287
|
+
removedLineNumberBg={uiTheme.diff.removedLineNumberBg}
|
|
288
|
+
/>
|
|
289
|
+
) : (
|
|
290
|
+
<code
|
|
291
|
+
content={renderedContent}
|
|
292
|
+
filetype={codeFiletype}
|
|
293
|
+
fg={uiTheme.codeBlock.text}
|
|
294
|
+
syntaxStyle={opencodeMarkdownSyntax}
|
|
295
|
+
wrapMode="char"
|
|
296
|
+
conceal={true}
|
|
297
|
+
drawUnstyledText={true}
|
|
298
|
+
selectable={true}
|
|
299
|
+
selectionBg={uiTheme.codeBlock.selectionBg}
|
|
300
|
+
selectionFg={uiTheme.codeBlock.selectionText}
|
|
301
|
+
/>
|
|
302
|
+
)}
|
|
303
|
+
</box>
|
|
304
|
+
{isTruncated ? (
|
|
305
|
+
<box marginTop={1} onMouseUp={clickHandler}>
|
|
306
|
+
<text fg={uiTheme.muted} attributes={uiTheme.typography.note}>
|
|
307
|
+
{toggleHint}
|
|
308
|
+
</text>
|
|
309
|
+
</box>
|
|
310
|
+
) : null}
|
|
311
|
+
</box>
|
|
312
|
+
);
|
|
313
|
+
};
|
|
@@ -0,0 +1,81 @@
|
|
|
1
|
+
import {
|
|
2
|
+
isAudioSelection,
|
|
3
|
+
isImageSelection,
|
|
4
|
+
isVideoSelection,
|
|
5
|
+
} from '../../files/attachment-capabilities';
|
|
6
|
+
import { uiTheme } from '../../ui/theme';
|
|
7
|
+
|
|
8
|
+
type PromptCardProps = {
|
|
9
|
+
prompt: string;
|
|
10
|
+
files?: string[];
|
|
11
|
+
createdAtMs: number;
|
|
12
|
+
isFirst?: boolean;
|
|
13
|
+
};
|
|
14
|
+
|
|
15
|
+
const formatTime = (timestamp: number) => {
|
|
16
|
+
return new Date(timestamp).toLocaleTimeString('en-US', {
|
|
17
|
+
hour: '2-digit',
|
|
18
|
+
minute: '2-digit',
|
|
19
|
+
second: '2-digit',
|
|
20
|
+
hour12: false,
|
|
21
|
+
});
|
|
22
|
+
};
|
|
23
|
+
|
|
24
|
+
export const PromptCard = ({
|
|
25
|
+
prompt,
|
|
26
|
+
files = [],
|
|
27
|
+
createdAtMs,
|
|
28
|
+
isFirst = false,
|
|
29
|
+
}: PromptCardProps) => {
|
|
30
|
+
const mediaFiles = files.filter(
|
|
31
|
+
file =>
|
|
32
|
+
isImageSelection({ relativePath: file, absolutePath: file, size: 0 }) ||
|
|
33
|
+
isAudioSelection({ relativePath: file, absolutePath: file, size: 0 }) ||
|
|
34
|
+
isVideoSelection({ relativePath: file, absolutePath: file, size: 0 })
|
|
35
|
+
);
|
|
36
|
+
|
|
37
|
+
return (
|
|
38
|
+
<box flexDirection="row" marginTop={isFirst ? 0 : 1} marginBottom={1}>
|
|
39
|
+
<box width={0.5} backgroundColor={uiTheme.accent} />
|
|
40
|
+
<box
|
|
41
|
+
flexGrow={1}
|
|
42
|
+
backgroundColor={uiTheme.userPromptBg}
|
|
43
|
+
paddingLeft={2}
|
|
44
|
+
paddingRight={1}
|
|
45
|
+
paddingTop={1}
|
|
46
|
+
paddingBottom={1}
|
|
47
|
+
>
|
|
48
|
+
<text
|
|
49
|
+
fg={uiTheme.userPromptText}
|
|
50
|
+
attributes={uiTheme.typography.heading}
|
|
51
|
+
wrapMode="word"
|
|
52
|
+
selectable={true}
|
|
53
|
+
>
|
|
54
|
+
{prompt}
|
|
55
|
+
</text>
|
|
56
|
+
{mediaFiles.length > 0 ? (
|
|
57
|
+
<box paddingTop={1} flexDirection="column">
|
|
58
|
+
<text fg={uiTheme.muted} attributes={uiTheme.typography.note}>
|
|
59
|
+
Media files
|
|
60
|
+
</text>
|
|
61
|
+
{mediaFiles.map(file => (
|
|
62
|
+
<text
|
|
63
|
+
key={file}
|
|
64
|
+
fg={uiTheme.text}
|
|
65
|
+
attributes={uiTheme.typography.note}
|
|
66
|
+
selectable={true}
|
|
67
|
+
>
|
|
68
|
+
{file}
|
|
69
|
+
</text>
|
|
70
|
+
))}
|
|
71
|
+
</box>
|
|
72
|
+
) : null}
|
|
73
|
+
<box paddingTop={1}>
|
|
74
|
+
<text fg={uiTheme.muted} attributes={uiTheme.typography.note} selectable={true}>
|
|
75
|
+
{formatTime(createdAtMs)}
|
|
76
|
+
</text>
|
|
77
|
+
</box>
|
|
78
|
+
</box>
|
|
79
|
+
</box>
|
|
80
|
+
);
|
|
81
|
+
};
|
|
@@ -0,0 +1,52 @@
|
|
|
1
|
+
import { describe, expect, it } from 'vitest';
|
|
2
|
+
|
|
3
|
+
import type { ReplySegment } from '../../types/chat';
|
|
4
|
+
import { buildReplyRenderItems } from './segment-groups';
|
|
5
|
+
|
|
6
|
+
describe('buildReplyRenderItems', () => {
|
|
7
|
+
it('groups tool segments by call id and keeps non-tool segments in place', () => {
|
|
8
|
+
const segments: ReplySegment[] = [
|
|
9
|
+
{ id: '1:thinking:1', type: 'thinking', content: 'thinking' },
|
|
10
|
+
{
|
|
11
|
+
id: '1:tool-use:call_a',
|
|
12
|
+
type: 'code',
|
|
13
|
+
content: '# Tool: bash (call_a)\n$ echo a\n',
|
|
14
|
+
data: { id: 'call_a', function: { name: 'bash', arguments: '{"command":"echo a"}' } },
|
|
15
|
+
},
|
|
16
|
+
{ id: '1:tool:call_a:stdout', type: 'code', content: 'a\n' },
|
|
17
|
+
{
|
|
18
|
+
id: '1:tool-result:call_a',
|
|
19
|
+
type: 'code',
|
|
20
|
+
content: '# Result: bash (call_a) success\na\n',
|
|
21
|
+
data: { result: { success: true, data: { output: 'a' } } },
|
|
22
|
+
},
|
|
23
|
+
{ id: '1:text:2', type: 'text', content: 'done' },
|
|
24
|
+
];
|
|
25
|
+
|
|
26
|
+
const items = buildReplyRenderItems(segments);
|
|
27
|
+
expect(items.map(item => item.type)).toEqual(['segment', 'tool', 'segment']);
|
|
28
|
+
expect(items[1]?.type === 'tool' ? items[1].group.toolCallId : '').toBe('call_a');
|
|
29
|
+
expect(items[1]?.type === 'tool' ? items[1].group.streams.length : 0).toBe(1);
|
|
30
|
+
expect(items[1]?.type === 'tool' ? items[1].group.use?.data : undefined).toEqual({
|
|
31
|
+
id: 'call_a',
|
|
32
|
+
function: { name: 'bash', arguments: '{"command":"echo a"}' },
|
|
33
|
+
});
|
|
34
|
+
expect(items[1]?.type === 'tool' ? items[1].group.result?.data : undefined).toEqual({
|
|
35
|
+
result: { success: true, data: { output: 'a' } },
|
|
36
|
+
});
|
|
37
|
+
});
|
|
38
|
+
|
|
39
|
+
it('starts a new tool group when call id changes', () => {
|
|
40
|
+
const segments: ReplySegment[] = [
|
|
41
|
+
{ id: '1:tool-use:call_a', type: 'code', content: '# Tool: bash (call_a)\n$ echo a\n' },
|
|
42
|
+
{ id: '1:tool-result:call_a', type: 'code', content: '# Result: bash (call_a) success\na\n' },
|
|
43
|
+
{ id: '1:tool-use:call_b', type: 'code', content: '# Tool: bash (call_b)\n$ echo b\n' },
|
|
44
|
+
{ id: '1:tool-result:call_b', type: 'code', content: '# Result: bash (call_b) success\nb\n' },
|
|
45
|
+
];
|
|
46
|
+
|
|
47
|
+
const items = buildReplyRenderItems(segments);
|
|
48
|
+
expect(items.length).toBe(2);
|
|
49
|
+
expect(items[0]?.type === 'tool' ? items[0].group.toolCallId : '').toBe('call_a');
|
|
50
|
+
expect(items[1]?.type === 'tool' ? items[1].group.toolCallId : '').toBe('call_b');
|
|
51
|
+
});
|
|
52
|
+
});
|