mir-analyzer 0.29.0

Analysis engine for the mir PHP static analyzer
Documentation
use std::sync::Arc;

use super::helpers::extract_simple_var;
use super::ExpressionAnalyzer;
use crate::flow_state::FlowState;
use crate::symbol::ReferenceKind;
use mir_issues::{IssueKind, Severity};
use mir_types::{Atomic, Type};
use php_ast::owned::Expr;

impl<'a> ExpressionAnalyzer<'a> {
    pub(super) fn analyze_variable(
        &mut self,
        name: &str,
        expr: &Expr,
        ctx: &mut FlowState,
    ) -> Type {
        let name_str = name.trim_start_matches('$');
        if !ctx.var_is_defined(name_str) {
            // isset() and empty() don't error on undefined variables
            if !self.suppress_undefined_errors {
                if ctx.var_possibly_defined(name_str) {
                    self.emit(
                        IssueKind::PossiblyUndefinedVariable {
                            name: name_str.to_string(),
                        },
                        Severity::Warning,
                        expr.span,
                    );
                } else if name_str == "this" {
                    self.emit(
                        IssueKind::InvalidScope {
                            in_class: ctx.self_fqcn.is_some(),
                        },
                        Severity::Error,
                        expr.span,
                    );
                } else {
                    self.emit(
                        IssueKind::UndefinedVariable {
                            name: name_str.to_string(),
                        },
                        Severity::Error,
                        expr.span,
                    );
                }
            }
        }
        ctx.read_vars.insert(mir_types::Name::from(name_str));
        let ty = if name_str == "this" && !ctx.var_is_defined("this") {
            Type::never()
        } else {
            ctx.get_var(name_str)
        };
        self.record_symbol(
            expr.span,
            ReferenceKind::Variable(Arc::from(name_str)),
            ty.clone(),
        );
        ty
    }

    pub(super) fn analyze_variable_variable(&mut self, inner: &Expr, ctx: &mut FlowState) -> Type {
        let inner_ty = self.analyze(inner, ctx);
        if let Some(var_name) = extract_simple_var(inner) {
            ctx.read_vars
                .insert(mir_types::Name::from(var_name.as_str()));
            for atomic in &inner_ty.types {
                if let Atomic::TLiteralString(accessed_var_name) = atomic {
                    ctx.read_vars
                        .insert(mir_types::Name::from(accessed_var_name.as_ref()));
                }
            }
        }
        Type::mixed()
    }

    pub(super) fn analyze_identifier(
        &mut self,
        name: &str,
        expr: &Expr,
        _ctx: &mut FlowState,
    ) -> Type {
        let name_str: &str = name;
        let name_str = name_str.strip_prefix('\\').unwrap_or(name_str);
        let ns_qualified = self
            .db
            .file_namespace(self.file.as_ref())
            .map(|ns| format!("{}\\{}", ns, name_str));

        let resolve_pull = |fqn: &str| -> Option<mir_types::Type> {
            let here = crate::db::Fqcn::from_str(self.db, fqn);
            crate::db::find_global_constant(self.db, here).map(|arc_union| (*arc_union).clone())
        };

        let ty = ns_qualified
            .as_deref()
            .and_then(resolve_pull)
            .or_else(|| resolve_pull(name_str));

        if let Some(ty) = ty {
            ty
        } else {
            self.emit(
                IssueKind::UndefinedConstant {
                    name: name_str.to_string(),
                },
                Severity::Error,
                expr.span,
            );
            Type::mixed()
        }
    }
}