ryo_executor/executor/registry/converters/
enum_.rs1use crate::engine::ASTRegApply;
4use crate::executor::registry::converters::ResolveTargetSymbol;
5use crate::executor::registry::{ConvertError, MutationConverter};
6use crate::executor::spec::{MutationSpec, VariantKind};
7use ryo_analysis::AnalysisContext;
8use ryo_mutations::{AddVariantMutation, RemoveVariantMutation};
9
10#[derive(Debug, Clone, Default)]
12pub struct EnumConverter;
13
14impl EnumConverter {
15 pub fn new() -> Self {
16 Self
17 }
18}
19
20impl ResolveTargetSymbol for EnumConverter {}
22
23impl MutationConverter for EnumConverter {
24 fn spec_kinds(&self) -> &'static [&'static str] {
25 &["AddVariant", "RemoveVariant"]
26 }
27
28 fn convert_v2(
29 &self,
30 spec: &MutationSpec,
31 ctx: &AnalysisContext,
32 ) -> Result<Vec<Box<dyn ASTRegApply>>, ConvertError> {
33 use ryo_source::pure::{PureField, PureFields, PureType, PureVis};
34
35 match spec {
36 MutationSpec::AddVariant {
37 target: target_symbol,
38 variant_name,
39 variant_kind,
40 } => {
41 let symbol_id = self.resolve_target_symbol(target_symbol, ctx)?;
43
44 let fields = match variant_kind {
45 VariantKind::Unit => PureFields::Unit,
46 VariantKind::Tuple { types } => {
47 PureFields::Tuple(types.iter().map(|t| PureType::Path(t.clone())).collect())
48 }
49 VariantKind::Struct { fields } => {
50 let pure_fields: Vec<PureField> = fields
51 .iter()
52 .map(|(n, t)| PureField {
53 attrs: Vec::new(),
54 vis: PureVis::Private,
55 name: n.clone(),
56 ty: PureType::Path(t.clone()),
57 })
58 .collect();
59 PureFields::Named(pure_fields)
60 }
61 };
62 let mutation = AddVariantMutation::new(symbol_id, variant_name, fields);
63 Ok(vec![Box::new(mutation)])
64 }
65 MutationSpec::RemoveVariant {
66 target: target_symbol,
67 variant_name,
68 } => {
69 let symbol_id = self.resolve_target_symbol(target_symbol, ctx)?;
71 let mutation = RemoveVariantMutation::new(symbol_id, variant_name);
72 Ok(vec![Box::new(mutation)])
73 }
74 _ => Err(ConvertError::TypeMismatch {
75 expected: "AddVariant or RemoveVariant",
76 actual: spec.kind_name().to_string(),
77 }),
78 }
79 }
80}
81
82#[cfg(test)]
83mod tests {
84 use super::*;
85 use crate::executor::spec::MutationTargetSymbol;
86 use ryo_symbol::{SymbolKind, SymbolPath, SymbolRegistry};
87
88 fn create_test_symbol_id() -> ryo_symbol::SymbolId {
89 let mut registry = SymbolRegistry::new();
90 let path = SymbolPath::parse("test_crate::Status").unwrap();
91 registry.register(path, SymbolKind::Enum).unwrap()
92 }
93
94 #[test]
95 fn test_enum_converter_spec_kinds() {
96 let converter = EnumConverter::new();
97 assert_eq!(converter.spec_kinds(), &["AddVariant", "RemoveVariant"]);
98 }
99
100 #[test]
101 fn test_enum_converter_can_handle_add_variant() {
102 let converter = EnumConverter::new();
103 let id = create_test_symbol_id();
104
105 let spec = MutationSpec::AddVariant {
106 target: MutationTargetSymbol::ById(id),
107 variant_name: "Pending".into(),
108 variant_kind: VariantKind::Unit,
109 };
110 assert!(converter.can_handle(&spec));
111 }
112
113 #[test]
114 fn test_enum_converter_can_handle_remove_variant() {
115 let converter = EnumConverter::new();
116 let id = create_test_symbol_id();
117
118 let spec = MutationSpec::RemoveVariant {
119 target: MutationTargetSymbol::ById(id),
120 variant_name: "Pending".into(),
121 };
122 assert!(converter.can_handle(&spec));
123 }
124}