1#[derive(Debug, Clone, serde::Serialize)]
8#[cfg_attr(feature = "schema", derive(schemars::JsonSchema))]
9#[allow(
10 clippy::struct_field_names,
11 reason = "triggered in bin but not lib — #[expect] would be unfulfilled in lib"
12)]
13pub struct TargetThresholds {
14 pub fan_in_p95: f64,
16 pub fan_in_p75: f64,
18 pub fan_out_p95: f64,
20 pub fan_out_p90: usize,
22}
23
24#[derive(Debug, Clone, serde::Serialize)]
26#[cfg_attr(feature = "schema", derive(schemars::JsonSchema))]
27#[serde(rename_all = "snake_case")]
28pub enum RecommendationCategory {
29 UrgentChurnComplexity,
31 BreakCircularDependency,
33 SplitHighImpact,
35 RemoveDeadCode,
37 ExtractComplexFunctions,
39 ExtractDependencies,
41 AddTestCoverage,
43}
44
45impl RecommendationCategory {
46 #[must_use]
48 pub const fn label(&self) -> &'static str {
49 match self {
50 Self::UrgentChurnComplexity => "churn+complexity",
51 Self::BreakCircularDependency => "circular dependency",
52 Self::SplitHighImpact => "high impact",
53 Self::RemoveDeadCode => "dead code",
54 Self::ExtractComplexFunctions => "complexity",
55 Self::ExtractDependencies => "coupling",
56 Self::AddTestCoverage => "untested risk",
57 }
58 }
59
60 #[must_use]
62 pub const fn compact_label(&self) -> &'static str {
63 match self {
64 Self::UrgentChurnComplexity => "churn_complexity",
65 Self::BreakCircularDependency => "circular_dep",
66 Self::SplitHighImpact => "high_impact",
67 Self::RemoveDeadCode => "dead_code",
68 Self::ExtractComplexFunctions => "complexity",
69 Self::ExtractDependencies => "coupling",
70 Self::AddTestCoverage => "untested_risk",
71 }
72 }
73}
74
75#[derive(Debug, Clone, serde::Serialize)]
77#[cfg_attr(feature = "schema", derive(schemars::JsonSchema))]
78pub struct ContributingFactor {
79 pub metric: &'static str,
81 pub value: f64,
83 pub threshold: f64,
85 pub detail: String,
87}
88
89#[derive(Debug, Clone, PartialEq, Eq, serde::Serialize)]
109#[cfg_attr(feature = "schema", derive(schemars::JsonSchema))]
110#[serde(rename_all = "snake_case")]
111pub enum EffortEstimate {
112 Low,
114 Medium,
116 High,
118}
119
120impl EffortEstimate {
121 #[must_use]
123 pub const fn label(&self) -> &'static str {
124 match self {
125 Self::Low => "low",
126 Self::Medium => "medium",
127 Self::High => "high",
128 }
129 }
130
131 #[must_use]
133 pub const fn numeric(&self) -> f64 {
134 match self {
135 Self::Low => 1.0,
136 Self::Medium => 2.0,
137 Self::High => 3.0,
138 }
139 }
140}
141
142#[derive(Debug, Clone, serde::Serialize)]
149#[cfg_attr(feature = "schema", derive(schemars::JsonSchema))]
150#[serde(rename_all = "snake_case")]
151pub enum Confidence {
152 High,
154 Medium,
156 Low,
158}
159
160impl Confidence {
161 #[must_use]
163 pub const fn label(&self) -> &'static str {
164 match self {
165 Self::High => "high",
166 Self::Medium => "medium",
167 Self::Low => "low",
168 }
169 }
170}
171
172#[derive(Debug, Clone, Default, serde::Serialize)]
177#[cfg_attr(feature = "schema", derive(schemars::JsonSchema))]
178pub struct TargetEvidence {
179 #[serde(default, skip_serializing_if = "Vec::is_empty")]
181 pub unused_exports: Vec<String>,
182 #[serde(default, skip_serializing_if = "Vec::is_empty")]
184 pub complex_functions: Vec<EvidenceFunction>,
185 #[serde(default, skip_serializing_if = "Vec::is_empty")]
187 pub cycle_path: Vec<String>,
188 #[serde(default, skip_serializing_if = "Vec::is_empty")]
190 pub direct_callers: Vec<DirectCallerEvidence>,
191 #[serde(default, skip_serializing_if = "Vec::is_empty")]
193 pub clone_siblings: Vec<CloneSiblingEvidence>,
194}
195
196#[derive(Debug, Clone, serde::Serialize)]
198#[cfg_attr(feature = "schema", derive(schemars::JsonSchema))]
199pub struct DirectCallerEvidence {
200 #[serde(serialize_with = "fallow_types::serde_path::serialize")]
202 pub path: std::path::PathBuf,
203 #[serde(default, skip_serializing_if = "Vec::is_empty")]
205 pub symbols: Vec<DirectCallerSymbolEvidence>,
206}
207
208#[derive(Debug, Clone, serde::Serialize)]
210#[cfg_attr(feature = "schema", derive(schemars::JsonSchema))]
211pub struct DirectCallerSymbolEvidence {
212 pub imported: String,
214 pub local: String,
216 pub type_only: bool,
218}
219
220#[derive(Debug, Clone, serde::Serialize)]
222#[cfg_attr(feature = "schema", derive(schemars::JsonSchema))]
223pub struct CloneSiblingEvidence {
224 #[serde(serialize_with = "fallow_types::serde_path::serialize")]
226 pub path: std::path::PathBuf,
227 pub start_line: usize,
229 pub end_line: usize,
231 pub fingerprint: String,
233}
234
235#[derive(Debug, Clone, serde::Serialize)]
237#[cfg_attr(feature = "schema", derive(schemars::JsonSchema))]
238pub struct EvidenceFunction {
239 pub name: String,
241 pub line: u32,
243 pub cognitive: u16,
245}
246
247#[derive(Debug, Clone, serde::Serialize)]
248#[cfg_attr(feature = "schema", derive(schemars::JsonSchema))]
249pub struct RefactoringTarget {
250 #[serde(serialize_with = "fallow_types::serde_path::serialize")]
252 pub path: std::path::PathBuf,
253 pub priority: f64,
255 pub efficiency: f64,
258 pub recommendation: String,
260 pub category: RecommendationCategory,
262 pub effort: EffortEstimate,
264 pub confidence: Confidence,
266 #[serde(default, skip_serializing_if = "Vec::is_empty")]
269 #[cfg_attr(feature = "schema", schemars(default))]
270 pub factors: Vec<ContributingFactor>,
271 #[serde(default, skip_serializing_if = "Option::is_none")]
273 pub evidence: Option<TargetEvidence>,
274}
275
276#[cfg(test)]
277mod tests {
278 use super::*;
279
280 #[test]
281 fn category_labels_are_non_empty() {
282 let categories = [
283 RecommendationCategory::UrgentChurnComplexity,
284 RecommendationCategory::BreakCircularDependency,
285 RecommendationCategory::SplitHighImpact,
286 RecommendationCategory::RemoveDeadCode,
287 RecommendationCategory::ExtractComplexFunctions,
288 RecommendationCategory::ExtractDependencies,
289 RecommendationCategory::AddTestCoverage,
290 ];
291 for cat in &categories {
292 assert!(!cat.label().is_empty(), "{cat:?} should have a label");
293 }
294 }
295
296 #[test]
297 fn category_labels_are_unique() {
298 let categories = [
299 RecommendationCategory::UrgentChurnComplexity,
300 RecommendationCategory::BreakCircularDependency,
301 RecommendationCategory::SplitHighImpact,
302 RecommendationCategory::RemoveDeadCode,
303 RecommendationCategory::ExtractComplexFunctions,
304 RecommendationCategory::ExtractDependencies,
305 RecommendationCategory::AddTestCoverage,
306 ];
307 let labels: Vec<&str> = categories
308 .iter()
309 .map(RecommendationCategory::label)
310 .collect();
311 let unique: rustc_hash::FxHashSet<&&str> = labels.iter().collect();
312 assert_eq!(labels.len(), unique.len(), "category labels must be unique");
313 }
314
315 #[test]
316 fn category_serializes_as_snake_case() {
317 let json = serde_json::to_string(&RecommendationCategory::UrgentChurnComplexity).unwrap();
318 assert_eq!(json, r#""urgent_churn_complexity""#);
319
320 let json = serde_json::to_string(&RecommendationCategory::BreakCircularDependency).unwrap();
321 assert_eq!(json, r#""break_circular_dependency""#);
322 }
323
324 #[test]
325 fn refactoring_target_skips_empty_factors() {
326 let target = RefactoringTarget {
327 path: std::path::PathBuf::from("/src/foo.ts"),
328 priority: 75.0,
329 efficiency: 75.0,
330 recommendation: "Test recommendation".into(),
331 category: RecommendationCategory::RemoveDeadCode,
332 effort: EffortEstimate::Low,
333 confidence: Confidence::High,
334 factors: vec![],
335 evidence: None,
336 };
337 let json = serde_json::to_string(&target).unwrap();
338 assert!(!json.contains("factors"));
339 assert!(!json.contains("evidence"));
340 }
341
342 #[test]
343 fn effort_numeric_values() {
344 assert!((EffortEstimate::Low.numeric() - 1.0).abs() < f64::EPSILON);
345 assert!((EffortEstimate::Medium.numeric() - 2.0).abs() < f64::EPSILON);
346 assert!((EffortEstimate::High.numeric() - 3.0).abs() < f64::EPSILON);
347 }
348
349 #[test]
350 fn confidence_labels_are_non_empty() {
351 let levels = [Confidence::High, Confidence::Medium, Confidence::Low];
352 for level in &levels {
353 assert!(!level.label().is_empty(), "{level:?} should have a label");
354 }
355 }
356
357 #[test]
358 fn confidence_serializes_as_snake_case() {
359 let json = serde_json::to_string(&Confidence::High).unwrap();
360 assert_eq!(json, r#""high""#);
361 let json = serde_json::to_string(&Confidence::Medium).unwrap();
362 assert_eq!(json, r#""medium""#);
363 let json = serde_json::to_string(&Confidence::Low).unwrap();
364 assert_eq!(json, r#""low""#);
365 }
366
367 #[test]
368 fn contributing_factor_serializes_correctly() {
369 let factor = ContributingFactor {
370 metric: "fan_in",
371 value: 15.0,
372 threshold: 10.0,
373 detail: "15 files depend on this".into(),
374 };
375 let json = serde_json::to_string(&factor).unwrap();
376 let parsed: serde_json::Value = serde_json::from_str(&json).unwrap();
377 assert_eq!(parsed["metric"], "fan_in");
378 assert_eq!(parsed["value"], 15.0);
379 assert_eq!(parsed["threshold"], 10.0);
380 }
381
382 #[test]
383 fn category_compact_labels_are_non_empty() {
384 let categories = [
385 RecommendationCategory::UrgentChurnComplexity,
386 RecommendationCategory::BreakCircularDependency,
387 RecommendationCategory::SplitHighImpact,
388 RecommendationCategory::RemoveDeadCode,
389 RecommendationCategory::ExtractComplexFunctions,
390 RecommendationCategory::ExtractDependencies,
391 RecommendationCategory::AddTestCoverage,
392 ];
393 for cat in &categories {
394 assert!(
395 !cat.compact_label().is_empty(),
396 "{cat:?} should have a compact_label"
397 );
398 }
399 }
400
401 #[test]
402 fn category_compact_labels_are_unique() {
403 let categories = [
404 RecommendationCategory::UrgentChurnComplexity,
405 RecommendationCategory::BreakCircularDependency,
406 RecommendationCategory::SplitHighImpact,
407 RecommendationCategory::RemoveDeadCode,
408 RecommendationCategory::ExtractComplexFunctions,
409 RecommendationCategory::ExtractDependencies,
410 RecommendationCategory::AddTestCoverage,
411 ];
412 let labels: Vec<&str> = categories
413 .iter()
414 .map(RecommendationCategory::compact_label)
415 .collect();
416 let unique: rustc_hash::FxHashSet<&&str> = labels.iter().collect();
417 assert_eq!(labels.len(), unique.len(), "compact labels must be unique");
418 }
419
420 #[test]
421 fn category_compact_labels_have_no_spaces() {
422 let categories = [
423 RecommendationCategory::UrgentChurnComplexity,
424 RecommendationCategory::BreakCircularDependency,
425 RecommendationCategory::SplitHighImpact,
426 RecommendationCategory::RemoveDeadCode,
427 RecommendationCategory::ExtractComplexFunctions,
428 RecommendationCategory::ExtractDependencies,
429 RecommendationCategory::AddTestCoverage,
430 ];
431 for cat in &categories {
432 assert!(
433 !cat.compact_label().contains(' '),
434 "compact_label for {:?} should not contain spaces: '{}'",
435 cat,
436 cat.compact_label()
437 );
438 }
439 }
440
441 #[test]
442 fn effort_labels_are_non_empty() {
443 let efforts = [
444 EffortEstimate::Low,
445 EffortEstimate::Medium,
446 EffortEstimate::High,
447 ];
448 for effort in &efforts {
449 assert!(!effort.label().is_empty(), "{effort:?} should have a label");
450 }
451 }
452
453 #[test]
454 fn effort_serializes_as_snake_case() {
455 assert_eq!(
456 serde_json::to_string(&EffortEstimate::Low).unwrap(),
457 r#""low""#
458 );
459 assert_eq!(
460 serde_json::to_string(&EffortEstimate::Medium).unwrap(),
461 r#""medium""#
462 );
463 assert_eq!(
464 serde_json::to_string(&EffortEstimate::High).unwrap(),
465 r#""high""#
466 );
467 }
468
469 #[test]
470 fn target_evidence_skips_empty_fields() {
471 let evidence = TargetEvidence {
472 unused_exports: vec![],
473 complex_functions: vec![],
474 cycle_path: vec![],
475 direct_callers: vec![],
476 clone_siblings: vec![],
477 };
478 let json = serde_json::to_string(&evidence).unwrap();
479 assert!(!json.contains("unused_exports"));
480 assert!(!json.contains("complex_functions"));
481 assert!(!json.contains("cycle_path"));
482 assert!(!json.contains("direct_callers"));
483 assert!(!json.contains("clone_siblings"));
484 }
485
486 #[test]
487 fn target_evidence_with_data() {
488 let evidence = TargetEvidence {
489 unused_exports: vec!["foo".to_string(), "bar".to_string()],
490 complex_functions: vec![EvidenceFunction {
491 name: "processData".into(),
492 line: 42,
493 cognitive: 30,
494 }],
495 cycle_path: vec![],
496 direct_callers: vec![DirectCallerEvidence {
497 path: "src/consumer.ts".into(),
498 symbols: vec![DirectCallerSymbolEvidence {
499 imported: "processData".into(),
500 local: "processData".into(),
501 type_only: false,
502 }],
503 }],
504 clone_siblings: vec![CloneSiblingEvidence {
505 path: "src/peer.ts".into(),
506 start_line: 12,
507 end_line: 20,
508 fingerprint: "dup:12345678".into(),
509 }],
510 };
511 let json = serde_json::to_string(&evidence).unwrap();
512 assert!(json.contains("unused_exports"));
513 assert!(json.contains("complex_functions"));
514 assert!(json.contains("processData"));
515 assert!(json.contains("direct_callers"));
516 assert!(json.contains("clone_siblings"));
517 assert!(!json.contains("cycle_path"));
518 }
519}