luaur-analysis 0.1.3

Luau type checker and type inference (Rust).
Documentation
use crate::functions::get_type_alt_j::get_type_id;
use crate::records::apply_mapped_generics::ApplyMappedGenerics;
use crate::records::generic_type::GenericType;
use crate::records::intersection_builder::IntersectionBuilder;
use crate::records::union_builder::UnionBuilder;
use crate::type_aliases::type_id::TypeId;
use luaur_common::macros::luau_assert::LUAU_ASSERT;

impl ApplyMappedGenerics {
    pub fn clean_type_id(&mut self, ty: TypeId) -> TypeId {
        let bounds = unsafe { (*self.env).get_mapped_type_bounds(ty, self.ice_reporter) };
        let lower_bound = &bounds.lower_bound;
        let upper_bound = &bounds.upper_bound;

        if upper_bound.empty() && lower_bound.empty() {
            // No bounds for the generic we're mapping.
            // In this case, unknown vs never is an arbitrary choice:
            // ie, does it matter if we map add<A, A> to add<unknown, unknown> or add<never, never> in the context of subtyping?
            // We choose unknown here, since it's closest to the original behavior.
            unsafe { (*self.builtin_types).unknownType }
        } else if !upper_bound.empty() {
            let mut ib = IntersectionBuilder::intersection_builder(self.arena, self.builtin_types);
            for &ub in &upper_bound.order {
                // NOTE: The original implementation skips over generic
                // types, but that seems incorrect to me.
                if unsafe { get_type_id::<GenericType>(ub).is_null() } {
                    ib.add(ub);
                }
            }
            ib.build()
        } else if !lower_bound.empty() {
            let mut ub_builder = UnionBuilder::union_builder(self.arena, self.builtin_types);
            for &lb in &lower_bound.order {
                // NOTE: The original implementation skips over generic
                // types, but that seems incorrect to me.
                if unsafe { get_type_id::<GenericType>(lb).is_null() } {
                    ub_builder.add(lb);
                }
            }
            ub_builder.build()
        } else {
            LUAU_ASSERT!(false);
            unsafe { (*self.builtin_types).unknownType }
        }
    }
}