2016-06-20 06:38:13 +02:00
|
|
|
<?php
|
|
|
|
|
2016-07-26 00:37:44 +02:00
|
|
|
namespace Psalm;
|
2016-06-20 06:38:13 +02:00
|
|
|
|
|
|
|
class Context
|
|
|
|
{
|
2016-08-08 17:28:14 +02:00
|
|
|
/** @var array<Type\Union> */
|
2016-06-20 06:38:13 +02:00
|
|
|
public $vars_in_scope = [];
|
|
|
|
|
|
|
|
public $vars_possibly_in_scope = [];
|
|
|
|
|
2016-08-08 17:28:14 +02:00
|
|
|
/** @var boolean */
|
2016-06-20 22:30:31 +02:00
|
|
|
public $in_loop = false;
|
|
|
|
|
2016-08-08 17:28:14 +02:00
|
|
|
/** @var string|null */
|
|
|
|
public $self;
|
|
|
|
|
|
|
|
/** @var string|null */
|
|
|
|
public $parent;
|
|
|
|
|
2016-06-20 06:38:13 +02:00
|
|
|
public function __clone()
|
|
|
|
{
|
|
|
|
foreach ($this->vars_in_scope as $key => &$type) {
|
|
|
|
$type = clone $type;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Updates the parent context, looking at the changes within a block
|
|
|
|
* and then applying those changes, where necessary, to the parent context
|
|
|
|
*
|
|
|
|
* @param Context $start_context
|
|
|
|
* @param Context $end_context
|
|
|
|
* @param bool $has_leaving_statements whether or not the parent scope is abandoned between $start_context and $end_context
|
|
|
|
* @return void
|
|
|
|
*/
|
2016-07-25 21:05:58 +02:00
|
|
|
public function update(Context $start_context, Context $end_context, $has_leaving_statements, array &$updated_vars)
|
2016-06-20 06:38:13 +02:00
|
|
|
{
|
|
|
|
foreach ($this->vars_in_scope as $var => &$context_type) {
|
|
|
|
$old_type = $start_context->vars_in_scope[$var];
|
|
|
|
// if we're leaving, we're effectively deleting the possibility of the if types
|
2016-07-25 21:05:58 +02:00
|
|
|
$new_type = !$has_leaving_statements ? $end_context->vars_in_scope[$var] : null;
|
2016-06-20 06:38:13 +02:00
|
|
|
|
2016-08-10 07:28:38 +02:00
|
|
|
// if the type changed within the block of statements, process the replacement
|
|
|
|
if ((string)$old_type !== (string)$new_type) {
|
|
|
|
$context_type->substitute($old_type, $new_type);
|
|
|
|
$updated_vars[$var] = true;
|
2016-06-20 06:38:13 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public static function getRedefinedVars(Context $original_context, Context $new_context)
|
|
|
|
{
|
|
|
|
$redefined_vars = [];
|
|
|
|
|
|
|
|
foreach ($original_context->vars_in_scope as $var => $context_type) {
|
|
|
|
if ((string)$new_context->vars_in_scope[$var] !== (string)$context_type) {
|
|
|
|
$redefined_vars[$var] = $new_context->vars_in_scope[$var];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return $redefined_vars;
|
|
|
|
}
|
|
|
|
}
|