1#![allow(
2 clippy::collapsible_if,
3 clippy::double_ended_iterator_last,
4 clippy::int_plus_one,
5 clippy::large_enum_variant,
6 clippy::len_without_is_empty,
7 clippy::let_and_return,
8 clippy::manual_contains,
9 clippy::manual_pattern_char_comparison,
10 clippy::manual_repeat_n,
11 clippy::manual_strip,
12 clippy::manual_unwrap_or_default,
13 clippy::map_clone,
14 clippy::needless_borrow,
15 clippy::needless_borrows_for_generic_args,
16 clippy::needless_range_loop,
17 clippy::new_without_default,
18 clippy::obfuscated_if_else,
19 clippy::ptr_arg,
20 clippy::question_mark,
21 clippy::same_item_push,
22 clippy::should_implement_trait,
23 clippy::single_match,
24 clippy::too_many_arguments,
25 clippy::type_complexity,
26 clippy::unnecessary_cast,
27 clippy::unnecessary_lazy_evaluations,
28 clippy::unnecessary_map_or
29)]
30
31pub mod ast_grep_lang;
47pub mod backup;
48pub mod bash_background;
49pub mod bash_permissions;
50pub mod bash_rewrite;
51pub mod callgraph;
52pub mod calls;
53pub mod checkpoint;
54pub mod commands;
55pub mod compress;
56pub mod config;
57pub mod context;
58pub mod edit;
59pub mod error;
60pub mod extract;
61pub mod format;
62pub mod fuzzy_match;
63pub mod imports;
64pub mod indent;
65pub mod language;
66pub mod log_ctx;
67pub mod lsp;
68pub mod lsp_hints;
69pub mod parser;
70pub mod protocol;
71pub mod search_index;
72pub mod semantic_index;
73pub mod symbols;
74
75#[cfg(test)]
76mod tests {
77 use super::*;
78 use config::Config;
79 use error::AftError;
80 use protocol::{RawRequest, Response};
81
82 #[test]
85 fn raw_request_deserializes_ping() {
86 let json = r#"{"id":"1","command":"ping"}"#;
87 let req: RawRequest = serde_json::from_str(json).unwrap();
88 assert_eq!(req.id, "1");
89 assert_eq!(req.command, "ping");
90 assert!(req.lsp_hints.is_none());
91 }
92
93 #[test]
94 fn raw_request_deserializes_echo_with_params() {
95 let json = r#"{"id":"2","command":"echo","message":"hello"}"#;
96 let req: RawRequest = serde_json::from_str(json).unwrap();
97 assert_eq!(req.id, "2");
98 assert_eq!(req.command, "echo");
99 assert_eq!(req.params["message"], "hello");
101 }
102
103 #[test]
104 fn raw_request_preserves_unknown_fields() {
105 let json = r#"{"id":"3","command":"ping","future_field":"abc","nested":{"x":1}}"#;
106 let req: RawRequest = serde_json::from_str(json).unwrap();
107 assert_eq!(req.params["future_field"], "abc");
108 assert_eq!(req.params["nested"]["x"], 1);
109 }
110
111 #[test]
112 fn raw_request_with_lsp_hints() {
113 let json = r#"{"id":"4","command":"ping","lsp_hints":{"completions":["foo","bar"]}}"#;
114 let req: RawRequest = serde_json::from_str(json).unwrap();
115 assert!(req.lsp_hints.is_some());
116 let hints = req.lsp_hints.unwrap();
117 assert_eq!(hints["completions"][0], "foo");
118 }
119
120 #[test]
121 fn response_success_round_trip() {
122 let resp = Response::success("42", serde_json::json!({"command": "pong"}));
123 let json_str = serde_json::to_string(&resp).unwrap();
124 let v: serde_json::Value = serde_json::from_str(&json_str).unwrap();
125 assert_eq!(v["id"], "42");
126 assert_eq!(v["success"], true);
127 assert_eq!(v["command"], "pong");
128 }
129
130 #[test]
131 fn response_error_round_trip() {
132 let resp = Response::error("99", "unknown_command", "unknown command: foo");
133 let json_str = serde_json::to_string(&resp).unwrap();
134 let v: serde_json::Value = serde_json::from_str(&json_str).unwrap();
135 assert_eq!(v["id"], "99");
136 assert_eq!(v["success"], false);
137 assert_eq!(v["code"], "unknown_command");
138 assert_eq!(v["message"], "unknown command: foo");
139 }
140
141 #[test]
144 fn error_display_symbol_not_found() {
145 let err = AftError::SymbolNotFound {
146 name: "foo".into(),
147 file: "bar.rs".into(),
148 };
149 assert_eq!(err.to_string(), "symbol 'foo' not found in bar.rs");
150 assert_eq!(err.code(), "symbol_not_found");
151 }
152
153 #[test]
154 fn error_display_ambiguous_symbol() {
155 let err = AftError::AmbiguousSymbol {
156 name: "Foo".into(),
157 candidates: vec!["a.rs:10".into(), "b.rs:20".into()],
158 };
159 let s = err.to_string();
160 assert!(s.contains("Foo"));
161 assert!(s.contains("a.rs:10, b.rs:20"));
162 }
163
164 #[test]
165 fn error_display_parse_error() {
166 let err = AftError::ParseError {
167 message: "unexpected token".into(),
168 };
169 assert_eq!(err.to_string(), "parse error: unexpected token");
170 }
171
172 #[test]
173 fn error_display_file_not_found() {
174 let err = AftError::FileNotFound {
175 path: "/tmp/missing.rs".into(),
176 };
177 assert_eq!(err.to_string(), "file not found: /tmp/missing.rs");
178 }
179
180 #[test]
181 fn error_display_invalid_request() {
182 let err = AftError::InvalidRequest {
183 message: "missing field".into(),
184 };
185 assert_eq!(err.to_string(), "invalid request: missing field");
186 }
187
188 #[test]
189 fn error_display_checkpoint_not_found() {
190 let err = AftError::CheckpointNotFound {
191 name: "pre-refactor".into(),
192 };
193 assert_eq!(err.to_string(), "checkpoint not found: pre-refactor");
194 assert_eq!(err.code(), "checkpoint_not_found");
195 }
196
197 #[test]
198 fn error_display_no_undo_history() {
199 let err = AftError::NoUndoHistory {
200 path: "src/main.rs".into(),
201 };
202 assert_eq!(err.to_string(), "no undo history for: src/main.rs");
203 assert_eq!(err.code(), "no_undo_history");
204 }
205
206 #[test]
207 fn error_display_ambiguous_match() {
208 let err = AftError::AmbiguousMatch {
209 pattern: "TODO".into(),
210 count: 5,
211 };
212 assert_eq!(
213 err.to_string(),
214 "pattern 'TODO' matches 5 occurrences, expected exactly 1"
215 );
216 assert_eq!(err.code(), "ambiguous_match");
217 }
218
219 #[test]
220 fn error_display_project_too_large() {
221 let err = AftError::ProjectTooLarge {
222 count: 20001,
223 max: 20000,
224 };
225 assert_eq!(
226 err.to_string(),
227 "project has 20001 source files, exceeding max_callgraph_files=20000. Call-graph operations (callers, trace_to, trace_data, impact) are disabled for this root. Open a specific subdirectory or raise max_callgraph_files in config."
228 );
229 assert_eq!(err.code(), "project_too_large");
230 }
231
232 #[test]
233 fn error_to_json_has_code_and_message() {
234 let err = AftError::FileNotFound { path: "/x".into() };
235 let j = err.to_error_json();
236 assert_eq!(j["code"], "file_not_found");
237 assert!(j["message"].as_str().unwrap().contains("/x"));
238 }
239
240 #[test]
243 fn config_default_values() {
244 let cfg = Config::default();
245 assert!(cfg.project_root.is_none());
246 assert_eq!(cfg.validation_depth, 1);
247 assert_eq!(cfg.checkpoint_ttl_hours, 24);
248 assert_eq!(cfg.max_symbol_depth, 10);
249 assert_eq!(cfg.formatter_timeout_secs, 10);
250 assert_eq!(cfg.type_checker_timeout_secs, 30);
251 assert_eq!(cfg.max_callgraph_files, 20_000);
252 }
253}