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
package/src/share/session.ts
DELETED
|
@@ -1,61 +0,0 @@
|
|
|
1
|
-
import { LayerNode } from "@opencode-ai/core/effect/layer-node"
|
|
2
|
-
import { Session } from "@/session/session"
|
|
3
|
-
import { SessionID } from "@/session/schema"
|
|
4
|
-
import { Effect, Layer, Scope, Context } from "effect"
|
|
5
|
-
import { Config } from "@/config/config"
|
|
6
|
-
import { RuntimeFlags } from "@/effect/runtime-flags"
|
|
7
|
-
import { ShareNext } from "./share-next"
|
|
8
|
-
|
|
9
|
-
export interface Interface {
|
|
10
|
-
readonly create: (input?: Session.CreateInput) => Effect.Effect<Session.Info>
|
|
11
|
-
readonly share: (sessionID: SessionID) => Effect.Effect<{ url: string }, unknown>
|
|
12
|
-
readonly unshare: (sessionID: SessionID) => Effect.Effect<void, unknown>
|
|
13
|
-
}
|
|
14
|
-
|
|
15
|
-
export class Service extends Context.Service<Service, Interface>()("@opencode/SessionShare") {}
|
|
16
|
-
|
|
17
|
-
export const layer = Layer.effect(
|
|
18
|
-
Service,
|
|
19
|
-
Effect.gen(function* () {
|
|
20
|
-
const cfg = yield* Config.Service
|
|
21
|
-
const session = yield* Session.Service
|
|
22
|
-
const shareNext = yield* ShareNext.Service
|
|
23
|
-
const scope = yield* Scope.Scope
|
|
24
|
-
const flags = yield* RuntimeFlags.Service
|
|
25
|
-
|
|
26
|
-
const share = Effect.fn("SessionShare.share")(function* (sessionID: SessionID) {
|
|
27
|
-
const conf = yield* cfg.get()
|
|
28
|
-
if (conf.share === "disabled") throw new Error("Sharing is disabled in configuration")
|
|
29
|
-
const result = yield* shareNext.create(sessionID)
|
|
30
|
-
yield* session.setShare({ sessionID, share: { url: result.url } })
|
|
31
|
-
return result
|
|
32
|
-
})
|
|
33
|
-
|
|
34
|
-
const unshare = Effect.fn("SessionShare.unshare")(function* (sessionID: SessionID) {
|
|
35
|
-
yield* shareNext.remove(sessionID)
|
|
36
|
-
yield* session.setShare({ sessionID, share: undefined })
|
|
37
|
-
})
|
|
38
|
-
|
|
39
|
-
const create = Effect.fn("SessionShare.create")(function* (input?: Session.CreateInput) {
|
|
40
|
-
const result = yield* session.create(input)
|
|
41
|
-
if (result.parentID) return result
|
|
42
|
-
const conf = yield* cfg.get()
|
|
43
|
-
if (!(flags.autoShare || conf.share === "auto")) return result
|
|
44
|
-
yield* share(result.id).pipe(Effect.ignore, Effect.forkIn(scope))
|
|
45
|
-
return result
|
|
46
|
-
})
|
|
47
|
-
|
|
48
|
-
return Service.of({ create, share, unshare })
|
|
49
|
-
}),
|
|
50
|
-
)
|
|
51
|
-
|
|
52
|
-
export const defaultLayer = layer.pipe(
|
|
53
|
-
Layer.provide(ShareNext.defaultLayer),
|
|
54
|
-
Layer.provide(Session.defaultLayer),
|
|
55
|
-
Layer.provide(Config.defaultLayer),
|
|
56
|
-
Layer.provide(RuntimeFlags.defaultLayer),
|
|
57
|
-
)
|
|
58
|
-
|
|
59
|
-
export const node = LayerNode.make(layer, [Config.node, Session.node, ShareNext.node, RuntimeFlags.node])
|
|
60
|
-
|
|
61
|
-
export * as SessionShare from "./session"
|
package/src/share/share-next.ts
DELETED
|
@@ -1,385 +0,0 @@
|
|
|
1
|
-
import { LayerNode } from "@opencode-ai/core/effect/layer-node"
|
|
2
|
-
import { httpClient } from "@opencode-ai/core/effect/layer-node-platform"
|
|
3
|
-
import type * as SDK from "@opencode-ai/sdk/v2"
|
|
4
|
-
import { serviceUse } from "@opencode-ai/core/effect/service-use"
|
|
5
|
-
import { Effect, Exit, Layer, Option, Schema, Scope, Context, Stream } from "effect"
|
|
6
|
-
import { FetchHttpClient, HttpClient, HttpClientRequest, HttpClientResponse } from "effect/unstable/http"
|
|
7
|
-
import { Account } from "@/account/account"
|
|
8
|
-
import { EventV2Bridge } from "@/event-v2-bridge"
|
|
9
|
-
import { InstanceState } from "@/effect/instance-state"
|
|
10
|
-
import { Provider } from "@/provider/provider"
|
|
11
|
-
|
|
12
|
-
import { Session } from "@/session/session"
|
|
13
|
-
import { MessageV2 } from "@/session/message-v2"
|
|
14
|
-
import type { SessionID } from "@/session/schema"
|
|
15
|
-
import { Database } from "@opencode-ai/core/database/database"
|
|
16
|
-
import { eq } from "drizzle-orm"
|
|
17
|
-
import { Config } from "@/config/config"
|
|
18
|
-
import { SessionShareTable } from "@opencode-ai/core/share/sql"
|
|
19
|
-
import { ProviderV2 } from "@opencode-ai/core/provider"
|
|
20
|
-
import { ModelV2 } from "@opencode-ai/core/model"
|
|
21
|
-
import { EventV2 } from "@opencode-ai/core/event"
|
|
22
|
-
|
|
23
|
-
const disabled = process.env["OPENCODE_DISABLE_SHARE"] === "true" || process.env["OPENCODE_DISABLE_SHARE"] === "1"
|
|
24
|
-
|
|
25
|
-
export type Api = {
|
|
26
|
-
create: string
|
|
27
|
-
sync: (shareID: string) => string
|
|
28
|
-
remove: (shareID: string) => string
|
|
29
|
-
data: (shareID: string) => string
|
|
30
|
-
}
|
|
31
|
-
|
|
32
|
-
export type Req = {
|
|
33
|
-
headers: Record<string, string>
|
|
34
|
-
api: Api
|
|
35
|
-
baseUrl: string
|
|
36
|
-
}
|
|
37
|
-
|
|
38
|
-
const ShareSchema = Schema.Struct({
|
|
39
|
-
id: Schema.String,
|
|
40
|
-
url: Schema.String,
|
|
41
|
-
secret: Schema.String,
|
|
42
|
-
})
|
|
43
|
-
export type Share = typeof ShareSchema.Type
|
|
44
|
-
|
|
45
|
-
type State = {
|
|
46
|
-
queue: Map<SessionID, Map<string, Data>>
|
|
47
|
-
scope: Scope.Closeable
|
|
48
|
-
shared: Map<SessionID, Share | null>
|
|
49
|
-
}
|
|
50
|
-
|
|
51
|
-
type Data =
|
|
52
|
-
| {
|
|
53
|
-
type: "session"
|
|
54
|
-
data: SDK.Session
|
|
55
|
-
}
|
|
56
|
-
| {
|
|
57
|
-
type: "message"
|
|
58
|
-
data: SDK.Message
|
|
59
|
-
}
|
|
60
|
-
| {
|
|
61
|
-
type: "part"
|
|
62
|
-
data: SDK.Part
|
|
63
|
-
}
|
|
64
|
-
| {
|
|
65
|
-
type: "session_diff"
|
|
66
|
-
data: SDK.SnapshotFileDiff[]
|
|
67
|
-
}
|
|
68
|
-
| {
|
|
69
|
-
type: "model"
|
|
70
|
-
data: SDK.Model[]
|
|
71
|
-
}
|
|
72
|
-
|
|
73
|
-
export interface Interface {
|
|
74
|
-
readonly init: () => Effect.Effect<void, unknown>
|
|
75
|
-
readonly url: () => Effect.Effect<string, unknown>
|
|
76
|
-
readonly request: () => Effect.Effect<Req, unknown>
|
|
77
|
-
readonly create: (sessionID: SessionID) => Effect.Effect<Share, unknown>
|
|
78
|
-
readonly remove: (sessionID: SessionID) => Effect.Effect<void, unknown>
|
|
79
|
-
}
|
|
80
|
-
|
|
81
|
-
export class Service extends Context.Service<Service, Interface>()("@opencode/ShareNext") {}
|
|
82
|
-
|
|
83
|
-
export const use = serviceUse(Service)
|
|
84
|
-
|
|
85
|
-
function api(resource: string): Api {
|
|
86
|
-
return {
|
|
87
|
-
create: `/api/${resource}`,
|
|
88
|
-
sync: (shareID) => `/api/${resource}/${shareID}/sync`,
|
|
89
|
-
remove: (shareID) => `/api/${resource}/${shareID}`,
|
|
90
|
-
data: (shareID) => `/api/${resource}/${shareID}/data`,
|
|
91
|
-
}
|
|
92
|
-
}
|
|
93
|
-
|
|
94
|
-
const legacyApi = api("share")
|
|
95
|
-
const consoleApi = api("shares")
|
|
96
|
-
|
|
97
|
-
function key(item: Data) {
|
|
98
|
-
switch (item.type) {
|
|
99
|
-
case "session":
|
|
100
|
-
return "session"
|
|
101
|
-
case "message":
|
|
102
|
-
return `message/${item.data.id}`
|
|
103
|
-
case "part":
|
|
104
|
-
return `part/${item.data.messageID}/${item.data.id}`
|
|
105
|
-
case "session_diff":
|
|
106
|
-
return "session_diff"
|
|
107
|
-
case "model":
|
|
108
|
-
return "model"
|
|
109
|
-
}
|
|
110
|
-
}
|
|
111
|
-
|
|
112
|
-
export const layer = Layer.effect(
|
|
113
|
-
Service,
|
|
114
|
-
Effect.gen(function* () {
|
|
115
|
-
const account = yield* Account.Service
|
|
116
|
-
const events = yield* EventV2Bridge.Service
|
|
117
|
-
const cfg = yield* Config.Service
|
|
118
|
-
const { db } = yield* Database.Service
|
|
119
|
-
const http = yield* HttpClient.HttpClient
|
|
120
|
-
const httpOk = HttpClient.filterStatusOk(http)
|
|
121
|
-
const provider = yield* Provider.Service
|
|
122
|
-
const session = yield* Session.Service
|
|
123
|
-
|
|
124
|
-
function sync(sessionID: SessionID, data: Data[]) {
|
|
125
|
-
return Effect.gen(function* () {
|
|
126
|
-
if (disabled) return
|
|
127
|
-
const share = yield* getCached(sessionID)
|
|
128
|
-
if (!share) return
|
|
129
|
-
|
|
130
|
-
const s = yield* InstanceState.get(state)
|
|
131
|
-
const existing = s.queue.get(sessionID)
|
|
132
|
-
if (existing) {
|
|
133
|
-
for (const item of data) {
|
|
134
|
-
existing.set(key(item), item)
|
|
135
|
-
}
|
|
136
|
-
return
|
|
137
|
-
}
|
|
138
|
-
|
|
139
|
-
const next = new Map(data.map((item) => [key(item), item]))
|
|
140
|
-
s.queue.set(sessionID, next)
|
|
141
|
-
yield* flush(sessionID).pipe(
|
|
142
|
-
Effect.delay(1000),
|
|
143
|
-
Effect.catchCause((cause) => Effect.logError("share flush failed", { sessionID: sessionID, cause: cause })),
|
|
144
|
-
Effect.forkIn(s.scope),
|
|
145
|
-
)
|
|
146
|
-
})
|
|
147
|
-
}
|
|
148
|
-
|
|
149
|
-
const state: InstanceState.InstanceState<State> = yield* InstanceState.make<State>(
|
|
150
|
-
Effect.fn("ShareNext.state")(function* (_ctx) {
|
|
151
|
-
const cache: State = { queue: new Map(), scope: yield* Scope.make(), shared: new Map() }
|
|
152
|
-
|
|
153
|
-
yield* Effect.addFinalizer(() =>
|
|
154
|
-
Scope.close(cache.scope, Exit.void).pipe(
|
|
155
|
-
Effect.andThen(
|
|
156
|
-
Effect.sync(() => {
|
|
157
|
-
cache.queue.clear()
|
|
158
|
-
cache.shared.clear()
|
|
159
|
-
}),
|
|
160
|
-
),
|
|
161
|
-
),
|
|
162
|
-
)
|
|
163
|
-
|
|
164
|
-
if (disabled) return cache
|
|
165
|
-
|
|
166
|
-
const watch = <D extends EventV2.Definition>(
|
|
167
|
-
def: D,
|
|
168
|
-
fn: (data: EventV2.Data<D>) => Effect.Effect<void, unknown>,
|
|
169
|
-
) =>
|
|
170
|
-
events.listen((event) => {
|
|
171
|
-
if (event.type !== def.type || event.location?.directory !== _ctx.directory) return Effect.void
|
|
172
|
-
return fn(event.data as EventV2.Data<D>).pipe(
|
|
173
|
-
Effect.catchCause((cause) =>
|
|
174
|
-
Effect.logError("share subscriber failed", { type: def.type, cause: cause }),
|
|
175
|
-
),
|
|
176
|
-
)
|
|
177
|
-
})
|
|
178
|
-
|
|
179
|
-
yield* watch(Session.Event.Updated, (data) =>
|
|
180
|
-
Effect.gen(function* () {
|
|
181
|
-
const info = data.info
|
|
182
|
-
yield* sync(info.id, [{ type: "session", data: structuredClone(info) as SDK.Session }])
|
|
183
|
-
}),
|
|
184
|
-
)
|
|
185
|
-
yield* watch(MessageV2.Event.Updated, (data) =>
|
|
186
|
-
Effect.gen(function* () {
|
|
187
|
-
const info = data.info
|
|
188
|
-
yield* sync(info.sessionID, [{ type: "message", data: structuredClone(info) as SDK.Message }])
|
|
189
|
-
if (info.role !== "user") return
|
|
190
|
-
const model = yield* provider.getModel(info.model.providerID, info.model.modelID)
|
|
191
|
-
yield* sync(info.sessionID, [{ type: "model", data: [model] }])
|
|
192
|
-
}),
|
|
193
|
-
)
|
|
194
|
-
yield* watch(MessageV2.Event.PartUpdated, (data) =>
|
|
195
|
-
sync(data.part.sessionID, [{ type: "part", data: structuredClone(data.part) as SDK.Part }]),
|
|
196
|
-
)
|
|
197
|
-
yield* watch(Session.Event.Diff, (data) =>
|
|
198
|
-
sync(data.sessionID, [{ type: "session_diff", data: structuredClone(data.diff) as SDK.SnapshotFileDiff[] }]),
|
|
199
|
-
)
|
|
200
|
-
yield* watch(Session.Event.Deleted, (data) => remove(data.sessionID))
|
|
201
|
-
|
|
202
|
-
return cache
|
|
203
|
-
}),
|
|
204
|
-
)
|
|
205
|
-
|
|
206
|
-
const request = Effect.fn("ShareNext.request")(function* () {
|
|
207
|
-
const headers: Record<string, string> = {}
|
|
208
|
-
const active = yield* account.active()
|
|
209
|
-
if (Option.isNone(active) || !active.value.active_org_id) {
|
|
210
|
-
const baseUrl = (yield* cfg.get()).enterprise?.url ?? "https://opncd.ai"
|
|
211
|
-
return { headers, api: legacyApi, baseUrl } satisfies Req
|
|
212
|
-
}
|
|
213
|
-
|
|
214
|
-
const token = yield* account.token(active.value.id)
|
|
215
|
-
if (Option.isNone(token)) {
|
|
216
|
-
throw new Error("No active account token available for sharing")
|
|
217
|
-
}
|
|
218
|
-
|
|
219
|
-
headers.authorization = `Bearer ${token.value}`
|
|
220
|
-
headers["x-org-id"] = active.value.active_org_id
|
|
221
|
-
return { headers, api: consoleApi, baseUrl: active.value.url } satisfies Req
|
|
222
|
-
})
|
|
223
|
-
|
|
224
|
-
const get = Effect.fnUntraced(function* (sessionID: SessionID) {
|
|
225
|
-
const row = yield* db
|
|
226
|
-
.select()
|
|
227
|
-
.from(SessionShareTable)
|
|
228
|
-
.where(eq(SessionShareTable.session_id, sessionID))
|
|
229
|
-
.get()
|
|
230
|
-
.pipe(Effect.orDie)
|
|
231
|
-
if (!row) return
|
|
232
|
-
return { id: row.id, secret: row.secret, url: row.url } satisfies Share
|
|
233
|
-
})
|
|
234
|
-
|
|
235
|
-
const getCached = Effect.fnUntraced(function* (sessionID: SessionID) {
|
|
236
|
-
const s = yield* InstanceState.get(state)
|
|
237
|
-
if (s.shared.has(sessionID)) {
|
|
238
|
-
const cached = s.shared.get(sessionID)
|
|
239
|
-
return cached === null ? undefined : cached
|
|
240
|
-
}
|
|
241
|
-
|
|
242
|
-
const share = yield* get(sessionID)
|
|
243
|
-
s.shared.set(sessionID, share ?? null)
|
|
244
|
-
return share
|
|
245
|
-
})
|
|
246
|
-
|
|
247
|
-
const flush = Effect.fn("ShareNext.flush")(function* (sessionID: SessionID) {
|
|
248
|
-
if (disabled) return
|
|
249
|
-
const s = yield* InstanceState.get(state)
|
|
250
|
-
const queued = s.queue.get(sessionID)
|
|
251
|
-
if (!queued) return
|
|
252
|
-
|
|
253
|
-
s.queue.delete(sessionID)
|
|
254
|
-
|
|
255
|
-
const share = yield* getCached(sessionID)
|
|
256
|
-
if (!share) return
|
|
257
|
-
|
|
258
|
-
const req = yield* request()
|
|
259
|
-
const res = yield* HttpClientRequest.post(`${req.baseUrl}${req.api.sync(share.id)}`).pipe(
|
|
260
|
-
HttpClientRequest.setHeaders(req.headers),
|
|
261
|
-
HttpClientRequest.bodyJson({ secret: share.secret, data: Array.from(queued.values()) }),
|
|
262
|
-
Effect.flatMap((r) => http.execute(r)),
|
|
263
|
-
)
|
|
264
|
-
|
|
265
|
-
if (res.status >= 400) {
|
|
266
|
-
yield* Effect.logWarning("failed to sync share", {
|
|
267
|
-
sessionID: sessionID,
|
|
268
|
-
shareID: share.id,
|
|
269
|
-
status: res.status,
|
|
270
|
-
})
|
|
271
|
-
}
|
|
272
|
-
})
|
|
273
|
-
|
|
274
|
-
const full = Effect.fn("ShareNext.full")(function* (sessionID: SessionID) {
|
|
275
|
-
yield* Effect.logInfo("full sync", { sessionID: sessionID })
|
|
276
|
-
const info = yield* session.get(sessionID)
|
|
277
|
-
const diffs = yield* session.diff(sessionID)
|
|
278
|
-
const messages = yield* session.messages({ sessionID })
|
|
279
|
-
const models = yield* Effect.forEach(
|
|
280
|
-
Array.from(
|
|
281
|
-
new Map(
|
|
282
|
-
messages
|
|
283
|
-
.filter((msg) => msg.info.role === "user")
|
|
284
|
-
.map((msg) => (msg.info as SDK.UserMessage).model)
|
|
285
|
-
.map((item) => [`${item.providerID}/${item.modelID}`, item] as const),
|
|
286
|
-
).values(),
|
|
287
|
-
),
|
|
288
|
-
(item) => provider.getModel(ProviderV2.ID.make(item.providerID), ModelV2.ID.make(item.modelID)),
|
|
289
|
-
{ concurrency: 8 },
|
|
290
|
-
)
|
|
291
|
-
|
|
292
|
-
yield* sync(sessionID, [
|
|
293
|
-
{ type: "session", data: info },
|
|
294
|
-
...messages.map((item) => ({ type: "message" as const, data: item.info })),
|
|
295
|
-
...messages.flatMap((item) => item.parts.map((part) => ({ type: "part" as const, data: part }))),
|
|
296
|
-
{ type: "session_diff", data: diffs },
|
|
297
|
-
{ type: "model", data: models },
|
|
298
|
-
])
|
|
299
|
-
})
|
|
300
|
-
|
|
301
|
-
const init = Effect.fn("ShareNext.init")(function* () {
|
|
302
|
-
if (disabled) return
|
|
303
|
-
yield* InstanceState.get(state)
|
|
304
|
-
})
|
|
305
|
-
|
|
306
|
-
const url = Effect.fn("ShareNext.url")(function* () {
|
|
307
|
-
return (yield* request()).baseUrl
|
|
308
|
-
})
|
|
309
|
-
|
|
310
|
-
const create = Effect.fn("ShareNext.create")(function* (sessionID: SessionID) {
|
|
311
|
-
if (disabled) return { id: "", url: "", secret: "" }
|
|
312
|
-
yield* Effect.logInfo("creating share", { sessionID: sessionID })
|
|
313
|
-
const req = yield* request()
|
|
314
|
-
const result = yield* HttpClientRequest.post(`${req.baseUrl}${req.api.create}`).pipe(
|
|
315
|
-
HttpClientRequest.setHeaders(req.headers),
|
|
316
|
-
HttpClientRequest.bodyJson({ sessionID }),
|
|
317
|
-
Effect.flatMap((r) => httpOk.execute(r)),
|
|
318
|
-
Effect.flatMap(HttpClientResponse.schemaBodyJson(ShareSchema)),
|
|
319
|
-
)
|
|
320
|
-
yield* db
|
|
321
|
-
.insert(SessionShareTable)
|
|
322
|
-
.values({ session_id: sessionID, id: result.id, secret: result.secret, url: result.url })
|
|
323
|
-
.onConflictDoUpdate({
|
|
324
|
-
target: SessionShareTable.session_id,
|
|
325
|
-
set: { id: result.id, secret: result.secret, url: result.url },
|
|
326
|
-
})
|
|
327
|
-
.run()
|
|
328
|
-
.pipe(Effect.orDie)
|
|
329
|
-
const s = yield* InstanceState.get(state)
|
|
330
|
-
s.shared.set(sessionID, result)
|
|
331
|
-
yield* full(sessionID).pipe(
|
|
332
|
-
Effect.catchCause((cause) => Effect.logError("share full sync failed", { sessionID: sessionID, cause: cause })),
|
|
333
|
-
Effect.forkIn(s.scope),
|
|
334
|
-
)
|
|
335
|
-
return result
|
|
336
|
-
})
|
|
337
|
-
|
|
338
|
-
const remove = Effect.fn("ShareNext.remove")(function* (sessionID: SessionID) {
|
|
339
|
-
if (disabled) return
|
|
340
|
-
yield* Effect.logInfo("removing share", { sessionID: sessionID })
|
|
341
|
-
const s = yield* InstanceState.get(state)
|
|
342
|
-
const share = yield* getCached(sessionID)
|
|
343
|
-
if (!share) {
|
|
344
|
-
s.shared.delete(sessionID)
|
|
345
|
-
s.queue.delete(sessionID)
|
|
346
|
-
return
|
|
347
|
-
}
|
|
348
|
-
|
|
349
|
-
const req = yield* request()
|
|
350
|
-
yield* HttpClientRequest.delete(`${req.baseUrl}${req.api.remove(share.id)}`).pipe(
|
|
351
|
-
HttpClientRequest.setHeaders(req.headers),
|
|
352
|
-
HttpClientRequest.bodyJson({ secret: share.secret }),
|
|
353
|
-
Effect.flatMap((r) => httpOk.execute(r)),
|
|
354
|
-
)
|
|
355
|
-
|
|
356
|
-
yield* db.delete(SessionShareTable).where(eq(SessionShareTable.session_id, sessionID)).run().pipe(Effect.orDie)
|
|
357
|
-
s.shared.delete(sessionID)
|
|
358
|
-
s.queue.delete(sessionID)
|
|
359
|
-
})
|
|
360
|
-
|
|
361
|
-
return Service.of({ init, url, request, create, remove })
|
|
362
|
-
}),
|
|
363
|
-
)
|
|
364
|
-
|
|
365
|
-
export const defaultLayer = layer.pipe(
|
|
366
|
-
Layer.provide(EventV2Bridge.defaultLayer),
|
|
367
|
-
Layer.provide(Account.defaultLayer),
|
|
368
|
-
Layer.provide(Config.defaultLayer),
|
|
369
|
-
Layer.provide(Database.defaultLayer),
|
|
370
|
-
Layer.provide(FetchHttpClient.layer),
|
|
371
|
-
Layer.provide(Provider.defaultLayer),
|
|
372
|
-
Layer.provide(Session.defaultLayer),
|
|
373
|
-
)
|
|
374
|
-
|
|
375
|
-
export const node = LayerNode.make(layer, [
|
|
376
|
-
Account.node,
|
|
377
|
-
EventV2Bridge.node,
|
|
378
|
-
Config.node,
|
|
379
|
-
Database.node,
|
|
380
|
-
httpClient,
|
|
381
|
-
Provider.node,
|
|
382
|
-
Session.node,
|
|
383
|
-
])
|
|
384
|
-
|
|
385
|
-
export * as ShareNext from "./share-next"
|
package/src/skill/discovery.ts
DELETED
|
@@ -1,109 +0,0 @@
|
|
|
1
|
-
import { LayerNode } from "@opencode-ai/core/effect/layer-node"
|
|
2
|
-
import { httpClient, path } from "@opencode-ai/core/effect/layer-node-platform"
|
|
3
|
-
import { NodePath } from "@effect/platform-node"
|
|
4
|
-
import { Effect, Layer, Path, Schema, Context } from "effect"
|
|
5
|
-
import { FetchHttpClient, HttpClient, HttpClientRequest, HttpClientResponse } from "effect/unstable/http"
|
|
6
|
-
import { withTransientReadRetry } from "@/util/effect-http-client"
|
|
7
|
-
import { FSUtil } from "@opencode-ai/core/fs-util"
|
|
8
|
-
import { Global } from "@opencode-ai/core/global"
|
|
9
|
-
|
|
10
|
-
const skillConcurrency = 4
|
|
11
|
-
const fileConcurrency = 8
|
|
12
|
-
|
|
13
|
-
class IndexSkill extends Schema.Class<IndexSkill>("IndexSkill")({
|
|
14
|
-
name: Schema.String,
|
|
15
|
-
files: Schema.Array(Schema.String),
|
|
16
|
-
}) {}
|
|
17
|
-
|
|
18
|
-
class Index extends Schema.Class<Index>("Index")({
|
|
19
|
-
skills: Schema.Array(IndexSkill),
|
|
20
|
-
}) {}
|
|
21
|
-
|
|
22
|
-
export interface Interface {
|
|
23
|
-
readonly pull: (url: string) => Effect.Effect<string[]>
|
|
24
|
-
}
|
|
25
|
-
|
|
26
|
-
export class Service extends Context.Service<Service, Interface>()("@opencode/SkillDiscovery") {}
|
|
27
|
-
|
|
28
|
-
export const layer: Layer.Layer<Service, never, FSUtil.Service | Path.Path | HttpClient.HttpClient> = Layer.effect(
|
|
29
|
-
Service,
|
|
30
|
-
Effect.gen(function* () {
|
|
31
|
-
const fs = yield* FSUtil.Service
|
|
32
|
-
const path = yield* Path.Path
|
|
33
|
-
const http = HttpClient.filterStatusOk(withTransientReadRetry(yield* HttpClient.HttpClient))
|
|
34
|
-
const cache = path.join(Global.Path.cache, "skills")
|
|
35
|
-
|
|
36
|
-
const download = Effect.fn("Discovery.download")(function* (url: string, dest: string) {
|
|
37
|
-
if (yield* fs.exists(dest).pipe(Effect.orDie)) return true
|
|
38
|
-
|
|
39
|
-
return yield* HttpClientRequest.get(url).pipe(
|
|
40
|
-
http.execute,
|
|
41
|
-
Effect.flatMap((res) => res.arrayBuffer),
|
|
42
|
-
Effect.flatMap((body) => fs.writeWithDirs(dest, new Uint8Array(body))),
|
|
43
|
-
Effect.as(true),
|
|
44
|
-
Effect.catch((err) => Effect.logError("failed to download", { url: url, error: err }).pipe(Effect.as(false))),
|
|
45
|
-
)
|
|
46
|
-
})
|
|
47
|
-
|
|
48
|
-
const pull = Effect.fn("Discovery.pull")(function* (url: string) {
|
|
49
|
-
const base = url.endsWith("/") ? url : `${url}/`
|
|
50
|
-
const index = new URL("index.json", base).href
|
|
51
|
-
const host = base.slice(0, -1)
|
|
52
|
-
|
|
53
|
-
yield* Effect.logInfo("fetching index", { url: index })
|
|
54
|
-
|
|
55
|
-
const data = yield* HttpClientRequest.get(index).pipe(
|
|
56
|
-
HttpClientRequest.acceptJson,
|
|
57
|
-
http.execute,
|
|
58
|
-
Effect.flatMap(HttpClientResponse.schemaBodyJson(Index)),
|
|
59
|
-
Effect.catch((err) =>
|
|
60
|
-
Effect.logError("failed to fetch index", { url: index, error: err }).pipe(Effect.as(null)),
|
|
61
|
-
),
|
|
62
|
-
)
|
|
63
|
-
|
|
64
|
-
if (!data) return []
|
|
65
|
-
|
|
66
|
-
const missing = data.skills.filter((skill) => !skill.files.includes("SKILL.md"))
|
|
67
|
-
yield* Effect.forEach(
|
|
68
|
-
missing,
|
|
69
|
-
(skill) => Effect.logWarning("skill entry missing SKILL.md", { url: index, skill: skill.name }),
|
|
70
|
-
{ discard: true },
|
|
71
|
-
)
|
|
72
|
-
const list = data.skills.filter((skill) => skill.files.includes("SKILL.md"))
|
|
73
|
-
|
|
74
|
-
const dirs = yield* Effect.forEach(
|
|
75
|
-
list,
|
|
76
|
-
(skill) =>
|
|
77
|
-
Effect.gen(function* () {
|
|
78
|
-
const root = path.join(cache, skill.name)
|
|
79
|
-
|
|
80
|
-
yield* Effect.forEach(
|
|
81
|
-
skill.files,
|
|
82
|
-
(file) => download(new URL(file, `${host}/${skill.name}/`).href, path.join(root, file)),
|
|
83
|
-
{
|
|
84
|
-
concurrency: fileConcurrency,
|
|
85
|
-
},
|
|
86
|
-
)
|
|
87
|
-
|
|
88
|
-
const md = path.join(root, "SKILL.md")
|
|
89
|
-
return (yield* fs.exists(md).pipe(Effect.orDie)) ? root : null
|
|
90
|
-
}),
|
|
91
|
-
{ concurrency: skillConcurrency },
|
|
92
|
-
)
|
|
93
|
-
|
|
94
|
-
return dirs.filter((dir): dir is string => dir !== null)
|
|
95
|
-
})
|
|
96
|
-
|
|
97
|
-
return Service.of({ pull })
|
|
98
|
-
}),
|
|
99
|
-
)
|
|
100
|
-
|
|
101
|
-
export const defaultLayer: Layer.Layer<Service> = layer.pipe(
|
|
102
|
-
Layer.provide(FetchHttpClient.layer),
|
|
103
|
-
Layer.provide(FSUtil.defaultLayer),
|
|
104
|
-
Layer.provide(NodePath.layer),
|
|
105
|
-
)
|
|
106
|
-
|
|
107
|
-
export const node = LayerNode.make(layer, [FSUtil.node, path, httpClient])
|
|
108
|
-
|
|
109
|
-
export * as Discovery from "./discovery"
|