codama-korok-visitors 0.9.2

Visitors for Korok trees
Documentation
use crate::KorokVisitor;
use codama_attributes::{Attributes, DefaultValueDirective, TryFromFilter};
use codama_errors::CodamaResult;
use codama_koroks::{KorokMut, KorokTrait};
use codama_nodes::{
    InstructionArgumentNode, Node, RegisteredTypeNode, StructFieldTypeNode, ValueNode,
};

#[derive(Default)]
pub struct SetDefaultValuesVisitor;

impl SetDefaultValuesVisitor {
    pub fn new() -> Self {
        Self
    }
}

impl KorokVisitor for SetDefaultValuesVisitor {
    fn visit_root(&mut self, korok: &mut codama_koroks::RootKorok) -> CodamaResult<()> {
        self.visit_children(korok)?;
        Ok(())
    }

    fn visit_crate(&mut self, korok: &mut codama_koroks::CrateKorok) -> CodamaResult<()> {
        self.visit_children(korok)?;
        set_default_values(korok.into())
    }

    fn visit_file_module(
        &mut self,
        korok: &mut codama_koroks::FileModuleKorok,
    ) -> CodamaResult<()> {
        self.visit_children(korok)?;
        set_default_values(korok.into())
    }

    fn visit_module(&mut self, korok: &mut codama_koroks::ModuleKorok) -> CodamaResult<()> {
        self.visit_children(korok)?;
        set_default_values(korok.into())
    }

    fn visit_struct(&mut self, korok: &mut codama_koroks::StructKorok) -> CodamaResult<()> {
        self.visit_children(korok)?;
        set_default_values(korok.into())
    }

    fn visit_enum(&mut self, korok: &mut codama_koroks::EnumKorok) -> CodamaResult<()> {
        self.visit_children(korok)?;
        set_default_values(korok.into())
    }

    fn visit_unsupported_item(
        &mut self,
        korok: &mut codama_koroks::UnsupportedItemKorok,
    ) -> CodamaResult<()> {
        self.visit_children(korok)?;
        set_default_values(korok.into())
    }

    fn visit_enum_variant(
        &mut self,
        korok: &mut codama_koroks::EnumVariantKorok,
    ) -> CodamaResult<()> {
        self.visit_children(korok)?;
        set_default_values(korok.into())
    }

    fn visit_field(&mut self, korok: &mut codama_koroks::FieldKorok) -> CodamaResult<()> {
        self.visit_children(korok)?;
        set_default_values(korok.into())
    }
}

fn set_default_values(mut korok: KorokMut) -> CodamaResult<()> {
    // Ensure it has attributes.
    let Some(attributes) = korok.attributes() else {
        return Ok(());
    };

    // Ensure there is a node to set a default value on.
    let Some(node) = get_node_with_default_value(korok.node(), attributes)? else {
        return Ok(());
    };

    korok.set_node(Some(node));
    Ok(())
}

fn get_node_with_default_value(
    node: &Option<Node>,
    attributes: &Attributes,
) -> CodamaResult<Option<Node>> {
    let Some(directive) = attributes.get_last(DefaultValueDirective::filter) else {
        return Ok(None);
    };
    let resolved_node = directive.node.try_resolved()?;

    match node {
        // Handle struct fields.
        Some(Node::Type(RegisteredTypeNode::StructField(field))) => {
            let value = ValueNode::try_from(resolved_node.clone()).ok();
            Ok(value.map(|value| {
                StructFieldTypeNode {
                    default_value: Some(value),
                    default_value_strategy: directive.default_value_strategy,
                    ..field.clone()
                }
                .into()
            }))
        }
        // Handle instruction arguments.
        Some(Node::InstructionArgument(argument)) => Ok(Some(
            InstructionArgumentNode {
                default_value: Some(resolved_node.clone()),
                default_value_strategy: directive.default_value_strategy,
                ..argument.clone()
            }
            .into(),
        )),
        _ => Ok(None),
    }
}