codama-nodes 0.9.0

Node specifications and helpers for the Codama standard
Documentation
use crate::{CamelCaseString, Docs, HasName, PdaSeedNode};
use codama_nodes_derive::node;

#[node]
pub struct PdaNode {
    // Data.
    pub name: CamelCaseString,
    #[serde(default, skip_serializing_if = "crate::is_default")]
    pub docs: Docs,
    #[serde(skip_serializing_if = "crate::is_default")]
    pub program_id: Option<String>,

    // Children.
    pub seeds: Vec<PdaSeedNode>,
}

impl PdaNode {
    pub fn new<T>(name: T, seeds: Vec<PdaSeedNode>) -> Self
    where
        T: Into<CamelCaseString>,
    {
        Self {
            name: name.into(),
            docs: Docs::default(),
            program_id: None,
            seeds,
        }
    }
}

impl HasName for PdaNode {
    fn name(&self) -> &CamelCaseString {
        &self.name
    }
}

#[cfg(test)]
mod tests {
    use super::*;
    use crate::{
        ConstantPdaSeedNode, NumberTypeNode, NumberValueNode, PublicKeyTypeNode,
        VariablePdaSeedNode, U8,
    };

    #[test]
    fn new() {
        let node = PdaNode::new(
            "associatedToken",
            vec![
                VariablePdaSeedNode::new("owner", PublicKeyTypeNode::new()).into(),
                ConstantPdaSeedNode::new(NumberTypeNode::le(U8), NumberValueNode::new(42u8)).into(),
                VariablePdaSeedNode::new("mint", PublicKeyTypeNode::new()).into(),
            ],
        );
        assert_eq!(node.name, CamelCaseString::new("associatedToken"));
        assert_eq!(node.docs, Docs::default());
        assert_eq!(node.program_id, None);
        assert_eq!(
            node.seeds,
            vec![
                PdaSeedNode::Variable(VariablePdaSeedNode::new("owner", PublicKeyTypeNode::new())),
                PdaSeedNode::Constant(ConstantPdaSeedNode::new(
                    NumberTypeNode::le(U8),
                    NumberValueNode::new(42u8)
                )),
                PdaSeedNode::Variable(VariablePdaSeedNode::new("mint", PublicKeyTypeNode::new())),
            ]
        );
    }

    #[test]
    fn direct_instantiation() {
        let node = PdaNode {
            name: "myPda".into(),
            docs: Docs::default(),
            program_id: Some("1234..5678".into()),
            seeds: vec![],
        };
        assert_eq!(node.name, CamelCaseString::new("myPda"));
        assert_eq!(node.docs, Docs::default());
        assert_eq!(node.program_id, Some("1234..5678".into()));
        assert_eq!(node.seeds, vec![]);
    }

    #[test]
    fn to_json() {
        let node = PdaNode::new("myPda", vec![]);
        let json = serde_json::to_string(&node).unwrap();
        assert_eq!(json, r#"{"kind":"pdaNode","name":"myPda","seeds":[]}"#);
    }

    #[test]
    fn from_json() {
        let json = r#"{"kind":"pdaNode","name":"myPda","seeds":[]}"#;
        let node: PdaNode = serde_json::from_str(json).unwrap();
        assert_eq!(node, PdaNode::new("myPda", vec![]));
    }
}