2016-12-27 19:58:58 +01:00
|
|
|
<?php
|
2021-12-15 04:58:32 +01:00
|
|
|
|
2018-11-06 03:57:36 +01:00
|
|
|
namespace Psalm\Internal;
|
2016-12-27 19:58:58 +01:00
|
|
|
|
2022-01-20 23:33:06 +01:00
|
|
|
use Psalm\Storage\Assertion;
|
2022-11-04 19:04:23 +01:00
|
|
|
use Psalm\Storage\ImmutableNonCloneableTrait;
|
2022-01-20 23:33:06 +01:00
|
|
|
use Psalm\Type\Atomic\TClassConstant;
|
|
|
|
use Psalm\Type\Atomic\TEnumCase;
|
|
|
|
use Psalm\Type\Atomic\TLiteralFloat;
|
|
|
|
use Psalm\Type\Atomic\TLiteralInt;
|
|
|
|
use Psalm\Type\Atomic\TLiteralString;
|
2021-12-03 20:11:20 +01:00
|
|
|
|
2019-06-26 22:52:29 +02:00
|
|
|
use function array_diff;
|
2019-07-05 22:24:00 +02:00
|
|
|
use function array_keys;
|
|
|
|
use function count;
|
2022-06-28 19:04:09 +02:00
|
|
|
use function hash;
|
2019-07-05 22:24:00 +02:00
|
|
|
use function implode;
|
|
|
|
use function ksort;
|
2021-12-03 21:07:25 +01:00
|
|
|
use function reset;
|
2022-04-27 07:42:37 +02:00
|
|
|
use function serialize;
|
2021-12-03 21:07:25 +01:00
|
|
|
use function substr;
|
2019-06-26 22:52:29 +02:00
|
|
|
|
2022-06-28 19:04:09 +02:00
|
|
|
use const PHP_VERSION_ID;
|
|
|
|
|
2018-12-02 00:37:49 +01:00
|
|
|
/**
|
|
|
|
* @internal
|
2020-08-26 16:41:47 +02:00
|
|
|
* @psalm-immutable
|
2018-12-02 00:37:49 +01:00
|
|
|
*/
|
2016-12-27 19:58:58 +01:00
|
|
|
class Clause
|
|
|
|
{
|
2022-11-04 19:04:23 +01:00
|
|
|
use ImmutableNonCloneableTrait;
|
2022-12-14 03:46:43 +01:00
|
|
|
|
2022-12-14 01:52:54 +01:00
|
|
|
public int $creating_conditional_id;
|
2020-08-26 21:35:29 +02:00
|
|
|
|
2022-12-14 01:52:54 +01:00
|
|
|
public int $creating_object_id;
|
2019-12-08 22:35:56 +01:00
|
|
|
|
2016-12-27 19:58:58 +01:00
|
|
|
/**
|
|
|
|
* An array of strings of the form
|
|
|
|
* [
|
2017-10-22 17:57:41 +02:00
|
|
|
* '$a' => ['falsy'],
|
|
|
|
* '$b' => ['!falsy'],
|
2016-12-27 19:58:58 +01:00
|
|
|
* '$c' => ['!null'],
|
|
|
|
* '$d' => ['string', 'int']
|
|
|
|
* ]
|
|
|
|
*
|
|
|
|
* representing the formula
|
|
|
|
*
|
|
|
|
* !$a || $b || $c !== null || is_string($d) || is_int($d)
|
|
|
|
*
|
2022-03-13 04:31:12 +01:00
|
|
|
* @var array<string, non-empty-array<string, Assertion>>
|
2016-12-27 19:58:58 +01:00
|
|
|
*/
|
2022-12-14 01:52:54 +01:00
|
|
|
public array $possibilities;
|
2016-12-27 19:58:58 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* An array of things that are not true
|
|
|
|
* [
|
2017-10-22 17:57:41 +02:00
|
|
|
* '$a' => ['!falsy'],
|
|
|
|
* '$b' => ['falsy'],
|
2016-12-27 19:58:58 +01:00
|
|
|
* '$c' => ['null'],
|
|
|
|
* '$d' => ['!string', '!int']
|
|
|
|
* ]
|
|
|
|
* represents the formula
|
|
|
|
*
|
|
|
|
* $a && !$b && $c === null && !is_string($d) && !is_int($d)
|
|
|
|
*
|
2022-01-20 23:33:06 +01:00
|
|
|
* @var array<string, non-empty-list<Assertion>>|null
|
2016-12-27 19:58:58 +01:00
|
|
|
*/
|
2022-12-14 01:52:54 +01:00
|
|
|
public ?array $impossibilities = null;
|
2016-12-27 19:58:58 +01:00
|
|
|
|
2022-12-14 01:52:54 +01:00
|
|
|
public bool $wedge;
|
2016-12-27 19:58:58 +01:00
|
|
|
|
2022-12-14 01:52:54 +01:00
|
|
|
public bool $reconcilable;
|
2016-12-27 19:58:58 +01:00
|
|
|
|
2022-12-14 01:52:54 +01:00
|
|
|
public bool $generated = false;
|
2018-05-07 20:52:45 +02:00
|
|
|
|
2019-12-08 16:17:40 +01:00
|
|
|
/** @var array<string, bool> */
|
2022-12-14 01:52:54 +01:00
|
|
|
public array $redefined_vars = [];
|
2019-12-08 16:17:40 +01:00
|
|
|
|
2022-12-14 01:52:54 +01:00
|
|
|
public string $hash;
|
2020-08-26 16:41:47 +02:00
|
|
|
|
2016-12-27 19:58:58 +01:00
|
|
|
/**
|
2022-03-13 04:31:12 +01:00
|
|
|
* @param array<string, non-empty-array<string, Assertion>> $possibilities
|
2019-12-08 16:17:40 +01:00
|
|
|
* @param array<string, bool> $redefined_vars
|
2016-12-27 19:58:58 +01:00
|
|
|
*/
|
2019-12-08 16:17:40 +01:00
|
|
|
public function __construct(
|
|
|
|
array $possibilities,
|
2020-08-26 21:35:29 +02:00
|
|
|
int $creating_conditional_id,
|
|
|
|
int $creating_object_id,
|
2020-10-12 21:46:47 +02:00
|
|
|
bool $wedge = false,
|
|
|
|
bool $reconcilable = true,
|
|
|
|
bool $generated = false,
|
2020-08-26 21:35:29 +02:00
|
|
|
array $redefined_vars = []
|
2019-12-08 16:17:40 +01:00
|
|
|
) {
|
2020-08-26 16:41:47 +02:00
|
|
|
if ($wedge || !$reconcilable) {
|
2020-08-26 21:35:29 +02:00
|
|
|
$this->hash = ($wedge ? 'w' : '') . $creating_object_id;
|
2020-08-26 16:41:47 +02:00
|
|
|
} else {
|
|
|
|
ksort($possibilities);
|
|
|
|
|
2022-03-13 04:31:12 +01:00
|
|
|
$possibility_strings = [];
|
|
|
|
|
2022-06-28 19:04:09 +02:00
|
|
|
foreach ($possibilities as $i => $v) {
|
2022-07-07 07:21:35 +02:00
|
|
|
if (count($v) > 1) {
|
|
|
|
ksort($v);
|
2022-06-28 19:04:09 +02:00
|
|
|
}
|
2022-07-07 07:21:35 +02:00
|
|
|
$possibility_strings[$i] = array_keys($v);
|
2020-08-26 16:41:47 +02:00
|
|
|
}
|
|
|
|
|
2022-11-08 10:45:21 +01:00
|
|
|
/** @psalm-suppress ImpureFunctionCall */
|
2022-07-07 07:21:35 +02:00
|
|
|
$data = serialize($possibility_strings);
|
|
|
|
$this->hash = PHP_VERSION_ID >= 8_01_00 ? hash('xxh128', $data) : hash('md4', $data);
|
2020-08-26 16:41:47 +02:00
|
|
|
}
|
2022-01-20 23:33:06 +01:00
|
|
|
|
|
|
|
$this->possibilities = $possibilities;
|
|
|
|
$this->wedge = $wedge;
|
|
|
|
$this->reconcilable = $reconcilable;
|
|
|
|
$this->generated = $generated;
|
|
|
|
$this->redefined_vars = $redefined_vars;
|
|
|
|
$this->creating_conditional_id = $creating_conditional_id;
|
|
|
|
$this->creating_object_id = $creating_object_id;
|
2016-12-27 19:58:58 +01:00
|
|
|
}
|
|
|
|
|
2020-09-04 22:26:33 +02:00
|
|
|
public function contains(Clause $other_clause): bool
|
2016-12-27 19:58:58 +01:00
|
|
|
{
|
2017-03-20 04:26:45 +01:00
|
|
|
if (count($other_clause->possibilities) > count($this->possibilities)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2022-06-28 15:21:29 +02:00
|
|
|
foreach ($other_clause->possibilities as $var => $_) {
|
|
|
|
if (!isset($this->possibilities[$var])) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-12-27 19:58:58 +01:00
|
|
|
foreach ($other_clause->possibilities as $var => $possible_types) {
|
2022-06-28 15:21:29 +02:00
|
|
|
if (count(array_diff($possible_types, $this->possibilities[$var]))) {
|
2016-12-27 19:58:58 +01:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2020-10-07 17:46:53 +02:00
|
|
|
/**
|
|
|
|
* @psalm-mutation-free
|
|
|
|
*/
|
2020-09-04 22:26:33 +02:00
|
|
|
public function __toString(): string
|
2018-05-07 07:26:06 +02:00
|
|
|
{
|
2022-04-09 11:58:26 +02:00
|
|
|
$clause_strings = [];
|
2020-10-07 17:46:53 +02:00
|
|
|
|
2022-04-09 11:58:26 +02:00
|
|
|
foreach ($this->possibilities as $var_id => $values) {
|
|
|
|
if ($var_id[0] === '*') {
|
|
|
|
$var_id = '<expr>';
|
|
|
|
}
|
2020-10-07 17:46:53 +02:00
|
|
|
|
2022-04-09 11:58:26 +02:00
|
|
|
$var_id_clauses = [];
|
|
|
|
foreach ($values as $value) {
|
|
|
|
$value = (string) $value;
|
|
|
|
if ($value === 'falsy') {
|
|
|
|
$var_id_clauses[] = '!'.$var_id;
|
|
|
|
continue;
|
|
|
|
}
|
2020-10-07 17:46:53 +02:00
|
|
|
|
2022-04-09 11:58:26 +02:00
|
|
|
if ($value === '!falsy') {
|
|
|
|
$var_id_clauses[] = $var_id;
|
|
|
|
continue;
|
|
|
|
}
|
2020-10-07 17:46:53 +02:00
|
|
|
|
2022-04-09 11:58:26 +02:00
|
|
|
$negate = false;
|
2020-10-07 17:46:53 +02:00
|
|
|
|
2022-04-09 11:58:26 +02:00
|
|
|
if ($value[0] === '!') {
|
|
|
|
$negate = true;
|
|
|
|
$value = substr($value, 1);
|
|
|
|
}
|
2020-10-07 17:46:53 +02:00
|
|
|
|
2022-04-09 11:58:26 +02:00
|
|
|
if ($value[0] === '=') {
|
|
|
|
$value = substr($value, 1);
|
|
|
|
}
|
2020-10-07 17:46:53 +02:00
|
|
|
|
2022-04-09 11:58:26 +02:00
|
|
|
if ($negate) {
|
|
|
|
$var_id_clauses[] = $var_id.' is not '.$value;
|
|
|
|
continue;
|
2020-10-07 17:46:53 +02:00
|
|
|
}
|
|
|
|
|
2022-04-09 11:58:26 +02:00
|
|
|
$var_id_clauses[] = $var_id.' is '.$value;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (count($var_id_clauses) > 1) {
|
|
|
|
$clause_strings[] = '('.implode(') || (', $var_id_clauses).')';
|
|
|
|
} else {
|
|
|
|
$clause_strings[] = reset($var_id_clauses);
|
|
|
|
}
|
|
|
|
}
|
2020-10-07 17:46:53 +02:00
|
|
|
|
|
|
|
if (count($clause_strings) > 1) {
|
|
|
|
return '(' . implode(') || (', $clause_strings) . ')';
|
|
|
|
}
|
|
|
|
|
2021-12-03 21:07:25 +01:00
|
|
|
return reset($clause_strings);
|
2018-05-07 07:26:06 +02:00
|
|
|
}
|
2020-08-26 16:41:47 +02:00
|
|
|
|
2021-12-05 18:51:26 +01:00
|
|
|
public function removePossibilities(string $var_id): ?self
|
2020-08-26 16:41:47 +02:00
|
|
|
{
|
|
|
|
$possibilities = $this->possibilities;
|
|
|
|
unset($possibilities[$var_id]);
|
|
|
|
|
2020-08-26 22:41:40 +02:00
|
|
|
if (!$possibilities) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2020-08-26 16:41:47 +02:00
|
|
|
return new self(
|
|
|
|
$possibilities,
|
2020-08-26 21:35:29 +02:00
|
|
|
$this->creating_conditional_id,
|
|
|
|
$this->creating_object_id,
|
2020-08-26 16:41:47 +02:00
|
|
|
$this->wedge,
|
|
|
|
$this->reconcilable,
|
|
|
|
$this->generated,
|
2022-12-18 17:15:15 +01:00
|
|
|
$this->redefined_vars,
|
2020-08-26 16:41:47 +02:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2022-03-13 04:31:12 +01:00
|
|
|
* @param non-empty-array<string, Assertion> $clause_var_possibilities
|
2020-08-26 16:41:47 +02:00
|
|
|
*/
|
2021-12-05 18:51:26 +01:00
|
|
|
public function addPossibilities(string $var_id, array $clause_var_possibilities): self
|
2020-08-26 16:41:47 +02:00
|
|
|
{
|
|
|
|
$possibilities = $this->possibilities;
|
|
|
|
$possibilities[$var_id] = $clause_var_possibilities;
|
|
|
|
|
|
|
|
return new self(
|
|
|
|
$possibilities,
|
2020-08-26 21:35:29 +02:00
|
|
|
$this->creating_conditional_id,
|
|
|
|
$this->creating_object_id,
|
2020-08-26 16:41:47 +02:00
|
|
|
$this->wedge,
|
|
|
|
$this->reconcilable,
|
|
|
|
$this->generated,
|
2022-12-18 17:15:15 +01:00
|
|
|
$this->redefined_vars,
|
2020-08-26 16:41:47 +02:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2021-12-05 18:51:26 +01:00
|
|
|
public function calculateNegation(): self
|
2020-08-26 16:41:47 +02:00
|
|
|
{
|
|
|
|
if ($this->impossibilities !== null) {
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
|
|
|
$impossibilities = [];
|
|
|
|
|
|
|
|
foreach ($this->possibilities as $var_id => $possibility) {
|
|
|
|
$impossibility = [];
|
|
|
|
|
|
|
|
foreach ($possibility as $type) {
|
2022-01-20 23:33:06 +01:00
|
|
|
if (!$type->hasEquality()
|
|
|
|
|| (($inner_type = $type->getAtomicType())
|
|
|
|
&& ($inner_type instanceof TLiteralInt
|
|
|
|
|| $inner_type instanceof TLiteralFloat
|
|
|
|
|| $inner_type instanceof TLiteralString
|
|
|
|
|| $inner_type instanceof TClassConstant
|
|
|
|
|| $inner_type instanceof TEnumCase))
|
2020-08-26 16:41:47 +02:00
|
|
|
) {
|
2022-01-20 23:33:06 +01:00
|
|
|
$impossibility[] = $type->getNegation();
|
2020-08-26 16:41:47 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($impossibility) {
|
|
|
|
$impossibilities[$var_id] = $impossibility;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$clause = clone $this;
|
|
|
|
|
|
|
|
$clause->impossibilities = $impossibilities;
|
|
|
|
|
|
|
|
return $clause;
|
|
|
|
}
|
2016-12-27 19:58:58 +01:00
|
|
|
}
|