1use serde::{Deserialize, Serialize};
7use strum::{EnumString, Display};
8
9#[derive(Debug, Clone, Serialize, Deserialize, PartialEq, Eq)]
11pub struct Span {
12 pub start_line: usize,
14 pub end_line: usize,
16}
17
18impl Span {
19 pub fn new(start_line: usize, start_col: usize, end_line: usize, end_col: usize) -> Self {
20 let _ = (start_col, end_col);
22 Self {
23 start_line,
24 end_line,
25 }
26 }
27}
28
29#[derive(Debug, Clone, Serialize, Deserialize, PartialEq, Eq, EnumString, Display)]
31#[strum(serialize_all = "PascalCase")]
32pub enum SymbolKind {
33 Function,
34 Class,
35 Struct,
36 Enum,
37 Interface,
38 Trait,
39 Constant,
40 Variable,
41 Method,
42 Module,
43 Namespace,
44 Type,
45 Macro,
46 Property,
47 Event,
48 Import,
49 Export,
50 Attribute,
51 #[strum(default)]
55 Unknown(String),
56}
57
58#[derive(Debug, Clone, Copy, Serialize, Deserialize, PartialEq, Eq, Default)]
60#[serde(rename_all = "lowercase")]
61pub enum Language {
62 #[default]
63 Rust,
64 Python,
65 JavaScript,
66 TypeScript,
67 Vue,
68 Svelte,
69 Go,
70 Java,
71 PHP,
72 C,
73 Cpp,
74 CSharp,
75 Ruby,
76 Kotlin,
77 Swift,
78 Zig,
79 Unknown,
80}
81
82impl Language {
83 pub fn from_extension(ext: &str) -> Self {
84 match ext {
85 "rs" => Language::Rust,
86 "py" => Language::Python,
87 "js" | "mjs" | "cjs" | "jsx" => Language::JavaScript,
88 "ts" | "mts" | "cts" | "tsx" => Language::TypeScript,
89 "vue" => Language::Vue,
90 "svelte" => Language::Svelte,
91 "go" => Language::Go,
92 "java" => Language::Java,
93 "php" => Language::PHP,
94 "c" | "h" => Language::C,
95 "cpp" | "cc" | "cxx" | "hpp" | "hxx" | "C" | "H" => Language::Cpp,
96 "cs" => Language::CSharp,
97 "rb" | "rake" | "gemspec" => Language::Ruby,
98 "kt" | "kts" => Language::Kotlin,
99 "swift" => Language::Swift,
100 "zig" => Language::Zig,
101 _ => Language::Unknown,
102 }
103 }
104
105 pub fn from_name(name: &str) -> Option<Self> {
110 match name.to_lowercase().as_str() {
111 "rust" | "rs" => Some(Language::Rust),
112 "python" | "py" => Some(Language::Python),
113 "javascript" | "js" => Some(Language::JavaScript),
114 "typescript" | "ts" => Some(Language::TypeScript),
115 "vue" => Some(Language::Vue),
116 "svelte" => Some(Language::Svelte),
117 "go" => Some(Language::Go),
118 "java" => Some(Language::Java),
119 "php" => Some(Language::PHP),
120 "c" => Some(Language::C),
121 "cpp" | "c++" => Some(Language::Cpp),
122 "csharp" | "cs" | "c#" => Some(Language::CSharp),
123 "ruby" | "rb" => Some(Language::Ruby),
124 "kotlin" | "kt" => Some(Language::Kotlin),
125 "zig" => Some(Language::Zig),
126 _ => None,
127 }
128 }
129
130 pub fn supported_names_help() -> &'static str {
132 "rust (rs), python (py), javascript (js), typescript (ts), vue, svelte, \
133 go, java, php, c, cpp (c++), csharp (cs, c#), ruby (rb), kotlin (kt), zig"
134 }
135
136 pub fn is_supported(&self) -> bool {
141 match self {
142 Language::Rust => true,
143 Language::TypeScript => true,
144 Language::JavaScript => true,
145 Language::Vue => true,
146 Language::Svelte => true,
147 Language::Python => true,
148 Language::Go => true,
149 Language::Java => true,
150 Language::PHP => true,
151 Language::C => true,
152 Language::Cpp => true,
153 Language::CSharp => true,
154 Language::Ruby => true,
155 Language::Kotlin => true,
156 Language::Swift => false, Language::Zig => true,
158 Language::Unknown => false,
159 }
160 }
161}
162
163#[derive(Debug, Clone, Serialize, Deserialize, PartialEq, Eq)]
165#[serde(rename_all = "lowercase")]
166pub enum ImportType {
167 Internal,
169 External,
171 Stdlib,
173 #[serde(rename = "mod_decl")]
175 ModDecl,
176}
177
178#[derive(Debug, Clone, Serialize, Deserialize)]
180pub struct DependencyInfo {
181 pub path: String,
183 #[serde(skip_serializing_if = "Option::is_none")]
185 pub line: Option<usize>,
186 #[serde(skip_serializing_if = "Option::is_none")]
188 pub symbols: Option<Vec<String>>,
189}
190
191#[derive(Debug, Clone)]
193pub struct Dependency {
194 pub file_id: i64,
196 pub imported_path: String,
198 pub resolved_file_id: Option<i64>,
200 pub import_type: ImportType,
202 pub line_number: usize,
204 pub imported_symbols: Option<Vec<String>>,
206}
207
208#[derive(Debug, Clone, Serialize, Deserialize, PartialEq, Eq)]
215pub struct SymbolRef {
216 pub name: String,
218 pub kind: SymbolKind,
220 pub span: Span,
222}
223
224fn is_unknown_kind(kind: &SymbolKind) -> bool {
226 matches!(kind, SymbolKind::Unknown(_))
227}
228
229#[derive(Debug, Clone, Serialize, Deserialize)]
231pub struct SearchResult {
232 pub path: String,
234 #[serde(skip)]
236 pub lang: Language,
237 #[serde(skip_serializing_if = "is_unknown_kind")]
239 pub kind: SymbolKind,
240 #[serde(skip_serializing_if = "Option::is_none")]
243 pub symbol: Option<String>,
244 pub span: Span,
246 pub preview: String,
248 #[serde(skip_serializing_if = "Option::is_none")]
251 pub dependencies: Option<Vec<DependencyInfo>>,
252}
253
254#[derive(Debug, Clone, Serialize, Deserialize)]
256pub struct MatchResult {
257 #[serde(skip_serializing_if = "is_unknown_kind")]
259 pub kind: SymbolKind,
260 #[serde(skip_serializing_if = "Option::is_none")]
262 pub symbol: Option<String>,
263 pub span: Span,
265 pub preview: String,
267 #[serde(skip_serializing_if = "Vec::is_empty")]
269 pub context_before: Vec<String>,
270 #[serde(skip_serializing_if = "Vec::is_empty")]
272 pub context_after: Vec<String>,
273}
274
275#[derive(Debug, Clone, Serialize, Deserialize)]
277pub struct FileGroupedResult {
278 pub path: String,
280 pub language: Language,
282 #[serde(skip_serializing_if = "Option::is_none")]
284 pub dependencies: Option<Vec<DependencyInfo>>,
285 pub matches: Vec<MatchResult>,
287}
288
289impl SearchResult {
290 pub fn new(
291 path: String,
292 lang: Language,
293 kind: SymbolKind,
294 symbol: Option<String>,
295 span: Span,
296 scope: Option<String>,
297 preview: String,
298 ) -> Self {
299 let _ = scope;
301 Self {
302 path,
303 lang,
304 kind,
305 symbol,
306 span,
307 preview,
308 dependencies: None,
309 }
310 }
311}
312
313#[derive(Debug, Clone, Serialize, Deserialize)]
315pub struct IndexConfig {
316 pub languages: Vec<Language>,
318 pub include_patterns: Vec<String>,
320 pub exclude_patterns: Vec<String>,
322 pub follow_symlinks: bool,
324 pub max_file_size: usize,
326 pub parallel_threads: usize,
328 pub query_timeout_secs: u64,
330 pub max_posting_list_entries: usize,
333}
334
335impl Default for IndexConfig {
336 fn default() -> Self {
337 Self {
338 languages: vec![],
339 include_patterns: vec![],
340 exclude_patterns: vec![],
341 follow_symlinks: false,
342 max_file_size: 10 * 1024 * 1024, parallel_threads: 0, query_timeout_secs: 30, max_posting_list_entries: 500_000, }
347 }
348}
349
350fn is_zero(v: &usize) -> bool { *v == 0 }
351fn is_zero_u64(v: &u64) -> bool { *v == 0 }
352
353#[derive(Debug, Clone, Serialize, Deserialize, Default)]
355pub struct IndexStats {
356 pub total_files: usize,
358 pub index_size_bytes: u64,
360 pub last_updated: String,
362 pub files_by_language: std::collections::HashMap<String, usize>,
364 pub lines_by_language: std::collections::HashMap<String, usize>,
366 #[serde(default, skip_serializing_if = "is_zero")]
368 pub new_files: usize,
369 #[serde(default, skip_serializing_if = "is_zero")]
371 pub modified_files: usize,
372 #[serde(default, skip_serializing_if = "is_zero")]
374 pub unchanged_files: usize,
375 #[serde(default, skip_serializing_if = "is_zero")]
377 pub skipped_too_large: usize,
378 #[serde(default, skip_serializing_if = "is_zero_u64")]
380 pub skipped_bytes_too_large: u64,
381}
382
383#[derive(Debug, Clone, Serialize, Deserialize)]
385pub struct IndexedFile {
386 pub path: String,
388 pub language: String,
390 pub last_indexed: String,
392}
393
394#[derive(Debug, Clone, Serialize, Deserialize, PartialEq, Eq)]
396#[serde(rename_all = "snake_case")]
397pub enum IndexStatus {
398 Fresh,
400 Stale,
402}
403
404#[derive(Debug, Clone, Serialize, Deserialize)]
406pub struct IndexWarning {
407 pub reason: String,
409 pub action_required: String,
411 #[serde(skip_serializing_if = "Option::is_none")]
413 pub files_modified: Option<u32>,
414 #[serde(skip_serializing_if = "Option::is_none")]
416 pub details: Option<IndexWarningDetails>,
417}
418
419#[derive(Debug, Clone, Serialize, Deserialize)]
421pub struct IndexWarningDetails {
422 #[serde(skip_serializing_if = "Option::is_none")]
424 pub current_branch: Option<String>,
425 #[serde(skip_serializing_if = "Option::is_none")]
427 pub indexed_branch: Option<String>,
428 #[serde(skip_serializing_if = "Option::is_none")]
430 pub current_commit: Option<String>,
431 #[serde(skip_serializing_if = "Option::is_none")]
433 pub indexed_commit: Option<String>,
434}
435
436#[derive(Debug, Clone, Serialize, Deserialize)]
438pub struct PaginationInfo {
439 pub total: usize,
441 pub count: usize,
443 pub offset: usize,
445 #[serde(skip_serializing_if = "Option::is_none")]
447 pub limit: Option<usize>,
448 pub has_more: bool,
450}
451
452#[derive(Debug, Clone, Serialize, Deserialize)]
454pub struct QueryResponse {
455 #[serde(skip_serializing_if = "Option::is_none")]
459 pub ai_instruction: Option<String>,
460 pub status: IndexStatus,
462 pub can_trust_results: bool,
464 #[serde(skip_serializing_if = "Option::is_none")]
466 pub warning: Option<IndexWarning>,
467 pub pagination: PaginationInfo,
469 pub results: Vec<FileGroupedResult>,
472}
473
474#[derive(Debug, Clone, Serialize, Deserialize)]
476pub struct CompactionReport {
477 pub files_removed: usize,
479 pub space_saved_bytes: u64,
481 pub duration_ms: u64,
483}
484
485#[cfg(test)]
486mod tests {
487 use super::*;
488
489 #[test]
490 fn test_symbol_ref_json_shape() {
491 let sym = SymbolRef {
492 name: "my_function".to_string(),
493 kind: SymbolKind::Function,
494 span: Span { start_line: 10, end_line: 20 },
495 };
496 let json = serde_json::to_value(&sym).unwrap();
497 assert_eq!(json["name"], "my_function");
498 assert_eq!(json["kind"], "Function");
499 assert_eq!(json["span"]["start_line"], 10);
500 assert_eq!(json["span"]["end_line"], 20);
501 assert!(json.as_array().is_none());
502 }
503
504 #[test]
505 fn test_symbol_ref_roundtrip() {
506 let original = SymbolRef {
507 name: "MyStruct".to_string(),
508 kind: SymbolKind::Struct,
509 span: Span { start_line: 1, end_line: 5 },
510 };
511 let json = serde_json::to_string(&original).unwrap();
512 let decoded: SymbolRef = serde_json::from_str(&json).unwrap();
513 assert_eq!(original, decoded);
514 }
515
516 #[test]
517 fn test_symbol_ref_exact_json() {
518 let sym = SymbolRef {
519 name: "Foo".to_string(),
520 kind: SymbolKind::Class,
521 span: Span { start_line: 3, end_line: 7 },
522 };
523 let json = serde_json::to_string(&sym).unwrap();
524 assert_eq!(
525 json,
526 r#"{"name":"Foo","kind":"Class","span":{"start_line":3,"end_line":7}}"#
527 );
528 }
529}