aver-lang 0.8.2

Interpreter and transpiler for Aver, a statically-typed language designed for AI-assisted development
Documentation
use super::*;

impl Interpreter {
    pub(super) fn import_type_def_runtime(&mut self, td: &TypeDef) {
        match td {
            TypeDef::Sum {
                name: type_name,
                variants,
                ..
            } => {
                let type_id = match self.arena.find_type_id(type_name) {
                    Some(id) => id,
                    None => self.arena.register_sum_type(type_name, vec![]),
                };
                for variant in variants {
                    if self.arena.find_variant_id(type_id, &variant.name).is_none() {
                        self.arena
                            .register_variant_name(type_id, variant.name.clone());
                    }
                }
            }
            TypeDef::Product { name, fields, .. } => {
                if self.arena.find_type_id(name).is_none() {
                    self.arena.register_record_type(
                        name,
                        fields.iter().map(|(field, _)| field.clone()).collect(),
                    );
                }
                self.record_schemas
                    .entry(name.clone())
                    .or_insert_with(|| fields.iter().map(|(field, _)| field.clone()).collect());
            }
        }
    }

    pub fn exec_items(&mut self, items: &[TopLevel]) -> Result<Value, RuntimeError> {
        for item in items {
            match item {
                TopLevel::FnDef(fd) => self.exec_fn_def(fd)?,
                TopLevel::Module(_) => {}
                TopLevel::Verify(_) => {}
                TopLevel::Decision(_) => {}
                TopLevel::TypeDef(td) => self.register_type_def(td),
                TopLevel::Stmt(s) => {
                    self.exec_stmt(s)?;
                }
            }
        }
        Ok(Value::Unit)
    }

    /// Register a user-defined type: sum type variants and record constructors.
    pub fn register_type_def(&mut self, td: &TypeDef) {
        match td {
            TypeDef::Sum {
                name: type_name,
                variants,
                ..
            } => {
                if self.arena.find_type_id(type_name).is_none() {
                    self.arena.register_sum_type(
                        type_name,
                        variants
                            .iter()
                            .map(|variant| variant.name.clone())
                            .collect(),
                    );
                }
                let mut members = HashMap::new();
                for variant in variants {
                    if variant.fields.is_empty() {
                        // Zero-arg variant: stored directly as a Value
                        members.insert(
                            variant.name.clone(),
                            Value::Variant {
                                type_name: type_name.clone(),
                                variant: variant.name.clone(),
                                fields: vec![].into(),
                            },
                        );
                    } else {
                        // Constructor function
                        members.insert(
                            variant.name.clone(),
                            Value::Builtin(format!("__ctor:{}:{}", type_name, variant.name)),
                        );
                    }
                }
                self.define(
                    type_name.clone(),
                    Value::Namespace {
                        name: type_name.clone(),
                        members,
                    },
                );
            }
            TypeDef::Product { name, fields, .. } => {
                if self.arena.find_type_id(name).is_none() {
                    self.arena.register_record_type(
                        name,
                        fields.iter().map(|(field, _)| field.clone()).collect(),
                    );
                }
                // Product types are constructed via Expr::RecordCreate.
                // Keep declaration field order so runtime records are canonicalized
                // and positional record matches stay stable.
                let schema = fields.iter().map(|(field, _)| field.clone()).collect();
                self.record_schemas.insert(name.clone(), schema);
            }
        }
    }

    pub fn exec_fn_def(&mut self, fd: &FnDef) -> Result<(), RuntimeError> {
        let lower_ctx = super::ir_bridge::InterpreterLowerCtx::new(self);
        let val = Value::Fn(Rc::new(crate::value::FunctionValue {
            name: Rc::new(fd.name.clone()),
            params: Rc::new(fd.params.clone()),
            return_type: Rc::new(fd.return_type.clone()),
            effects: Rc::new(fd.effects.iter().map(|e| e.node.clone()).collect()),
            body: Rc::clone(&fd.body),
            lowered_body: super::lowered::lower_fn_body(fd.body.as_ref(), &lower_ctx, &fd.name),
            resolution: fd.resolution.clone(),
            memo_eligible: self.memo_fns.contains(&fd.name),
            home_globals: None,
        }));
        self.define(fd.name.clone(), val);
        Ok(())
    }

    pub fn exec_stmt(&mut self, stmt: &Stmt) -> Result<Value, RuntimeError> {
        match stmt {
            Stmt::Binding(name, _, expr) => {
                let val = self.eval_expr(expr)?;
                self.define(name.clone(), val);
                Ok(Value::Unit)
            }
            Stmt::Expr(expr) => self.eval_expr(expr),
        }
    }

    pub fn exec_body(&mut self, stmts: &[Stmt]) -> Result<Value, RuntimeError> {
        let mut last = Value::Unit;
        for stmt in stmts {
            last = self.exec_stmt(stmt)?;
        }
        Ok(last)
    }

    pub fn run_file(&mut self, source: &str) -> Result<Value, RuntimeError> {
        let mut items = parse_source(source).map_err(RuntimeError::Error)?;
        crate::resolver::resolve_program(&mut items);

        // First pass: register all top-level definitions
        for item in &items {
            match item {
                TopLevel::FnDef(fd) => self.exec_fn_def(fd)?,
                TopLevel::Stmt(s) => {
                    self.exec_stmt(s)?;
                }
                _ => {}
            }
        }

        // Second pass: run main() if it exists
        let main_fn = self.lookup("main");
        if let Ok(fn_val) = main_fn {
            let allowed = Self::callable_declared_effects(&fn_val);
            self.call_value_with_effects_pub(fn_val, vec![], "<main>", allowed)?;
        }

        Ok(Value::Unit)
    }
}