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