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