@zhijiewang/openharness 0.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (135) hide show
  1. package/CHANGELOG.md +18 -0
  2. package/CONTRIBUTING.md +55 -0
  3. package/LICENSE +21 -0
  4. package/README.md +154 -0
  5. package/data/models.json +74 -0
  6. package/data/prompts/system.md +25 -0
  7. package/data/skills/code-review.md +19 -0
  8. package/data/skills/commit.md +17 -0
  9. package/data/skills/debug.md +24 -0
  10. package/data/skills/tdd.md +22 -0
  11. package/dist/Tool.d.ts +45 -0
  12. package/dist/Tool.d.ts.map +1 -0
  13. package/dist/Tool.js +62 -0
  14. package/dist/Tool.js.map +1 -0
  15. package/dist/components/App.d.ts +16 -0
  16. package/dist/components/App.d.ts.map +1 -0
  17. package/dist/components/App.js +25 -0
  18. package/dist/components/App.js.map +1 -0
  19. package/dist/components/Messages.d.ts +9 -0
  20. package/dist/components/Messages.d.ts.map +1 -0
  21. package/dist/components/Messages.js +23 -0
  22. package/dist/components/Messages.js.map +1 -0
  23. package/dist/components/PermissionPrompt.d.ts +9 -0
  24. package/dist/components/PermissionPrompt.d.ts.map +1 -0
  25. package/dist/components/PermissionPrompt.js +18 -0
  26. package/dist/components/PermissionPrompt.js.map +1 -0
  27. package/dist/components/REPL.d.ts +15 -0
  28. package/dist/components/REPL.d.ts.map +1 -0
  29. package/dist/components/REPL.js +114 -0
  30. package/dist/components/REPL.js.map +1 -0
  31. package/dist/components/Spinner.d.ts +7 -0
  32. package/dist/components/Spinner.d.ts.map +1 -0
  33. package/dist/components/Spinner.js +7 -0
  34. package/dist/components/Spinner.js.map +1 -0
  35. package/dist/components/TextInput.d.ts +7 -0
  36. package/dist/components/TextInput.d.ts.map +1 -0
  37. package/dist/components/TextInput.js +37 -0
  38. package/dist/components/TextInput.js.map +1 -0
  39. package/dist/components/ToolCallDisplay.d.ts +12 -0
  40. package/dist/components/ToolCallDisplay.d.ts.map +1 -0
  41. package/dist/components/ToolCallDisplay.js +16 -0
  42. package/dist/components/ToolCallDisplay.js.map +1 -0
  43. package/dist/harness/cost.d.ts +33 -0
  44. package/dist/harness/cost.d.ts.map +1 -0
  45. package/dist/harness/cost.js +68 -0
  46. package/dist/harness/cost.js.map +1 -0
  47. package/dist/harness/onboarding.d.ts +17 -0
  48. package/dist/harness/onboarding.d.ts.map +1 -0
  49. package/dist/harness/onboarding.js +99 -0
  50. package/dist/harness/onboarding.js.map +1 -0
  51. package/dist/harness/rules.d.ts +8 -0
  52. package/dist/harness/rules.d.ts.map +1 -0
  53. package/dist/harness/rules.js +66 -0
  54. package/dist/harness/rules.js.map +1 -0
  55. package/dist/harness/session.d.ts +24 -0
  56. package/dist/harness/session.d.ts.map +1 -0
  57. package/dist/harness/session.js +56 -0
  58. package/dist/harness/session.js.map +1 -0
  59. package/dist/main.d.ts +12 -0
  60. package/dist/main.d.ts.map +1 -0
  61. package/dist/main.js +177 -0
  62. package/dist/main.js.map +1 -0
  63. package/dist/providers/anthropic.d.ts +27 -0
  64. package/dist/providers/anthropic.d.ts.map +1 -0
  65. package/dist/providers/anthropic.js +291 -0
  66. package/dist/providers/anthropic.js.map +1 -0
  67. package/dist/providers/base.d.ts +41 -0
  68. package/dist/providers/base.d.ts.map +1 -0
  69. package/dist/providers/base.js +5 -0
  70. package/dist/providers/base.js.map +1 -0
  71. package/dist/providers/index.d.ts +12 -0
  72. package/dist/providers/index.d.ts.map +1 -0
  73. package/dist/providers/index.js +57 -0
  74. package/dist/providers/index.js.map +1 -0
  75. package/dist/providers/ollama.d.ts +19 -0
  76. package/dist/providers/ollama.d.ts.map +1 -0
  77. package/dist/providers/ollama.js +233 -0
  78. package/dist/providers/ollama.js.map +1 -0
  79. package/dist/providers/openai.d.ts +21 -0
  80. package/dist/providers/openai.d.ts.map +1 -0
  81. package/dist/providers/openai.js +242 -0
  82. package/dist/providers/openai.js.map +1 -0
  83. package/dist/providers/openrouter.d.ts +25 -0
  84. package/dist/providers/openrouter.d.ts.map +1 -0
  85. package/dist/providers/openrouter.js +278 -0
  86. package/dist/providers/openrouter.js.map +1 -0
  87. package/dist/query.d.ts +35 -0
  88. package/dist/query.d.ts.map +1 -0
  89. package/dist/query.js +185 -0
  90. package/dist/query.js.map +1 -0
  91. package/dist/tools/BashTool/index.d.ts +15 -0
  92. package/dist/tools/BashTool/index.d.ts.map +1 -0
  93. package/dist/tools/BashTool/index.js +78 -0
  94. package/dist/tools/BashTool/index.js.map +1 -0
  95. package/dist/tools/FileEditTool/index.d.ts +21 -0
  96. package/dist/tools/FileEditTool/index.d.ts.map +1 -0
  97. package/dist/tools/FileEditTool/index.js +70 -0
  98. package/dist/tools/FileEditTool/index.js.map +1 -0
  99. package/dist/tools/FileReadTool/index.d.ts +18 -0
  100. package/dist/tools/FileReadTool/index.d.ts.map +1 -0
  101. package/dist/tools/FileReadTool/index.js +63 -0
  102. package/dist/tools/FileReadTool/index.js.map +1 -0
  103. package/dist/tools/FileWriteTool/index.d.ts +15 -0
  104. package/dist/tools/FileWriteTool/index.d.ts.map +1 -0
  105. package/dist/tools/FileWriteTool/index.js +42 -0
  106. package/dist/tools/FileWriteTool/index.js.map +1 -0
  107. package/dist/tools/GlobTool/index.d.ts +15 -0
  108. package/dist/tools/GlobTool/index.d.ts.map +1 -0
  109. package/dist/tools/GlobTool/index.js +126 -0
  110. package/dist/tools/GlobTool/index.js.map +1 -0
  111. package/dist/tools/GrepTool/index.d.ts +21 -0
  112. package/dist/tools/GrepTool/index.d.ts.map +1 -0
  113. package/dist/tools/GrepTool/index.js +125 -0
  114. package/dist/tools/GrepTool/index.js.map +1 -0
  115. package/dist/tools/WebFetchTool/index.d.ts +12 -0
  116. package/dist/tools/WebFetchTool/index.d.ts.map +1 -0
  117. package/dist/tools/WebFetchTool/index.js +98 -0
  118. package/dist/tools/WebFetchTool/index.js.map +1 -0
  119. package/dist/tools.d.ts +9 -0
  120. package/dist/tools.d.ts.map +1 -0
  121. package/dist/tools.js +25 -0
  122. package/dist/tools.js.map +1 -0
  123. package/dist/types/events.d.ts +49 -0
  124. package/dist/types/events.d.ts.map +1 -0
  125. package/dist/types/events.js +5 -0
  126. package/dist/types/events.js.map +1 -0
  127. package/dist/types/message.d.ts +27 -0
  128. package/dist/types/message.d.ts.map +1 -0
  129. package/dist/types/message.js +22 -0
  130. package/dist/types/message.js.map +1 -0
  131. package/dist/types/permissions.d.ts +22 -0
  132. package/dist/types/permissions.d.ts.map +1 -0
  133. package/dist/types/permissions.js +27 -0
  134. package/dist/types/permissions.js.map +1 -0
  135. package/package.json +55 -0
