1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
//! Background compinit pre-warm — extension; no zsh C counterpart.
use crate::compsys::cache::CompsysCache;
use crate::compsys::CompInitResult;
#[allow(unused_imports)]
use crate::ported::vm_helper::ShellExecutor;
#[allow(unused_imports)]
use std::{collections::HashMap, env, path::PathBuf};
/// Result from background compinit thread.
/// Outcome of background `compinit` autoload.
/// zshrs-original — Src/Modules/complete.c blocks on `compinit`
/// inline. The Rust port runs it on the worker pool.
pub struct CompInitBgResult {
/// `result` field.
pub result: CompInitResult,
/// `cache` field.
pub cache: CompsysCache,
}
// ===========================================================
// Methods moved verbatim from src/ported/vm_helper because their
// C counterpart's source file maps 1:1 to this Rust module.
// Phase: drift
// ===========================================================
// BEGIN moved-from-exec-rs
impl crate::ported::vm_helper::ShellExecutor {
/// Non-blocking drain of background compinit results.
/// Call this before any completion lookup (prompt, tab-complete, etc.).
/// If the background thread hasn't finished yet, this is a no-op.
pub fn drain_compinit_bg(&mut self) {
if let Some((rx, start)) = self.compinit_pending.take() {
match rx.try_recv() {
Ok(bg) => {
let comps = bg.result.comps.len();
self.set_assoc("_comps".to_string(), bg.result.comps.into_iter().collect());
self.set_assoc(
"_services".to_string(),
bg.result.services.into_iter().collect(),
);
self.set_assoc(
"_patcomps".to_string(),
bg.result.patcomps.into_iter().collect(),
);
self.compsys_cache = Some(bg.cache);
tracing::info!(
wall_ms = start.elapsed().as_millis() as u64,
comps,
"compinit: background results merged"
);
}
Err(std::sync::mpsc::TryRecvError::Empty) => {
// Not ready yet — put the receiver back for next poll
self.compinit_pending = Some((rx, start));
}
Err(std::sync::mpsc::TryRecvError::Disconnected) => {
tracing::warn!("compinit: background thread died without sending results");
}
}
}
}
/// Traditional zsh compinit (--zsh-compat mode)
/// Uses fpath scanning, .zcompdump files, no SQLite
pub(crate) fn compinit_compat(
&mut self,
quiet: bool,
no_dump: bool,
dump_file: Option<String>,
use_cache: bool,
) -> i32 {
let zdotdir = self
.scalar("ZDOTDIR")
.or_else(|| std::env::var("ZDOTDIR").ok())
.unwrap_or_else(|| std::env::var("HOME").unwrap_or_else(|_| "/tmp".to_string()));
let dump_path = dump_file
.map(PathBuf::from)
.unwrap_or_else(|| PathBuf::from(&zdotdir).join(".zcompdump"));
// -C: Try to use existing .zcompdump if valid
if use_cache
&& dump_path.exists()
&& crate::compsys::check_dump(&dump_path, &self.fpath, "zshrs-0.1.0")
{
// Valid dump - source it to load _comps
// For now, just rescan (proper impl would source the dump file)
if !quiet {
tracing::info!("compinit: .zcompdump valid, rescanning for compat");
}
}
// Full fpath scan (traditional zsh algorithm)
let result = crate::compsys::compinit(&self.fpath);
if !quiet {
tracing::info!(
functions = result.files_scanned,
comps = result.comps.len(),
dirs = result.dirs_scanned,
ms = result.scan_time_ms,
"compinit: fpath scan complete"
);
}
// Write .zcompdump unless -D
if !no_dump {
let _ = crate::compsys::compdump(&result, &dump_path, "zshrs-0.1.0");
}
// Set up _comps associative array
self.set_assoc(
"_comps".to_string(),
result.comps.clone().into_iter().collect(),
);
self.set_assoc(
"_services".to_string(),
result.services.clone().into_iter().collect(),
);
self.set_assoc(
"_patcomps".to_string(),
result.patcomps.clone().into_iter().collect(),
);
// No SQLite cache in compat mode
self.compsys_cache = None;
0
}
}
// END moved-from-exec-rs