onlycode 1.18.0 → 1.20.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/bin/{opencode.cjs → onlycode.cjs} +8 -42
- package/package.json +12 -147
- package/src/account/account.ts +0 -463
- package/src/account/repo.ts +0 -173
- package/src/account/schema.ts +0 -99
- package/src/account/url.ts +0 -8
- package/src/acp/agent.ts +0 -95
- package/src/acp/config-option.ts +0 -203
- package/src/acp/content.ts +0 -250
- package/src/acp/directory.ts +0 -210
- package/src/acp/error.ts +0 -90
- package/src/acp/event.ts +0 -342
- package/src/acp/permission.ts +0 -124
- package/src/acp/profile.ts +0 -42
- package/src/acp/service.ts +0 -1048
- package/src/acp/session.ts +0 -231
- package/src/acp/tool.ts +0 -367
- package/src/acp/usage.ts +0 -232
- package/src/agent/agent.ts +0 -459
- package/src/agent/generate.txt +0 -75
- package/src/agent/prompt/compaction.txt +0 -9
- package/src/agent/prompt/explore.txt +0 -18
- package/src/agent/prompt/summary.txt +0 -11
- package/src/agent/prompt/title.txt +0 -44
- package/src/agent/subagent-permissions.ts +0 -27
- package/src/audio.d.ts +0 -14
- package/src/auth/index.ts +0 -99
- package/src/background/job.ts +0 -39
- package/src/bus/global.ts +0 -22
- package/src/cli/bootstrap.ts +0 -11
- package/src/cli/cmd/account.ts +0 -264
- package/src/cli/cmd/acp.ts +0 -73
- package/src/cli/cmd/agent.ts +0 -259
- package/src/cli/cmd/attach.ts +0 -97
- package/src/cli/cmd/cmd.ts +0 -7
- package/src/cli/cmd/db.ts +0 -62
- package/src/cli/cmd/debug/agent.handler.ts +0 -193
- package/src/cli/cmd/debug/agent.ts +0 -27
- package/src/cli/cmd/debug/config.ts +0 -14
- package/src/cli/cmd/debug/file.ts +0 -73
- package/src/cli/cmd/debug/index.ts +0 -87
- package/src/cli/cmd/debug/lsp.ts +0 -50
- package/src/cli/cmd/debug/ripgrep.ts +0 -79
- package/src/cli/cmd/debug/scrap.ts +0 -15
- package/src/cli/cmd/debug/skill.ts +0 -15
- package/src/cli/cmd/debug/snapshot.ts +0 -50
- package/src/cli/cmd/debug/startup.ts +0 -11
- package/src/cli/cmd/debug/v2.ts +0 -44
- package/src/cli/cmd/export.ts +0 -292
- package/src/cli/cmd/generate.ts +0 -54
- package/src/cli/cmd/github.handler.ts +0 -1593
- package/src/cli/cmd/github.shared.ts +0 -30
- package/src/cli/cmd/github.ts +0 -42
- package/src/cli/cmd/import.ts +0 -224
- package/src/cli/cmd/mcp.ts +0 -849
- package/src/cli/cmd/models.ts +0 -66
- package/src/cli/cmd/plug.ts +0 -230
- package/src/cli/cmd/pr.ts +0 -115
- package/src/cli/cmd/prompt-display.ts +0 -1
- package/src/cli/cmd/providers.ts +0 -534
- package/src/cli/cmd/run/demo.ts +0 -1274
- package/src/cli/cmd/run/entry.body.ts +0 -205
- package/src/cli/cmd/run/footer.command.tsx +0 -1064
- package/src/cli/cmd/run/footer.menu.tsx +0 -351
- package/src/cli/cmd/run/footer.permission.tsx +0 -472
- package/src/cli/cmd/run/footer.prompt.tsx +0 -1306
- package/src/cli/cmd/run/footer.question.tsx +0 -573
- package/src/cli/cmd/run/footer.subagent.tsx +0 -173
- package/src/cli/cmd/run/footer.ts +0 -1129
- package/src/cli/cmd/run/footer.view.tsx +0 -943
- package/src/cli/cmd/run/footer.width.ts +0 -27
- package/src/cli/cmd/run/permission.shared.ts +0 -256
- package/src/cli/cmd/run/prompt.editor.ts +0 -157
- package/src/cli/cmd/run/prompt.shared.ts +0 -153
- package/src/cli/cmd/run/question.shared.ts +0 -340
- package/src/cli/cmd/run/runtime.boot.ts +0 -202
- package/src/cli/cmd/run/runtime.lifecycle.ts +0 -406
- package/src/cli/cmd/run/runtime.queue.ts +0 -349
- package/src/cli/cmd/run/runtime.shared.ts +0 -17
- package/src/cli/cmd/run/runtime.stdin.ts +0 -37
- package/src/cli/cmd/run/runtime.ts +0 -814
- package/src/cli/cmd/run/scrollback.shared.ts +0 -92
- package/src/cli/cmd/run/scrollback.surface.ts +0 -431
- package/src/cli/cmd/run/scrollback.writer.tsx +0 -352
- package/src/cli/cmd/run/session-data.ts +0 -1113
- package/src/cli/cmd/run/session-replay.ts +0 -374
- package/src/cli/cmd/run/session.shared.ts +0 -196
- package/src/cli/cmd/run/splash.ts +0 -275
- package/src/cli/cmd/run/stream.transport.ts +0 -1462
- package/src/cli/cmd/run/stream.ts +0 -175
- package/src/cli/cmd/run/subagent-data.ts +0 -876
- package/src/cli/cmd/run/theme.ts +0 -690
- package/src/cli/cmd/run/tool.ts +0 -1489
- package/src/cli/cmd/run/trace.ts +0 -94
- package/src/cli/cmd/run/turn-summary.ts +0 -47
- package/src/cli/cmd/run/types.ts +0 -350
- package/src/cli/cmd/run/variant.shared.ts +0 -215
- package/src/cli/cmd/run.ts +0 -894
- package/src/cli/cmd/serve.ts +0 -24
- package/src/cli/cmd/session.ts +0 -147
- package/src/cli/cmd/stats.ts +0 -393
- package/src/cli/cmd/tui.ts +0 -224
- package/src/cli/cmd/uninstall.ts +0 -353
- package/src/cli/cmd/upgrade.ts +0 -74
- package/src/cli/cmd/web.ts +0 -84
- package/src/cli/effect/prompt.ts +0 -37
- package/src/cli/effect-cmd.ts +0 -96
- package/src/cli/error.ts +0 -130
- package/src/cli/heap.ts +0 -45
- package/src/cli/logo-pixel.ts +0 -35
- package/src/cli/logo.ts +0 -1
- package/src/cli/network.ts +0 -64
- package/src/cli/tui/layer.ts +0 -7
- package/src/cli/tui/validate-session.ts +0 -29
- package/src/cli/tui/worker.ts +0 -71
- package/src/cli/ui.ts +0 -148
- package/src/cli/upgrade.ts +0 -53
- package/src/command/index.ts +0 -184
- package/src/command/template/initialize.txt +0 -66
- package/src/command/template/review.txt +0 -101
- package/src/config/agent.ts +0 -59
- package/src/config/command.ts +0 -39
- package/src/config/config.ts +0 -686
- package/src/config/entry-name.ts +0 -19
- package/src/config/managed.ts +0 -69
- package/src/config/markdown.ts +0 -36
- package/src/config/parse.ts +0 -79
- package/src/config/paths.ts +0 -45
- package/src/config/plugin.ts +0 -79
- package/src/config/tui-cwd.ts +0 -5
- package/src/config/tui-host-attention.ts +0 -21
- package/src/config/tui-migrate.ts +0 -132
- package/src/config/tui.ts +0 -274
- package/src/config/variable.ts +0 -91
- package/src/control-plane/adapters/index.ts +0 -41
- package/src/control-plane/adapters/worktree.ts +0 -96
- package/src/control-plane/dev/README.md +0 -19
- package/src/control-plane/dev/debug-workspace-plugin.ts +0 -73
- package/src/control-plane/types.ts +0 -59
- package/src/control-plane/util.ts +0 -39
- package/src/control-plane/workspace-adapter-runtime.ts +0 -51
- package/src/control-plane/workspace-context.ts +0 -26
- package/src/control-plane/workspace.ts +0 -989
- package/src/effect/app-runtime.ts +0 -132
- package/src/effect/bootstrap-runtime.ts +0 -23
- package/src/effect/bridge.ts +0 -84
- package/src/effect/config-service.ts +0 -67
- package/src/effect/instance-ref.ts +0 -11
- package/src/effect/instance-registry.ts +0 -12
- package/src/effect/instance-state.ts +0 -69
- package/src/effect/promise.ts +0 -17
- package/src/effect/run-service.ts +0 -47
- package/src/effect/runner.ts +0 -217
- package/src/effect/runtime-flags.ts +0 -79
- package/src/env/index.ts +0 -43
- package/src/event-v2-bridge.ts +0 -79
- package/src/format/formatter.ts +0 -404
- package/src/format/index.ts +0 -205
- package/src/git/index.ts +0 -350
- package/src/id/id.ts +0 -80
- package/src/ide/index.ts +0 -61
- package/src/image/image.ts +0 -174
- package/src/index.ts +0 -142
- package/src/installation/index.ts +0 -350
- package/src/lsp/client.ts +0 -650
- package/src/lsp/diagnostic.ts +0 -29
- package/src/lsp/language.ts +0 -121
- package/src/lsp/launch.ts +0 -21
- package/src/lsp/lsp.ts +0 -511
- package/src/lsp/server.ts +0 -1983
- package/src/markdown.d.ts +0 -4
- package/src/mcp/auth.ts +0 -174
- package/src/mcp/catalog.ts +0 -153
- package/src/mcp/index.ts +0 -953
- package/src/mcp/oauth-callback.ts +0 -233
- package/src/mcp/oauth-provider.ts +0 -206
- package/src/node.ts +0 -4
- package/src/patch/index.ts +0 -686
- package/src/permission/arity.ts +0 -163
- package/src/permission/evaluate.ts +0 -1
- package/src/permission/index.ts +0 -230
- package/src/plugin/azure.ts +0 -26
- package/src/plugin/cloudflare.ts +0 -76
- package/src/plugin/digitalocean.ts +0 -383
- package/src/plugin/github-copilot/copilot.ts +0 -414
- package/src/plugin/github-copilot/models.ts +0 -246
- package/src/plugin/index.ts +0 -316
- package/src/plugin/install.ts +0 -439
- package/src/plugin/loader.ts +0 -237
- package/src/plugin/meta.ts +0 -188
- package/src/plugin/openai/README.md +0 -31
- package/src/plugin/openai/codex.ts +0 -641
- package/src/plugin/openai/ws-pool.ts +0 -270
- package/src/plugin/openai/ws.ts +0 -381
- package/src/plugin/pty-environment.ts +0 -24
- package/src/plugin/shared.ts +0 -323
- package/src/plugin/snowflake-cortex.ts +0 -529
- package/src/plugin/tui/internal.ts +0 -10
- package/src/plugin/tui/runtime.ts +0 -1130
- package/src/plugin/xai.ts +0 -716
- package/src/project/bootstrap-service.ts +0 -9
- package/src/project/bootstrap.ts +0 -76
- package/src/project/instance-context.ts +0 -24
- package/src/project/instance-layer.ts +0 -11
- package/src/project/instance-runtime.ts +0 -16
- package/src/project/instance-store.ts +0 -209
- package/src/project/project.ts +0 -519
- package/src/project/vcs.ts +0 -431
- package/src/provider/auth.ts +0 -233
- package/src/provider/error.ts +0 -188
- package/src/provider/model-status.ts +0 -8
- package/src/provider/provider.ts +0 -1975
- package/src/provider/transform.ts +0 -1543
- package/src/question/index.ts +0 -229
- package/src/question/schema.ts +0 -10
- package/src/server/auth.ts +0 -48
- package/src/server/event.ts +0 -13
- package/src/server/global-lifecycle.ts +0 -28
- package/src/server/init-projectors.ts +0 -3
- package/src/server/mdns.ts +0 -47
- package/src/server/projectors.ts +0 -1
- package/src/server/proxy-util.ts +0 -48
- package/src/server/routes/instance/httpapi/AGENTS.md +0 -39
- package/src/server/routes/instance/httpapi/api.ts +0 -78
- package/src/server/routes/instance/httpapi/errors.ts +0 -193
- package/src/server/routes/instance/httpapi/groups/config.ts +0 -65
- package/src/server/routes/instance/httpapi/groups/control-plane.ts +0 -35
- package/src/server/routes/instance/httpapi/groups/control.ts +0 -76
- package/src/server/routes/instance/httpapi/groups/event.ts +0 -29
- package/src/server/routes/instance/httpapi/groups/experimental.ts +0 -275
- package/src/server/routes/instance/httpapi/groups/file.ts +0 -185
- package/src/server/routes/instance/httpapi/groups/global.ts +0 -138
- package/src/server/routes/instance/httpapi/groups/instance.ts +0 -206
- package/src/server/routes/instance/httpapi/groups/mcp.ts +0 -156
- package/src/server/routes/instance/httpapi/groups/metadata.ts +0 -18
- package/src/server/routes/instance/httpapi/groups/permission.ts +0 -61
- package/src/server/routes/instance/httpapi/groups/project-copy.ts +0 -32
- package/src/server/routes/instance/httpapi/groups/project.ts +0 -93
- package/src/server/routes/instance/httpapi/groups/provider.ts +0 -101
- package/src/server/routes/instance/httpapi/groups/pty.ts +0 -172
- package/src/server/routes/instance/httpapi/groups/query.ts +0 -12
- package/src/server/routes/instance/httpapi/groups/question.ts +0 -74
- package/src/server/routes/instance/httpapi/groups/session.ts +0 -462
- package/src/server/routes/instance/httpapi/groups/sync.ts +0 -113
- package/src/server/routes/instance/httpapi/groups/tui.ts +0 -208
- package/src/server/routes/instance/httpapi/groups/workspace.ts +0 -141
- package/src/server/routes/instance/httpapi/handlers/config.ts +0 -34
- package/src/server/routes/instance/httpapi/handlers/control-plane.ts +0 -37
- package/src/server/routes/instance/httpapi/handlers/control.ts +0 -43
- package/src/server/routes/instance/httpapi/handlers/event.ts +0 -99
- package/src/server/routes/instance/httpapi/handlers/experimental.ts +0 -192
- package/src/server/routes/instance/httpapi/handlers/file.ts +0 -139
- package/src/server/routes/instance/httpapi/handlers/global.ts +0 -156
- package/src/server/routes/instance/httpapi/handlers/instance.ts +0 -110
- package/src/server/routes/instance/httpapi/handlers/mcp.ts +0 -111
- package/src/server/routes/instance/httpapi/handlers/permission.ts +0 -41
- package/src/server/routes/instance/httpapi/handlers/project-copy.ts +0 -83
- package/src/server/routes/instance/httpapi/handlers/project.ts +0 -63
- package/src/server/routes/instance/httpapi/handlers/provider.ts +0 -113
- package/src/server/routes/instance/httpapi/handlers/pty.ts +0 -273
- package/src/server/routes/instance/httpapi/handlers/question.ts +0 -54
- package/src/server/routes/instance/httpapi/handlers/session-errors.ts +0 -21
- package/src/server/routes/instance/httpapi/handlers/session.ts +0 -440
- package/src/server/routes/instance/httpapi/handlers/sync.ts +0 -89
- package/src/server/routes/instance/httpapi/handlers/tui.ts +0 -131
- package/src/server/routes/instance/httpapi/handlers/workspace.ts +0 -102
- package/src/server/routes/instance/httpapi/lifecycle.ts +0 -54
- package/src/server/routes/instance/httpapi/middleware/authorization.ts +0 -150
- package/src/server/routes/instance/httpapi/middleware/compression.ts +0 -64
- package/src/server/routes/instance/httpapi/middleware/cors-vary.ts +0 -29
- package/src/server/routes/instance/httpapi/middleware/error.ts +0 -43
- package/src/server/routes/instance/httpapi/middleware/fence.ts +0 -25
- package/src/server/routes/instance/httpapi/middleware/instance-context.ts +0 -43
- package/src/server/routes/instance/httpapi/middleware/proxy.ts +0 -108
- package/src/server/routes/instance/httpapi/middleware/schema-error.ts +0 -41
- package/src/server/routes/instance/httpapi/middleware/workspace-routing.ts +0 -250
- package/src/server/routes/instance/httpapi/public.ts +0 -535
- package/src/server/routes/instance/httpapi/server.ts +0 -298
- package/src/server/routes/instance/httpapi/websocket-tracker.ts +0 -57
- package/src/server/server.ts +0 -225
- package/src/server/shared/fence.ts +0 -60
- package/src/server/shared/pty-ticket.ts +0 -15
- package/src/server/shared/public-ui.ts +0 -12
- package/src/server/shared/tui-control.ts +0 -28
- package/src/server/shared/ui.ts +0 -108
- package/src/server/shared/workspace-routing.ts +0 -38
- package/src/server/tui-event.ts +0 -53
- package/src/session/compaction.ts +0 -620
- package/src/session/instruction.ts +0 -241
- package/src/session/llm/AGENTS.md +0 -90
- package/src/session/llm/ai-sdk.ts +0 -288
- package/src/session/llm/native-request.ts +0 -196
- package/src/session/llm/native-runtime.ts +0 -195
- package/src/session/llm/request.ts +0 -216
- package/src/session/llm.ts +0 -415
- package/src/session/message-error.ts +0 -14
- package/src/session/message-v2.ts +0 -747
- package/src/session/message.ts +0 -148
- package/src/session/overflow.ts +0 -34
- package/src/session/processor.ts +0 -1084
- package/src/session/prompt/anthropic.txt +0 -109
- package/src/session/prompt/beast.txt +0 -151
- package/src/session/prompt/build-switch.txt +0 -9
- package/src/session/prompt/codex.txt +0 -83
- package/src/session/prompt/copilot-gpt-5.txt +0 -147
- package/src/session/prompt/default.txt +0 -99
- package/src/session/prompt/gemini.txt +0 -159
- package/src/session/prompt/gpt.txt +0 -111
- package/src/session/prompt/kimi.txt +0 -99
- package/src/session/prompt/plan-mode.txt +0 -74
- package/src/session/prompt/plan-reminder-anthropic.txt +0 -71
- package/src/session/prompt/plan.txt +0 -30
- package/src/session/prompt/trinity.txt +0 -101
- package/src/session/prompt.ts +0 -1707
- package/src/session/reminders.ts +0 -92
- package/src/session/retry.ts +0 -201
- package/src/session/revert.ts +0 -160
- package/src/session/run-state.ts +0 -156
- package/src/session/schema.ts +0 -26
- package/src/session/session.ts +0 -1119
- package/src/session/status.ts +0 -97
- package/src/session/summary.ts +0 -165
- package/src/session/system.ts +0 -117
- package/src/session/todo.ts +0 -90
- package/src/session/tools.ts +0 -207
- package/src/share/session.ts +0 -61
- package/src/share/share-next.ts +0 -385
- package/src/skill/discovery.ts +0 -109
- package/src/skill/index.ts +0 -366
- package/src/snapshot/index.ts +0 -808
- package/src/sql.d.ts +0 -4
- package/src/storage/schema.ts +0 -5
- package/src/storage/storage.ts +0 -329
- package/src/sync/README.md +0 -179
- package/src/sync/schema.ts +0 -11
- package/src/temporary.ts +0 -31
- package/src/tool/apply_patch.ts +0 -315
- package/src/tool/apply_patch.txt +0 -33
- package/src/tool/apply_patch.zh.txt +0 -33
- package/src/tool/description.ts +0 -100
- package/src/tool/edit.ts +0 -739
- package/src/tool/edit.txt +0 -10
- package/src/tool/edit.zh.txt +0 -10
- package/src/tool/external-directory.ts +0 -49
- package/src/tool/glob.ts +0 -78
- package/src/tool/glob.txt +0 -6
- package/src/tool/glob.zh.txt +0 -6
- package/src/tool/grep.ts +0 -114
- package/src/tool/grep.txt +0 -8
- package/src/tool/grep.zh.txt +0 -8
- package/src/tool/invalid.ts +0 -21
- package/src/tool/json-schema.ts +0 -164
- package/src/tool/lsp.ts +0 -115
- package/src/tool/lsp.txt +0 -24
- package/src/tool/lsp.zh.txt +0 -24
- package/src/tool/mcp-websearch.ts +0 -96
- package/src/tool/plan-enter.txt +0 -14
- package/src/tool/plan-enter.zh.txt +0 -14
- package/src/tool/plan-exit.txt +0 -13
- package/src/tool/plan-exit.zh.txt +0 -13
- package/src/tool/plan.ts +0 -81
- package/src/tool/question.ts +0 -46
- package/src/tool/question.txt +0 -10
- package/src/tool/question.zh.txt +0 -10
- package/src/tool/read.ts +0 -388
- package/src/tool/read.txt +0 -14
- package/src/tool/read.zh.txt +0 -14
- package/src/tool/registry.ts +0 -440
- package/src/tool/schema.ts +0 -14
- package/src/tool/shell/id.ts +0 -19
- package/src/tool/shell/prompt.ts +0 -307
- package/src/tool/shell/shell.txt +0 -21
- package/src/tool/shell.ts +0 -657
- package/src/tool/skill.ts +0 -73
- package/src/tool/skill.txt +0 -5
- package/src/tool/skill.zh.txt +0 -5
- package/src/tool/task.ts +0 -348
- package/src/tool/task.txt +0 -19
- package/src/tool/task.zh.txt +0 -19
- package/src/tool/todo.ts +0 -59
- package/src/tool/todowrite.txt +0 -44
- package/src/tool/todowrite.zh.txt +0 -44
- package/src/tool/tool.ts +0 -183
- package/src/tool/truncate.ts +0 -158
- package/src/tool/truncation-dir.ts +0 -4
- package/src/tool/webfetch.ts +0 -194
- package/src/tool/webfetch.txt +0 -13
- package/src/tool/webfetch.zh.txt +0 -13
- package/src/tool/websearch.ts +0 -145
- package/src/tool/websearch.txt +0 -14
- package/src/tool/websearch.zh.txt +0 -14
- package/src/tool/write.ts +0 -106
- package/src/tool/write.txt +0 -8
- package/src/tool/write.zh.txt +0 -8
- package/src/util/archive.ts +0 -17
- package/src/util/bom.ts +0 -27
- package/src/util/data-url.ts +0 -9
- package/src/util/defer.ts +0 -10
- package/src/util/effect-http-client.ts +0 -11
- package/src/util/error.ts +0 -1
- package/src/util/filesystem.ts +0 -251
- package/src/util/html.ts +0 -8
- package/src/util/iife.ts +0 -3
- package/src/util/lazy.ts +0 -20
- package/src/util/local-context.ts +0 -25
- package/src/util/locale.ts +0 -2
- package/src/util/media.ts +0 -26
- package/src/util/process.ts +0 -177
- package/src/util/proxy-env.ts +0 -72
- package/src/util/queue.ts +0 -32
- package/src/util/record.ts +0 -1
- package/src/util/repository.ts +0 -232
- package/src/util/rpc.ts +0 -66
- package/src/util/signal.ts +0 -12
- package/src/util/timeout.ts +0 -13
- package/src/util/token.ts +0 -1
- package/src/util/wildcard.ts +0 -59
- package/src/worktree/index.ts +0 -654
|
@@ -1,196 +0,0 @@
|
|
|
1
|
-
import type { JsonSchema, LLMRequest, ProviderMetadata } from "@opencode-ai/llm"
|
|
2
|
-
import { LLM, Message, SystemPart, ToolCallPart, ToolDefinition, ToolResultPart } from "@opencode-ai/llm"
|
|
3
|
-
import {
|
|
4
|
-
AmazonBedrock,
|
|
5
|
-
Anthropic,
|
|
6
|
-
Azure,
|
|
7
|
-
Google,
|
|
8
|
-
OpenAI,
|
|
9
|
-
OpenAICompatible,
|
|
10
|
-
OpenRouter,
|
|
11
|
-
} from "@opencode-ai/llm/providers"
|
|
12
|
-
import type { ModelMessage } from "ai"
|
|
13
|
-
import type { Provider } from "@/provider/provider"
|
|
14
|
-
import { isRecord } from "@/util/record"
|
|
15
|
-
|
|
16
|
-
type ToolInput = {
|
|
17
|
-
readonly description?: string
|
|
18
|
-
readonly inputSchema?: unknown
|
|
19
|
-
}
|
|
20
|
-
|
|
21
|
-
export type RequestInput = {
|
|
22
|
-
readonly model: Provider.Model
|
|
23
|
-
readonly apiKey?: string
|
|
24
|
-
readonly baseURL?: string
|
|
25
|
-
readonly system?: readonly string[]
|
|
26
|
-
readonly messages: readonly ModelMessage[]
|
|
27
|
-
readonly tools?: Record<string, ToolInput>
|
|
28
|
-
readonly toolChoice?: "auto" | "required" | "none"
|
|
29
|
-
readonly temperature?: number
|
|
30
|
-
readonly topP?: number
|
|
31
|
-
readonly topK?: number
|
|
32
|
-
readonly maxOutputTokens?: number
|
|
33
|
-
readonly providerOptions?: LLMRequest["providerOptions"]
|
|
34
|
-
readonly headers?: Record<string, string>
|
|
35
|
-
}
|
|
36
|
-
|
|
37
|
-
const providerMetadata = (value: unknown): ProviderMetadata | undefined => {
|
|
38
|
-
if (!isRecord(value)) return undefined
|
|
39
|
-
const result = Object.fromEntries(
|
|
40
|
-
Object.entries(value).filter((entry): entry is [string, Record<string, unknown>] => isRecord(entry[1])),
|
|
41
|
-
)
|
|
42
|
-
return Object.keys(result).length === 0 ? undefined : result
|
|
43
|
-
}
|
|
44
|
-
|
|
45
|
-
// Stored AI SDK parts historically kept provider-owned continuation metadata in
|
|
46
|
-
// `providerOptions`; native parts now use `providerMetadata` directly.
|
|
47
|
-
const partProviderMetadata = (part: Record<string, unknown>) =>
|
|
48
|
-
providerMetadata(part.providerMetadata) ?? providerMetadata(part.providerOptions)
|
|
49
|
-
|
|
50
|
-
const textPart = (part: Record<string, unknown>) => ({
|
|
51
|
-
type: "text" as const,
|
|
52
|
-
text: typeof part.text === "string" ? part.text : "",
|
|
53
|
-
providerMetadata: partProviderMetadata(part),
|
|
54
|
-
})
|
|
55
|
-
|
|
56
|
-
const mediaPart = (part: Record<string, unknown>) => {
|
|
57
|
-
if (typeof part.data !== "string" && !(part.data instanceof Uint8Array))
|
|
58
|
-
throw new Error("Native LLM request adapter only supports file parts with string or Uint8Array data")
|
|
59
|
-
return {
|
|
60
|
-
type: "media" as const,
|
|
61
|
-
mediaType: typeof part.mediaType === "string" ? part.mediaType : "application/octet-stream",
|
|
62
|
-
data: part.data,
|
|
63
|
-
filename: typeof part.filename === "string" ? part.filename : undefined,
|
|
64
|
-
}
|
|
65
|
-
}
|
|
66
|
-
|
|
67
|
-
const toolResult = (part: Record<string, unknown>) => {
|
|
68
|
-
const output = isRecord(part.output) ? part.output : { type: "json", value: part.output }
|
|
69
|
-
const type = output.type === "text" ? "text" : output.type === "error-text" ? "error" : "json"
|
|
70
|
-
return ToolResultPart.make({
|
|
71
|
-
id: typeof part.toolCallId === "string" ? part.toolCallId : "",
|
|
72
|
-
name: typeof part.toolName === "string" ? part.toolName : "",
|
|
73
|
-
result: "value" in output ? output.value : output,
|
|
74
|
-
resultType: type,
|
|
75
|
-
providerExecuted: typeof part.providerExecuted === "boolean" ? part.providerExecuted : undefined,
|
|
76
|
-
providerMetadata: partProviderMetadata(part),
|
|
77
|
-
})
|
|
78
|
-
}
|
|
79
|
-
|
|
80
|
-
const contentPart = (part: unknown) => {
|
|
81
|
-
if (!isRecord(part)) throw new Error("Native LLM request adapter only supports object content parts")
|
|
82
|
-
if (part.type === "text") return textPart(part)
|
|
83
|
-
if (part.type === "file") return mediaPart(part)
|
|
84
|
-
if (part.type === "reasoning")
|
|
85
|
-
return {
|
|
86
|
-
type: "reasoning" as const,
|
|
87
|
-
text: typeof part.text === "string" ? part.text : "",
|
|
88
|
-
providerMetadata: partProviderMetadata(part),
|
|
89
|
-
}
|
|
90
|
-
if (part.type === "tool-call")
|
|
91
|
-
return ToolCallPart.make({
|
|
92
|
-
id: typeof part.toolCallId === "string" ? part.toolCallId : "",
|
|
93
|
-
name: typeof part.toolName === "string" ? part.toolName : "",
|
|
94
|
-
input: part.input,
|
|
95
|
-
providerExecuted: typeof part.providerExecuted === "boolean" ? part.providerExecuted : undefined,
|
|
96
|
-
providerMetadata: partProviderMetadata(part),
|
|
97
|
-
})
|
|
98
|
-
if (part.type === "tool-result") return toolResult(part)
|
|
99
|
-
throw new Error(`Native LLM request adapter does not support ${String(part.type)} content parts`)
|
|
100
|
-
}
|
|
101
|
-
|
|
102
|
-
const content = (value: ModelMessage["content"]) =>
|
|
103
|
-
typeof value === "string" ? [{ type: "text" as const, text: value }] : value.map(contentPart)
|
|
104
|
-
|
|
105
|
-
const messages = (input: readonly ModelMessage[]) => {
|
|
106
|
-
const system = input.flatMap((message) => (message.role === "system" ? [SystemPart.make(message.content)] : []))
|
|
107
|
-
const messages = input.flatMap((message) => {
|
|
108
|
-
if (message.role === "system") return []
|
|
109
|
-
return [
|
|
110
|
-
Message.make({
|
|
111
|
-
role: message.role,
|
|
112
|
-
content: content(message.content),
|
|
113
|
-
native: isRecord(message.providerOptions) ? { providerOptions: message.providerOptions } : undefined,
|
|
114
|
-
}),
|
|
115
|
-
]
|
|
116
|
-
})
|
|
117
|
-
return { system, messages }
|
|
118
|
-
}
|
|
119
|
-
|
|
120
|
-
const schema = (value: unknown): JsonSchema => {
|
|
121
|
-
if (!isRecord(value)) return { type: "object", properties: {} }
|
|
122
|
-
if (isRecord(value.jsonSchema)) return value.jsonSchema
|
|
123
|
-
return value
|
|
124
|
-
}
|
|
125
|
-
|
|
126
|
-
const tools = (input: Record<string, ToolInput> | undefined): ToolDefinition[] =>
|
|
127
|
-
Object.entries(input ?? {}).map(([name, item]) =>
|
|
128
|
-
ToolDefinition.make({
|
|
129
|
-
name,
|
|
130
|
-
description: item.description ?? "",
|
|
131
|
-
inputSchema: schema(item.inputSchema),
|
|
132
|
-
}),
|
|
133
|
-
)
|
|
134
|
-
|
|
135
|
-
const generation = (input: RequestInput) => {
|
|
136
|
-
const result = {
|
|
137
|
-
temperature: input.temperature,
|
|
138
|
-
topP: input.topP,
|
|
139
|
-
topK: input.topK,
|
|
140
|
-
maxTokens: input.maxOutputTokens,
|
|
141
|
-
}
|
|
142
|
-
return Object.values(result).some((value) => value !== undefined) ? result : undefined
|
|
143
|
-
}
|
|
144
|
-
|
|
145
|
-
const baseURL = (input: Provider.Model | RequestInput) =>
|
|
146
|
-
"model" in input ? (input.baseURL ?? (input.model.api.url || undefined)) : input.api.url || undefined
|
|
147
|
-
|
|
148
|
-
const requireBaseURL = (model: Provider.Model, url: string | undefined) => {
|
|
149
|
-
if (url) return url
|
|
150
|
-
throw new Error(`Native LLM request adapter requires a base URL for ${model.providerID}/${model.id}`)
|
|
151
|
-
}
|
|
152
|
-
|
|
153
|
-
export const model = (input: Provider.Model | RequestInput, headers?: Record<string, string>) => {
|
|
154
|
-
const model = "model" in input ? input.model : input
|
|
155
|
-
const url = baseURL(input)
|
|
156
|
-
const options = {
|
|
157
|
-
...("model" in input && input.apiKey ? { apiKey: input.apiKey } : {}),
|
|
158
|
-
...(url ? { baseURL: url } : {}),
|
|
159
|
-
headers: Object.keys({ ...model.headers, ...headers }).length === 0 ? undefined : { ...model.headers, ...headers },
|
|
160
|
-
limits: {
|
|
161
|
-
context: model.limit.context,
|
|
162
|
-
output: model.limit.output,
|
|
163
|
-
},
|
|
164
|
-
}
|
|
165
|
-
if (model.api.npm === "@ai-sdk/openai") return OpenAI.configure(options).responses(model.api.id)
|
|
166
|
-
if (model.api.npm === "@ai-sdk/azure")
|
|
167
|
-
return Azure.configure({ ...options, baseURL: requireBaseURL(model, url) }).responses(model.api.id)
|
|
168
|
-
if (model.api.npm === "@ai-sdk/anthropic") return Anthropic.configure(options).model(model.api.id)
|
|
169
|
-
if (model.api.npm === "@ai-sdk/google") return Google.configure(options).model(model.api.id)
|
|
170
|
-
if (model.api.npm === "@ai-sdk/amazon-bedrock") return AmazonBedrock.configure(options).model(model.api.id)
|
|
171
|
-
if (model.api.npm === "@ai-sdk/openai-compatible")
|
|
172
|
-
return OpenAICompatible.configure({
|
|
173
|
-
...options,
|
|
174
|
-
provider: String(model.providerID),
|
|
175
|
-
baseURL: requireBaseURL(model, url),
|
|
176
|
-
}).model(model.api.id)
|
|
177
|
-
if (model.api.npm === "@openrouter/ai-sdk-provider") return OpenRouter.configure(options).model(model.api.id)
|
|
178
|
-
throw new Error(`Native LLM request adapter does not support provider package ${model.api.npm}`)
|
|
179
|
-
}
|
|
180
|
-
|
|
181
|
-
export const request = (input: RequestInput) => {
|
|
182
|
-
const converted = messages(input.messages)
|
|
183
|
-
// This is the only native adapter boundary that should construct canonical
|
|
184
|
-
// @opencode-ai/llm request objects from opencode's session/AI SDK-shaped data.
|
|
185
|
-
return LLM.request({
|
|
186
|
-
model: model(input, input.headers),
|
|
187
|
-
system: [...(input.system ?? []).map(SystemPart.make), ...converted.system],
|
|
188
|
-
messages: converted.messages,
|
|
189
|
-
tools: tools(input.tools),
|
|
190
|
-
toolChoice: input.toolChoice,
|
|
191
|
-
generation: generation(input),
|
|
192
|
-
providerOptions: input.providerOptions,
|
|
193
|
-
})
|
|
194
|
-
}
|
|
195
|
-
|
|
196
|
-
export * as LLMNative from "./native-request"
|
|
@@ -1,195 +0,0 @@
|
|
|
1
|
-
import type { Auth } from "@/auth"
|
|
2
|
-
import type { Provider } from "@/provider/provider"
|
|
3
|
-
import { ProviderTransform } from "@/provider/transform"
|
|
4
|
-
import { errorMessage } from "@/util/error"
|
|
5
|
-
import { isRecord } from "@/util/record"
|
|
6
|
-
import { asSchema, type ModelMessage, type Tool } from "ai"
|
|
7
|
-
import { Cause, Effect, FiberSet, Queue } from "effect"
|
|
8
|
-
import * as Stream from "effect/Stream"
|
|
9
|
-
import { FetchHttpClient } from "effect/unstable/http"
|
|
10
|
-
import {
|
|
11
|
-
LLMRequest,
|
|
12
|
-
Tool as NativeTool,
|
|
13
|
-
ToolFailure,
|
|
14
|
-
ToolRuntime,
|
|
15
|
-
toDefinitions,
|
|
16
|
-
type JsonSchema,
|
|
17
|
-
type LLMEvent,
|
|
18
|
-
} from "@opencode-ai/llm"
|
|
19
|
-
import type { LLMClientShape } from "@opencode-ai/llm/route"
|
|
20
|
-
import { LLMNative } from "./native-request"
|
|
21
|
-
|
|
22
|
-
export type RuntimeStatus =
|
|
23
|
-
| { readonly type: "supported"; readonly apiKey: string; readonly baseURL?: string }
|
|
24
|
-
| { readonly type: "unsupported"; readonly reason: string }
|
|
25
|
-
export type StreamResult =
|
|
26
|
-
| { readonly type: "supported"; readonly stream: Stream.Stream<LLMEvent, unknown> }
|
|
27
|
-
| { readonly type: "unsupported"; readonly reason: string }
|
|
28
|
-
|
|
29
|
-
type StreamInput = {
|
|
30
|
-
readonly model: Provider.Model
|
|
31
|
-
readonly provider: Provider.Info
|
|
32
|
-
readonly auth: Auth.Info | undefined
|
|
33
|
-
readonly llmClient: LLMClientShape
|
|
34
|
-
readonly messages: ModelMessage[]
|
|
35
|
-
readonly tools: Record<string, Tool>
|
|
36
|
-
readonly toolChoice?: "auto" | "required" | "none"
|
|
37
|
-
readonly temperature?: number
|
|
38
|
-
readonly topP?: number
|
|
39
|
-
readonly topK?: number
|
|
40
|
-
readonly maxOutputTokens?: number
|
|
41
|
-
readonly providerOptions?: Record<string, any>
|
|
42
|
-
readonly headers: Record<string, string>
|
|
43
|
-
readonly abort: AbortSignal
|
|
44
|
-
}
|
|
45
|
-
|
|
46
|
-
export function status(input: Pick<StreamInput, "model" | "provider" | "auth">): RuntimeStatus {
|
|
47
|
-
return statusWithFetch(input, providerFetch(input))
|
|
48
|
-
}
|
|
49
|
-
|
|
50
|
-
function statusWithFetch(
|
|
51
|
-
input: Pick<StreamInput, "model" | "provider" | "auth">,
|
|
52
|
-
fetch: typeof globalThis.fetch | undefined,
|
|
53
|
-
): RuntimeStatus {
|
|
54
|
-
const providerID = input.model.providerID
|
|
55
|
-
if (providerID !== "openai" && providerID !== "anthropic" && !providerID.startsWith("opencode"))
|
|
56
|
-
return { type: "unsupported", reason: "provider is not openai, opencode, or anthropic" }
|
|
57
|
-
const npm = input.model.api.npm
|
|
58
|
-
if (npm !== "@ai-sdk/openai" && npm !== "@ai-sdk/openai-compatible" && npm !== "@ai-sdk/anthropic")
|
|
59
|
-
return { type: "unsupported", reason: "provider package is not OpenAI, OpenAI-compatible, or Anthropic" }
|
|
60
|
-
if (input.auth?.type === "oauth" && !(input.provider.id === "openai" && fetch)) {
|
|
61
|
-
return { type: "unsupported", reason: "OAuth auth requires a provider fetch override" }
|
|
62
|
-
}
|
|
63
|
-
|
|
64
|
-
const apiKey = typeof input.provider.options.apiKey === "string" ? input.provider.options.apiKey : input.provider.key
|
|
65
|
-
if (!apiKey) return { type: "unsupported", reason: "API key is not configured" }
|
|
66
|
-
|
|
67
|
-
return {
|
|
68
|
-
type: "supported",
|
|
69
|
-
apiKey,
|
|
70
|
-
baseURL: typeof input.provider.options.baseURL === "string" ? input.provider.options.baseURL : undefined,
|
|
71
|
-
}
|
|
72
|
-
}
|
|
73
|
-
|
|
74
|
-
export function stream(input: StreamInput): StreamResult {
|
|
75
|
-
const fetch = providerFetch(input)
|
|
76
|
-
const current = statusWithFetch(input, fetch)
|
|
77
|
-
if (current.type === "unsupported") return current
|
|
78
|
-
|
|
79
|
-
// Integration point with @opencode-ai/llm: native-request lowers session data
|
|
80
|
-
// into an LLMRequest, then LLMClient handles route selection and transport.
|
|
81
|
-
//
|
|
82
|
-
// ProviderTransform.providerOptions builds AI-SDK-shaped options for the
|
|
83
|
-
// selected SDK key (e.g. "openai") and the native LLM SDK reads the same
|
|
84
|
-
// keys via OpenAIOptions.* (store, reasoningEffort, reasoningSummary,
|
|
85
|
-
// include, textVerbosity, promptCacheKey). Both sides intentionally use
|
|
86
|
-
// OpenAI's official wire field names, so this is identity, not translation
|
|
87
|
-
// — if a field ever needs to differ between the two surfaces, the
|
|
88
|
-
// translation belongs here, not split across both packages.
|
|
89
|
-
const tools = nativeTools(input.tools, input)
|
|
90
|
-
const request = LLMNative.request({
|
|
91
|
-
model: input.model,
|
|
92
|
-
apiKey: current.apiKey,
|
|
93
|
-
baseURL: current.baseURL,
|
|
94
|
-
messages: ProviderTransform.message(input.messages, input.model, input.providerOptions ?? {}),
|
|
95
|
-
toolChoice: input.toolChoice,
|
|
96
|
-
temperature: input.temperature,
|
|
97
|
-
topP: input.topP,
|
|
98
|
-
topK: input.topK,
|
|
99
|
-
maxOutputTokens: input.maxOutputTokens,
|
|
100
|
-
providerOptions: ProviderTransform.providerOptions(input.model, input.providerOptions ?? {}),
|
|
101
|
-
headers: { ...providerHeaders(input.provider.options.headers), ...input.headers },
|
|
102
|
-
})
|
|
103
|
-
const stream = Stream.scoped(
|
|
104
|
-
Stream.unwrap(
|
|
105
|
-
Effect.gen(function* () {
|
|
106
|
-
const settlements = yield* FiberSet.make<void>()
|
|
107
|
-
const results = yield* Queue.unbounded<LLMEvent, Cause.Done>()
|
|
108
|
-
const provider = input.llmClient
|
|
109
|
-
.stream(
|
|
110
|
-
LLMRequest.update(request, {
|
|
111
|
-
tools: [...request.tools, ...toDefinitions(tools)],
|
|
112
|
-
}),
|
|
113
|
-
)
|
|
114
|
-
.pipe(
|
|
115
|
-
Stream.flatMap((event) =>
|
|
116
|
-
event.type !== "tool-call" || event.providerExecuted
|
|
117
|
-
? Stream.make(event)
|
|
118
|
-
: Stream.make(event).pipe(
|
|
119
|
-
Stream.concat(
|
|
120
|
-
Stream.fromEffectDrain(
|
|
121
|
-
ToolRuntime.dispatch(tools, event).pipe(
|
|
122
|
-
Effect.flatMap((dispatched) => Queue.offerAll(results, dispatched.events)),
|
|
123
|
-
Effect.catchCause((cause) => Queue.failCause(results, cause)),
|
|
124
|
-
Effect.asVoid,
|
|
125
|
-
FiberSet.run(settlements, { startImmediately: true }),
|
|
126
|
-
),
|
|
127
|
-
),
|
|
128
|
-
),
|
|
129
|
-
),
|
|
130
|
-
),
|
|
131
|
-
Stream.concat(
|
|
132
|
-
Stream.fromEffectDrain(
|
|
133
|
-
FiberSet.awaitEmpty(settlements).pipe(Effect.andThen(Queue.end(results)), Effect.asVoid),
|
|
134
|
-
),
|
|
135
|
-
),
|
|
136
|
-
)
|
|
137
|
-
return provider.pipe(Stream.concat(Stream.fromQueue(results)))
|
|
138
|
-
}),
|
|
139
|
-
),
|
|
140
|
-
)
|
|
141
|
-
|
|
142
|
-
return {
|
|
143
|
-
...current,
|
|
144
|
-
stream: fetch ? stream.pipe(Stream.provideService(FetchHttpClient.Fetch, fetch)) : stream,
|
|
145
|
-
}
|
|
146
|
-
}
|
|
147
|
-
|
|
148
|
-
function providerFetch(input: Pick<StreamInput, "provider" | "auth">): typeof globalThis.fetch | undefined {
|
|
149
|
-
if (input.provider.id !== "openai" || input.auth?.type !== "oauth") return undefined
|
|
150
|
-
const value: unknown = input.provider.options.fetch
|
|
151
|
-
if (typeof value !== "function") return undefined
|
|
152
|
-
return value as typeof globalThis.fetch
|
|
153
|
-
}
|
|
154
|
-
|
|
155
|
-
function providerHeaders(value: unknown): Record<string, string> | undefined {
|
|
156
|
-
if (!isRecord(value)) return undefined
|
|
157
|
-
return Object.fromEntries(
|
|
158
|
-
Object.entries(value).filter((entry): entry is [string, string] => typeof entry[1] === "string"),
|
|
159
|
-
)
|
|
160
|
-
}
|
|
161
|
-
|
|
162
|
-
function nativeSchema(value: unknown): JsonSchema {
|
|
163
|
-
if (!value || typeof value !== "object") return { type: "object", properties: {} }
|
|
164
|
-
if ("jsonSchema" in value && value.jsonSchema && typeof value.jsonSchema === "object")
|
|
165
|
-
return value.jsonSchema as JsonSchema
|
|
166
|
-
return asSchema(value as Parameters<typeof asSchema>[0]).jsonSchema as JsonSchema
|
|
167
|
-
}
|
|
168
|
-
|
|
169
|
-
export function nativeTools(tools: Record<string, Tool>, input: Pick<StreamInput, "messages" | "abort">) {
|
|
170
|
-
return Object.fromEntries(
|
|
171
|
-
Object.entries(tools).map(([name, item]) => [
|
|
172
|
-
name,
|
|
173
|
-
// Tool execution remains opencode-owned. The native runtime only adapts
|
|
174
|
-
// the @opencode-ai/llm tool call back into the AI SDK Tool.execute shape.
|
|
175
|
-
NativeTool.make({
|
|
176
|
-
description: item.description ?? "",
|
|
177
|
-
jsonSchema: nativeSchema(item.inputSchema),
|
|
178
|
-
execute: (args: unknown, ctx) =>
|
|
179
|
-
Effect.tryPromise({
|
|
180
|
-
try: () => {
|
|
181
|
-
if (!item.execute) throw new Error(`Tool has no execute handler: ${name}`)
|
|
182
|
-
return item.execute(args, {
|
|
183
|
-
toolCallId: ctx?.id ?? name,
|
|
184
|
-
messages: input.messages,
|
|
185
|
-
abortSignal: input.abort,
|
|
186
|
-
})
|
|
187
|
-
},
|
|
188
|
-
catch: (error) => new ToolFailure({ message: errorMessage(error), error }),
|
|
189
|
-
}),
|
|
190
|
-
}),
|
|
191
|
-
]),
|
|
192
|
-
)
|
|
193
|
-
}
|
|
194
|
-
|
|
195
|
-
export * as LLMNativeRuntime from "./native-runtime"
|
|
@@ -1,216 +0,0 @@
|
|
|
1
|
-
import { PermissionV1 } from "@opencode-ai/core/v1/permission"
|
|
2
|
-
import type { Auth } from "@/auth"
|
|
3
|
-
import { SessionV1 } from "@opencode-ai/core/v1/session"
|
|
4
|
-
import type { RuntimeFlags } from "@/effect/runtime-flags"
|
|
5
|
-
import { InstanceState } from "@/effect/instance-state"
|
|
6
|
-
import { Permission } from "@/permission"
|
|
7
|
-
import type { Agent } from "@/agent/agent"
|
|
8
|
-
import type { MessageV2 } from "../message-v2"
|
|
9
|
-
import type { Provider } from "@/provider/provider"
|
|
10
|
-
import { ProviderTransform } from "@/provider/transform"
|
|
11
|
-
import { SystemPrompt } from "../system"
|
|
12
|
-
import { InstallationVersion } from "@opencode-ai/core/installation/version"
|
|
13
|
-
import { Effect, Record } from "effect"
|
|
14
|
-
import { jsonSchema, tool as aiTool, type ModelMessage, type Tool } from "ai"
|
|
15
|
-
import type { Plugin } from "@/plugin"
|
|
16
|
-
import { mergeDeep } from "remeda"
|
|
17
|
-
|
|
18
|
-
const USER_AGENT = `opencode/${InstallationVersion}`
|
|
19
|
-
|
|
20
|
-
type PrepareInput = {
|
|
21
|
-
readonly user: SessionV1.User
|
|
22
|
-
readonly sessionID: string
|
|
23
|
-
readonly parentSessionID?: string
|
|
24
|
-
readonly model: Provider.Model
|
|
25
|
-
readonly agent: Agent.Info
|
|
26
|
-
readonly permission?: PermissionV1.Ruleset
|
|
27
|
-
readonly system: string[]
|
|
28
|
-
readonly messages: ModelMessage[]
|
|
29
|
-
readonly small?: boolean
|
|
30
|
-
readonly tools: Record<string, Tool>
|
|
31
|
-
readonly provider: Provider.Info
|
|
32
|
-
readonly auth: Auth.Info | undefined
|
|
33
|
-
readonly plugin: Plugin.Interface
|
|
34
|
-
readonly flags: RuntimeFlags.Info
|
|
35
|
-
readonly isWorkflow: boolean
|
|
36
|
-
}
|
|
37
|
-
|
|
38
|
-
export type Prepared = {
|
|
39
|
-
readonly system: string[]
|
|
40
|
-
readonly messages: ModelMessage[]
|
|
41
|
-
readonly tools: Record<string, Tool>
|
|
42
|
-
readonly params: {
|
|
43
|
-
readonly temperature?: number
|
|
44
|
-
readonly topP?: number
|
|
45
|
-
readonly topK?: number
|
|
46
|
-
readonly maxOutputTokens?: number
|
|
47
|
-
readonly options: Record<string, any>
|
|
48
|
-
}
|
|
49
|
-
readonly messageTransformOptions: Record<string, any>
|
|
50
|
-
readonly headers: Record<string, string>
|
|
51
|
-
}
|
|
52
|
-
|
|
53
|
-
const mergeOptions = (target: Record<string, any>, source: Record<string, any> | undefined): Record<string, any> =>
|
|
54
|
-
mergeDeep(target, source ?? {}) as Record<string, any>
|
|
55
|
-
|
|
56
|
-
export const prepare = Effect.fn("LLMRequestPrep.prepare")(function* (input: PrepareInput) {
|
|
57
|
-
const isOpenaiOauth = input.provider.id === "openai" && input.auth?.type === "oauth"
|
|
58
|
-
const system = [
|
|
59
|
-
[
|
|
60
|
-
...(input.agent.prompt ? [input.agent.prompt] : SystemPrompt.provider(input.model)),
|
|
61
|
-
...input.system,
|
|
62
|
-
...(input.user.system ? [input.user.system] : []),
|
|
63
|
-
]
|
|
64
|
-
.filter((x) => x)
|
|
65
|
-
.join("\n"),
|
|
66
|
-
]
|
|
67
|
-
|
|
68
|
-
const header = system[0]
|
|
69
|
-
yield* input.plugin.trigger(
|
|
70
|
-
"experimental.chat.system.transform",
|
|
71
|
-
{ sessionID: input.sessionID, model: input.model },
|
|
72
|
-
{ system },
|
|
73
|
-
)
|
|
74
|
-
if (system.length > 2 && system[0] === header) {
|
|
75
|
-
const rest = system.slice(1)
|
|
76
|
-
system.length = 0
|
|
77
|
-
system.push(header, rest.join("\n"))
|
|
78
|
-
}
|
|
79
|
-
|
|
80
|
-
const variant =
|
|
81
|
-
!input.small && input.model.variants && input.user.model.variant
|
|
82
|
-
? input.model.variants[input.user.model.variant]
|
|
83
|
-
: {}
|
|
84
|
-
const base = input.small
|
|
85
|
-
? ProviderTransform.smallOptions(input.model)
|
|
86
|
-
: ProviderTransform.options({
|
|
87
|
-
model: input.model,
|
|
88
|
-
sessionID: input.sessionID,
|
|
89
|
-
providerOptions: input.provider.options,
|
|
90
|
-
})
|
|
91
|
-
const options = mergeOptions(mergeOptions(mergeOptions(base, input.model.options), input.agent.options), variant)
|
|
92
|
-
if (
|
|
93
|
-
input.model.api.npm === "@ai-sdk/azure" &&
|
|
94
|
-
(input.provider.options.useCompletionUrls || input.model.options.useCompletionUrls || options.useCompletionUrls)
|
|
95
|
-
) {
|
|
96
|
-
delete options.reasoningSummary
|
|
97
|
-
delete options.include
|
|
98
|
-
}
|
|
99
|
-
if (isOpenaiOauth) options.instructions = system.join("\n")
|
|
100
|
-
|
|
101
|
-
const messages =
|
|
102
|
-
isOpenaiOauth || input.isWorkflow
|
|
103
|
-
? input.messages
|
|
104
|
-
: [
|
|
105
|
-
...system.map(
|
|
106
|
-
(x): ModelMessage => ({
|
|
107
|
-
role: "system",
|
|
108
|
-
content: x,
|
|
109
|
-
}),
|
|
110
|
-
),
|
|
111
|
-
...input.messages,
|
|
112
|
-
]
|
|
113
|
-
|
|
114
|
-
const params = yield* input.plugin.trigger(
|
|
115
|
-
"chat.params",
|
|
116
|
-
{
|
|
117
|
-
sessionID: input.sessionID,
|
|
118
|
-
agent: input.agent.name,
|
|
119
|
-
model: input.model,
|
|
120
|
-
provider: input.provider,
|
|
121
|
-
message: input.user,
|
|
122
|
-
},
|
|
123
|
-
{
|
|
124
|
-
temperature: input.model.capabilities.temperature
|
|
125
|
-
? (input.agent.temperature ?? ProviderTransform.temperature(input.model))
|
|
126
|
-
: undefined,
|
|
127
|
-
topP: input.agent.topP ?? ProviderTransform.topP(input.model),
|
|
128
|
-
topK: ProviderTransform.topK(input.model),
|
|
129
|
-
maxOutputTokens: ProviderTransform.maxOutputTokens(input.model, input.flags.outputTokenMax),
|
|
130
|
-
options,
|
|
131
|
-
},
|
|
132
|
-
)
|
|
133
|
-
|
|
134
|
-
const { headers } = yield* input.plugin.trigger(
|
|
135
|
-
"chat.headers",
|
|
136
|
-
{
|
|
137
|
-
sessionID: input.sessionID,
|
|
138
|
-
agent: input.agent.name,
|
|
139
|
-
model: input.model,
|
|
140
|
-
provider: input.provider,
|
|
141
|
-
message: input.user,
|
|
142
|
-
},
|
|
143
|
-
{
|
|
144
|
-
headers: {},
|
|
145
|
-
},
|
|
146
|
-
)
|
|
147
|
-
|
|
148
|
-
const tools = resolveTools(input)
|
|
149
|
-
if (
|
|
150
|
-
input.model.providerID.includes("github-copilot") &&
|
|
151
|
-
Object.keys(tools).length === 0 &&
|
|
152
|
-
hasToolCalls(input.messages)
|
|
153
|
-
) {
|
|
154
|
-
// Copilot needs a tools field when replaying prior tool calls, even if no tools are currently enabled.
|
|
155
|
-
tools["_noop"] = aiTool({
|
|
156
|
-
description: "Do not call this tool. It exists only for API compatibility and must never be invoked.",
|
|
157
|
-
inputSchema: jsonSchema({
|
|
158
|
-
type: "object",
|
|
159
|
-
properties: {
|
|
160
|
-
reason: { type: "string", description: "Unused" },
|
|
161
|
-
},
|
|
162
|
-
}),
|
|
163
|
-
execute: async () => ({ output: "", title: "", metadata: {} }),
|
|
164
|
-
})
|
|
165
|
-
}
|
|
166
|
-
|
|
167
|
-
const opencodeProjectID = input.model.providerID.startsWith("opencode")
|
|
168
|
-
? (yield* InstanceState.context).project.id
|
|
169
|
-
: undefined
|
|
170
|
-
|
|
171
|
-
return {
|
|
172
|
-
system,
|
|
173
|
-
messages,
|
|
174
|
-
tools: Object.fromEntries(Object.entries(tools).toSorted(([a], [b]) => a.localeCompare(b))),
|
|
175
|
-
params,
|
|
176
|
-
messageTransformOptions: options,
|
|
177
|
-
headers: {
|
|
178
|
-
...(input.model.providerID.startsWith("opencode")
|
|
179
|
-
? {
|
|
180
|
-
...(opencodeProjectID ? { "x-opencode-project": opencodeProjectID } : {}),
|
|
181
|
-
"x-opencode-session": input.sessionID,
|
|
182
|
-
"x-opencode-request": input.user.id,
|
|
183
|
-
"x-opencode-client": input.flags.client,
|
|
184
|
-
"User-Agent": USER_AGENT,
|
|
185
|
-
}
|
|
186
|
-
: {
|
|
187
|
-
"x-session-affinity": input.sessionID,
|
|
188
|
-
"X-Session-Id": input.sessionID,
|
|
189
|
-
...(input.parentSessionID ? { "x-parent-session-id": input.parentSessionID } : {}),
|
|
190
|
-
"User-Agent": USER_AGENT,
|
|
191
|
-
}),
|
|
192
|
-
...input.model.headers,
|
|
193
|
-
...headers,
|
|
194
|
-
},
|
|
195
|
-
}
|
|
196
|
-
})
|
|
197
|
-
|
|
198
|
-
function resolveTools(input: Pick<PrepareInput, "tools" | "agent" | "permission" | "user">) {
|
|
199
|
-
const disabled = Permission.disabled(
|
|
200
|
-
Object.keys(input.tools),
|
|
201
|
-
Permission.merge(input.agent.permission, input.permission ?? []),
|
|
202
|
-
)
|
|
203
|
-
return Record.filter(input.tools, (_, k) => input.user.tools?.[k] !== false && !disabled.has(k))
|
|
204
|
-
}
|
|
205
|
-
|
|
206
|
-
export function hasToolCalls(messages: ModelMessage[]): boolean {
|
|
207
|
-
for (const msg of messages) {
|
|
208
|
-
if (!Array.isArray(msg.content)) continue
|
|
209
|
-
for (const part of msg.content) {
|
|
210
|
-
if (part.type === "tool-call" || part.type === "tool-result") return true
|
|
211
|
-
}
|
|
212
|
-
}
|
|
213
|
-
return false
|
|
214
|
-
}
|
|
215
|
-
|
|
216
|
-
export * as LLMRequestPrep from "./request"
|