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
34pub 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 calls;
58pub mod checkpoint;
59pub mod commands;
60pub mod compress;
61pub mod config;
62pub mod context;
63pub mod db;
64pub mod edit;
65pub mod error;
66pub mod extract;
67pub mod format;
68pub mod fs_lock;
69pub mod fuzzy_match;
70pub mod grep_executor;
71pub mod harness;
72pub mod imports;
73pub mod indent;
74pub mod language;
75pub mod log_ctx;
76pub mod lsp;
77pub mod lsp_hints;
78pub mod migrate_storage;
79pub mod parser;
80pub mod pattern_compile;
81pub mod protocol;
82pub mod query_shape;
83pub mod search_index;
84pub mod semantic_index;
85pub mod symbol_cache_disk;
86pub mod symbols;
87pub mod url_fetch;
88pub mod windows_shell;
93
94#[cfg(test)]
95mod tests {
96 use super::*;
97 use config::Config;
98 use error::AftError;
99 use protocol::{RawRequest, Response};
100
101 #[test]
104 fn raw_request_deserializes_ping() {
105 let json = r#"{"id":"1","command":"ping"}"#;
106 let req: RawRequest = serde_json::from_str(json).unwrap();
107 assert_eq!(req.id, "1");
108 assert_eq!(req.command, "ping");
109 assert!(req.lsp_hints.is_none());
110 }
111
112 #[test]
113 fn raw_request_deserializes_echo_with_params() {
114 let json = r#"{"id":"2","command":"echo","message":"hello"}"#;
115 let req: RawRequest = serde_json::from_str(json).unwrap();
116 assert_eq!(req.id, "2");
117 assert_eq!(req.command, "echo");
118 assert_eq!(req.params["message"], "hello");
120 }
121
122 #[test]
123 fn raw_request_preserves_unknown_fields() {
124 let json = r#"{"id":"3","command":"ping","future_field":"abc","nested":{"x":1}}"#;
125 let req: RawRequest = serde_json::from_str(json).unwrap();
126 assert_eq!(req.params["future_field"], "abc");
127 assert_eq!(req.params["nested"]["x"], 1);
128 }
129
130 #[test]
131 fn raw_request_with_lsp_hints() {
132 let json = r#"{"id":"4","command":"ping","lsp_hints":{"completions":["foo","bar"]}}"#;
133 let req: RawRequest = serde_json::from_str(json).unwrap();
134 assert!(req.lsp_hints.is_some());
135 let hints = req.lsp_hints.unwrap();
136 assert_eq!(hints["completions"][0], "foo");
137 }
138
139 #[test]
140 fn response_success_round_trip() {
141 let resp = Response::success("42", serde_json::json!({"command": "pong"}));
142 let json_str = serde_json::to_string(&resp).unwrap();
143 let v: serde_json::Value = serde_json::from_str(&json_str).unwrap();
144 assert_eq!(v["id"], "42");
145 assert_eq!(v["success"], true);
146 assert_eq!(v["command"], "pong");
147 }
148
149 #[test]
150 fn response_error_round_trip() {
151 let resp = Response::error("99", "unknown_command", "unknown command: foo");
152 let json_str = serde_json::to_string(&resp).unwrap();
153 let v: serde_json::Value = serde_json::from_str(&json_str).unwrap();
154 assert_eq!(v["id"], "99");
155 assert_eq!(v["success"], false);
156 assert_eq!(v["code"], "unknown_command");
157 assert_eq!(v["message"], "unknown command: foo");
158 }
159
160 #[test]
163 fn error_display_symbol_not_found() {
164 let err = AftError::SymbolNotFound {
165 name: "foo".into(),
166 file: "bar.rs".into(),
167 };
168 assert_eq!(err.to_string(), "symbol 'foo' not found in bar.rs");
169 assert_eq!(err.code(), "symbol_not_found");
170 }
171
172 #[test]
173 fn error_display_ambiguous_symbol() {
174 let err = AftError::AmbiguousSymbol {
175 name: "Foo".into(),
176 candidates: vec!["a.rs:10".into(), "b.rs:20".into()],
177 };
178 let s = err.to_string();
179 assert!(s.contains("Foo"));
180 assert!(s.contains("a.rs:10, b.rs:20"));
181 }
182
183 #[test]
184 fn error_display_parse_error() {
185 let err = AftError::ParseError {
186 message: "unexpected token".into(),
187 };
188 assert_eq!(err.to_string(), "parse error: unexpected token");
189 }
190
191 #[test]
192 fn error_display_file_not_found() {
193 let err = AftError::FileNotFound {
194 path: "/tmp/missing.rs".into(),
195 };
196 assert_eq!(err.to_string(), "file not found: /tmp/missing.rs");
197 }
198
199 #[test]
200 fn error_display_invalid_request() {
201 let err = AftError::InvalidRequest {
202 message: "missing field".into(),
203 };
204 assert_eq!(err.to_string(), "invalid request: missing field");
205 }
206
207 #[test]
208 fn error_display_checkpoint_not_found() {
209 let err = AftError::CheckpointNotFound {
210 name: "pre-refactor".into(),
211 };
212 assert_eq!(err.to_string(), "checkpoint not found: pre-refactor");
213 assert_eq!(err.code(), "checkpoint_not_found");
214 }
215
216 #[test]
217 fn error_display_no_undo_history() {
218 let err = AftError::NoUndoHistory {
219 path: "src/main.rs".into(),
220 };
221 assert_eq!(err.to_string(), "no undo history for: src/main.rs");
222 assert_eq!(err.code(), "no_undo_history");
223 }
224
225 #[test]
226 fn error_display_ambiguous_match() {
227 let err = AftError::AmbiguousMatch {
228 pattern: "TODO".into(),
229 count: 5,
230 };
231 assert_eq!(
232 err.to_string(),
233 "pattern 'TODO' matches 5 occurrences, expected exactly 1"
234 );
235 assert_eq!(err.code(), "ambiguous_match");
236 }
237
238 #[test]
239 fn error_display_project_too_large() {
240 let err = AftError::ProjectTooLarge {
241 count: 20001,
242 max: 20000,
243 };
244 assert_eq!(
245 err.to_string(),
246 "project has 20001 source files, exceeding max_callgraph_files=20000. Call-graph operations (callers, trace_to, trace_to_symbol, trace_data, impact) are disabled for this root. Open a specific subdirectory or raise max_callgraph_files in config."
247 );
248 assert_eq!(err.code(), "project_too_large");
249 }
250
251 #[test]
252 fn error_to_json_has_code_and_message() {
253 let err = AftError::FileNotFound { path: "/x".into() };
254 let j = err.to_error_json();
255 assert_eq!(j["code"], "file_not_found");
256 assert!(j["message"].as_str().unwrap().contains("/x"));
257 }
258
259 #[test]
262 fn config_default_values() {
263 let cfg = Config::default();
264 assert!(cfg.project_root.is_none());
265 assert_eq!(cfg.validation_depth, 1);
266 assert_eq!(cfg.checkpoint_ttl_hours, 24);
267 assert_eq!(cfg.max_symbol_depth, 10);
268 assert_eq!(cfg.formatter_timeout_secs, 10);
269 assert_eq!(cfg.type_checker_timeout_secs, 30);
270 assert_eq!(cfg.max_callgraph_files, 5_000);
271 }
272}