luaur-analysis 0.1.3

Luau type checker and type inference (Rust).
Documentation
use crate::functions::as_mutable_type_pack_alt_d::as_mutable_type_pack;
use crate::functions::flatten_type_pack::flatten_type_pack_id;
use crate::functions::parse_pattern_string::parse_pattern_string;
use crate::records::function_type::FunctionType;
use crate::records::magic_function_call_context::MagicFunctionCallContext;
use crate::records::magic_gmatch::MagicGmatch;
use crate::records::type_pack::TypePack;
use crate::type_aliases::type_pack_variant::TypePackVariant;
use luaur_ast::records::ast_expr_constant_string::AstExprConstantString;
use luaur_ast::records::ast_node::AstNode;
use luaur_ast::rtti::ast_node_as;

impl MagicGmatch {
    pub fn infer(&self, context: &MagicFunctionCallContext) -> bool {
        let (params, _tail) = flatten_type_pack_id(context.arguments);

        if params.len() != 2 {
            return false;
        }

        let solver = unsafe { context.solver.as_ref() };
        let arena = unsafe { &mut *solver.arena };
        let call_site = unsafe { context.call_site.as_ref() };

        let index = if call_site.self_ { 0 } else { 1 };
        let pattern = if call_site.args.size > index {
            let expr = unsafe { *call_site.args.data.add(index) };
            unsafe { ast_node_as::<AstExprConstantString>(expr as *mut AstNode) }
        } else {
            core::ptr::null_mut()
        };

        if pattern.is_null() {
            return false;
        }

        let return_types = unsafe {
            let pattern = &*pattern;
            parse_pattern_string(
                core::ptr::NonNull::new_unchecked(solver.builtin_types),
                pattern.value.data,
                pattern.value.size,
            )
        };

        if return_types.is_empty() {
            return false;
        }

        unsafe {
            let builtin_types = &*solver.builtin_types;
            (*context.solver.as_ptr()).constraint_solver_unify(
                context.constraint.as_ptr(),
                params[0],
                builtin_types.stringType,
            );
        }

        let empty_pack = arena.add_type_pack_t(TypePack {
            head: Vec::new(),
            tail: None,
        });
        let return_list = arena.add_type_pack_t(TypePack {
            head: return_types,
            tail: None,
        });
        let iterator_type = arena.add_type(FunctionType::function_type_new(
            empty_pack,
            return_list,
            None,
            false,
        ));
        let res_type_pack = arena.add_type_pack_t(TypePack {
            head: vec![iterator_type],
            tail: None,
        });

        let result_mut = as_mutable_type_pack(context.result);
        unsafe {
            (*result_mut).ty = TypePackVariant::Bound(res_type_pack);
        }

        true
    }
}