This document tracks the plan to make jcode's self-dev / refactor loop much faster without sacrificing full-feature builds.
See also:
- Keep full-featured builds available for normal usage and self-dev reloads.
- Make common self-dev edits significantly cheaper to compile.
- Reduce how often customizations require recompilation at all.
- Measure improvements after each phase and stop churn that does not pay off.
Measured locally on the current tree:
- Warm
cargo check --quiet: ~8.5s - Warm
scripts/dev_cargo.sh build --release -p jcode --bin jcode --quiet: ~47.3s
Additional observations from this audit:
- A previous warm-ish
cargo checkrun landed around ~12.3s. - A less-warm
cargo check --timingsrun landed around ~23.8s. - The previous local default
clang + moldsetup failed during release linking on this machine. clang + lldlinks the releasejcodebinary successfully here.
For common self-dev edits that do not touch broad shared interfaces:
- Warm
cargo check: < 5s - Warm
cargo build/ reload-oriented build: < 20–30s
For shared/core edits we should still aim to stay materially below today's baseline, even if they cannot reach the same fast path.
- Workspace / crate boundaries
- Rust caches best at the crate boundary.
- Heavy untouched subsystems should remain compiled and reusable in full builds.
- Good boundary design
- High-churn logic should not live in broad fanout crates or unstable shared types.
sccache- Practical win for repeated local builds and CI.
- Fast, reliable linker configuration
- Especially important for
cargo buildand release/self-dev reload builds.
- Especially important for
- Heavy subsystem isolation
- Embeddings, provider implementations, and large TUI/rendering code should stop churning unrelated builds.
- Narrower build targets for inner loops
- Avoid rebuilding extra bins/targets when not needed.
- Reduce the need to recompile at all
- Issue #32's customization records and extension points should make many changes config/hook/skill/data driven rather than source driven.
- Keep
.cargo/config.tomlconservative for local contributors. - Use
scripts/dev_cargo.shfor local self-dev builds:- enables
sccacheautomatically if installed - prefers
clang + lldon Linux x86_64 - uses the dedicated Cargo
selfdevprofile forjcodeself-dev build/reload paths - can still opt into
moldviaJCODE_FAST_LINKER=mold
- enables
- Route refactor-shadow builds through that wrapper.
Standard self-dev checkpoints now live behind scripts/bench_selfdev_checkpoints.sh, which runs:
- cold
cargo check - warm touched-file
cargo check - cold self-dev
jcodebuild - warm touched-file self-dev
jcodebuild
Use it when capturing comparable before/after numbers for refactors.
- Add documented commands for cold/warm
checkandbuildtiming. - Prefer touched-file timings (for example
scripts/bench_compile.sh check --touch src/server.rs) over no-op hot-cache reruns when judging ROI. - Track timing deltas after each structural phase.
- Fix build/link blockers before treating any timing data as authoritative.
- 2026-03-25: upgraded
scripts/bench_compile.shto support repeated runs, summary stats, JSON output, and extra cargo-arg passthrough so compile-speed work can use consistent touched-file measurements instead of one-off ad hoc timings. - 2026-03-25: upgraded
scripts/dev_cargo.shwith--print-setupplus clearer cache/linker diagnostics so developers can confirm whethersccache/ fast-linker paths are actually active. - 2026-03-30: removed the per-build
build.rstimestamp/build-number churn from local source builds.JCODE_VERSIONfor source builds is now stable perCargo.tomlversion + git hash, while UI/version build-time display comes from the binary mtime at runtime. Validation on this machine: two no-op release-jcode runs measured 221.688s then 0.559s, confirming the main crate no longer recompiles just because build metadata changed. - 2026-04-09: introduced a dedicated Cargo
selfdevprofile for self-dev iteration. On this machine, the warm localjcodeself-dev build path dropped from about 56.1s forscripts/dev_cargo.sh build --release -p jcode --bin jcode --quietto about 16.0s forscripts/dev_cargo.sh build --profile selfdev -p jcode --bin jcode --quiet, while keeping the normal release/distribution profile unchanged. - 2026-04-18: added
scripts/bench_selfdev_checkpoints.shto standardize cold/warm self-dev checkpoints. First local checkpoint attempt on this machine surfaced two environment blockers:- cold checkpoints failed because
cargo cleancould not remove part oftarget/release(Permission deniedon a fingerprint timestamp file) - warm
selfdev-jcodetouched-file measurement onsrc/tool/read.rsfailed because thesccache-wrapped rustc process terminated with signal 15 during thejcodecrate build - warm touched-file
cargo checkonsrc/tool/read.rscompleted in 93.115s then 9.430s, which is useful as a rough upper/lower bound but not yet stable enough to treat as an authoritative checkpoint - follow-up required: fix the
target/releasepermission issue, rerun cold checkpoints, and rerun warm self-dev measurements until they are stable enough to compare against future waves
- cold checkpoints failed because
- 2026-04-18: updated
scripts/bench_selfdev_checkpoints.shto keep running after individual checkpoint failures and report them in JSON/text output instead of aborting early. Verified local output on this machine with--touch src/tool/read.rs --runs 1:- warm touched-file
cargo check: 9.582s - warm touched-file
selfdev-jcodebuild: 59.898s - failed checkpoints reported cleanly:
cold_check,cold_selfdev_build
- warm touched-file
- 2026-04-18: added
--skip-coldtoscripts/bench_selfdev_checkpoints.shso warm-only checkpoints remain usable while cold-path cleanup is blocked locally. Verified local output on this machine with--skip-cold --touch src/tool/read.rs --runs 1:- warm touched-file
cargo check: 9.339s - warm touched-file
selfdev-jcodebuild: 18.844s - skipped checkpoints reported explicitly:
cold_check,cold_selfdev_build
- warm touched-file
- 2026-04-18: additional warm-only checkpoint on a broader shared edit target with
--skip-cold --touch src/server.rs --runs 1:- warm touched-file
cargo check: 8.711s - warm touched-file
selfdev-jcodebuild: 18.969s
- warm touched-file
- 2026-04-18: additional warm-only checkpoint on a heavy tool-path file with
--skip-cold --touch src/tool/communicate.rs --runs 1:- warm touched-file
cargo check: 8.496s - warm touched-file
selfdev-jcodebuild: 21.400s
- warm touched-file
- 2026-04-18: additional warm-only checkpoint on a provider-heavy file with
--skip-cold --touch src/provider/openai.rs --runs 1:- warm touched-file
cargo check: 8.750s - warm touched-file
selfdev-jcodebuild: 21.386s
- warm touched-file
- 2026-04-18: additional warm-only checkpoint on the shared provider module with
--skip-cold --touch src/provider/mod.rs --runs 1:- warm touched-file
cargo check: 9.772s - warm touched-file
selfdev-jcodebuild: 17.917s
- warm touched-file
- 2026-04-18: additional warm-only checkpoint on the agent entry module with
--skip-cold --touch src/agent.rs --runs 1:- warm touched-file
cargo check: 7.318s - warm touched-file
selfdev-jcodebuild: 30.928s
- warm touched-file
- 2026-04-18: additional warm-only checkpoint on the memory tool with
--skip-cold --touch src/tool/memory.rs --runs 1:- warm touched-file
cargo check: 7.787s - warm touched-file
selfdev-jcodebuild: 12.798s
- warm touched-file
- 2026-04-18: additional warm-only checkpoint on session search with
--skip-cold --touch src/tool/session_search.rs --runs 1:- warm touched-file
cargo check: 7.009s - warm touched-file
selfdev-jcodebuild: 12.874s
- warm touched-file
- 2026-04-18: additional warm-only checkpoint on the browser tool with
--skip-cold --touch src/tool/browser.rs --runs 1:- warm touched-file
cargo check: 13.693s - warm touched-file
selfdev-jcodebuild: 18.874s
- warm touched-file
- 2026-04-28: diagnosed the repeated self-dev
jcodelib buildSIGTERMon this 16 GiB, no-swap workstation.journalctl -u earlyoomshowed earlyoom sendingSIGTERMto the rootrustcwhen available memory crossed the 10% threshold. A direct no-sccachebuild reproduced the same signal, sosccachewas only reporting the termination.scripts/dev_cargo.shnow enables adaptive low-memory overrides for--profile selfdevwhen Linux + earlyoom + no swap + <24 GiB RAM + <8 GiB currently available RAM are detected:CARGO_INCREMENTAL=1,CARGO_PROFILE_SELFDEV_INCREMENTAL=true, andCARGO_PROFILE_SELFDEV_CODEGEN_UNITS=256. UseJCODE_SELFDEV_LOW_MEMORY=offto disable, orJCODE_SELFDEV_LOW_MEMORY=onto force. Initial validation completed under the earlier settings in 2m34s after an interrupted partial build reused artifacts; a later benchmark with 9.4 GiB available showed that preserving the inherited selfdev profile can reduce warm edit builds from about 60s to about 14s when there is enough headroom. - 2026-05-21: rechecked the same failure mode during overnight TUI test triage.
journalctl -u earlyoomshowed repeatedSIGTERMevents against rootjcoderustcprocesses at about 2.7-3.3 GiB RSS.CARGO_PROFILE_SELFDEV_CODEGEN_UNITS=16still failed under current browser and desktop-session memory pressure, while a direct no-sccacheselfdev build with incremental enabled andCARGO_PROFILE_SELFDEV_CODEGEN_UNITS=256completed. The adaptive low-memory default was changed to match that passing profile, disablessccacheby default becausesccacherejects Cargo incremental builds, andscripts/dev_cargo.shnow honorsSCCACHE_DISABLE=1before auto-enabling the wrapper. Fresh lib-test compilation remains heavier than the binary build and may still require a remote builder, more free memory, or swap. - 2026-05-05: trimmed root compile surface by replacing broad
tokio/fullwith explicit used features, aligning Jcode-ownedcrosstermdependencies on 0.29, and replacingqr2termwith directqrcoderendering. This removed the duplicatecrossterm 0.28path from thejcodetree while preserving login QR output. Validation:cargo check --profile selfdev -p jcode --bin jcode,cargo test --profile selfdev login_qr --lib -- --nocapture, and coordinatedselfdev buildpassed. - 2026-05-05: removed unused
reqwest/blockingfromjcode-provider-core; static search showed no blocking API usage in that crate. Validation:cargo check --profile selfdev -p jcode-provider-coreand fullcargo check --profile selfdev -p jcode --bin jcodepassed. - 2026-05-03: added
JCODE_DEV_FEATURE_PROFILEtoscripts/dev_cargo.shso compile-speed probes and narrow inner-loop builds can consistently select feature sets without repeating Cargo flags. Profiles:default,minimal/none(--no-default-features),pdf(--no-default-features --features pdf),embeddings(--no-default-features --features embeddings), andfull(--features embeddings,pdf). The wrapper leaves explicit--features/--no-default-featurescargo args untouched. Validation on this machine:JCODE_DEV_FEATURE_PROFILE=minimal scripts/dev_cargo.sh check -p jcode --lib --quietpassed. - 2026-05-03: disabled Cargo auto-discovery for root binary targets and moved developer-only helper
binaries (
tui_bench,session_memory_bench,mermaid_side_panel_probe) behind the opt-indev-binsfeature. This keeps broad normal checks focused on production/test targets while preserving explicit probe coverage viacargo check --all-targets -p jcode --features dev-bins. Validation showedcargo check --all-targets -p jcodeskips those three bins, while adding--features dev-binsincludes them. - 2026-05-03: moved the self-dev build/version/channel support implementation out of the root crate and
into
crates/jcode-build-support, leavingsrc/build.rsas a re-export facade. This cuts another stable, high-fanout support subsystem out of the root compile unit while preserving existing call sites (crate::build::*). Validation:cargo check -p jcode-build-support,cargo test -p jcode-build-support, andcargo check -p jcode --libpassed during the split. - 2026-05-03: moved the pure keybinding parser/matcher/types from
src/tui/keybind.rsintojcode-tui-core::keybind, leaving root TUI config-loading wrappers in place. This creates a reusable cache boundary for a low-coupling TUI helper module while preserving the existingcrate::tui::keybind::*API. Validation:cargo check -p jcode-tui-core,cargo test -p jcode-tui-core, andcargo check -p jcode --libpassed.
Warm-only touched-file checkpoints captured so far on this machine:
| Touched file | Warm cargo check |
Warm selfdev-jcode build |
|---|---|---|
src/tool/session_search.rs |
7.009s | 12.874s |
src/agent.rs |
7.318s | 30.928s |
src/tool/memory.rs |
7.787s | 12.798s |
src/tool/communicate.rs |
8.496s | 21.400s |
src/server.rs |
8.711s | 18.969s |
src/provider/openai.rs |
8.750s | 21.386s |
src/tool/read.rs |
9.339s | 18.844s |
src/provider/mod.rs |
9.772s | 17.917s |
src/tool/browser.rs |
13.693s | 18.874s |
Observed spread from these warm-only checkpoints:
- warm touched-file
cargo check: 7.009s to 13.693s - warm touched-file
selfdev-jcodebuild: 12.798s to 30.928s - fastest measured warm self-dev rebuilds so far are on smaller tool-path edits
src/agent.rscurrently stands out as the most expensive warm self-dev rebuild in this sample setsrc/tool/browser.rscurrently stands out as the slowest warmcargo checkin this sample set
The refined layered target, dependency rules, and migration guidance live in
docs/MODULAR_ARCHITECTURE_RFC.md. The crate list
below is the compile-performance-oriented destination sketch and should be read
as compatible with that RFC, not as the only acceptable final packaging.
Proposed destination layout:
jcode-core- protocol, ids, message types, config primitives, shared utility types
jcode-server- server lifecycle, reload, socket, swarm, daemon behaviors
jcode-agent- agent turn loop, tool orchestration, stream handling
jcode-provider- provider traits, shared provider types, routing/catalog support
jcode-embedding- embedding model integration and related heavy inference dependencies
jcode-tui- TUI rendering, widgets, state reduction, terminal UI support
jcode-tui-core- low-level TUI helpers with minimal root coupling, including stream buffers and keybinding parsing
jcode-selfdev- customization records, migration logic, self-dev productization
jcode-build-support- self-dev build commands, source-state fingerprints, binary channel paths/manifests
Start with the highest-leverage cache boundaries:
jcode-embedding- provider support / provider implementation splits
- self-dev/customization system once the new extension-point work lands
- server / agent split along the seams already being extracted
-
2026-03-24: moved the heavy ONNX/tokenizer implementation into the new
crates/jcode-embeddingworkspace crate. -
The main
src/embedding.rsmodule now acts as a facade for process-local cache/stats/path/logging integration. -
This preserves the public
crate::embeddingAPI while creating a real Cargo cache boundary for the heaviest embedding dependencies. -
Follow-up: gather more realistic before/after timing data using controlled touched-file benchmarks rather than fully hot no-op rebuilds.
-
2026-05-05: made the
embeddingsfeature opt-in instead of part of default features for faster ordinarycargo check/cargo buildloops. -
2026-05-23: reverted that default-feature split because embedding-backed memory recall and semantic retrieval should work out of the box in normal builds. Default builds now enable both
pdfandembeddings; developers who need compile-speed probes can useJCODE_DEV_FEATURE_PROFILE=minimalorJCODE_DEV_FEATURE_PROFILE=pdfto skip the local inference stack. Full local inference remains available explicitly via--features embeddingsorJCODE_DEV_FEATURE_PROFILE=fullwhen testing non-default feature paths. Validation target:cargo tree -p jcode --edges normal --depth 1should include bothjcode-pdfandjcode-embedding;--no-default-featuresshould include neither. -
2026-03-24: moved PDF extraction behind the new
crates/jcode-pdfworkspace crate and fixed the--no-default-featuresbuild path by making PDF support degrade gracefully when the feature is disabled. -
2026-03-24: moved Azure bearer-token retrieval behind the new
crates/jcode-azure-authworkspace crate so the Azure SDK no longer lives directly in the main crate. -
Note: touched-file timing for
src/auth/azure.rsneeds more instrumentation cleanup; one post-split sample was anomalous and should not be treated as a trustworthy ROI datapoint yet. -
2026-03-24: moved email notification / IMAP reply transport behind the new
crates/jcode-notify-emailworkspace crate. -
The main
src/notifications.rsmodule now keeps the higher-level ambient, safety, and channel integration while SMTP/IMAP/mail parsing lives behind a dedicated crate boundary. -
This split is primarily meant to keep
lettre,imap,mail-parser, andnative-tlsout of unrelated self-dev rebuilds; edits tonotifications.rsitself still invalidate the main crate and are not the right sole ROI metric. -
2026-03-25: landed the first provider boundary slice with
crates/jcode-provider-metadata. -
Boundary decision: provider metadata / profile catalogs / pure selection helpers move into their own crate first, while env mutation, config-file I/O, and runtime integration remain in
src/provider_catalog.rsas a facade. -
This is intentionally narrower than a full
Providertrait split: it creates a real provider-side compile boundary without prematurely dragging streaming/message/runtime dependencies into a shared crate that would likely stay high-churn. -
2026-03-25: landed the next provider-core slice with
crates/jcode-provider-core. -
Boundary decision: move shared HTTP client + route/cost/core provider value types first, but keep the
Providertrait itself insrc/provider/mod.rsfor now. -
Reason: the trait currently still mixes in
message.rs, runtime/auth behavior, and provider-specific streaming/compaction concerns; moving it too early would likely create a noisy, still-high-churn core crate. -
2026-03-25: landed the first provider-implementation support crate with
crates/jcode-provider-openrouter. -
Boundary decision: move OpenRouter-specific model catalog / endpoint cache / provider ranking / model-spec parsing support into a dedicated crate, while keeping the actual
Providertrait impl, auth wiring, and message/stream translation insrc/provider/openrouter.rs. -
Reason: this creates a real provider-implementation compile boundary now, without introducing a crate cycle through
Provider,EventStream, ormessage.rs. -
2026-03-25: landed the next provider-implementation support crate with
crates/jcode-provider-gemini. -
Boundary decision: move Gemini Code Assist schema/types, model-list constants, and pure support helpers into a dedicated crate, while keeping the actual
Providertrait impl, auth calls, and runtime/network orchestration insrc/provider/gemini.rs. -
Reason: this creates another real provider-side compile boundary without forcing the
Provider/EventStreamseam prematurely. -
2026-03-30: moved the pure OpenAI tool-schema normalization helpers into
crates/jcode-provider-core/src/openai_schema.rs. -
Boundary decision: move pure schema adaptation / strict-normalization helpers first, while keeping
build_tools(...)and request-history rewriting insrc/provider/openai_request.rsbecause those still depend on local tool/message types. -
Reason: this creates another provider-side cache boundary now without prematurely pulling
Message,ToolDefinition, or theProvidertrait into a shared crate. -
2026-05-05: moved provider catalog-refresh diffing into
jcode-provider-core::catalog_refreshand re-exported it from the root provider facade. -
Boundary decision: move the pure
ModelRoutesummary/diff logic first because it has no root-crate auth/runtime/config dependencies. -
2026-05-05: split the stable provider pricing tables/helpers into
jcode-provider-core::pricing, leavingsrc/provider/pricing.rsas a thin facade for root-only auth/env/OpenRouter-cache lookups. -
Reason: provider pricing is relatively stable table/math code, but it previously lived in the main crate beside high-churn provider runtime code. This creates a reusable cache boundary without moving the
Providertrait or network implementations prematurely. -
Validation:
cargo test -p jcode-provider-core --quiet,cargo test -p jcode pricing:: --quiet,cargo check -p jcode --quiet, andcargo check -p jcode --features embeddings --quietpass. -
2026-05-05: moved provider failover prompt/decision/classifier contracts and provider selection/fallback-order contracts into
jcode-provider-core, leaving root provider modules as facades for env/runtime/account state. This continues shrinkingsrc/provider/mod.rssupport surfaces toward an eventualjcode-providerruntime crate. -
Validation:
cargo test -p jcode-provider-core --quiet, focused root provider selection/failover tests, andcargo check -p jcode --quietpass. -
2026-05-05: moved the Copilot
PremiumModeprovider-control enum intojcode-provider-coreand re-exported it from the root/Copilot facades. TheProvidertrait no longer needs to name the rootcopilotmodule for this control surface. -
Validation:
cargo check -p jcode-provider-core --quietandcargo check -p jcode --quietpass. -
2026-05-05: moved provider-native tool result DTOs/sender aliases into
jcode-provider-core. The globalProvidertrait no longer has to expose types owned by the root Claude module. -
Validation:
cargo check -p jcode-provider-core --quietandcargo check -p jcode --quietpass. -
2026-05-05: moved stable provider model constants, static provider/model classification, Copilot model-name normalization, and fallback context-window heuristics into
jcode-provider-core::models. Rootsrc/provider/models.rsnow layers dynamic account catalogs, runtime availability, and cache hydration on top of those core helpers. -
Validation:
cargo test -p jcode-provider-core models:: --quiet,cargo check -p jcode-provider-core --quiet, andcargo check -p jcode --quietpass. -
2026-05-05: moved the global
Providertrait andEventStreamalias intojcode-provider-core. Rootsrc/provider/mod.rsnow re-exports the contract while continuing to own concrete provider implementations andMultiProvidercomposition. This is the main provider seam needed before a futurejcode-providerruntime crate can be introduced safely. -
Validation:
cargo check -p jcode-provider-core --quietandcargo check -p jcode --quietpass. -
Warm-only touched-file benchmark on
src/provider/mod.rsafter the provider-core seam: first self-dev build was a noisy artifact-producing 140.739s, then the immediate rerun measured 12.101s warmcargo checkand 27.433s warm self-dev build. Treat the rerun as the comparable steady-state datapoint. -
2026-05-05: moved the stable provider-facing
ToolDefinitioncontract fromsrc/message.rsintojcode-message-typesand re-exported it from the root message facade. This is a prerequisite for shrinking the provider trait and tool registry surfaces away from root-crate-only message types. -
Validation:
cargo test -p jcode-message-types --quietandcargo check -p jcode --quietpass. -
2026-05-05: introduced
jcode-tool-typesfor stable tool execution output DTOs and movedToolOutput/ToolImageout ofsrc/tool/mod.rs. Root tool modules continue using the same names via a facade re-export, but provider/agent/server seams can now depend on a narrow tool result contract without depending on the root tool registry. -
Validation:
cargo check -p jcode-tool-types --quiet,cargo test -p jcode-tool-types --quiet, andcargo check -p jcode --quietpass. -
2026-05-05: added
jcode-tool-corefor runtime tool contracts and movedTool,ToolContext,ToolExecutionMode, andStdinInputRequestout ofsrc/tool/mod.rs.jcode-tool-typesstays DTO-only, while channel/runtime-bearing context lives in the runtime-contract crate instead of contaminating pure type crates. -
2026-05-05: also moved the shared tool intent schema helper into
jcode-tool-core, keeping the rootsrc/tool/mod.rsmodule focused on registry composition rather than shared schema contracts. -
Validation:
cargo check -p jcode-tool-core --quiet,cargo check -p jcode-tool-types --quiet, andcargo check -p jcode --quietpass. -
2026-05-05: moved provider streaming contracts
StreamEventandConnectionPhasefromsrc/message.rsintojcode-message-types, again preserving root facade re-exports. Together withToolDefinition, this materially reduces the root-only surface of the provider trait and prepares a futurejcode-providercrate. -
Validation:
cargo check -p jcode-message-types --quiet,cargo test -p jcode-message-types --quiet, andcargo check -p jcode --quietpass. -
2026-05-05: moved core conversation DTOs
Message,ContentBlock,Role, andCacheControlintojcode-message-types, while keeping root-only redaction/generated-image/session helpers insrc/message.rs. Provider and agent contracts can now refer to message data through the lower type crate rather than the root crate facade. -
Validation:
cargo check -p jcode-message-types --quiet,cargo test -p jcode-message-types --quiet, andcargo check -p jcode --quietpass. -
2026-05-05: moved pure message helpers for fresh-user-turn detection, stable message hashing, tool ID sanitization, and the missing-tool-output constant into
jcode-message-types. Root keeps secret redaction and generated-image visual context because those still depend on regex/env/fs/base64 integration details. -
Validation:
cargo check -p jcode-message-types --quiet, focused root message helper tests, andcargo check -p jcode --quietpass. -
2026-05-05: moved the provider split-system dynamic-context insertion helper and its tests into
jcode-message-types. This removes another pure message transformation fromsrc/provider/mod.rsand keeps preparing the provider trait for an eventual runtime crate split. -
Validation:
cargo test -p jcode-message-types dynamic_context --quiet,cargo check -p jcode-message-types --quiet, andcargo check -p jcode --quietpass. -
2026-05-05: moved the server lightweight-control request classifier from
src/server/client_lifecycle.rsintojcode-protocol::Request::is_lightweight_control_request. This is a small but directionally important server seam: protocol-shape policy belongs with the protocol contract, while the large client lifecycle module keeps runtime dispatch. -
Validation:
cargo check -p jcode-protocol --quietandcargo check -p jcode --quietpass. -
2026-05-05: moved swarm task-control action parsing, assignment-message formatting, and status eligibility/error policy from
src/server/comm_control.rsintojcode-plan. This keeps plan/task policy next to the plan graph/status helpers and leaves server comm control focused on runtime I/O and mutation orchestration. -
Validation:
cargo test -p jcode-plan --quietandcargo check -p jcode --quietpass. -
2026-03-30: moved the workspace-map subsystem into the new
crates/jcode-tui-workspacecrate. -
Boundary decision: move workspace map data/model + widget rendering first, while keeping the surrounding
info_widget, app state, and higher-level TUI composition in the main crate. -
Reason: this is a safe first
jcode-tuifoothold because the workspace map code is already mostly self-contained and avoids the much riskierApp/ renderer / markdown / mermaid seams.
- Continue shrinking giant hotspot files.
- Keep high-churn code out of stable low-level crates.
- Avoid changing shared broad fanout types casually.
- Store customization intent, provenance, validation, and migration hints.
- Add extension points so more user changes live in:
- config
- hooks
- skills
- prompt overlays
- routing/theme/layout data
- Prefer those over direct Rust source edits whenever possible.
- 2026-03-30: landed the first prompt-overlay seam for system-prompt customization without a rebuild.
jcode now loads
~/.jcode/prompt-overlay.mdand./.jcode/prompt-overlay.mdinto the static prompt, which is a low-risk first step toward the broader issue #32 customization plan.
Touched-file cargo check samples gathered during this batch:
src/server.rs: ~8.7ssrc/tool/read.rs: ~8.8ssrc/auth/azure.rsbefore Azure crate split: ~7.0ssrc/provider/openrouter.rsbefore Azure crate split: ~6.5ssrc/provider/openrouter.rsafter Azure crate split: ~6.0ssrc/notifications.rsafter notification-email crate split: ~11.4ssrc/channel.rsafter notification-email crate split: ~4.8ssrc/provider_catalog.rsafter provider-metadata split: ~5.8ssrc/provider/mod.rsafter provider-core type split: ~50.1ssrc/provider/openrouter.rsafter openrouter-support crate split: ~5.6ssrc/provider/gemini.rsafter gemini-support crate split: ~5.5s
Notes:
- The post-split touched-file measurement for
src/auth/azure.rsproduced an anomalous result and should not be treated as a reliable ROI datapoint yet. - The post-split
src/notifications.rstiming is not by itself a negative signal: touching that root module still rebuilds the main crate, while the intended win is that unrelated edits stop dragging mail transport dependencies through the same compile unit. - No-op fully hot-cache reruns can look unrealistically fast; use touched-file scenarios when evaluating structural compile-speed changes.
- Provider metadata timings should be interpreted as a first provider-side foothold, not the final provider ROI story; the larger wins should come from future provider-core / implementation splits.
- The
src/provider/mod.rstouched-file timing remains high because touching that root file still rebuilds the main crate and the auth/runtime-heavy trait logic. This stage is about carving out stable reusable pieces first, not claiming that the provider root is solved. - The
src/provider/openrouter.rstouched-file sample is more encouraging because the heavy OpenRouter-specific catalog/ranking/cache support now lives in its own crate while the main module stays a thinner wrapper. - The
src/provider/gemini.rstouched-file sample is similarly encouraging: the serde-heavy Code Assist schema and pure model-list/support helpers now live outside the main crate while the runtime wrapper remains local.
global-hotkeyis now gated behindtarget_os = "macos"instead of being compiled on all platforms.- This is a smaller win than a crate split, but it removes an unnecessary dependency subtree from Linux self-dev builds because the hotkey listener implementation is macOS-only.
- Validation: on Linux,
cargo tree -i global-hotkeyis now empty.
The next obvious heavy dependency boundaries are less clearly safe/local than the ones already landed:
- provider support remains high-value, but
src/provider/mod.rsand related implementations are broad enough that the next split should be designed carefully instead of rushed. - a future
jcode-provider-core/ provider-implementation split is still the most promising next compile-speed move, but it needs boundary design first so high-churn shared types do not create a new invalidation hotspot.
Current provider-boundary stance:
- Done:
jcode-provider-metadatafor stable login/profile catalog data and pure selection logic. - Done:
jcode-provider-corefor shared HTTP client plus route/cost/core provider value types. - Done:
jcode-provider-openrouterfor OpenRouter-specific catalog/cache/ranking/model-spec support. - Done:
jcode-provider-geminifor Gemini Code Assist schema/types and pure model support helpers. - Done:
jcode-provider-core::openai_schemafor pure OpenAI schema adaptation / strict-normalization helpers. - Not done yet:
Providertrait /EventStreamextraction and fully independent provider impl crates. - Reason: the trait side still depends on
message.rs, auth flows, runtime behavior, and provider-specific streaming logic; the current staged split avoids turning that unstable seam into a low-value high-churn crate.
That means the best next batch should likely target either:
- a carefully designed trait seam, or
- another provider implementation support split with similarly clean boundaries.
Current TUI-boundary stance:
- Done:
jcode-tui-workspacefor workspace-map model + widget rendering. - Not done yet: broader
jcode-tuiextraction for markdown, mermaid, info widgets, and the shared renderer. - Reason: the remaining high-value TUI files are larger but still more tightly coupled to
App, config, images, side-panel state, and rendering orchestration, so they need staged extraction rather than a rushed top-level split.
The previous splits created many *-types / *-core crates that the root re-exports from, but
scripts/analyze_root_crate.py shows 334K of 336K root-crate lines are still genuinely in-root:
the heavy logic stayed behind thin facades. The analyzer (committed alongside this section) gives the
objective map needed to finish the job. Run it any time:
python3 scripts/analyze_root_crate.py # ranked modules + blockers + cycles + feedback arcs
python3 scripts/analyze_root_crate.py --full # full feedback-arc-set listing
python3 scripts/analyze_root_crate.py --json # machine-readableThe library-only module graph (test code excluded) has a single strongly-connected component of 42
modules / ~310K lines (92% of the crate): tui, server, provider, tool, cli, auth, agent, session, ….
You cannot peel tui/server/provider into independent crates while they mutually reference each
other. This is the structural blocker, and it is why "just move the big dirs out" never works.
The 42-module cycle is held together by only 46 back-edges totaling 178 references, and most are single-reference "accidental" couplings. Breaking this small feedback arc set turns the graph into a DAG, after which modules peel off bottom-up. Cheapest-first (from the analyzer):
- 1-ref edges (≈24 of them): e.g.
agent -> tui(onewrite_generated_image_side_panel_pagecall),tool -> tui(onetui::image::display_imageimport),config -> auth,config -> tool,telemetry -> cli,bus -> provider,browser -> provider. Each is a single call/import that can move to a shared lower-level crate or be inverted behind a trait/callback. - Mid-weight edges:
usage -> auth(4),tool -> provider(5),tool -> server(5),sidecar -> provider(7),agent -> tool(9),import -> tui(9),usage -> provider(9). - Heavy structural seams (design carefully):
agent -> provider(20),cli -> tui(21),auth -> provider(39). These are the genuine architectural couplings worth a trait seam.
- Baseline metrics. Record peak rustc RSS for the largest current unit and full-build wall time
(
scripts/bench_compile.sh), so each extraction's memory/compile win is measurable. - Break the cheap back-edges first. Eliminate the 1-2 ref couplings (image helpers, single config
lookups, telemetry/cli) by moving shared primitives down into existing low-level crates
(
jcode-core,jcode-tui-*) or inverting them behind small traits. Re-run the analyzer; watch the SCC shrink. - Extract already-clean leaves. Modules the analyzer marks "extractable now" (no in-root blockers):
background,prompt,safety,transport,replay,browser,perf, plus the many <400 loc leaves. These need no cycle-breaking and immediately shrink the root crate. - Address the heavy seams (
auth↔provider,cli↔tui,agent↔provider) with deliberate trait boundaries once the cheap edges are gone. - Lift the big subsystems (
tool29K,provider35K,server38K, thentui125K) once they are no longer in the cycle.tuigoes last: almost nothing depends on it (sink of the DAG), so once its own outbound deps are crates it lifts cleanly and removes the single largest compilation unit.
rustc holds an entire crate's IR in memory at its codegen peak, so peak RSS scales with the size of the largest compilation unit. Today that unit is the 336K-line root crate (~2.5-3 GiB/process). Every module moved into its own crate shrinks the largest unit, lowering peak per-process RSS, which is exactly what lets more parallel jobs run without OOM (complementing the memory-adaptive job count above). It also sharpens incremental caching: editing one file rebuilds one small crate instead of the whole monolith.
Use:
scripts/dev_cargo.sh check --quiet
scripts/dev_cargo.sh build --release -p jcode --bin jcode --quiet
scripts/dev_cargo.sh build --profile selfdev -p jcode --bin jcode --quiet
scripts/dev_cargo.sh --print-setupFor narrower feature-set probes, set JCODE_DEV_FEATURE_PROFILE instead of spelling out Cargo flags:
JCODE_DEV_FEATURE_PROFILE=minimal scripts/dev_cargo.sh check -p jcode --lib --quiet
JCODE_DEV_FEATURE_PROFILE=pdf scripts/dev_cargo.sh build --profile selfdev -p jcode --bin jcode --quiet
JCODE_DEV_FEATURE_PROFILE=full scripts/dev_cargo.sh check -p jcode --lib --quietThis is especially useful because default jcode enables both embeddings and pdf; in the current
dependency graph, the root tree is about 3740 lines with defaults, 1133 with PDF-only, and 1106
with no default features. Use these profiles for measurements and local probes, while keeping full/default
builds in CI and release paths where feature coverage matters.
Developer-only root binaries are opt-in to keep --all-targets inner loops from compiling extra probe
entrypoints by default:
cargo run --features dev-bins --bin tui_bench -- --help
cargo run --features dev-bins --bin session_memory_bench -- --help
cargo run --features dev-bins --bin mermaid_side_panel_probe -- --help
cargo check --all-targets -p jcode --features dev-bins --quietThe wrapper:
- uses
sccacheautomatically when available for non-incremental builds only - prefers
lldlocally on Linux x86_64 - uses the fast
selfdevCargo profile for self-dev build/reload workflows - can inject a named feature profile via
JCODE_DEV_FEATURE_PROFILEunless explicit feature args are present - avoids hard-forcing a linker mode that may be broken on a given machine
- can print the currently selected cache/linker setup with
--print-setup
Override linker mode explicitly when needed:
JCODE_FAST_LINKER=lld scripts/dev_cargo.sh build --release -p jcode --bin jcode
JCODE_FAST_LINKER=mold scripts/dev_cargo.sh build --release -p jcode --bin jcode
JCODE_FAST_LINKER=system scripts/dev_cargo.sh build --release -p jcode --bin jcodesccache cannot cache incremental compilation units. All of jcode's common
profiles (selfdev, dev, release, test) set incremental = true, so on
the inner loop sccache produced a 0% hit rate (measured across 272 real
compilations and clean workspace/dep rebuilds) while still adding per-rustc
wrapper overhead and a misleading "enabled" status.
dev_cargo.sh now decides automatically:
- Incremental builds (selfdev/dev/release/test, or
CARGO_INCREMENTAL=1): sccache is skipped (sccache_status=skipped-incremental), since it can never hit. - Non-incremental builds (
release-lto, orCARGO_INCREMENTAL=0): sccache is enabled, where it genuinely produces cache hits (CI, distribution builds).
Overrides:
JCODE_SCCACHE=on scripts/dev_cargo.sh build --profile selfdev ... # force-enable
JCODE_SCCACHE=off scripts/dev_cargo.sh build --profile release-lto ... # force-disable
CARGO_INCREMENTAL=0 scripts/dev_cargo.sh build --profile selfdev ... # makes it cacheable- 2026-05-29: made sccache incremental-aware. Validation on this machine:
--print-setupreportsskipped-incrementalfor--profile selfdevandenabledfor--profile release-lto;JCODE_SCCACHE=onandCARGO_INCREMENTAL=0both re-enable it for selfdev. A cleanjcode-azure-authrebuild under the old always-on sccache showed0/54cache hits, confirming the prior wasted overhead.
When JCODE_REMOTE_CARGO=1 (commonly set in ~/.config/jcode/remote-build.env),
dev_cargo.sh offloads builds to JCODE_REMOTE_HOST via scripts/remote_build.sh.
The preflight is designed so that remote builds "just work" when the host is up,
without paying a slow timeout when it is down:
- A cheap
bash/dev/tcpreachability probe runs before the SSH preflight, so an offline host fails over to local cargo in about 1s instead of waiting for the full SSHConnectTimeout(previously ~5s on every build). - After a recent failure, subsequent builds use a shorter recovery probe timeout (default 0.3s) so repeated builds during an outage stay cheap.
- Recovery is detected on the very next build: an up host answers the TCP probe in a few milliseconds, so it immediately reverts to remote builds with no cooldown.
- The probe is skipped automatically when the SSH config uses
ProxyJump/ProxyCommand(where a direct TCP probe to the final host would be misleading), falling back to the normal SSH preflight.
Tunables (all optional):
JCODE_REMOTE_TCP_TIMEOUT=1 # first-probe TCP timeout (seconds, fractional ok)
JCODE_REMOTE_RECOVERY_TCP_TIMEOUT=0.3 # probe timeout while host was recently down
JCODE_REMOTE_DOWN_TTL=300 # how long to keep using the recovery timeout
JCODE_REMOTE_TCP_PROBE=0 # disable the pre-probe; use SSH preflight only
JCODE_REMOTE_CARGO=0 # disable remote builds entirely for one command- 2026-05-29: added the TCP pre-probe + recovery-timeout logic above. Validation on this
machine (remote host offline): warm self-dev edit-build preflight dropped from about
5.0s to about 1.0s on the first build and about 0.3s on subsequent builds,
while an up host still reconnects on the next build (TCP probe answered in ~10ms in
unit tests). Function-level unit tests covered reachable/unreachable probes, the
recent-failure window, and
desktop-tailscaleendpoint resolution.
The target/ directory grows without bound across profiles (dev/selfdev/release)
and cross-compile caches (*-apple-darwin, *-pc-windows-*, linux-compat). On
this machine it reached ~84GB. A bloated target dir does not slow compilation
directly, but it can exhaust disk and force full rebuilds when space runs out.
scripts/clean_target.sh reclaims space safely while parallel builds are running:
- It never touches a
target/<profile>dir that has an activerustc/cargoprocess (scanned via/proc/<pid>/cmdline) or that was written to within a recent activity window (default 20min,JCODE_CLEAN_ACTIVE_WINDOW_MIN). - Default mode removes only cross-compile/compat caches (regenerated on demand) and reports what it would free.
--aggressiveadditionally runscargo clean --profile <p>on stale profiles (still subject to the active-process / recent-write guards).
scripts/clean_target.sh # dry-run: report reclaimable space
scripts/clean_target.sh --apply # remove cross-compile/compat caches
scripts/clean_target.sh --apply --aggressive # also cargo-clean stale profiles- 2026-05-29: reclaimed a stale
target/aarch64-apple-darwin(2.5G) and added this script. Dry-runs verified it correctly SKIPs profiles with recent writes / active rustc while a parallel agent was building ondebug/selfdev.
The biggest day-to-day pain on the 8-core/15 GiB dev machine was memory
pressure: several self-dev agents build at once, and .cargo/config.toml pinned
a static jobs = 6. rustc on the large root jcode crate peaks around 2.5-3 GiB
RSS, so 6 concurrent rustc processes (across one or several builds) overshoot RAM
on a no-swap box and earlyoom kills the build. A static job count is wrong in both
directions: it wastes cores when the machine is idle and oversubscribes memory when
it is busy.
scripts/dev_cargo.sh (the path selfdev build uses) now sizes the job count from
currently-available memory each time it runs:
select_build_jobs()readsMemAvailableand divides by a per-job memory budget (default 2048 MiB,JCODE_BUILD_MIB_PER_JOB), then clamps into[1, nproc].- It exports
CARGO_BUILD_JOBS, which overridesbuild.jobsfrom.cargo/config.toml. - An idle machine still uses every core; under pressure a fresh build self-throttles (e.g. it picked 2 jobs at ~5.9 GiB available during a parallel-agent build).
- Explicit
JCODE_BUILD_JOBS/CARGO_BUILD_JOBSalways win; invalid values warn and fall back to adaptive sizing. Non-Linux hosts keep the cargo/.cargodefault.
The committed static fallback in .cargo/config.toml was also lowered from 6 to
4 for direct cargo invocations that bypass the wrapper (still memory-safe for a
single build on ~15 GiB, but no longer assuming a near-full core count).
JCODE_BUILD_JOBS=2 # hard override the job count for one command
JCODE_BUILD_MIB_PER_JOB=2048 # memory budget per rustc job (default)
scripts/dev_cargo.sh --print-setup # shows build_jobs_status + cargo_build_jobs- 2026-05-29: added adaptive sizing. Verified via
--print-setupand a stubbed-cargo harness that overrides win, invalid input falls through to adaptive, budget extremes clamp to[1, nproc], and the chosenCARGO_BUILD_JOBSis exported to the child cargo process. A realdev_cargo.sh check -p jcode-loggingbuild succeeded.
For compile timing, prefer repeatable touched-file measurements over no-op hot-cache reruns:
scripts/bench_compile.sh check --runs 3 --touch src/server.rs
scripts/bench_compile.sh check --runs 3 --touch src/tool/read.rs
scripts/bench_compile.sh release-jcode --runs 3
scripts/bench_compile.sh selfdev-jcode --runs 3
scripts/bench_compile.sh build -- --package jcode --bin test_api
scripts/bench_selfdev_checkpoints.sh --touch src/server.rs --runs 3bench_compile.sh now supports:
--runs <n>for repeated timings with min/median/avg/max summaries--touch <path>to simulate a local edit before each timed run--jsonfor scriptable output-- <extra cargo args>to narrow the measured target/package/bin/features
bench_selfdev_checkpoints.sh builds on that foundation to produce a single standard
self-dev checkpoint bundle for cold/warm check + build comparisons.
After each structural phase we should re-measure and ask:
- Did warm
checktime improve materially? - Did warm
build/ reload-oriented build time improve materially? - Did we reduce rebuild scope for common self-dev edits?
If not, we should avoid continuing high-churn refactors on compile-time grounds alone.