sdk/ARCHITECTURE.md
This document is the architecture source of truth for the Cline SDK repository. It describes how the system is organized, how components interact, and the design principles that guide development decisions.
Who should read this?
@cline/coreWhat this covers:
What this is NOT:
The workspace is organized as a layered runtime stack.
flowchart LR
shared["@cline/shared"]
llms["@cline/llms"]
agents["@cline/agents"]
core["@cline/core"]
apps["Host Apps"]
llms --> shared
agents --> llms
agents --> shared
core --> agents
core --> llms
core --> shared
apps --> core
@cline/sharedOwns reusable low-level contracts and infrastructure:
Design rule:
shared should not depend on higher-level runtime packages.@cline/llmsOwns model/provider runtime concerns:
Design rule:
core or apps.@cline/agentsOwns the stateless runtime loop:
Design rule:
agents should not own persistent storage or host lifecycle concerns.@cline/coreOwns stateful orchestration:
src/hub/@cline/core/hub/daemon-entry subpathNodeHubClient, HubSessionClient, HubUIClient, connectToHub) exported from @cline/core/hubDesign rules:
core is the app-facing orchestration layer over agents.packages/core/src/hub/, grouped by service:
client/ contains host-facing hub clients and browser connection helpersdaemon/ contains detached daemon startup, entrypoint, and local runtime handler wiringdiscovery/ contains endpoint defaults, discovery records, and workspace owner resolutionserver/ contains WebSocket server startup, native/browser socket adapters, server transport, server helpers, and handlers/ for hub command dispatchsettings.* hub command family and react to settings.changed.RuntimeHost through @cline/core.@cline/core selects LocalRuntimeHost through packages/core/src/runtime/host.ts.RuntimeSessionConfig plus localRuntime overrides before calling RuntimeHost.start(...).@cline/core prepares a local bootstrap artifact from localRuntime, then builds the runtime from it.@cline/core creates an Agent from @cline/agents.@cline/agents runs the loop using @cline/llms handlers.@cline/core persists state, artifacts, and metadata.Completion telemetry is anchored to the assistant's explicit completion
declaration, not session shutdown. After each agent turn, the local
runtime inspects AgentResult.toolCalls and emits task.completed the
moment a successful submit_and_exit (the SDK analog of original
Cline's attempt_completion) is observed. shutdownSession(...)
retains a fallback emission for completed sessions that finished
without an explicit completion-tool observation, so non-interactive
runs not using the yolo preset still produce a task.completed signal.
Each session emits at most one task.completed. See DOC.md for the
event payload and source field.
RuntimeHost through @cline/core.@cline/core selects HubRuntimeHost or RemoteRuntimeHost through packages/core/src/runtime/host.ts.@cline/core can spawn a detached hub daemon and reconnect through discovery.@cline/agents and @cline/llms.@cline/core hub services broker sessions, events, approvals, schedules, and client-owned runtime capabilities such as session-local tool executors.@cline/core/hub (NodeHubClient, HubSessionClient, HubUIClient, connectToHub) translate command/reply and event streams into host-facing APIs.session.get records include both canonical root-session usage and explicit aggregate usage from the hub-owned RuntimeHost, so attached clients can intentionally render either root-only or root-plus-teammate costs without replaying event streams.Detached daemon startup retries transient ETXTBSY spawn failures before
polling discovery. This covers package-manager updates that replace the CLI
binary immediately before a command restarts the shared hub.
Local hub discovery also carries the authentication contract for the shared
daemon. On startup, the hub server generates a cryptographically random
per-process auth token, stores it in the owner discovery record, and writes that
record with owner-only file permissions. Local clients resolve the token from
the discovery file at connection time rather than embedding it in endpoint URLs.
The server validates the token with a constant-time comparison before accepting
/hub WebSocket upgrades or /shutdown requests; WebSocket clients send it via
the Sec-WebSocket-Protocol header and shutdown requests use an
Authorization: Bearer header. Unauthenticated local processes can still probe
public health/build metadata, but they cannot attach to sessions, issue
commands, or stop the daemon.
Local hub rediscovery is limited to managed shared-daemon endpoints obtained
through discovery or ensure*HubServer(...) startup paths. Explicit endpoints,
including loopback URLs such as ws://127.0.0.1:<port>/hub, are sticky exact
targets: reconnects may retry the same socket URL, but command recovery and
startup-deadlock recovery must not replace them with the workspace-discovered
hub. This keeps custom local hubs and remote hubs from silently drifting to a
different process.
apps/cli owns OpenTUI startup and must render the first frame without waiting for detached hub startup.backendMode: "auto" so an already-compatible hub can be reused immediately, while a missing hub is only prewarmed in the background and the TUI falls back to a local runtime for responsiveness.cline hub, schedules, connectors, and --zen may still call the explicit ensure path because those commands require a live hub before proceeding.renderOpenTui() so loading previous messages cannot block initial TUI paint.RemoteConfigBundle.@cline/shared/remote-config caches the bundle when configured..cline/<plugin>/.@cline/core exposes the app-facing integration wrapper that applies extensions, telemetry, and session metadata to StartSessionInput.@cline/core consumes the prepared local overrides during local bootstrap.This keeps reusable remote-config behavior in shared while the session-specific bridge remains in core.
The codebase relies on a few repeated seams instead of one-off integration paths.
Core uses file-based discovery and watchers for:
Design implication:
packages/core, config-facing discovery, parsing, watching, and slash-command projection live under src/extensions/configDefaultRuntimeBuilder composes a runtime from generic inputs:
Design implication:
packages/core/src/services/local-runtime-bootstrap.ts and feeds the builder rather than bypassing itCore exposes one shared execution boundary: RuntimeHost.
Concrete implementations:
LocalRuntimeHost for in-process executionHubRuntimeHost for shared local hub executionRemoteRuntimeHost for explicit remote hub endpointsDesign implication:
packages/core/src/runtime/host.tsClineCore delegates uniformly to RuntimeHost and does not branch on local vs hub behaviorRuntimeHost inputs stay transport-safe, while ClineCore.start(...) is the app-facing facade that normalizes broad local config before delegationRuntimeSessionConfig is transport-neutral across local, shared hub, and remote hub modes; host-local bootstrap concerns stay under localRuntimedefaultToolExecutors, are attached at session start and proxied through hub capability requests instead of changing host selectionClineCore.pendingPrompts service. Usage summary lookup and active-session
model switching are also service-style capabilities exposed through
ClineCore when the concrete transport implements them. These service APIs
are intentionally outside the minimal RuntimeHost primitive vocabulary.getAccumulatedUsage(sessionId) method returns a summary
with two explicit buckets: usage for the root/lead agent and
aggregateUsage for root plus teammates/subagents. Local execution tracks
root usage and teammate usage as separate buckets, then derives aggregate
totals from those buckets while telemetry remains scoped to the primary
lead/root agent.Core owns settings snapshots and mutations through packages/core/src/settings.
The hub exposes the same path through settings.list and settings.toggle.
Design implication:
settings.changed with the changed settings typesClineCore.create(...) exposes a generic prepare(input) hook.
Design implication:
Cross-package logging uses a small injected interface exported from @cline/shared:
BasicLogger — required debug and log; optional error. Hosts map these to their backend (Pino, VS Code OutputChannel, etc.). Many runtime options take logger?: BasicLogger; when omitted, components skip logging or use noopBasicLogger where a full object is required.BasicLogMetadata — optional structured fields (sessionId, runId, providerId, toolName, durationMs, …) plus severity on log when a single method must represent both informational and warning-style messages (for example the CLI Pino bridge maps severity: "warn" to Pino warn).Naming clarity:
CliLoggerAdapter (CLI) — a host bundle: holds the raw pino logger (for file paths, rotation, and CLI-only concerns) and exposes .core: BasicLogger for anything that consumes the SDK contract. It is not an ITelemetryAdapter.TelemetryLoggerSink (@cline/core) — an ITelemetryAdapter that mirrors telemetry events and metrics into a BasicLogger. It is a telemetry sink, not a host logging implementation.The agent and other call sites route former info / warn semantics through log (warnings include severity: "warn" in metadata). Errors prefer error when implemented; otherwise log with severity: "error" is used as a fallback.
Design implication:
logger?: BasicLogger parameter insteadStateful persistence should be isolated behind adapter/service layers.
Design implication:
Extensibility is split deliberately:
Design implication:
Context compaction is owned by core.
@cline/agents owns the generic turn-preparation seam:
@cline/core owns compaction policy:
Design implications:
coreagents stays focused on the stateless loop and provider/tool orchestrationpackages/core/src/extensions is split by concern:
extensions/config: config loaders, parsers, watchers, and watcher projections such as runtime slash-command expansionextensions/plugin: runtime plugin discovery, loading, and sandboxingextensions/context: core-owned context/message pipeline concerns such as compactionDesign implications:
agents StatelessDo not move these concerns into @cline/agents:
core GenericDo not make @cline/core organization- or provider-specific.
If a capability is truly generic and app-facing, add a generic core seam. Reusable remote-config parsing, materialization, and upload primitives belong in @cline/shared/remote-config.
Optional higher-level integrations may depend on lower layers. Lower layers should not depend on optional feature packages.
For remote config, that means shared owns the reusable bundle/materialization/blob primitives and core owns only the session-oriented wrapper exported to apps.
ClineCore / CronService)@cline/core ships a file-based automation subsystem under
packages/core/src/cron/. It lets operators author recurring and one-off
tasks as Markdown files under global ~/.cline/cron/ by default, and
event-driven tasks as events/*.event.md specs. All trigger kinds run
through the same durable queue and runtime handlers. ClineCore exposes the
SDK-facing cline.automation.* entry points; CronService is the internal
orchestrator used by core and hub layers.
cron/specs/cron-spec-parser.ts): parses YAML frontmatter + body
into a CronSpec discriminated union (one_off | schedule | event).
Types live in @cline/shared under src/cron/cron-spec-types.ts
so other packages can consume them without the YAML parser. Schedule
expressions and timezones are validated before a spec can become
runnable.cron/store/sqlite-cron-store.ts): owns cron.db at
resolveCronDbPath() (default .cline/data/db/cron.db). Schema is
bootstrapped from cron/store/cron-schema.ts — sessions and cron live in separate
DBs so their lifecycles stay decoupled.cron/specs/cron-reconciler.ts): scans the configured cron specs
directory (global ~/.cline/cron/ by default, or workspace-scoped when
configured), parses each file independently, and upserts spec state.
Invalid specs are recorded
with parse_status='invalid' so state is durable rather than silently
dropped. Files that disappear between scans get removed=1 and their
queued runs are cancelled.cron/specs/cron-watcher.ts): node:fs watch({ recursive: true })
with a ~250ms per-path debounce. Watcher events always trigger a
re-reconcile — the reconciler is always the source of truth, not the
watcher stream.cron/runner/cron-materializer.ts): turns file-triggered specs into
queued cron_runs. One-off: at most one run record per (spec_id, revision), including failed runs so specs do not retry accidentally.
Schedule: "one overdue catch-up on startup then advance" using
timezone-aware getNextCronTime.cron/events/cron-event-ingress.ts): accepts already-normalized
AutomationEventEnvelope values, persists them into cron_event_log,
matches enabled event specs by event_type plus declarative filters,
applies dedupe/debounce/cooldown policy, and enqueues cron_runs with
trigger_kind='event'. It never executes agents directly. Plugins can
declare automationEvents and submit normalized events through
ctx.automation.ingestEvent(...); sandboxed plugins forward those events
through the core plugin event bridge.cron/runner/cron-runner.ts): polls cron.db, atomically claims
queued runs, executes them via the existing HubScheduleRuntimeHandlers
(startSession → sendSession → stopSession / abortSession),
renews the run claim while execution is active, writes a markdown report
per run, and transactionally updates status. File specs can constrain
tool availability, config extension loading (rules, skills,
plugins), session source, and a notes directory that is injected into
the system prompt. Event runs include the normalized trigger event context
in the prompt.cron/reports/cron-report-writer.ts): writes
.cline/cron/reports/<run-id>.md with run frontmatter plus
## Summary, ## Usage, ## Tool Calls, and, for event runs,
## Trigger Event sections.cron/service/cron-service.ts): orchestrates all of the above.
ClineCore.create({ automation }) owns the SDK-facing lifecycle and exposes
cline.automation.* methods. Hub-side callers can submit normalized events
through the cron.event.ingest command.The detached hub daemon passes its workspace root as cronOptions, so
normal CLI/hub startup watches ${workspaceRoot}/.cline/cron/ without a
custom host needing to opt in.
Programmatic hub schedules are stored as cron_specs with source
hub-schedule and execute through the same cron_runs
claim/requeue/report flow as file-backed one-off, recurring, and
event-driven specs. The hub schedule command surface remains a thin adapter;
there is no separate schedules table, schedule store, or schedule runner.
I want to understand the agent loop and tool execution:
packages/agents/src/agent.ts — the stateless runtime looppackages/agents/src/agent-step.ts — individual iteration stepspackages/core/src/extensions/plugin/ — plugin discovery and sandboxingI want to understand session persistence and state:
packages/core/src/runtime/host/local-runtime-host.ts — local session lifecyclepackages/core/src/runtime/orchestration/ — session orchestrationpackages/core/src/settings/ — settings mutation and stateI want to understand the hub system:
packages/core/src/hub/server/ — WebSocket server and hub command handlerspackages/core/src/hub/client/ — host-side hub clientspackages/core/src/hub/runtime-host/ — hub-backed runtime hostsI want to add a new tool:
packages/core/src/extensions/tools/ — built-in tool definitionspackages/agents/src/tool-use.ts — how tools are calledpackages/core/src/extensions/plugin/ — plugin-registered toolsI want to understand settings and configuration:
packages/core/src/extensions/config/ — file watching and loadingpackages/core/src/runtime/config/ — provider settings resolutionpackages/core/src/settings/ — settings state and mutationI want to add a new runtime feature (hook/extension):
packages/shared/src/hooks/ — hook types and enginepackages/core/src/extensions/plugin/ — plugin discovery and executionpackages/core/src/services/local-runtime-bootstrap.ts — how runtime is composed*.ts — TypeScript source*.test.ts — unit tests (Vitest)*.e2e.test.ts — end-to-end tests requiring full integration*.ts in examples — runnable example files (plugins, hooks)*.md files in apps/examples/ — documentation and markdown-based specs (cron, events)ClineCore — packages/core/src/index.ts — the main SDK orchestratorAgent — packages/agents/src/agent.ts — the agent loopRuntimeHost — packages/core/src/runtime/host/runtime-host.ts — execution abstractionAgentPlugin — packages/shared/src/plugin/ — plugin contractCronSpec — packages/shared/src/cron/cron-spec-types.ts — automation specsThis repo has both publishable SDK packages and internal workspace packages.
Architectural consequence:
The following packages are published to npm:
@cline/shared — shared types, contracts, and low-level utilities@cline/llms — provider integrations and model manifests@cline/agents — the agent loop and tool orchestration@cline/core — the main SDK with session management, hub, and configurationThe following workspace apps are internal and not published as SDK packages:
apps/cli — CLI implementationapps/webview — VS Code webviewapps/examples — example plugins and integrations