pub struct AppContext {
pub harness: RefCell<Option<Harness>>,
/* private fields */
}Expand description
Shared application context threaded through all command handlers.
Holds the language provider, backup/checkpoint stores, configuration,
and call graph engine. Constructed once at startup and passed by
reference to dispatch.
Stores use RefCell for interior mutability — the binary is single-threaded
(one request at a time on the stdin read loop) so runtime borrow checking
is safe and never contended.
Fields§
§harness: RefCell<Option<Harness>>Implementations§
Source§impl AppContext
impl AppContext
pub fn build_status_snapshot(&self) -> StatusPayload
pub fn build_status_snapshot_for_session( &self, session_id: &str, ) -> StatusPayload
Source§impl AppContext
impl AppContext
pub fn new(provider: Box<dyn LanguageProvider>, config: Config) -> Self
Sourcepub fn status_bar_counts(&self) -> Option<StatusBarCounts>
pub fn status_bar_counts(&self) -> Option<StatusBarCounts>
Current agent status-bar counts. errors/warnings are read LIVE from
the LSP diagnostics store (continuously drained, no round-trip); the
Tier-2 + todos counts are the last-known cached values. Returns None
until the Tier-2 cache has been populated at least once, so we never
surface a bar that misleadingly claims “0 dead code” before any scan.
Sourcepub fn clear_tsconfig_membership_cache(&self)
pub fn clear_tsconfig_membership_cache(&self)
Invalidate the status-bar tsconfig-membership cache. Called from the
watcher seam when a tsconfig-like file changes and from configure
when the project root changes, so the next bar count re-reads from disk.
Sourcepub fn mark_status_bar_tier2_stale(&self) -> bool
pub fn mark_status_bar_tier2_stale(&self) -> bool
Mark the status-bar Tier-2 counts stale (rendered with ~) without
changing the numbers — called when the watcher sees a source-file change,
so the bar honestly signals the counts predate the latest edit until the
next background scan completes. Returns true only when the visible stale
bit flips. No-op before the first populate.
Sourcepub fn update_status_bar_tier2(
&self,
dead_code: Option<usize>,
unused_exports: Option<usize>,
duplicates: Option<usize>,
todos: Option<usize>,
stale: bool,
)
pub fn update_status_bar_tier2( &self, dead_code: Option<usize>, unused_exports: Option<usize>, duplicates: Option<usize>, todos: Option<usize>, stale: bool, )
Refresh the cached Tier-2 + todos counts for the status bar. Each count
is Option: None preserves the last-known value (the category wasn’t
recomputed or has no real aggregate yet) so we never overwrite a real
count with a fabricated 0. stale marks the Tier-2 numbers as
not-yet-reconciled with the latest edits.
Sourcepub fn gitignore(&self) -> Option<Arc<Gitignore>>
pub fn gitignore(&self) -> Option<Arc<Gitignore>>
Borrow the cached project gitignore matcher. Returns None when no
project_root is configured or when the project has no gitignore files.
Sourcepub fn clear_gitignore(&self)
pub fn clear_gitignore(&self)
Rebuild the gitignore matcher from the current project_root and
cache it. Called by the configure handler whenever the project root
changes, and by the watcher event drain when a .gitignore file
itself is modified.
The builder honors:
<project_root>/.gitignore- Git’s global excludes file (the same source used by
ignore::WalkBuilder) - the repository’s real
info/excludefile, resolved through Git’s common dir for linked worktrees - nested
.gitignorefiles (each.gitignorediscovered during the recursive walk)
Stores None if there’s no project_root or no matchable gitignore
files. Logs build errors but never fails configure.
Clear any cached gitignore matcher without rebuilding.
Used by handle_configure in degraded mode (e.g. project_root == $HOME)
where running the gitignore-discovery walk would exceed the configure
budget. The watcher event filter falls back to the hardcoded infra-dir
skip list when no matcher is present.
pub fn rebuild_gitignore(&self)
Sourcepub fn bash_compress_flag(&self) -> Arc<AtomicBool> ⓘ
pub fn bash_compress_flag(&self) -> Arc<AtomicBool> ⓘ
Shared atomic mirror of experimental.bash.compress. Updated by the
configure handler. Read by the BgTaskRegistry compressor closure.
Sourcepub fn sync_bash_compress_flag(&self)
pub fn sync_bash_compress_flag(&self)
Update the shared bash_compress_flag mirror. Call this from the
configure handler whenever experimental.bash.compress changes so the
BgTaskRegistry watchdog sees the new value on the next completion.
pub fn set_bash_compress_enabled(&self, enabled: bool)
Sourcepub fn filter_registry(&self) -> RwLockReadGuard<'_, FilterRegistry>
pub fn filter_registry(&self) -> RwLockReadGuard<'_, FilterRegistry>
Read-only access to the TOML filter registry, building it lazily on
first use. Returns an RwLockReadGuard that callers can lookup
against directly.
Returns the shared Arc<RwLock<FilterRegistry>> handle so threads
outside AppContext (notably the bash watchdog) can read it without
touching the rest of the context.
Sourcepub fn reset_filter_registry(&self)
pub fn reset_filter_registry(&self)
Force a fresh load of the TOML filter registry. Called when configure
changes the project root, storage_dir, or trust state so subsequent
compress::compress calls pick up new filters.
Sourcepub fn lsp_child_registry(&self) -> LspChildRegistry
pub fn lsp_child_registry(&self) -> LspChildRegistry
Clone the LSP child registry handle. Used by main.rs to give the signal handler thread a way to SIGKILL LSP children on shutdown.
pub fn stdout_writer(&self) -> SharedStdoutWriter
pub fn set_progress_sender(&self, sender: Option<ProgressSender>)
pub fn emit_progress(&self, frame: ProgressFrame)
pub fn status_emitter(&self) -> &StatusEmitter
Sourcepub fn progress_sender_handle(&self) -> Option<ProgressSender>
pub fn progress_sender_handle(&self) -> Option<ProgressSender>
Get a clone of the current progress sender for use from background
threads. Returns None when the main loop hasn’t installed one (tests,
CLI without push frames).
Used by configure’s deferred file-walk thread to push warnings after
configure has already returned, so configure latency stays sub-100 ms
even on huge directories.
pub fn advance_configure_generation(&self) -> u64
pub fn configure_generation(&self) -> u64
pub fn configure_warnings_sender(&self) -> Sender<(u64, ConfigureWarningsFrame)>
pub fn drain_configure_warnings(&self) -> Vec<(u64, ConfigureWarningsFrame)>
pub fn bash_background(&self) -> &BgTaskRegistry
pub fn drain_bg_completions(&self) -> Vec<BgCompletion>
Sourcepub fn provider(&self) -> &dyn LanguageProvider
pub fn provider(&self) -> &dyn LanguageProvider
Access the language provider.
Sourcepub fn backup(&self) -> &RefCell<BackupStore>
pub fn backup(&self) -> &RefCell<BackupStore>
Access the backup store.
Sourcepub fn checkpoint(&self) -> &RefCell<CheckpointStore>
pub fn checkpoint(&self) -> &RefCell<CheckpointStore>
Access the checkpoint store.
pub fn set_db(&self, conn: Arc<Mutex<Connection>>)
pub fn clear_db(&self)
pub fn db(&self) -> Option<Arc<Mutex<Connection>>>
Sourcepub fn config_mut(&self) -> RefMut<'_, Config>
pub fn config_mut(&self) -> RefMut<'_, Config>
Access the configuration (mutable borrow).
pub fn set_harness(&self, harness: Harness)
pub fn harness_opt(&self) -> Option<Harness>
pub fn harness(&self) -> Harness
pub fn storage_dir(&self) -> PathBuf
pub fn harness_dir(&self) -> PathBuf
pub fn inspect_dir(&self) -> PathBuf
pub fn bash_tasks_dir(&self, session_id: &str) -> PathBuf
pub fn backups_dir(&self, session_id: &str, path_hash: &str) -> PathBuf
pub fn filters_dir(&self) -> PathBuf
Sourcepub fn trust_file(&self) -> PathBuf
pub fn trust_file(&self) -> PathBuf
HOST-GLOBAL — NOT under harness_dir. Read by trust.rs across both harnesses.
pub fn set_canonical_cache_root(&self, root: PathBuf)
pub fn canonical_cache_root(&self) -> PathBuf
pub fn canonical_cache_root_opt(&self) -> Option<PathBuf>
pub fn set_cache_role( &self, is_worktree_bridge: bool, git_common_dir: Option<PathBuf>, )
pub fn is_worktree_bridge(&self) -> bool
pub fn git_common_dir(&self) -> Option<PathBuf>
Sourcepub fn set_degraded_reasons(&self, reasons: Vec<String>)
pub fn set_degraded_reasons(&self, reasons: Vec<String>)
Replace the current degraded-mode reasons. Empty vec = full-featured
mode (no degradation). Called by handle_configure after deciding
which subsystems to disable for this project root.
pub fn add_degraded_reason(&self, reason: impl Into<String>) -> bool
Sourcepub fn degraded_reasons(&self) -> Vec<String>
pub fn degraded_reasons(&self) -> Vec<String>
Snapshot of current degraded-mode reasons. Order is stable
(insertion order from set_degraded_reasons) so UI rendering and
snapshot diffs are deterministic.
Sourcepub fn is_degraded(&self) -> bool
pub fn is_degraded(&self) -> bool
True iff at least one degraded reason is recorded.
pub fn cache_role(&self) -> &'static str
Sourcepub fn callgraph_store(&self) -> &RefCell<Option<CallGraphStore>>
pub fn callgraph_store(&self) -> &RefCell<Option<CallGraphStore>>
Access the persisted call graph store.
pub fn mark_callgraph_store_force_rebuild(&self)
pub fn callgraph_store_dir(&self) -> PathBuf
pub fn ensure_callgraph_store( &self, ) -> Result<Option<RefMut<'_, CallGraphStore>>, CallGraphStoreError>
Sourcepub fn revalidate_callgraph_store_generation(&self)
pub fn revalidate_callgraph_store_generation(&self)
Access the persisted callgraph store for the five store-backed edge-query ops without ever blocking the request thread on a cold build.
- Store resident ->
Ready. - Warm on-disk DB present -> opened synchronously (cheap) ->
Ready. - Genuine cold build needed -> kicked off in the background, returns
Building; the watcher keeps the store fresh once it lands. - Worktree without a built store, or not configured ->
Unavailable.
A build already in flight (callgraph_store_rx set) also returns
Building without starting a second build.
Drop the resident callgraph store when another process (or a local cold
rebuild) has published a newer generation, so the next access reopens via
the pointer. No-op when no store is resident, a build is in flight, or the
store is still current. Must run before serving ops AND before any
incremental write, so every process converges on the current generation
rather than writing to a stale one.
pub fn callgraph_store_for_ops(&self) -> CallgraphStoreAccess<'_>
Sourcepub fn callgraph_store_rx(&self) -> &RefCell<Option<Receiver<CallGraphStore>>>
pub fn callgraph_store_rx(&self) -> &RefCell<Option<Receiver<CallGraphStore>>>
Access the callgraph-store background-build receiver (drained by the main loop once the cold build completes).
Sourcepub fn add_pending_callgraph_store_paths<I>(&self, paths: I)where
I: IntoIterator<Item = PathBuf>,
pub fn add_pending_callgraph_store_paths<I>(&self, paths: I)where
I: IntoIterator<Item = PathBuf>,
Record source-file paths that changed while a cold build was in flight, so they can be refreshed once the freshly-built store is installed.
Sourcepub fn take_pending_callgraph_store_paths(&self) -> Vec<PathBuf>
pub fn take_pending_callgraph_store_paths(&self) -> Vec<PathBuf>
Take and clear the paths that changed during a background cold build.
Sourcepub fn search_index(&self) -> &RefCell<Option<SearchIndex>>
pub fn search_index(&self) -> &RefCell<Option<SearchIndex>>
Access the search index.
Sourcepub fn search_index_rx(&self) -> &RefCell<Option<Receiver<SearchIndex>>>
pub fn search_index_rx(&self) -> &RefCell<Option<Receiver<SearchIndex>>>
Access the search-index build receiver.
pub fn add_pending_search_index_paths<I>(&self, paths: I)where
I: IntoIterator<Item = PathBuf>,
pub fn take_pending_search_index_paths(&self) -> Vec<PathBuf>
pub fn add_pending_semantic_index_paths<I>(&self, paths: I)where
I: IntoIterator<Item = PathBuf>,
pub fn take_pending_semantic_index_paths(&self) -> Vec<PathBuf>
pub fn mark_pending_semantic_corpus_refresh(&self)
pub fn take_pending_semantic_corpus_refresh(&self) -> bool
pub fn clear_pending_index_updates(&self)
pub fn inspect_manager(&self) -> Arc<InspectManager> ⓘ
Sourcepub fn take_new_reuse_completions(&self) -> bool
pub fn take_new_reuse_completions(&self) -> bool
Returns true when one or more watcher-driven (reuse-path) Tier-2 scans
have completed since the last call, advancing the last-seen marker. The
per-request inspect drain uses this to refresh the status bar after a
background scan — those completions bypass drain_completions.
pub fn reset_tier2_refresh_scheduler(&self)
pub fn request_tier2_refresh_pull(&self) -> bool
pub fn tick_tier2_refresh_scheduler( &self, changed_path_count: usize, ) -> Option<Tier2TriggerReason>
pub fn note_tier2_refresh_started(&self)
pub fn tier2_trigger_reason(&self) -> Option<&'static str>
Sourcepub fn symbol_cache(&self) -> SharedSymbolCache
pub fn symbol_cache(&self) -> SharedSymbolCache
Access the shared symbol cache.
Sourcepub fn reset_symbol_cache(&self) -> u64
pub fn reset_symbol_cache(&self) -> u64
Clear the shared symbol cache and return the new active generation.
Sourcepub fn semantic_index(&self) -> &RefCell<Option<SemanticIndex>>
pub fn semantic_index(&self) -> &RefCell<Option<SemanticIndex>>
Access the semantic search index.
Sourcepub fn semantic_index_rx(
&self,
) -> &RefCell<Option<Receiver<SemanticIndexEvent>>>
pub fn semantic_index_rx( &self, ) -> &RefCell<Option<Receiver<SemanticIndexEvent>>>
Access the semantic-index build receiver.
pub fn semantic_index_status(&self) -> &RefCell<SemanticIndexStatus>
pub fn install_semantic_refresh_worker( &self, sender: Sender<SemanticRefreshRequest>, event_rx: Receiver<SemanticRefreshEvent>, worker_slot: SemanticRefreshWorkerSlot, )
pub fn clear_semantic_refresh_worker(&self)
pub fn semantic_refresh_sender(&self) -> Option<Sender<SemanticRefreshRequest>>
pub fn semantic_refresh_event_rx( &self, ) -> &RefCell<Option<Receiver<SemanticRefreshEvent>>>
Sourcepub fn semantic_embedding_model(&self) -> &RefCell<Option<EmbeddingModel>>
pub fn semantic_embedding_model(&self) -> &RefCell<Option<EmbeddingModel>>
Access the cached semantic embedding model.
Sourcepub fn watcher(&self) -> &RefCell<Option<RecommendedWatcher>>
pub fn watcher(&self) -> &RefCell<Option<RecommendedWatcher>>
Access the file watcher handle (kept alive to continue watching).
Sourcepub fn watcher_rx(&self) -> &RefCell<Option<Receiver<Result<Event>>>>
pub fn watcher_rx(&self) -> &RefCell<Option<Receiver<Result<Event>>>>
Access the watcher event receiver.
Sourcepub fn lsp(&self) -> RefMut<'_, LspManager>
pub fn lsp(&self) -> RefMut<'_, LspManager>
Access the LSP manager.
Sourcepub fn lsp_notify_file_changed(&self, file_path: &Path, content: &str)
pub fn lsp_notify_file_changed(&self, file_path: &Path, content: &str)
Notify LSP servers that a file was written. Call this after write_format_validate in command handlers.
Sourcepub fn lsp_clear_diagnostics_for_file(&self, file_path: &Path) -> bool
pub fn lsp_clear_diagnostics_for_file(&self, file_path: &Path) -> bool
Drop cached LSP diagnostics for a deleted/renamed-away file so its
errors/warnings don’t linger in the warm set (no server republishes for
a vanished path), keeping the status bar and aft_inspect honest.
Returns true if any entry was removed. Best-effort: a contended borrow is
skipped silently (the watcher drain retries on subsequent events).
Sourcepub fn lsp_notify_and_collect_diagnostics(
&self,
file_path: &Path,
content: &str,
timeout: Duration,
) -> PostEditWaitOutcome
pub fn lsp_notify_and_collect_diagnostics( &self, file_path: &Path, content: &str, timeout: Duration, ) -> PostEditWaitOutcome
Notify LSP and optionally wait for diagnostics.
Call this after write_format_validate when the request has "diagnostics": true.
Sends didChange to the server, waits briefly for publishDiagnostics, and returns
any diagnostics for the file. If no server is running, returns empty immediately.
v0.17.3: this is the version-aware path. Pre-edit cached diagnostics
are NEVER returned — only entries whose version matches the
post-edit document version (or, for unversioned servers, whose
epoch advanced past the pre-edit snapshot).
pub fn lsp_notify_watched_config_file( &self, file_path: &Path, change_type: FileChangeType, )
Sourcepub fn lsp_post_multi_file_write(
&self,
file_path: &Path,
content: &str,
file_paths: &[PathBuf],
params: &Value,
) -> Option<PostEditWaitOutcome>
pub fn lsp_post_multi_file_write( &self, file_path: &Path, content: &str, file_paths: &[PathBuf], params: &Value, ) -> Option<PostEditWaitOutcome>
Post-write LSP hook for multi-file edits. When the patch includes
config-file edits, notify active workspace servers via
workspace/didChangeWatchedFiles before sending the per-document
didOpen/didChange for the current file.
Sourcepub fn lsp_post_write(
&self,
file_path: &Path,
content: &str,
params: &Value,
) -> Option<PostEditWaitOutcome>
pub fn lsp_post_write( &self, file_path: &Path, content: &str, params: &Value, ) -> Option<PostEditWaitOutcome>
Post-write LSP hook: notify server and optionally collect diagnostics.
This is the single call site for all command handlers after write_format_validate.
Behavior:
- When
diagnostics: trueis inparams, notifies the server, waits until matching diagnostics arrive or the timeout expires, and returnsSome(outcome)with the verified-fresh diagnostics + per-server status. - When
diagnostics: false(or absent), just notifies (fire-and-forget) and returnsNone. Callers must NOT wrap this inSome(...); theNoneis what tells the response builder to omit the LSP fields entirely (preserves the no-diagnostics-requested response shape).
v0.17.3: default wait_ms raised from 1500 to 3000 because real-world
tsserver re-analysis on monorepo files routinely takes 2-5s. Still
capped at 10000ms.
Sourcepub fn validate_path(
&self,
req_id: &str,
path: &Path,
) -> Result<PathBuf, Response>
pub fn validate_path( &self, req_id: &str, path: &Path, ) -> Result<PathBuf, Response>
Validate that a file path falls within the configured project root.
When project_root is configured (normal plugin usage), this resolves the
path and checks it starts with the root. Returns the canonicalized path on
success, or an error response on violation.
When no project_root is configured (direct CLI usage), all paths pass
through unrestricted for backward compatibility.
Sourcepub fn lsp_server_count(&self) -> usize
pub fn lsp_server_count(&self) -> usize
Count active LSP server instances.
Sourcepub fn symbol_cache_stats(&self) -> Value
pub fn symbol_cache_stats(&self) -> Value
Symbol cache statistics from the language provider.
Auto Trait Implementations§
impl !Freeze for AppContext
impl !RefUnwindSafe for AppContext
impl !Send for AppContext
impl !Sync for AppContext
impl !UnwindSafe for AppContext
impl Unpin for AppContext
impl UnsafeUnpin for AppContext
Blanket Implementations§
Source§impl<T> BorrowMut<T> for Twhere
T: ?Sized,
impl<T> BorrowMut<T> for Twhere
T: ?Sized,
Source§fn borrow_mut(&mut self) -> &mut T
fn borrow_mut(&mut self) -> &mut T
Source§impl<T> Downcast for Twhere
T: Any,
impl<T> Downcast for Twhere
T: Any,
Source§fn into_any(self: Box<T>) -> Box<dyn Any>
fn into_any(self: Box<T>) -> Box<dyn Any>
Box<dyn Trait> (where Trait: Downcast) to Box<dyn Any>. Box<dyn Any> can
then be further downcast into Box<ConcreteType> where ConcreteType implements Trait.Source§fn into_any_rc(self: Rc<T>) -> Rc<dyn Any>
fn into_any_rc(self: Rc<T>) -> Rc<dyn Any>
Rc<Trait> (where Trait: Downcast) to Rc<Any>. Rc<Any> can then be
further downcast into Rc<ConcreteType> where ConcreteType implements Trait.Source§fn as_any(&self) -> &(dyn Any + 'static)
fn as_any(&self) -> &(dyn Any + 'static)
&Trait (where Trait: Downcast) to &Any. This is needed since Rust cannot
generate &Any’s vtable from &Trait’s.Source§fn as_any_mut(&mut self) -> &mut (dyn Any + 'static)
fn as_any_mut(&mut self) -> &mut (dyn Any + 'static)
&mut Trait (where Trait: Downcast) to &Any. This is needed since Rust cannot
generate &mut Any’s vtable from &mut Trait’s.Source§impl<T> Instrument for T
impl<T> Instrument for T
Source§fn instrument(self, span: Span) -> Instrumented<Self>
fn instrument(self, span: Span) -> Instrumented<Self>
Source§fn in_current_span(self) -> Instrumented<Self>
fn in_current_span(self) -> Instrumented<Self>
Source§impl<T> IntoEither for T
impl<T> IntoEither for T
Source§fn into_either(self, into_left: bool) -> Either<Self, Self>
fn into_either(self, into_left: bool) -> Either<Self, Self>
self into a Left variant of Either<Self, Self>
if into_left is true.
Converts self into a Right variant of Either<Self, Self>
otherwise. Read moreSource§fn into_either_with<F>(self, into_left: F) -> Either<Self, Self>
fn into_either_with<F>(self, into_left: F) -> Either<Self, Self>
self into a Left variant of Either<Self, Self>
if into_left(&self) returns true.
Converts self into a Right variant of Either<Self, Self>
otherwise. Read more