Skip to main content

aft/
lib.rs

1#![allow(
2    clippy::collapsible_if,
3    clippy::collapsible_match,
4    clippy::double_ended_iterator_last,
5    clippy::int_plus_one,
6    clippy::large_enum_variant,
7    clippy::len_without_is_empty,
8    clippy::let_and_return,
9    clippy::manual_contains,
10    clippy::manual_pattern_char_comparison,
11    clippy::manual_repeat_n,
12    clippy::manual_strip,
13    clippy::manual_unwrap_or_default,
14    clippy::map_clone,
15    clippy::iter_kv_map,
16    clippy::needless_borrow,
17    clippy::needless_borrows_for_generic_args,
18    clippy::needless_range_loop,
19    clippy::new_without_default,
20    clippy::obfuscated_if_else,
21    clippy::ptr_arg,
22    clippy::question_mark,
23    clippy::same_item_push,
24    clippy::should_implement_trait,
25    clippy::single_match,
26    clippy::too_many_arguments,
27    clippy::type_complexity,
28    clippy::unnecessary_sort_by,
29    clippy::unnecessary_cast,
30    clippy::unnecessary_lazy_evaluations,
31    clippy::unnecessary_map_or
32)]
33
34// ## Note on `.unwrap()` / `.expect()` usage
35//
36// The remaining `.unwrap()` and `.expect()` calls in `src/` are in:
37// - **Tree-sitter query operations** (parser.rs, zoom.rs, extract.rs, inline.rs,
38//   outline.rs): These operate on AFT's own compiled grammars and query patterns, which
39//   are compile-time constants. Pattern captures and node kinds are guaranteed to exist.
40// - **Checkpoint serialization** (checkpoint.rs): serde_json::to_value on known-good
41//   HashMap<PathBuf, String> types cannot fail.
42// - **lib.rs main loop**: JSON parsing of stdin lines — a malformed line is logged and
43//   skipped, not unwrapped.
44//
45// All production command handlers that process user/agent input return Result or
46// Response::error instead of panicking. Confirmed zero .unwrap()/.expect() in
47// production error paths as of v0.6.3 audit.
48
49pub mod ast_grep_hints;
50pub mod ast_grep_lang;
51pub mod backup;
52pub mod bash_background;
53pub mod bash_permissions;
54pub mod bash_rewrite;
55pub mod cache_freshness;
56pub mod callgraph;
57pub mod callgraph_store;
58pub mod calls;
59pub mod checkpoint;
60pub mod commands;
61pub mod compress;
62pub mod config;
63pub mod context;
64pub mod db;
65pub mod edit;
66pub mod error;
67pub mod extract;
68pub mod format;
69pub mod fs_lock;
70pub mod fuzzy_match;
71pub mod grep_executor;
72pub mod harness;
73pub mod imports;
74pub mod indent;
75pub mod inspect;
76pub mod language;
77pub mod local_embed;
78pub mod log_ctx;
79pub mod lsp;
80pub mod lsp_hints;
81pub mod migrate_storage;
82pub mod parser;
83pub mod pattern_compile;
84pub mod protocol;
85pub mod query_shape;
86pub mod search_index;
87pub mod semantic_index;
88pub mod symbol_cache_disk;
89pub mod symbols;
90pub mod tool_path;
91pub mod url_fetch;
92pub mod watcher_filter;
93// Compiled on all platforms so cross-platform unit tests in
94// `commands::bash::try_spawn_with_fallback` can exercise the retry
95// decision logic without a real Windows runtime. The module itself only
96// uses portable APIs; only its callers are Windows-gated.
97pub mod windows_shell;
98
99#[cfg(test)]
100mod tests {
101    use super::*;
102    use config::Config;
103    use error::AftError;
104    use protocol::{RawRequest, Response};
105
106    // --- Protocol serialization ---
107
108    #[test]
109    fn raw_request_deserializes_ping() {
110        let json = r#"{"id":"1","command":"ping"}"#;
111        let req: RawRequest = serde_json::from_str(json).unwrap();
112        assert_eq!(req.id, "1");
113        assert_eq!(req.command, "ping");
114        assert!(req.lsp_hints.is_none());
115    }
116
117    #[test]
118    fn raw_request_deserializes_echo_with_params() {
119        let json = r#"{"id":"2","command":"echo","message":"hello"}"#;
120        let req: RawRequest = serde_json::from_str(json).unwrap();
121        assert_eq!(req.id, "2");
122        assert_eq!(req.command, "echo");
123        // "message" is captured in the flattened params
124        assert_eq!(req.params["message"], "hello");
125    }
126
127    #[test]
128    fn raw_request_preserves_unknown_fields() {
129        let json = r#"{"id":"3","command":"ping","future_field":"abc","nested":{"x":1}}"#;
130        let req: RawRequest = serde_json::from_str(json).unwrap();
131        assert_eq!(req.params["future_field"], "abc");
132        assert_eq!(req.params["nested"]["x"], 1);
133    }
134
135    #[test]
136    fn raw_request_with_lsp_hints() {
137        let json = r#"{"id":"4","command":"ping","lsp_hints":{"completions":["foo","bar"]}}"#;
138        let req: RawRequest = serde_json::from_str(json).unwrap();
139        assert!(req.lsp_hints.is_some());
140        let hints = req.lsp_hints.unwrap();
141        assert_eq!(hints["completions"][0], "foo");
142    }
143
144    #[test]
145    fn response_success_round_trip() {
146        let resp = Response::success("42", serde_json::json!({"command": "pong"}));
147        let json_str = serde_json::to_string(&resp).unwrap();
148        let v: serde_json::Value = serde_json::from_str(&json_str).unwrap();
149        assert_eq!(v["id"], "42");
150        assert_eq!(v["success"], true);
151        assert_eq!(v["command"], "pong");
152    }
153
154    #[test]
155    fn response_error_round_trip() {
156        let resp = Response::error("99", "unknown_command", "unknown command: foo");
157        let json_str = serde_json::to_string(&resp).unwrap();
158        let v: serde_json::Value = serde_json::from_str(&json_str).unwrap();
159        assert_eq!(v["id"], "99");
160        assert_eq!(v["success"], false);
161        assert_eq!(v["code"], "unknown_command");
162        assert_eq!(v["message"], "unknown command: foo");
163    }
164
165    // --- Error formatting ---
166
167    #[test]
168    fn error_display_symbol_not_found() {
169        let err = AftError::SymbolNotFound {
170            name: "foo".into(),
171            file: "bar.rs".into(),
172        };
173        assert_eq!(err.to_string(), "symbol 'foo' not found in bar.rs");
174        assert_eq!(err.code(), "symbol_not_found");
175    }
176
177    #[test]
178    fn error_display_ambiguous_symbol() {
179        let err = AftError::AmbiguousSymbol {
180            name: "Foo".into(),
181            candidates: vec!["a.rs:10".into(), "b.rs:20".into()],
182        };
183        let s = err.to_string();
184        assert!(s.contains("Foo"));
185        assert!(s.contains("a.rs:10, b.rs:20"));
186    }
187
188    #[test]
189    fn error_display_parse_error() {
190        let err = AftError::ParseError {
191            message: "unexpected token".into(),
192        };
193        assert_eq!(err.to_string(), "parse error: unexpected token");
194    }
195
196    #[test]
197    fn error_display_file_not_found() {
198        let err = AftError::FileNotFound {
199            path: "/tmp/missing.rs".into(),
200        };
201        assert_eq!(err.to_string(), "file not found: /tmp/missing.rs");
202    }
203
204    #[test]
205    fn error_display_invalid_request() {
206        let err = AftError::InvalidRequest {
207            message: "missing field".into(),
208        };
209        assert_eq!(err.to_string(), "invalid request: missing field");
210    }
211
212    #[test]
213    fn error_display_checkpoint_not_found() {
214        let err = AftError::CheckpointNotFound {
215            name: "pre-refactor".into(),
216        };
217        assert_eq!(err.to_string(), "checkpoint not found: pre-refactor");
218        assert_eq!(err.code(), "checkpoint_not_found");
219    }
220
221    #[test]
222    fn error_display_no_undo_history() {
223        let err = AftError::NoUndoHistory {
224            path: "src/main.rs".into(),
225        };
226        assert_eq!(err.to_string(), "no undo history for: src/main.rs");
227        assert_eq!(err.code(), "no_undo_history");
228    }
229
230    #[test]
231    fn error_display_ambiguous_match() {
232        let err = AftError::AmbiguousMatch {
233            pattern: "TODO".into(),
234            count: 5,
235        };
236        assert_eq!(
237            err.to_string(),
238            "pattern 'TODO' matches 5 occurrences, expected exactly 1"
239        );
240        assert_eq!(err.code(), "ambiguous_match");
241    }
242
243    #[test]
244    fn error_display_project_too_large() {
245        let err = AftError::ProjectTooLarge {
246            count: 20001,
247            max: 20000,
248        };
249        assert_eq!(
250            err.to_string(),
251            "project has 20001 source files, exceeding max_callgraph_files=20000. Legacy in-memory call-graph operations (trace_data, dead_code snapshots, and symbol move analysis) are disabled for this root. Open a specific subdirectory or raise max_callgraph_files in config."
252        );
253        assert_eq!(err.code(), "project_too_large");
254    }
255
256    #[test]
257    fn error_to_json_has_code_and_message() {
258        let err = AftError::FileNotFound { path: "/x".into() };
259        let j = err.to_error_json();
260        assert_eq!(j["code"], "file_not_found");
261        assert!(j["message"].as_str().unwrap().contains("/x"));
262    }
263
264    // --- Config defaults ---
265
266    #[test]
267    fn config_default_values() {
268        let cfg = Config::default();
269        assert!(cfg.project_root.is_none());
270        assert_eq!(cfg.validation_depth, 1);
271        assert_eq!(cfg.checkpoint_ttl_hours, 24);
272        assert_eq!(cfg.max_symbol_depth, 10);
273        assert_eq!(cfg.formatter_timeout_secs, 10);
274        assert_eq!(cfg.type_checker_timeout_secs, 30);
275        assert_eq!(cfg.max_callgraph_files, 5_000);
276    }
277}