@@ -0,0 +1,242 @@
1
+ /**
2
+ * OpenAI-compatible provider — works with OpenAI, DeepSeek, Groq, Together, etc.
3
+ */
4
+ import { createAssistantMessage } from "../types/message.js";
5
+ export class OpenAIProvider {
6
+ name;
7
+ apiKey;
8
+ baseUrl;
9
+ defaultModel;
10
+ constructor(config) {
11
+ this.name = config.name || "openai";
12
+ this.apiKey = config.apiKey ?? "";
13
+ this.baseUrl = (config.baseUrl ?? "https://api.openai.com/v1").replace(/\/$/, "");
14
+ this.defaultModel = config.defaultModel ?? "gpt-4o";
15
+ }
16
+ convertMessages(messages, systemPrompt) {
17
+ const out = [{ role: "system", content: systemPrompt }];
18
+ for (const msg of messages) {
19
+ if (msg.role === "system")
20
+ continue;
21
+ if (msg.role === "assistant" && msg.toolCalls?.length) {
22
+ out.push({
23
+ role: "assistant",
24
+ content: msg.content || null,
25
+ tool_calls: msg.toolCalls.map((tc) => ({
26
+ id: tc.id,
27
+ type: "function",
28
+ function: {
29
+ name: tc.toolName,
30
+ arguments: JSON.stringify(tc.arguments),
31
+ },
32
+ })),
33
+ });
34
+ }
35
+ else if (msg.role === "tool" && msg.toolResults?.length) {
36
+ for (const tr of msg.toolResults) {
37
+ out.push({
38
+ role: "tool",
39
+ tool_call_id: tr.callId,
40
+ content: tr.output,
41
+ });
42
+ }
43
+ }
44
+ else {
45
+ out.push({ role: msg.role, content: msg.content });
46
+ }
47
+ }
48
+ return out;
49
+ }
50
+ headers() {
51
+ return {
52
+ "Content-Type": "application/json",
53
+ Authorization: `Bearer ${this.apiKey}`,
54
+ };
55
+ }
56
+ async *stream(messages, systemPrompt, tools, model) {
57
+ const m = model ?? this.defaultModel;
58
+ const body = {
59
+ model: m,
60
+ messages: this.convertMessages(messages, systemPrompt),
61
+ stream: true,
62
+ stream_options: { include_usage: true },
63
+ };
64
+ if (tools?.length)
65
+ body.tools = tools;
66
+ let res;
67
+ try {
68
+ res = await fetch(`${this.baseUrl}/chat/completions`, {
69
+ method: "POST",
70
+ headers: this.headers(),
71
+ body: JSON.stringify(body),
72
+ });
73
+ }
74
+ catch (err) {
75
+ yield { type: "error", message: `OpenAI request failed: ${err}` };
76
+ return;
77
+ }
78
+ if (!res.ok) {
79
+ yield { type: "error", message: `OpenAI HTTP ${res.status}: ${await res.text()}` };
80
+ return;
81
+ }
82
+ const reader = res.body?.getReader();
83
+ if (!reader)
84
+ return;
85
+ const decoder = new TextDecoder();
86
+ let buffer = "";
87
+ // Track tool call argument deltas by index
88
+ const toolCallAccumulators = new Map();
89
+ while (true) {
90
+ const { done, value } = await reader.read();
91
+ if (done)
92
+ break;
93
+ buffer += decoder.decode(value, { stream: true });
94
+ const lines = buffer.split("\n");
95
+ buffer = lines.pop() ?? "";
96
+ for (const line of lines) {
97
+ const trimmed = line.trim();
98
+ if (!trimmed.startsWith("data:"))
99
+ continue;
100
+ const payload = trimmed.slice(5).trim();
101
+ if (payload === "[DONE]")
102
+ continue;
103
+ let chunk;
104
+ try {
105
+ chunk = JSON.parse(payload);
106
+ }
107
+ catch {
108
+ continue;
109
+ }
110
+ // Usage info (final chunk with stream_options)
111
+ if (chunk.usage) {
112
+ const inputTokens = chunk.usage.prompt_tokens ?? 0;
113
+ const outputTokens = chunk.usage.completion_tokens ?? 0;
114
+ const info = this.getModelInfo(m);
115
+ const cost = (inputTokens * (info?.inputCostPerMtok ?? 0) +
116
+ outputTokens * (info?.outputCostPerMtok ?? 0)) /
117
+ 1_000_000;
118
+ yield { type: "cost_update", inputTokens, outputTokens, cost, model: m };
119
+ }
120
+ const delta = chunk.choices?.[0]?.delta;
121
+ if (!delta)
122
+ continue;
123
+ if (delta.content) {
124
+ yield { type: "text_delta", content: delta.content };
125
+ }
126
+ if (delta.tool_calls) {
127
+ for (const tc of delta.tool_calls) {
128
+ const idx = tc.index ?? 0;
129
+ if (tc.id) {
130
+ // New tool call
131
+ toolCallAccumulators.set(idx, {
132
+ id: tc.id,
133
+ name: tc.function?.name ?? "",
134
+ args: tc.function?.arguments ?? "",
135
+ });
136
+ yield {
137
+ type: "tool_call_start",
138
+ toolName: tc.function?.name ?? "unknown",
139
+ callId: tc.id,
140
+ };
141
+ }
142
+ else {
143
+ // Continuation delta
144
+ const acc = toolCallAccumulators.get(idx);
145
+ if (acc && tc.function?.arguments) {
146
+ acc.args += tc.function.arguments;
147
+ }
148
+ }
149
+ }
150
+ }
151
+ }
152
+ }
153
+ // Emit tool_call_complete for each accumulated tool call
154
+ for (const [, acc] of toolCallAccumulators) {
155
+ yield {
156
+ type: "tool_call_complete",
157
+ callId: acc.id,
158
+ toolName: acc.name,
159
+ arguments: acc.args ? JSON.parse(acc.args) : {},
160
+ };
161
+ }
162
+ }
163
+ async complete(messages, systemPrompt, tools, model) {
164
+ const m = model ?? this.defaultModel;
165
+ const body = {
166
+ model: m,
167
+ messages: this.convertMessages(messages, systemPrompt),
168
+ stream: false,
169
+ };
170
+ if (tools?.length)
171
+ body.tools = tools;
172
+ const res = await fetch(`${this.baseUrl}/chat/completions`, {
173
+ method: "POST",
174
+ headers: this.headers(),
175
+ body: JSON.stringify(body),
176
+ });
177
+ if (!res.ok) {
178
+ throw new Error(`OpenAI HTTP ${res.status}: ${await res.text()}`);
179
+ }
180
+ const data = await res.json();
181
+ const choice = data.choices?.[0]?.message;
182
+ const content = choice?.content ?? "";
183
+ let toolCalls;
184
+ if (choice?.tool_calls?.length) {
185
+ toolCalls = choice.tool_calls.map((tc) => ({
186
+ id: tc.id,
187
+ toolName: tc.function.name,
188
+ arguments: JSON.parse(tc.function.arguments),
189
+ }));
190
+ }
191
+ return createAssistantMessage(content, toolCalls);
192
+ }
193
+ getModelInfo(id) {
194
+ return this.listModels().find((m) => m.id === id);
195
+ }
196
+ listModels() {
197
+ return [
198
+ {
199
+ id: "gpt-4o",
200
+ provider: this.name,
201
+ contextWindow: 128_000,
202
+ supportsTools: true,
203
+ supportsStreaming: true,
204
+ supportsVision: true,
205
+ inputCostPerMtok: 2.5,
206
+ outputCostPerMtok: 10,
207
+ },
208
+ {
209
+ id: "gpt-4o-mini",
210
+ provider: this.name,
211
+ contextWindow: 128_000,
212
+ supportsTools: true,
213
+ supportsStreaming: true,
214
+ supportsVision: true,
215
+ inputCostPerMtok: 0.15,
216
+ outputCostPerMtok: 0.6,
217
+ },
218
+ {
219
+ id: "o3-mini",
220
+ provider: this.name,
221
+ contextWindow: 200_000,
222
+ supportsTools: true,
223
+ supportsStreaming: true,
224
+ supportsVision: false,
225
+ inputCostPerMtok: 1.1,
226
+ outputCostPerMtok: 4.4,
227
+ },
228
+ ];
229
+ }
230
+ async healthCheck() {
231
+ try {
232
+ const res = await fetch(`${this.baseUrl}/models`, {
233
+ headers: this.headers(),
234
+ });
235
+ return res.ok;
236
+ }
237
+ catch {
238
+ return false;
239
+ }
240
+ }
241
+ }
242
+ //# sourceMappingURL=openai.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"openai.js","sourceRoot":"","sources":["../../src/providers/openai.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,25 @@
1
+ /**
2
+ * OpenRouter provider — aggregated LLM access via OpenAI-compatible API.
3
+ */
4
+ import type { Message } from "../types/message.js";
5
+ import type { StreamEvent } from "../types/events.js";
6
+ import type { Provider, APIToolDef, ModelInfo, ProviderConfig } from "./base.js";
7
+ export declare class OpenRouterProvider implements Provider {
8
+ readonly name = "openrouter";
9
+ private apiKey;
10
+ private baseUrl;
11
+ private defaultModel;
12
+ private cachedModels;
13
+ constructor(config: ProviderConfig);
14
+ private headers;
15
+ private convertMessages;
16
+ stream(messages: Message[], systemPrompt: string, tools?: APIToolDef[], model?: string): AsyncGenerator<StreamEvent, void>;
17
+ complete(messages: Message[], systemPrompt: string, tools?: APIToolDef[], model?: string): Promise<Message>;
18
+ listModels(): ModelInfo[];
19
+ /**
20
+ * Fetch and cache the full model list from OpenRouter.
21
+ */
22
+ fetchModels(): Promise<ModelInfo[]>;
23
+ healthCheck(): Promise<boolean>;
24
+ }
25
+ //# sourceMappingURL=openrouter.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"openrouter.d.ts","sourceRoot":"","sources":["../../src/providers/openrouter.ts"],"names":[],"mappings":"AAAA;;GAEG;AAEH,OAAO,KAAK,EAAE,OAAO,EAAY,MAAM,qBAAqB,CAAC;AAC7D,OAAO,KAAK,EAAE,WAAW,EAAoB,MAAM,oBAAoB,CAAC;AAExE,OAAO,KAAK,EAAE,QAAQ,EAAE,UAAU,EAAE,SAAS,EAAE,cAAc,EAAE,MAAM,WAAW,CAAC;AAEjF,qBAAa,kBAAmB,YAAW,QAAQ;IACjD,QAAQ,CAAC,IAAI,gBAAgB;IAC7B,OAAO,CAAC,MAAM,CAAS;IACvB,OAAO,CAAC,OAAO,CAAS;IACxB,OAAO,CAAC,YAAY,CAAS;IAC7B,OAAO,CAAC,YAAY,CAA4B;gBAEpC,MAAM,EAAE,cAAc;IAMlC,OAAO,CAAC,OAAO;IASf,OAAO,CAAC,eAAe;IAkChB,MAAM,CACX,QAAQ,EAAE,OAAO,EAAE,EACnB,YAAY,EAAE,MAAM,EACpB,KAAK,CAAC,EAAE,UAAU,EAAE,EACpB,KAAK,CAAC,EAAE,MAAM,GACb,cAAc,CAAC,WAAW,EAAE,IAAI,CAAC;IA6G9B,QAAQ,CACZ,QAAQ,EAAE,OAAO,EAAE,EACnB,YAAY,EAAE,MAAM,EACpB,KAAK,CAAC,EAAE,UAAU,EAAE,EACpB,KAAK,CAAC,EAAE,MAAM,GACb,OAAO,CAAC,OAAO,CAAC;IAmCnB,UAAU,IAAI,SAAS,EAAE;IA+CzB;;OAEG;IACG,WAAW,IAAI,OAAO,CAAC,SAAS,EAAE,CAAC;IAwBnC,WAAW,IAAI,OAAO,CAAC,OAAO,CAAC;CAUtC"}
@@ -0,0 +1,278 @@
1
+ /**
2
+ * OpenRouter provider — aggregated LLM access via OpenAI-compatible API.
3
+ */
4
+ import { createAssistantMessage } from "../types/message.js";
5
+ export class OpenRouterProvider {
6
+ name = "openrouter";
7
+ apiKey;
8
+ baseUrl;
9
+ defaultModel;
10
+ cachedModels = null;
11
+ constructor(config) {
12
+ this.apiKey = config.apiKey ?? "";
13
+ this.baseUrl = (config.baseUrl ?? "https://openrouter.ai/api/v1").replace(/\/$/, "");
14
+ this.defaultModel = config.defaultModel ?? "anthropic/claude-sonnet-4-6";
15
+ }
16
+ headers() {
17
+ return {
18
+ "Content-Type": "application/json",
19
+ Authorization: `Bearer ${this.apiKey}`,
20
+ "HTTP-Referer": "https://github.com/openharness",
21
+ "X-Title": "OpenHarness CLI",
22
+ };
23
+ }
24
+ convertMessages(messages, systemPrompt) {
25
+ const out = [{ role: "system", content: systemPrompt }];
26
+ for (const msg of messages) {
27
+ if (msg.role === "system")
28
+ continue;
29
+ if (msg.role === "assistant" && msg.toolCalls?.length) {
30
+ out.push({
31
+ role: "assistant",
32
+ content: msg.content || null,
33
+ tool_calls: msg.toolCalls.map((tc) => ({
34
+ id: tc.id,
35
+ type: "function",
36
+ function: {
37
+ name: tc.toolName,
38
+ arguments: JSON.stringify(tc.arguments),
39
+ },
40
+ })),
41
+ });
42
+ }
43
+ else if (msg.role === "tool" && msg.toolResults?.length) {
44
+ for (const tr of msg.toolResults) {
45
+ out.push({
46
+ role: "tool",
47
+ tool_call_id: tr.callId,
48
+ content: tr.output,
49
+ });
50
+ }
51
+ }
52
+ else {
53
+ out.push({ role: msg.role, content: msg.content });
54
+ }
55
+ }
56
+ return out;
57
+ }
58
+ async *stream(messages, systemPrompt, tools, model) {
59
+ const m = model ?? this.defaultModel;
60
+ const body = {
61
+ model: m,
62
+ messages: this.convertMessages(messages, systemPrompt),
63
+ stream: true,
64
+ };
65
+ if (tools?.length)
66
+ body.tools = tools;
67
+ let res;
68
+ try {
69
+ res = await fetch(`${this.baseUrl}/chat/completions`, {
70
+ method: "POST",
71
+ headers: this.headers(),
72
+ body: JSON.stringify(body),
73
+ });
74
+ }
75
+ catch (err) {
76
+ yield { type: "error", message: `OpenRouter request failed: ${err}` };
77
+ return;
78
+ }
79
+ if (!res.ok) {
80
+ yield { type: "error", message: `OpenRouter HTTP ${res.status}: ${await res.text()}` };
81
+ return;
82
+ }
83
+ const reader = res.body?.getReader();
84
+ if (!reader)
85
+ return;
86
+ const decoder = new TextDecoder();
87
+ let buffer = "";
88
+ const toolCallAccumulators = new Map();
89
+ while (true) {
90
+ const { done, value } = await reader.read();
91
+ if (done)
92
+ break;
93
+ buffer += decoder.decode(value, { stream: true });
94
+ const lines = buffer.split("\n");
95
+ buffer = lines.pop() ?? "";
96
+ for (const line of lines) {
97
+ const trimmed = line.trim();
98
+ if (!trimmed.startsWith("data:"))
99
+ continue;
100
+ const payload = trimmed.slice(5).trim();
101
+ if (payload === "[DONE]")
102
+ continue;
103
+ let chunk;
104
+ try {
105
+ chunk = JSON.parse(payload);
106
+ }
107
+ catch {
108
+ continue;
109
+ }
110
+ if (chunk.usage) {
111
+ const inputTokens = chunk.usage.prompt_tokens ?? 0;
112
+ const outputTokens = chunk.usage.completion_tokens ?? 0;
113
+ yield {
114
+ type: "cost_update",
115
+ inputTokens,
116
+ outputTokens,
117
+ cost: 0, // OpenRouter pricing varies; let caller resolve
118
+ model: m,
119
+ };
120
+ }
121
+ const delta = chunk.choices?.[0]?.delta;
122
+ if (!delta)
123
+ continue;
124
+ if (delta.content) {
125
+ yield { type: "text_delta", content: delta.content };
126
+ }
127
+ if (delta.tool_calls) {
128
+ for (const tc of delta.tool_calls) {
129
+ const idx = tc.index ?? 0;
130
+ if (tc.id) {
131
+ toolCallAccumulators.set(idx, {
132
+ id: tc.id,
133
+ name: tc.function?.name ?? "",
134
+ args: tc.function?.arguments ?? "",
135
+ });
136
+ yield {
137
+ type: "tool_call_start",
138
+ toolName: tc.function?.name ?? "unknown",
139
+ callId: tc.id,
140
+ };
141
+ }
142
+ else {
143
+ const acc = toolCallAccumulators.get(idx);
144
+ if (acc && tc.function?.arguments) {
145
+ acc.args += tc.function.arguments;
146
+ }
147
+ }
148
+ }
149
+ }
150
+ }
151
+ }
152
+ // Emit tool_call_complete for each accumulated tool call
153
+ for (const [, acc] of toolCallAccumulators) {
154
+ yield {
155
+ type: "tool_call_complete",
156
+ callId: acc.id,
157
+ toolName: acc.name,
158
+ arguments: acc.args ? JSON.parse(acc.args) : {},
159
+ };
160
+ }
161
+ }
162
+ async complete(messages, systemPrompt, tools, model) {
163
+ const m = model ?? this.defaultModel;
164
+ const body = {
165
+ model: m,
166
+ messages: this.convertMessages(messages, systemPrompt),
167
+ stream: false,
168
+ };
169
+ if (tools?.length)
170
+ body.tools = tools;
171
+ const res = await fetch(`${this.baseUrl}/chat/completions`, {
172
+ method: "POST",
173
+ headers: this.headers(),
174
+ body: JSON.stringify(body),
175
+ });
176
+ if (!res.ok) {
177
+ throw new Error(`OpenRouter HTTP ${res.status}: ${await res.text()}`);
178
+ }
179
+ const data = await res.json();
180
+ const choice = data.choices?.[0]?.message;
181
+ const content = choice?.content ?? "";
182
+ let toolCalls;
183
+ if (choice?.tool_calls?.length) {
184
+ toolCalls = choice.tool_calls.map((tc) => ({
185
+ id: tc.id,
186
+ toolName: tc.function.name,
187
+ arguments: JSON.parse(tc.function.arguments),
188
+ }));
189
+ }
190
+ return createAssistantMessage(content, toolCalls);
191
+ }
192
+ listModels() {
193
+ // Return cached models if fetched, otherwise a curated default set
194
+ if (this.cachedModels)
195
+ return this.cachedModels;
196
+ return [
197
+ {
198
+ id: "anthropic/claude-sonnet-4-6",
199
+ provider: "openrouter",
200
+ contextWindow: 200_000,
201
+ supportsTools: true,
202
+ supportsStreaming: true,
203
+ supportsVision: true,
204
+ inputCostPerMtok: 3,
205
+ outputCostPerMtok: 15,
206
+ },
207
+ {
208
+ id: "openai/gpt-4o",
209
+ provider: "openrouter",
210
+ contextWindow: 128_000,
211
+ supportsTools: true,
212
+ supportsStreaming: true,
213
+ supportsVision: true,
214
+ inputCostPerMtok: 2.5,
215
+ outputCostPerMtok: 10,
216
+ },
217
+ {
218
+ id: "google/gemini-2.5-pro",
219
+ provider: "openrouter",
220
+ contextWindow: 1_000_000,
221
+ supportsTools: true,
222
+ supportsStreaming: true,
223
+ supportsVision: true,
224
+ inputCostPerMtok: 1.25,
225
+ outputCostPerMtok: 10,
226
+ },
227
+ {
228
+ id: "meta-llama/llama-3.1-405b-instruct",
229
+ provider: "openrouter",
230
+ contextWindow: 128_000,
231
+ supportsTools: true,
232
+ supportsStreaming: true,
233
+ supportsVision: false,
234
+ inputCostPerMtok: 2,
235
+ outputCostPerMtok: 2,
236
+ },
237
+ ];
238
+ }
239
+ /**
240
+ * Fetch and cache the full model list from OpenRouter.
241
+ */
242
+ async fetchModels() {
243
+ try {
244
+ const res = await fetch(`${this.baseUrl}/models`, {
245
+ headers: this.headers(),
246
+ });
247
+ if (!res.ok)
248
+ return this.listModels();
249
+ const data = await res.json();
250
+ this.cachedModels = (data.data ?? []).map((m) => ({
251
+ id: m.id,
252
+ provider: "openrouter",
253
+ contextWindow: m.context_length ?? 4096,
254
+ supportsTools: true,
255
+ supportsStreaming: true,
256
+ supportsVision: false,
257
+ inputCostPerMtok: parseFloat(m.pricing?.prompt ?? "0") * 1_000_000,
258
+ outputCostPerMtok: parseFloat(m.pricing?.completion ?? "0") * 1_000_000,
259
+ }));
260
+ return this.cachedModels;
261
+ }
262
+ catch {
263
+ return this.listModels();
264
+ }
265
+ }
266
+ async healthCheck() {
267
+ try {
268
+ const res = await fetch(`${this.baseUrl}/models`, {
269
+ headers: this.headers(),
270
+ });
271
+ return res.ok;
272
+ }
273
+ catch {
274
+ return false;
275
+ }
276
+ }
277
+ }
278
+ //# sourceMappingURL=openrouter.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"openrouter.js","sourceRoot":"","sources":["../../src/providers/openrouter.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,35 @@
1
+ /**
2
+ * Agent loop — the core LLM-to-Tool orchestration cycle.
3
+ *
4
+ * Mirrors Claude Code's query.ts while(true) pattern:
5
+ * 1. Send messages to LLM
6
+ * 2. If LLM requests tool calls → execute them → loop back
7
+ * 3. If LLM returns text only → yield to UI → done
8
+ *
9
+ * Uses async generators for streaming events to the React UI.
10
+ */
11
+ import type { Tools } from "./Tool.js";
12
+ import type { StreamEvent } from "./types/events.js";
13
+ import type { Message } from "./types/message.js";
14
+ import type { AskUserFn, PermissionMode } from "./types/permissions.js";
15
+ import type { Provider } from "./providers/base.js";
16
+ export type QueryConfig = {
17
+ provider: Provider;
18
+ tools: Tools;
19
+ systemPrompt: string;
20
+ permissionMode: PermissionMode;
21
+ askUser?: AskUserFn;
22
+ maxTurns?: number;
23
+ maxCost?: number;
24
+ };
25
+ type State = {
26
+ messages: Message[];
27
+ turn: number;
28
+ totalCost: number;
29
+ };
30
+ /**
31
+ * Main agent loop. Yields streaming events to the UI.
32
+ */
33
+ export declare function query(userMessage: string, config: QueryConfig, existingMessages?: Message[]): AsyncGenerator<StreamEvent, void>;
34
+ export { type State };
35
+ //# sourceMappingURL=query.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"query.d.ts","sourceRoot":"","sources":["../src/query.ts"],"names":[],"mappings":"AAAA;;;;;;;;;GASG;AAEH,OAAO,KAAK,EAAiC,KAAK,EAAE,MAAM,WAAW,CAAC;AAEtE,OAAO,KAAK,EAAE,WAAW,EAAE,MAAM,mBAAmB,CAAC;AACrD,OAAO,KAAK,EACV,OAAO,EAGR,MAAM,oBAAoB,CAAC;AAM5B,OAAO,KAAK,EAAE,SAAS,EAAE,cAAc,EAAE,MAAM,wBAAwB,CAAC;AAExE,OAAO,KAAK,EAAE,QAAQ,EAAE,MAAM,qBAAqB,CAAC;AAEpD,MAAM,MAAM,WAAW,GAAG;IACxB,QAAQ,EAAE,QAAQ,CAAC;IACnB,KAAK,EAAE,KAAK,CAAC;IACb,YAAY,EAAE,MAAM,CAAC;IACrB,cAAc,EAAE,cAAc,CAAC;IAC/B,OAAO,CAAC,EAAE,SAAS,CAAC;IACpB,QAAQ,CAAC,EAAE,MAAM,CAAC;IAClB,OAAO,CAAC,EAAE,MAAM,CAAC;CAClB,CAAC;AAEF,KAAK,KAAK,GAAG;IACX,QAAQ,EAAE,OAAO,EAAE,CAAC;IACpB,IAAI,EAAE,MAAM,CAAC;IACb,SAAS,EAAE,MAAM,CAAC;CACnB,CAAC;AAIF;;GAEG;AACH,wBAAuB,KAAK,CAC1B,WAAW,EAAE,MAAM,EACnB,MAAM,EAAE,WAAW,EACnB,gBAAgB,GAAE,OAAO,EAAO,GAC/B,cAAc,CAAC,WAAW,EAAE,IAAI,CAAC,CA0FnC;AAqJD,OAAO,EAAE,KAAK,KAAK,EAAE,CAAC"}