2016-06-20 06:38:13 +02:00
|
|
|
<?php
|
2021-12-15 04:58:32 +01:00
|
|
|
|
2016-07-26 00:37:44 +02:00
|
|
|
namespace Psalm;
|
2016-06-20 06:38:13 +02:00
|
|
|
|
2022-05-27 00:46:20 +02:00
|
|
|
use InvalidArgumentException;
|
2021-06-08 04:55:21 +02:00
|
|
|
use Psalm\Internal\Analyzer\StatementsAnalyzer;
|
|
|
|
use Psalm\Internal\Clause;
|
2021-12-04 03:37:19 +01:00
|
|
|
use Psalm\Internal\ReferenceConstraint;
|
|
|
|
use Psalm\Internal\Scope\CaseScope;
|
|
|
|
use Psalm\Internal\Scope\FinallyScope;
|
|
|
|
use Psalm\Internal\Scope\LoopScope;
|
2021-06-08 04:55:21 +02:00
|
|
|
use Psalm\Internal\Type\AssertionReconciler;
|
|
|
|
use Psalm\Storage\FunctionLikeStorage;
|
2021-12-13 16:28:14 +01:00
|
|
|
use Psalm\Type\Atomic\DependentType;
|
2021-12-13 04:45:57 +01:00
|
|
|
use Psalm\Type\Atomic\TArray;
|
2021-06-08 04:55:21 +02:00
|
|
|
use Psalm\Type\Union;
|
2022-01-07 23:16:36 +01:00
|
|
|
use RuntimeException;
|
2021-06-08 04:55:21 +02:00
|
|
|
|
2019-07-05 22:24:00 +02:00
|
|
|
use function array_keys;
|
2021-06-08 04:55:21 +02:00
|
|
|
use function array_search;
|
2022-01-11 18:36:37 +01:00
|
|
|
use function array_shift;
|
|
|
|
use function assert;
|
2019-07-05 22:24:00 +02:00
|
|
|
use function count;
|
|
|
|
use function in_array;
|
2021-06-08 04:55:21 +02:00
|
|
|
use function is_int;
|
2019-07-05 22:24:00 +02:00
|
|
|
use function json_encode;
|
|
|
|
use function preg_match;
|
|
|
|
use function preg_quote;
|
|
|
|
use function preg_replace;
|
2019-06-26 22:52:29 +02:00
|
|
|
use function strpos;
|
|
|
|
use function strtolower;
|
2017-04-02 21:26:10 +02:00
|
|
|
|
2022-01-05 18:32:43 +01:00
|
|
|
use const JSON_THROW_ON_ERROR;
|
|
|
|
|
2022-02-04 03:51:18 +01:00
|
|
|
final class Context
|
2016-06-20 06:38:13 +02:00
|
|
|
{
|
2016-10-18 22:14:52 +02:00
|
|
|
/**
|
2021-12-13 16:28:14 +01:00
|
|
|
* @var array<string, Union>
|
2016-10-18 22:14:52 +02:00
|
|
|
*/
|
2016-06-20 06:38:13 +02:00
|
|
|
public $vars_in_scope = [];
|
|
|
|
|
2016-10-18 22:14:52 +02:00
|
|
|
/**
|
2017-12-02 19:32:20 +01:00
|
|
|
* @var array<string, bool>
|
2016-10-18 22:14:52 +02:00
|
|
|
*/
|
2016-06-20 06:38:13 +02:00
|
|
|
public $vars_possibly_in_scope = [];
|
|
|
|
|
2022-01-07 23:16:36 +01:00
|
|
|
/**
|
|
|
|
* Keeps track of how many times a var_in_scope has been referenced. May not be set for all $vars_in_scope.
|
|
|
|
*
|
|
|
|
* @var array<string, int<0, max>>
|
|
|
|
*/
|
|
|
|
public $referenced_counts = [];
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Maps references to referenced variables for the current scope.
|
|
|
|
* With `$b = &$a`, this will contain `['$b' => '$a']`.
|
|
|
|
*
|
|
|
|
* All keys and values in this array are guaranteed to be set in $vars_in_scope.
|
|
|
|
*
|
|
|
|
* To check if a variable was passed or returned by reference, or
|
|
|
|
* references an object property or array item, see Union::$by_ref.
|
|
|
|
*
|
|
|
|
* @var array<string, string>
|
|
|
|
*/
|
|
|
|
public $references_in_scope = [];
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Set of references to variables in another scope. These references will be marked as used if they are assigned to.
|
|
|
|
*
|
|
|
|
* @var array<string, true>
|
|
|
|
*/
|
|
|
|
public $references_to_external_scope = [];
|
|
|
|
|
|
|
|
/**
|
|
|
|
* A set of globals that are referenced somewhere.
|
|
|
|
*
|
|
|
|
* @var array<string, true>
|
|
|
|
*/
|
|
|
|
public $referenced_globals = [];
|
|
|
|
|
|
|
|
/**
|
|
|
|
* A set of references that might still be in scope from a scope likely to cause confusion. This applies
|
|
|
|
* to references set inside a loop or if statement, since it's easy to forget about PHP's weird scope
|
|
|
|
* rules, and assinging to a reference will change the referenced variable rather than shadowing it.
|
|
|
|
*
|
|
|
|
* @var array<string, CodeLocation>
|
|
|
|
*/
|
|
|
|
public $references_possibly_from_confusing_scope = [];
|
|
|
|
|
2017-03-13 16:23:26 +01:00
|
|
|
/**
|
|
|
|
* Whether or not we're inside the conditional of an if/where etc.
|
|
|
|
*
|
|
|
|
* This changes whether or not the context is cloned
|
|
|
|
*
|
2017-05-27 02:16:18 +02:00
|
|
|
* @var bool
|
2017-03-13 16:23:26 +01:00
|
|
|
*/
|
|
|
|
public $inside_conditional = false;
|
2016-06-20 22:30:31 +02:00
|
|
|
|
2018-01-08 05:59:17 +01:00
|
|
|
/**
|
|
|
|
* Whether or not we're inside an isset call
|
|
|
|
*
|
2020-07-24 15:34:05 +02:00
|
|
|
* Inside issets Psalm is more lenient about certain things
|
2018-01-08 05:59:17 +01:00
|
|
|
*
|
|
|
|
* @var bool
|
|
|
|
*/
|
|
|
|
public $inside_isset = false;
|
|
|
|
|
2018-02-10 16:30:08 +01:00
|
|
|
/**
|
2018-04-19 18:16:00 +02:00
|
|
|
* Whether or not we're inside an unset call, where
|
|
|
|
* we don't care about possibly undefined variables
|
2018-02-10 16:30:08 +01:00
|
|
|
*
|
|
|
|
* @var bool
|
|
|
|
*/
|
|
|
|
public $inside_unset = false;
|
|
|
|
|
2018-04-19 18:16:00 +02:00
|
|
|
/**
|
|
|
|
* Whether or not we're inside an class_exists call, where
|
|
|
|
* we don't care about possibly undefined classes
|
|
|
|
*
|
|
|
|
* @var bool
|
|
|
|
*/
|
|
|
|
public $inside_class_exists = false;
|
|
|
|
|
2019-05-07 02:47:55 +02:00
|
|
|
/**
|
|
|
|
* Whether or not we're inside a function/method call
|
|
|
|
*
|
|
|
|
* @var bool
|
|
|
|
*/
|
|
|
|
public $inside_call = false;
|
|
|
|
|
2020-09-30 18:28:13 +02:00
|
|
|
/**
|
|
|
|
* Whether or not we're inside any other situation that treats a variable as used
|
|
|
|
*
|
|
|
|
* @var bool
|
|
|
|
*/
|
2021-06-25 15:54:39 +02:00
|
|
|
public $inside_general_use = false;
|
2020-09-30 18:28:13 +02:00
|
|
|
|
2021-06-10 20:18:15 +02:00
|
|
|
/**
|
|
|
|
* Whether or not we're inside a return expression
|
|
|
|
*
|
|
|
|
* @var bool
|
|
|
|
*/
|
|
|
|
public $inside_return = false;
|
|
|
|
|
2020-02-22 04:15:25 +01:00
|
|
|
/**
|
|
|
|
* Whether or not we're inside a throw
|
|
|
|
*
|
|
|
|
* @var bool
|
|
|
|
*/
|
|
|
|
public $inside_throw = false;
|
|
|
|
|
2019-08-13 19:15:23 +02:00
|
|
|
/**
|
|
|
|
* Whether or not we're inside an assignment
|
|
|
|
*
|
|
|
|
* @var bool
|
|
|
|
*/
|
|
|
|
public $inside_assignment = false;
|
|
|
|
|
2022-02-13 22:02:46 +01:00
|
|
|
/**
|
|
|
|
* Whether or not we're inside a try block.
|
|
|
|
*
|
|
|
|
* @var bool
|
|
|
|
*/
|
|
|
|
public $inside_try = false;
|
|
|
|
|
2017-06-21 20:22:52 +02:00
|
|
|
/**
|
2018-01-28 22:52:57 +01:00
|
|
|
* @var null|CodeLocation
|
2017-06-21 20:22:52 +02:00
|
|
|
*/
|
2021-09-26 23:24:07 +02:00
|
|
|
public $include_location;
|
2017-06-21 20:22:52 +02:00
|
|
|
|
2016-10-18 22:14:52 +02:00
|
|
|
/**
|
|
|
|
* @var string|null
|
2021-12-01 23:38:45 +01:00
|
|
|
* The name of the current class. Null if outside a class.
|
2016-10-18 22:14:52 +02:00
|
|
|
*/
|
2016-08-08 17:28:14 +02:00
|
|
|
public $self;
|
|
|
|
|
2016-10-18 22:14:52 +02:00
|
|
|
/**
|
|
|
|
* @var string|null
|
|
|
|
*/
|
2016-08-08 17:28:14 +02:00
|
|
|
public $parent;
|
|
|
|
|
2016-10-18 22:14:52 +02:00
|
|
|
/**
|
2017-05-27 02:16:18 +02:00
|
|
|
* @var bool
|
2016-10-18 22:14:52 +02:00
|
|
|
*/
|
|
|
|
public $check_classes = true;
|
|
|
|
|
|
|
|
/**
|
2017-05-27 02:16:18 +02:00
|
|
|
* @var bool
|
2016-10-18 22:14:52 +02:00
|
|
|
*/
|
|
|
|
public $check_variables = true;
|
|
|
|
|
|
|
|
/**
|
2017-05-27 02:16:18 +02:00
|
|
|
* @var bool
|
2016-10-18 22:14:52 +02:00
|
|
|
*/
|
|
|
|
public $check_methods = true;
|
|
|
|
|
|
|
|
/**
|
2017-05-27 02:16:18 +02:00
|
|
|
* @var bool
|
2016-10-18 22:14:52 +02:00
|
|
|
*/
|
|
|
|
public $check_consts = true;
|
|
|
|
|
|
|
|
/**
|
2017-05-27 02:16:18 +02:00
|
|
|
* @var bool
|
2016-10-18 22:14:52 +02:00
|
|
|
*/
|
|
|
|
public $check_functions = true;
|
|
|
|
|
2016-10-22 19:23:18 +02:00
|
|
|
/**
|
|
|
|
* A list of classes checked with class_exists
|
2016-11-02 07:29:00 +01:00
|
|
|
*
|
2021-03-29 06:11:45 +02:00
|
|
|
* @var array<lowercase-string,true>
|
2016-10-22 19:23:18 +02:00
|
|
|
*/
|
2018-08-10 05:29:30 +02:00
|
|
|
public $phantom_classes = [];
|
|
|
|
|
|
|
|
/**
|
|
|
|
* A list of files checked with file_exists
|
|
|
|
*
|
|
|
|
* @var array<string,bool>
|
|
|
|
*/
|
|
|
|
public $phantom_files = [];
|
2016-10-22 19:23:18 +02:00
|
|
|
|
2016-12-27 19:58:58 +01:00
|
|
|
/**
|
|
|
|
* A list of clauses in Conjunctive Normal Form
|
|
|
|
*
|
2019-10-09 00:44:46 +02:00
|
|
|
* @var list<Clause>
|
2016-12-27 19:58:58 +01:00
|
|
|
*/
|
|
|
|
public $clauses = [];
|
|
|
|
|
2019-12-08 06:49:34 +01:00
|
|
|
/**
|
|
|
|
* A list of hashed clauses that have already been factored in
|
|
|
|
*
|
2020-08-26 21:35:29 +02:00
|
|
|
* @var list<string|int>
|
2019-12-08 06:49:34 +01:00
|
|
|
*/
|
|
|
|
public $reconciled_expression_clauses = [];
|
|
|
|
|
2017-01-12 03:37:53 +01:00
|
|
|
/**
|
|
|
|
* Whether or not to do a deep analysis and collect mutations to this context
|
|
|
|
*
|
2017-05-27 02:16:18 +02:00
|
|
|
* @var bool
|
2017-01-12 03:37:53 +01:00
|
|
|
*/
|
|
|
|
public $collect_mutations = false;
|
|
|
|
|
2017-01-27 07:23:12 +01:00
|
|
|
/**
|
2020-04-21 06:04:47 +02:00
|
|
|
* Whether or not to do a deep analysis and collect initializations from private or final methods
|
2017-01-27 07:23:12 +01:00
|
|
|
*
|
2017-05-27 02:16:18 +02:00
|
|
|
* @var bool
|
2017-01-27 07:23:12 +01:00
|
|
|
*/
|
|
|
|
public $collect_initializations = false;
|
|
|
|
|
2020-04-21 06:04:47 +02:00
|
|
|
/**
|
|
|
|
* Whether or not to do a deep analysis and collect initializations from public non-final methods
|
|
|
|
*
|
|
|
|
* @var bool
|
|
|
|
*/
|
|
|
|
public $collect_nonprivate_initializations = false;
|
|
|
|
|
2018-01-24 19:11:23 +01:00
|
|
|
/**
|
|
|
|
* Stored to prevent re-analysing methods when checking for initialised properties
|
|
|
|
*
|
|
|
|
* @var array<string, bool>|null
|
|
|
|
*/
|
2021-09-26 23:24:07 +02:00
|
|
|
public $initialized_methods;
|
2018-01-24 19:11:23 +01:00
|
|
|
|
2017-01-15 21:58:40 +01:00
|
|
|
/**
|
2021-12-13 16:28:14 +01:00
|
|
|
* @var array<string, Union>
|
2017-01-15 21:58:40 +01:00
|
|
|
*/
|
|
|
|
public $constants = [];
|
|
|
|
|
2018-06-22 07:13:49 +02:00
|
|
|
/**
|
|
|
|
* Whether or not to track exceptions
|
|
|
|
*
|
|
|
|
* @var bool
|
|
|
|
*/
|
|
|
|
public $collect_exceptions = false;
|
|
|
|
|
2017-02-01 05:24:33 +01:00
|
|
|
/**
|
2022-03-15 23:40:31 +01:00
|
|
|
* A list of variables that have been referenced in conditionals
|
2017-02-01 05:24:33 +01:00
|
|
|
*
|
|
|
|
* @var array<string, bool>
|
|
|
|
*/
|
2022-03-15 23:40:31 +01:00
|
|
|
public $cond_referenced_var_ids = [];
|
2017-02-01 05:24:33 +01:00
|
|
|
|
2017-02-23 06:25:28 +01:00
|
|
|
/**
|
|
|
|
* A list of variables that have been passed by reference (where we know their type)
|
|
|
|
*
|
2021-12-04 03:37:19 +01:00
|
|
|
* @var array<string, ReferenceConstraint>
|
2017-02-23 06:25:28 +01:00
|
|
|
*/
|
2020-09-30 18:28:13 +02:00
|
|
|
public $byref_constraints = [];
|
2017-02-23 06:25:28 +01:00
|
|
|
|
2017-09-03 00:15:52 +02:00
|
|
|
/**
|
|
|
|
* A list of vars that have been assigned to
|
|
|
|
*
|
2020-11-01 17:26:42 +01:00
|
|
|
* @var array<string, int>
|
2017-09-03 00:15:52 +02:00
|
|
|
*/
|
2017-11-25 17:21:45 +01:00
|
|
|
public $assigned_var_ids = [];
|
2017-09-03 00:15:52 +02:00
|
|
|
|
2018-05-18 17:02:50 +02:00
|
|
|
/**
|
|
|
|
* A list of vars that have been may have been assigned to
|
|
|
|
*
|
|
|
|
* @var array<string, bool>
|
|
|
|
*/
|
|
|
|
public $possibly_assigned_var_ids = [];
|
|
|
|
|
2018-06-22 07:13:49 +02:00
|
|
|
/**
|
|
|
|
* A list of classes or interfaces that may have been thrown
|
|
|
|
*
|
2019-04-03 01:42:23 +02:00
|
|
|
* @var array<string, array<array-key, CodeLocation>>
|
2018-06-22 07:13:49 +02:00
|
|
|
*/
|
|
|
|
public $possibly_thrown_exceptions = [];
|
|
|
|
|
2017-12-06 06:56:00 +01:00
|
|
|
/**
|
|
|
|
* @var bool
|
|
|
|
*/
|
|
|
|
public $is_global = false;
|
|
|
|
|
2017-12-17 16:58:03 +01:00
|
|
|
/**
|
|
|
|
* @var array<string, bool>
|
|
|
|
*/
|
|
|
|
public $protected_var_ids = [];
|
|
|
|
|
2018-01-21 22:24:20 +01:00
|
|
|
/**
|
|
|
|
* If we've branched from the main scope, a byte offset for where that branch happened
|
|
|
|
*
|
|
|
|
* @var int|null
|
|
|
|
*/
|
|
|
|
public $branch_point;
|
|
|
|
|
2018-01-24 06:01:08 +01:00
|
|
|
/**
|
2020-01-27 18:17:12 +01:00
|
|
|
* What does break mean in this context?
|
2018-01-24 06:01:08 +01:00
|
|
|
*
|
2020-01-27 18:17:12 +01:00
|
|
|
* 'loop' means we're breaking out of a loop,
|
|
|
|
* 'switch' means we're breaking out of a switch
|
|
|
|
*
|
|
|
|
* @var list<'loop'|'switch'>
|
2018-01-24 06:01:08 +01:00
|
|
|
*/
|
2020-01-27 18:17:12 +01:00
|
|
|
public $break_types = [];
|
2018-01-24 06:01:08 +01:00
|
|
|
|
2018-05-03 19:56:30 +02:00
|
|
|
/**
|
|
|
|
* @var bool
|
|
|
|
*/
|
|
|
|
public $inside_loop = false;
|
|
|
|
|
2018-06-17 03:54:44 +02:00
|
|
|
/**
|
2021-12-04 03:37:19 +01:00
|
|
|
* @var LoopScope|null
|
2018-06-17 03:54:44 +02:00
|
|
|
*/
|
2021-09-26 23:24:07 +02:00
|
|
|
public $loop_scope;
|
2018-06-17 03:54:44 +02:00
|
|
|
|
|
|
|
/**
|
2021-12-04 03:37:19 +01:00
|
|
|
* @var CaseScope|null
|
2018-06-17 03:54:44 +02:00
|
|
|
*/
|
2021-09-26 23:24:07 +02:00
|
|
|
public $case_scope;
|
2018-06-17 03:54:44 +02:00
|
|
|
|
2020-09-21 21:16:19 +02:00
|
|
|
/**
|
2021-12-04 03:37:19 +01:00
|
|
|
* @var FinallyScope|null
|
2020-09-21 21:16:19 +02:00
|
|
|
*/
|
2021-09-26 23:24:07 +02:00
|
|
|
public $finally_scope;
|
2020-09-21 21:16:19 +02:00
|
|
|
|
2019-12-08 06:49:34 +01:00
|
|
|
/**
|
|
|
|
* @var Context|null
|
|
|
|
*/
|
2022-01-29 00:30:47 +01:00
|
|
|
public $if_body_context;
|
2019-12-08 06:49:34 +01:00
|
|
|
|
2018-08-28 23:42:39 +02:00
|
|
|
/**
|
|
|
|
* @var bool
|
|
|
|
*/
|
|
|
|
public $strict_types = false;
|
|
|
|
|
2018-09-26 00:37:24 +02:00
|
|
|
/**
|
|
|
|
* @var string|null
|
|
|
|
*/
|
2019-12-29 00:37:55 +01:00
|
|
|
public $calling_function_id;
|
2018-09-26 00:37:24 +02:00
|
|
|
|
2020-03-26 17:35:27 +01:00
|
|
|
/**
|
|
|
|
* @var lowercase-string|null
|
|
|
|
*/
|
|
|
|
public $calling_method_id;
|
|
|
|
|
2019-02-26 07:03:33 +01:00
|
|
|
/**
|
|
|
|
* @var bool
|
|
|
|
*/
|
|
|
|
public $inside_negation = false;
|
|
|
|
|
2019-04-20 23:49:49 +02:00
|
|
|
/**
|
|
|
|
* @var bool
|
|
|
|
*/
|
|
|
|
public $ignore_variable_property = false;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @var bool
|
|
|
|
*/
|
|
|
|
public $ignore_variable_method = false;
|
|
|
|
|
2019-07-18 07:31:48 +02:00
|
|
|
/**
|
|
|
|
* @var bool
|
|
|
|
*/
|
|
|
|
public $pure = false;
|
|
|
|
|
2019-08-30 18:36:35 +02:00
|
|
|
/**
|
|
|
|
* @var bool
|
2022-01-01 11:13:45 +01:00
|
|
|
* Set by @psalm-immutable
|
2019-08-30 18:36:35 +02:00
|
|
|
*/
|
|
|
|
public $mutation_free = false;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @var bool
|
2022-01-01 11:13:45 +01:00
|
|
|
* Set by @psalm-external-mutation-free
|
2019-08-30 18:36:35 +02:00
|
|
|
*/
|
|
|
|
public $external_mutation_free = false;
|
|
|
|
|
2019-10-22 16:40:37 +02:00
|
|
|
/**
|
|
|
|
* @var bool
|
|
|
|
*/
|
|
|
|
public $error_suppressing = false;
|
|
|
|
|
2020-03-15 19:31:41 +01:00
|
|
|
/**
|
|
|
|
* @var bool
|
|
|
|
*/
|
|
|
|
public $has_returned = false;
|
|
|
|
|
2022-01-24 22:35:42 +01:00
|
|
|
/**
|
|
|
|
* @var array<string, true>
|
|
|
|
*/
|
|
|
|
public $parent_remove_vars = [];
|
|
|
|
|
2022-02-04 03:51:18 +01:00
|
|
|
/** @internal */
|
2020-09-07 01:36:47 +02:00
|
|
|
public function __construct(?string $self = null)
|
2016-08-14 03:14:32 +02:00
|
|
|
{
|
|
|
|
$this->self = $self;
|
|
|
|
}
|
|
|
|
|
2019-06-30 03:06:21 +02:00
|
|
|
public function __destruct()
|
|
|
|
{
|
|
|
|
$this->case_scope = null;
|
|
|
|
}
|
|
|
|
|
2016-06-20 06:38:13 +02:00
|
|
|
/**
|
2016-11-02 07:29:00 +01:00
|
|
|
* Updates the parent context, looking at the changes within a block and then applying those changes, where
|
|
|
|
* necessary, to the parent context
|
2016-06-20 06:38:13 +02:00
|
|
|
*
|
2016-11-02 07:29:00 +01:00
|
|
|
* @param bool $has_leaving_statements whether or not the parent scope is abandoned between
|
|
|
|
* $start_context and $end_context
|
2020-01-22 15:57:59 +01:00
|
|
|
* @param array<string, bool> $updated_vars
|
2016-06-20 06:38:13 +02:00
|
|
|
*/
|
2016-11-02 07:29:00 +01:00
|
|
|
public function update(
|
|
|
|
Context $start_context,
|
|
|
|
Context $end_context,
|
2020-09-07 01:36:47 +02:00
|
|
|
bool $has_leaving_statements,
|
2016-11-02 07:29:00 +01:00
|
|
|
array $vars_to_update,
|
|
|
|
array &$updated_vars
|
2020-09-12 17:24:05 +02:00
|
|
|
): void {
|
2022-07-31 01:20:05 +02:00
|
|
|
foreach ($start_context->vars_in_scope as $var_id => $old_type) {
|
2018-02-17 17:24:08 +01:00
|
|
|
// this is only true if there was some sort of type negation
|
|
|
|
if (in_array($var_id, $vars_to_update, true)) {
|
|
|
|
// if we're leaving, we're effectively deleting the possibility of the if types
|
|
|
|
$new_type = !$has_leaving_statements && $end_context->hasVariable($var_id)
|
|
|
|
? $end_context->vars_in_scope[$var_id]
|
|
|
|
: null;
|
|
|
|
|
2021-09-26 22:06:24 +02:00
|
|
|
$existing_type = $this->vars_in_scope[$var_id] ?? null;
|
2018-02-17 17:24:08 +01:00
|
|
|
|
2022-07-31 01:20:05 +02:00
|
|
|
if (!$existing_type) {
|
|
|
|
if ($new_type) {
|
2022-11-04 19:04:23 +01:00
|
|
|
$this->vars_in_scope[$var_id] = $new_type;
|
2022-07-31 01:20:05 +02:00
|
|
|
$updated_vars[$var_id] = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// if the type changed within the block of statements, process the replacement
|
|
|
|
// also never allow ourselves to remove all types from a union
|
|
|
|
if ((!$new_type || !$old_type->equals($new_type))
|
|
|
|
&& ($new_type || count($existing_type->getAtomicTypes()) > 1)
|
|
|
|
) {
|
2022-10-03 10:45:36 +02:00
|
|
|
$existing_type = $existing_type
|
|
|
|
->getBuilder()
|
2022-10-03 15:13:47 +02:00
|
|
|
->substitute($old_type, $new_type);
|
2022-07-31 01:20:05 +02:00
|
|
|
|
|
|
|
if ($new_type && $new_type->from_docblock) {
|
2022-10-03 15:13:47 +02:00
|
|
|
$existing_type = $existing_type->setFromDocblock();
|
2022-07-31 01:20:05 +02:00
|
|
|
}
|
2022-10-03 15:13:47 +02:00
|
|
|
$existing_type = $existing_type->freeze();
|
2022-07-31 01:20:05 +02:00
|
|
|
|
2018-02-17 17:24:08 +01:00
|
|
|
$updated_vars[$var_id] = true;
|
2016-08-10 07:54:45 +02:00
|
|
|
}
|
2022-07-31 01:20:05 +02:00
|
|
|
|
|
|
|
$this->vars_in_scope[$var_id] = $existing_type;
|
2016-06-20 06:38:13 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-01-07 23:16:36 +01:00
|
|
|
/**
|
|
|
|
* Updates the list of possible references from a confusing scope,
|
2022-01-07 23:57:22 +01:00
|
|
|
* such as a reference created in an if that might later be reused.
|
2022-01-07 23:16:36 +01:00
|
|
|
*/
|
|
|
|
public function updateReferencesPossiblyFromConfusingScope(
|
|
|
|
Context $confusing_scope_context,
|
|
|
|
StatementsAnalyzer $statements_analyzer
|
|
|
|
): void {
|
2022-01-08 00:41:39 +01:00
|
|
|
$references = $confusing_scope_context->references_in_scope
|
|
|
|
+ $confusing_scope_context->references_to_external_scope;
|
|
|
|
foreach ($references as $reference_id => $_) {
|
2022-01-07 23:16:36 +01:00
|
|
|
if (!isset($this->references_in_scope[$reference_id])
|
|
|
|
&& !isset($this->references_to_external_scope[$reference_id])
|
|
|
|
&& $reference_location = $statements_analyzer->getFirstAppearance($reference_id)
|
|
|
|
) {
|
|
|
|
$this->references_possibly_from_confusing_scope[$reference_id] = $reference_location;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
$this->references_possibly_from_confusing_scope +=
|
|
|
|
$confusing_scope_context->references_possibly_from_confusing_scope;
|
|
|
|
}
|
|
|
|
|
2016-10-09 23:54:58 +02:00
|
|
|
/**
|
2021-12-13 16:28:14 +01:00
|
|
|
* @param array<string, Union> $new_vars_in_scope
|
|
|
|
* @return array<string, Union>
|
2016-10-09 23:54:58 +02:00
|
|
|
*/
|
2020-10-12 21:46:47 +02:00
|
|
|
public function getRedefinedVars(array $new_vars_in_scope, bool $include_new_vars = false): array
|
2016-06-20 06:38:13 +02:00
|
|
|
{
|
|
|
|
$redefined_vars = [];
|
|
|
|
|
2018-02-17 17:24:08 +01:00
|
|
|
foreach ($this->vars_in_scope as $var_id => $this_type) {
|
|
|
|
if (!isset($new_vars_in_scope[$var_id])) {
|
|
|
|
if ($include_new_vars) {
|
|
|
|
$redefined_vars[$var_id] = $this_type;
|
|
|
|
}
|
2017-06-29 06:28:37 +02:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2018-02-17 17:24:08 +01:00
|
|
|
$new_type = $new_vars_in_scope[$var_id];
|
2017-06-29 06:28:37 +02:00
|
|
|
|
2022-11-04 19:04:23 +01:00
|
|
|
if (!$this_type->equals(
|
|
|
|
$new_type,
|
|
|
|
true,
|
2022-12-18 17:15:15 +01:00
|
|
|
!($this_type->propagate_parent_nodes || $new_type->propagate_parent_nodes),
|
2022-11-04 19:04:23 +01:00
|
|
|
)
|
|
|
|
) {
|
2018-02-17 17:24:08 +01:00
|
|
|
$redefined_vars[$var_id] = $this_type;
|
2016-06-20 06:38:13 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return $redefined_vars;
|
|
|
|
}
|
2016-09-17 17:57:44 +02:00
|
|
|
|
2017-03-15 01:14:25 +01:00
|
|
|
/**
|
2020-10-17 18:36:44 +02:00
|
|
|
* @return list<string>
|
2017-03-15 01:14:25 +01:00
|
|
|
*/
|
2020-09-04 22:26:33 +02:00
|
|
|
public static function getNewOrUpdatedVarIds(Context $original_context, Context $new_context): array
|
2017-03-15 01:14:25 +01:00
|
|
|
{
|
|
|
|
$redefined_var_ids = [];
|
|
|
|
|
|
|
|
foreach ($new_context->vars_in_scope as $var_id => $context_type) {
|
2018-03-17 22:35:36 +01:00
|
|
|
if (!isset($original_context->vars_in_scope[$var_id])
|
2020-11-01 17:26:42 +01:00
|
|
|
|| ($original_context->assigned_var_ids[$var_id] ?? 0)
|
|
|
|
!== ($new_context->assigned_var_ids[$var_id] ?? 0)
|
2018-05-18 17:02:50 +02:00
|
|
|
|| !$original_context->vars_in_scope[$var_id]->equals($context_type)
|
2017-03-15 01:14:25 +01:00
|
|
|
) {
|
|
|
|
$redefined_var_ids[] = $var_id;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return $redefined_var_ids;
|
|
|
|
}
|
|
|
|
|
2022-01-07 23:16:36 +01:00
|
|
|
public function remove(string $remove_var_id, bool $removeDescendents = true): void
|
2016-09-17 17:57:44 +02:00
|
|
|
{
|
|
|
|
if (isset($this->vars_in_scope[$remove_var_id])) {
|
2017-04-02 21:26:10 +02:00
|
|
|
$existing_type = $this->vars_in_scope[$remove_var_id];
|
2016-09-17 17:57:44 +02:00
|
|
|
unset($this->vars_in_scope[$remove_var_id]);
|
|
|
|
|
2022-01-07 23:16:36 +01:00
|
|
|
if ($removeDescendents) {
|
|
|
|
$this->removeDescendents($remove_var_id, $existing_type);
|
|
|
|
}
|
2016-09-17 17:57:44 +02:00
|
|
|
}
|
2022-01-07 23:16:36 +01:00
|
|
|
$this->removePossibleReference($remove_var_id);
|
|
|
|
unset($this->vars_possibly_in_scope[$remove_var_id]);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2022-01-11 18:36:37 +01:00
|
|
|
* Remove a variable from the context which might be a reference to another variable, or
|
|
|
|
* referenced by another variable. Leaves the variable as possibly-in-scope, unlike remove().
|
2022-01-07 23:16:36 +01:00
|
|
|
*/
|
|
|
|
public function removePossibleReference(string $remove_var_id): void
|
|
|
|
{
|
2022-01-11 18:36:37 +01:00
|
|
|
if (isset($this->referenced_counts[$remove_var_id]) && $this->referenced_counts[$remove_var_id] > 0) {
|
|
|
|
// If a referenced variable goes out of scope, we need to update the references.
|
|
|
|
// All of the references to this variable are still references to the same value,
|
|
|
|
// so we pick the first one and make the rest of the references point to it.
|
|
|
|
$references = [];
|
|
|
|
foreach ($this->references_in_scope as $reference => $referenced) {
|
|
|
|
if ($referenced === $remove_var_id) {
|
|
|
|
$references[] = $reference;
|
|
|
|
unset($this->references_in_scope[$reference]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
assert(!empty($references));
|
|
|
|
$first_reference = array_shift($references);
|
|
|
|
if (!empty($references)) {
|
|
|
|
$this->referenced_counts[$first_reference] = count($references);
|
|
|
|
foreach ($references as $reference) {
|
|
|
|
$this->references_in_scope[$reference] = $first_reference;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2022-01-07 23:16:36 +01:00
|
|
|
if (isset($this->references_in_scope[$remove_var_id])) {
|
2022-05-27 00:46:20 +02:00
|
|
|
$this->decrementReferenceCount($remove_var_id);
|
2022-01-07 23:16:36 +01:00
|
|
|
}
|
|
|
|
unset(
|
|
|
|
$this->vars_in_scope[$remove_var_id],
|
2022-03-15 23:40:31 +01:00
|
|
|
$this->cond_referenced_var_ids[$remove_var_id],
|
2022-01-11 18:36:37 +01:00
|
|
|
$this->referenced_counts[$remove_var_id],
|
2022-01-07 23:16:36 +01:00
|
|
|
$this->references_in_scope[$remove_var_id],
|
|
|
|
$this->references_to_external_scope[$remove_var_id],
|
|
|
|
);
|
2016-09-17 17:57:44 +02:00
|
|
|
}
|
|
|
|
|
2022-05-27 00:46:20 +02:00
|
|
|
/**
|
|
|
|
* Decrement the reference count of the variable that $ref_id is referring to. This needs to
|
|
|
|
* be done before $ref_id is changed to no longer reference its currently referenced variable,
|
|
|
|
* for example by unsetting, reassigning to another reference, or being shadowed by a global.
|
|
|
|
*/
|
|
|
|
public function decrementReferenceCount(string $ref_id): void
|
|
|
|
{
|
|
|
|
if (!isset($this->referenced_counts[$this->references_in_scope[$ref_id]])) {
|
|
|
|
throw new InvalidArgumentException("$ref_id is not a reference");
|
|
|
|
}
|
|
|
|
$reference_count = $this->referenced_counts[$this->references_in_scope[$ref_id]];
|
|
|
|
if ($reference_count < 1) {
|
|
|
|
throw new RuntimeException("Incorrect referenced count found");
|
|
|
|
}
|
|
|
|
--$reference_count;
|
|
|
|
$this->referenced_counts[$this->references_in_scope[$ref_id]] = $reference_count;
|
|
|
|
}
|
|
|
|
|
2019-12-08 06:49:34 +01:00
|
|
|
/**
|
2020-08-26 16:41:47 +02:00
|
|
|
* @param Clause[] $clauses
|
|
|
|
* @param array<string, bool> $changed_var_ids
|
2022-11-12 02:14:21 +01:00
|
|
|
* @return array{list<Clause>, list<Clause>}
|
2020-08-26 16:41:47 +02:00
|
|
|
* @psalm-pure
|
2019-12-08 06:49:34 +01:00
|
|
|
*/
|
2020-09-04 22:26:33 +02:00
|
|
|
public static function removeReconciledClauses(array $clauses, array $changed_var_ids): array
|
2019-12-08 06:49:34 +01:00
|
|
|
{
|
|
|
|
$included_clauses = [];
|
|
|
|
$rejected_clauses = [];
|
|
|
|
|
|
|
|
foreach ($clauses as $c) {
|
|
|
|
if ($c->wedge) {
|
|
|
|
$included_clauses[] = $c;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
foreach ($c->possibilities as $key => $_) {
|
|
|
|
if (isset($changed_var_ids[$key])) {
|
|
|
|
$rejected_clauses[] = $c;
|
|
|
|
continue 2;
|
2019-10-09 00:44:46 +02:00
|
|
|
}
|
2019-12-08 06:49:34 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
$included_clauses[] = $c;
|
|
|
|
}
|
|
|
|
|
|
|
|
return [$included_clauses, $rejected_clauses];
|
2017-11-28 06:46:41 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param Clause[] $clauses
|
2019-10-09 00:44:46 +02:00
|
|
|
* @return list<Clause>
|
2017-11-28 06:46:41 +01:00
|
|
|
*/
|
|
|
|
public static function filterClauses(
|
2020-09-07 01:36:47 +02:00
|
|
|
string $remove_var_id,
|
2017-11-28 06:46:41 +01:00
|
|
|
array $clauses,
|
2020-09-07 01:36:47 +02:00
|
|
|
?Union $new_type = null,
|
|
|
|
?StatementsAnalyzer $statements_analyzer = null
|
2020-09-12 17:24:05 +02:00
|
|
|
): array {
|
2017-12-03 18:44:08 +01:00
|
|
|
$new_type_string = $new_type ? $new_type->getId() : '';
|
2017-11-28 06:46:41 +01:00
|
|
|
$clauses_to_keep = [];
|
|
|
|
|
|
|
|
foreach ($clauses as $clause) {
|
2020-08-26 16:41:47 +02:00
|
|
|
$clause = $clause->calculateNegation();
|
2017-04-02 21:26:10 +02:00
|
|
|
|
2018-07-10 06:39:33 +02:00
|
|
|
$quoted_remove_var_id = preg_quote($remove_var_id, '/');
|
2017-12-15 23:34:21 +01:00
|
|
|
|
|
|
|
foreach ($clause->possibilities as $var_id => $_) {
|
2018-07-10 06:39:33 +02:00
|
|
|
if (preg_match('/' . $quoted_remove_var_id . '[\]\[\-]/', $var_id)) {
|
2017-12-15 23:34:21 +01:00
|
|
|
break 2;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-01-20 23:33:06 +01:00
|
|
|
if (!isset($clause->possibilities[$remove_var_id])
|
|
|
|
|| (count($clause->possibilities[$remove_var_id]) === 1
|
2022-03-13 04:31:12 +01:00
|
|
|
&& array_keys($clause->possibilities[$remove_var_id])[0] === $new_type_string)
|
2017-04-02 21:26:10 +02:00
|
|
|
) {
|
2016-12-28 20:20:16 +01:00
|
|
|
$clauses_to_keep[] = $clause;
|
2018-11-11 18:01:14 +01:00
|
|
|
} elseif ($statements_analyzer &&
|
2017-04-02 22:51:27 +02:00
|
|
|
$new_type &&
|
2018-12-08 19:18:55 +01:00
|
|
|
!$new_type->hasMixed()
|
2017-04-02 22:51:27 +02:00
|
|
|
) {
|
2017-04-02 21:26:10 +02:00
|
|
|
$type_changed = false;
|
|
|
|
|
|
|
|
// if the clause contains any possibilities that would be altered
|
2017-12-02 23:57:58 +01:00
|
|
|
// by the new type
|
2022-01-20 23:33:06 +01:00
|
|
|
foreach ($clause->possibilities[$remove_var_id] as $assertion) {
|
2019-08-16 16:32:03 +02:00
|
|
|
// if we're negating a type, we generally don't need the clause anymore
|
2022-01-20 23:33:06 +01:00
|
|
|
if ($assertion->isNegation()) {
|
2019-08-16 16:32:03 +02:00
|
|
|
$type_changed = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2019-08-10 19:22:21 +02:00
|
|
|
$result_type = AssertionReconciler::reconcile(
|
2022-01-20 23:33:06 +01:00
|
|
|
$assertion,
|
2022-11-04 19:04:23 +01:00
|
|
|
$new_type,
|
2017-04-02 21:26:10 +02:00
|
|
|
null,
|
2018-11-11 18:01:14 +01:00
|
|
|
$statements_analyzer,
|
2018-12-08 19:18:55 +01:00
|
|
|
false,
|
2019-01-23 05:42:54 +01:00
|
|
|
[],
|
2017-04-02 21:26:10 +02:00
|
|
|
null,
|
|
|
|
[],
|
2022-12-18 17:15:15 +01:00
|
|
|
$failed_reconciliation,
|
2017-04-02 21:26:10 +02:00
|
|
|
);
|
|
|
|
|
2017-12-03 18:44:08 +01:00
|
|
|
if ($result_type->getId() !== $new_type_string) {
|
2017-04-02 21:26:10 +02:00
|
|
|
$type_changed = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!$type_changed) {
|
|
|
|
$clauses_to_keep[] = $clause;
|
|
|
|
}
|
2016-12-28 20:20:16 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-11-28 06:46:41 +01:00
|
|
|
return $clauses_to_keep;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function removeVarFromConflictingClauses(
|
2020-09-07 01:36:47 +02:00
|
|
|
string $remove_var_id,
|
|
|
|
?Union $new_type = null,
|
|
|
|
?StatementsAnalyzer $statements_analyzer = null
|
2020-09-12 17:24:05 +02:00
|
|
|
): void {
|
2018-11-11 18:01:14 +01:00
|
|
|
$this->clauses = self::filterClauses($remove_var_id, $this->clauses, $new_type, $statements_analyzer);
|
2022-01-24 22:35:42 +01:00
|
|
|
$this->parent_remove_vars[$remove_var_id] = true;
|
2016-12-28 20:20:16 +01:00
|
|
|
}
|
|
|
|
|
2021-01-17 20:28:28 +01:00
|
|
|
/**
|
|
|
|
* This method is used after assignments to variables to remove any existing
|
|
|
|
* items in $vars_in_scope that are now made redundant by an update to some data
|
|
|
|
*/
|
2017-04-02 21:26:10 +02:00
|
|
|
public function removeDescendents(
|
2020-09-07 01:36:47 +02:00
|
|
|
string $remove_var_id,
|
2022-01-24 16:34:48 +01:00
|
|
|
Union $existing_type,
|
2020-09-07 01:36:47 +02:00
|
|
|
?Union $new_type = null,
|
|
|
|
?StatementsAnalyzer $statements_analyzer = null
|
2020-10-12 21:02:52 +02:00
|
|
|
): void {
|
2020-01-11 05:46:45 +01:00
|
|
|
$this->removeVarFromConflictingClauses(
|
|
|
|
$remove_var_id,
|
|
|
|
$existing_type->hasMixed()
|
|
|
|
|| ($new_type && $existing_type->from_docblock !== $new_type->from_docblock)
|
|
|
|
? null
|
|
|
|
: $new_type,
|
2022-12-18 17:15:15 +01:00
|
|
|
$statements_analyzer,
|
2020-01-11 05:46:45 +01:00
|
|
|
);
|
2016-12-27 19:58:58 +01:00
|
|
|
|
2022-10-03 10:45:36 +02:00
|
|
|
foreach ($this->vars_in_scope as $var_id => &$type) {
|
2018-07-07 01:16:31 +02:00
|
|
|
if (preg_match('/' . preg_quote($remove_var_id, '/') . '[\]\[\-]/', $var_id)) {
|
2022-01-07 23:16:36 +01:00
|
|
|
$this->remove($var_id, false);
|
2016-09-17 17:57:44 +02:00
|
|
|
}
|
2020-12-07 00:14:21 +01:00
|
|
|
|
2022-10-03 10:45:36 +02:00
|
|
|
$builder = null;
|
2020-12-07 00:14:21 +01:00
|
|
|
foreach ($type->getAtomicTypes() as $atomic_type) {
|
2021-12-13 16:28:14 +01:00
|
|
|
if ($atomic_type instanceof DependentType
|
2020-12-07 00:14:21 +01:00
|
|
|
&& $atomic_type->getVarId() === $remove_var_id
|
|
|
|
) {
|
2022-10-03 10:45:36 +02:00
|
|
|
$builder ??= $type->getBuilder();
|
|
|
|
$builder->addType($atomic_type->getReplacement());
|
2020-12-07 00:14:21 +01:00
|
|
|
}
|
|
|
|
}
|
2022-10-03 10:45:36 +02:00
|
|
|
if ($builder) {
|
|
|
|
$type = $builder->freeze();
|
|
|
|
}
|
2018-02-17 17:24:08 +01:00
|
|
|
}
|
2016-09-17 17:57:44 +02:00
|
|
|
}
|
2016-10-18 22:14:52 +02:00
|
|
|
|
2021-03-05 06:39:19 +01:00
|
|
|
public function removeMutableObjectVars(bool $methods_only = false): void
|
2017-04-15 03:32:14 +02:00
|
|
|
{
|
|
|
|
$vars_to_remove = [];
|
|
|
|
|
2020-12-08 22:39:06 +01:00
|
|
|
foreach ($this->vars_in_scope as $var_id => $type) {
|
|
|
|
if ($type->has_mutations
|
|
|
|
&& (strpos($var_id, '->') !== false || strpos($var_id, '::') !== false)
|
2021-03-05 06:39:19 +01:00
|
|
|
&& (!$methods_only || strpos($var_id, '()'))
|
2020-12-08 22:39:06 +01:00
|
|
|
) {
|
2017-04-15 03:32:14 +02:00
|
|
|
$vars_to_remove[] = $var_id;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!$vars_to_remove) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
foreach ($vars_to_remove as $var_id) {
|
2022-01-07 23:16:36 +01:00
|
|
|
$this->remove($var_id, false);
|
2017-04-15 03:32:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
$clauses_to_keep = [];
|
|
|
|
|
|
|
|
foreach ($this->clauses as $clause) {
|
|
|
|
$abandon_clause = false;
|
|
|
|
|
|
|
|
foreach (array_keys($clause->possibilities) as $key) {
|
2021-03-05 06:39:19 +01:00
|
|
|
if ((strpos($key, '->') !== false || strpos($key, '::') !== false)
|
|
|
|
&& (!$methods_only || strpos($key, '()'))
|
|
|
|
) {
|
2017-04-15 03:32:14 +02:00
|
|
|
$abandon_clause = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!$abandon_clause) {
|
|
|
|
$clauses_to_keep[] = $clause;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$this->clauses = $clauses_to_keep;
|
|
|
|
}
|
|
|
|
|
2020-09-12 17:24:05 +02:00
|
|
|
public function updateChecks(Context $op_context): void
|
2016-10-18 22:14:52 +02:00
|
|
|
{
|
|
|
|
$this->check_classes = $this->check_classes && $op_context->check_classes;
|
|
|
|
$this->check_variables = $this->check_variables && $op_context->check_variables;
|
|
|
|
$this->check_methods = $this->check_methods && $op_context->check_methods;
|
|
|
|
$this->check_functions = $this->check_functions && $op_context->check_functions;
|
|
|
|
$this->check_consts = $this->check_consts && $op_context->check_consts;
|
|
|
|
}
|
2016-10-22 19:23:18 +02:00
|
|
|
|
2020-09-07 01:36:47 +02:00
|
|
|
public function isPhantomClass(string $class_name): bool
|
2016-10-22 19:23:18 +02:00
|
|
|
{
|
2017-03-01 17:56:36 +01:00
|
|
|
return isset($this->phantom_classes[strtolower($class_name)]);
|
2016-10-22 19:23:18 +02:00
|
|
|
}
|
|
|
|
|
2021-09-25 16:05:47 +02:00
|
|
|
public function hasVariable(string $var_name): bool
|
2017-02-01 05:24:33 +01:00
|
|
|
{
|
2019-02-26 07:03:33 +01:00
|
|
|
if (!$var_name) {
|
2017-11-28 06:46:41 +01:00
|
|
|
return false;
|
|
|
|
}
|
2017-02-01 05:24:33 +01:00
|
|
|
|
2022-09-08 18:51:33 +02:00
|
|
|
$stripped_var = preg_replace('/(->|\[).*$/', '', $var_name, 1);
|
2017-02-01 05:24:33 +01:00
|
|
|
|
2019-10-04 00:13:04 +02:00
|
|
|
if ($stripped_var !== '$this' || $var_name !== $stripped_var) {
|
2022-03-15 23:40:31 +01:00
|
|
|
$this->cond_referenced_var_ids[$var_name] = true;
|
2017-02-01 05:24:33 +01:00
|
|
|
}
|
|
|
|
|
2017-11-28 06:46:41 +01:00
|
|
|
return isset($this->vars_in_scope[$var_name]);
|
2017-02-01 05:24:33 +01:00
|
|
|
}
|
2019-01-31 18:55:48 +01:00
|
|
|
|
2021-12-05 18:51:26 +01:00
|
|
|
public function getScopeSummary(): string
|
2019-01-31 18:55:48 +01:00
|
|
|
{
|
|
|
|
$summary = [];
|
|
|
|
foreach ($this->vars_possibly_in_scope as $k => $_) {
|
|
|
|
$summary[$k] = true;
|
|
|
|
}
|
|
|
|
foreach ($this->vars_in_scope as $k => $v) {
|
|
|
|
$summary[$k] = $v->getId();
|
|
|
|
}
|
2019-07-05 22:24:00 +02:00
|
|
|
|
2022-01-05 18:32:43 +01:00
|
|
|
return json_encode($summary, JSON_THROW_ON_ERROR);
|
2019-01-31 18:55:48 +01:00
|
|
|
}
|
2019-03-11 14:54:41 +01:00
|
|
|
|
2020-09-12 17:24:05 +02:00
|
|
|
public function defineGlobals(): void
|
2019-03-11 14:54:41 +01:00
|
|
|
{
|
|
|
|
$globals = [
|
2021-12-13 16:28:14 +01:00
|
|
|
'$argv' => new Union([
|
2021-12-13 04:45:57 +01:00
|
|
|
new TArray([Type::getInt(), Type::getString()]),
|
2019-03-11 14:54:41 +01:00
|
|
|
]),
|
2019-03-14 01:15:29 +01:00
|
|
|
'$argc' => Type::getInt(),
|
2019-03-11 14:54:41 +01:00
|
|
|
];
|
|
|
|
|
|
|
|
$config = Config::getInstance();
|
|
|
|
|
2019-03-14 01:15:29 +01:00
|
|
|
foreach ($config->globals as $global_id => $type_string) {
|
|
|
|
$globals[$global_id] = Type::parseString($type_string);
|
2019-03-11 14:54:41 +01:00
|
|
|
}
|
|
|
|
|
2019-03-14 01:15:29 +01:00
|
|
|
foreach ($globals as $global_id => $type) {
|
|
|
|
$this->vars_in_scope[$global_id] = $type;
|
|
|
|
$this->vars_possibly_in_scope[$global_id] = true;
|
2019-03-11 14:54:41 +01:00
|
|
|
}
|
|
|
|
}
|
2019-03-29 00:43:14 +01:00
|
|
|
|
2020-09-12 17:24:05 +02:00
|
|
|
public function mergeExceptions(Context $other_context): void
|
2019-03-29 00:43:14 +01:00
|
|
|
{
|
2019-04-03 01:42:23 +02:00
|
|
|
foreach ($other_context->possibly_thrown_exceptions as $possibly_thrown_exception => $codelocations) {
|
|
|
|
foreach ($codelocations as $hash => $codelocation) {
|
|
|
|
$this->possibly_thrown_exceptions[$possibly_thrown_exception][$hash] = $codelocation;
|
|
|
|
}
|
|
|
|
}
|
2019-03-29 00:43:14 +01:00
|
|
|
}
|
2020-09-21 21:16:19 +02:00
|
|
|
|
2020-09-04 22:26:33 +02:00
|
|
|
public function isSuppressingExceptions(StatementsAnalyzer $statements_analyzer): bool
|
2019-03-29 00:50:29 +01:00
|
|
|
{
|
|
|
|
if (!$this->collect_exceptions) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
$issue_type = $this->is_global ? 'UncaughtThrowInGlobalScope' : 'MissingThrowsDocblock';
|
|
|
|
$suppressed_issues = $statements_analyzer->getSuppressedIssues();
|
2019-10-04 20:01:58 +02:00
|
|
|
$suppressed_issue_position = array_search($issue_type, $suppressed_issues, true);
|
|
|
|
if ($suppressed_issue_position !== false) {
|
|
|
|
if (is_int($suppressed_issue_position)) {
|
|
|
|
$file = $statements_analyzer->getFileAnalyzer()->getFilePath();
|
|
|
|
IssueBuffer::addUsedSuppressions([
|
|
|
|
$file => [$suppressed_issue_position => true],
|
|
|
|
]);
|
|
|
|
}
|
2019-03-29 00:50:29 +01:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2019-03-29 00:43:14 +01:00
|
|
|
public function mergeFunctionExceptions(
|
|
|
|
FunctionLikeStorage $function_storage,
|
|
|
|
CodeLocation $codelocation
|
2020-09-12 17:24:05 +02:00
|
|
|
): void {
|
2019-04-03 01:42:23 +02:00
|
|
|
$hash = $codelocation->getHash();
|
2019-03-29 00:43:14 +01:00
|
|
|
foreach ($function_storage->throws as $possibly_thrown_exception => $_) {
|
2019-04-03 01:42:23 +02:00
|
|
|
$this->possibly_thrown_exceptions[$possibly_thrown_exception][$hash] = $codelocation;
|
2019-03-29 00:43:14 +01:00
|
|
|
}
|
|
|
|
}
|
2021-06-25 16:14:49 +02:00
|
|
|
|
|
|
|
public function insideUse(): bool
|
|
|
|
{
|
|
|
|
return $this->inside_assignment
|
|
|
|
|| $this->inside_return
|
|
|
|
|| $this->inside_call
|
|
|
|
|| $this->inside_general_use
|
|
|
|
|| $this->inside_conditional
|
|
|
|
|| $this->inside_throw
|
|
|
|
|| $this->inside_isset;
|
|
|
|
}
|
2016-06-20 06:38:13 +02:00
|
|
|
}
|