Skip to content

Commit

Permalink
Introduce nested type exceptions with paths
Browse files Browse the repository at this point in the history
  • Loading branch information
veewee committed Mar 27, 2024
1 parent 6559c40 commit 474de7e
Show file tree
Hide file tree
Showing 9 changed files with 111 additions and 44 deletions.
1 change: 0 additions & 1 deletion src/Psl/Internal/Loader.php
Original file line number Diff line number Diff line change
Expand Up @@ -690,7 +690,6 @@ final class Loader
'Psl\\Type\\Internal\\LiteralScalarType' => 'Psl/Type/Internal/LiteralScalarType.php',
'Psl\\Type\\Internal\\BackedEnumType' => 'Psl/Type/Internal/BackedEnumType.php',
'Psl\\Type\\Internal\\UnitEnumType' => 'Psl/Type/Internal/UnitEnumType.php',
'Psl\\Type\\Exception\\TypeTrace' => 'Psl/Type/Exception/TypeTrace.php',
'Psl\\Type\\Exception\\AssertException' => 'Psl/Type/Exception/AssertException.php',
'Psl\\Type\\Exception\\CoercionException' => 'Psl/Type/Exception/CoercionException.php',
'Psl\\Type\\Exception\\Exception' => 'Psl/Type/Exception/Exception.php',
Expand Down
21 changes: 18 additions & 3 deletions src/Psl/Type/Exception/AssertException.php
Original file line number Diff line number Diff line change
Expand Up @@ -5,16 +5,27 @@
namespace Psl\Type\Exception;

use Psl\Str;
use Psl\Vec;

use Throwable;
use function get_debug_type;

final class AssertException extends Exception
{
private string $expected;

public function __construct(string $actual, string $expected)
/**
* @param list<string> $paths
*/
public function __construct(string $actual, string $expected, array $paths = [], ?Throwable $previous = null)
{
parent::__construct(Str\format('Expected "%s", got "%s".', $expected, $actual), $actual);
parent::__construct(
Str\format(
'Expected "%s", got "%s"%s.',
$expected,
$actual,
$paths ? ' at path "' . Str\join($paths, '.') . '"' : ''
), $actual, $paths, $previous);

$this->expected = $expected;
}
Expand All @@ -27,7 +38,11 @@ public function getExpectedType(): string
public static function withValue(
mixed $value,
string $expected_type,
?string $path = null,
?Throwable $previous = null
): self {
return new self(get_debug_type($value), $expected_type);
$paths = $previous instanceof Exception ? [$path, ...$previous->getPaths()] : [$path];

return new self(get_debug_type($value), $expected_type, Vec\filter_nulls($paths), $previous);
}
}
29 changes: 13 additions & 16 deletions src/Psl/Type/Exception/CoercionException.php
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,7 @@
namespace Psl\Type\Exception;

use Psl\Str;
use Psl\Vec;
use Throwable;

use function get_debug_type;
Expand All @@ -13,17 +14,21 @@ final class CoercionException extends Exception
{
private string $target;

public function __construct(string $actual, string $target, string $additionalInfo = '')
/**
* @param list<string> $paths
*/
public function __construct(string $actual, string $target, array $paths = [], ?Throwable $previous = null)
{
parent::__construct(
Str\format(
'Could not coerce "%s" to type "%s"%s%s',
'Could not coerce "%s" to type "%s"%s.',
$actual,
$target,
$additionalInfo ? ': ' : '.',
$additionalInfo
$paths ? ' at path "' . Str\join($paths, '.') . '"' : ''
),
$actual,
$paths,
$previous
);

$this->target = $target;
Expand All @@ -37,19 +42,11 @@ public function getTargetType(): string
public static function withValue(
mixed $value,
string $target,
?string $path = null,
?Throwable $previous = null
): self {
return new self(get_debug_type($value), $target);
}
$paths = $previous instanceof Exception ? [$path, ...$previous->getPaths()] : [$path];

public static function withConversionFailureOnValue(
mixed $value,
string $target,
Throwable $failure,
): self {
return new self(
get_debug_type($value),
$target,
$failure->getMessage()
);
return new self(get_debug_type($value), $target, Vec\filter_nulls($paths), $previous);
}
}
23 changes: 21 additions & 2 deletions src/Psl/Type/Exception/Exception.php
Original file line number Diff line number Diff line change
Expand Up @@ -10,13 +10,32 @@ abstract class Exception extends RuntimeException implements ExceptionInterface
{
private string $actual;

/**
* @var list<string>
*/
private array $paths;

/**
* @param list<string> $paths
*/
public function __construct(
string $message,
string $actual,
array $paths,
?\Throwable $previous = null
) {
parent::__construct($message);
parent::__construct($message, 0, $previous);

$this->paths = $paths;
$this->actual = $actual;
}

$this->actual = $actual;
/**
* @return list<string>
*/
public function getPaths(): array
{
return $this->paths;
}

public function getActualType(): string
Expand Down
2 changes: 1 addition & 1 deletion src/Psl/Type/Internal/ConvertedType.php
Original file line number Diff line number Diff line change
Expand Up @@ -53,7 +53,7 @@ public function coerce(mixed $value): mixed
try {
$converted = ($this->converter)($coercedInput);
} catch (Throwable $failure) {
throw CoercionException::withConversionFailureOnValue($value, $this->toString(), $failure);
throw CoercionException::withValue($value, $this->toString(), previous: $failure);
}

return $this->into->coerce($converted);
Expand Down
12 changes: 4 additions & 8 deletions src/Psl/Type/Internal/DictType.php
Original file line number Diff line number Diff line change
Expand Up @@ -42,17 +42,15 @@ public function coerce(mixed $value): array
throw CoercionException::withValue($value, $this->toString());
}

$key_type = $this->key_type;
$value_type = $this->value_type;

$result = [];

/**
* @var Tk $k
* @var Tv $v
*/
foreach ($value as $k => $v) {
$result[$key_type->coerce($k)] = $value_type->coerce($v);
// TODO : how to specify a 'key' path??
$result[$this->coerceChildType($this->key_type, $k, 'key')] = $this->coerceChildType($this->value_type, $v, $k);
}

return $result;
Expand All @@ -71,17 +69,15 @@ public function assert(mixed $value): array
throw AssertException::withValue($value, $this->toString());
}

$key_type = $this->key_type;
$value_type = $this->value_type;

$result = [];

/**
* @var Tk $k
* @var Tv $v
*/
foreach ($value as $k => $v) {
$result[$key_type->assert($k)] = $value_type->assert($v);
// TODO : how to specify a 'key' path??
$result[$this->assertChildType($this->key_type, $k, 'key')] = $this->assertChildType($this->value_type, $v, (string) $k);
}

return $result;
Expand Down
7 changes: 4 additions & 3 deletions src/Psl/Type/Internal/ShapeType.php
Original file line number Diff line number Diff line change
Expand Up @@ -119,7 +119,7 @@ private function coerceIterable(mixed $value): array
$result = [];
foreach ($this->elements_types as $element => $type) {
if (Iter\contains_key($array, $element)) {
$result[$element] = $type->coerce($array[$element]);
$result[$element] = $this->coerceChildType($type, $array[$element], $element);

continue;
}
Expand All @@ -128,7 +128,7 @@ private function coerceIterable(mixed $value): array
continue;
}

throw CoercionException::withValue($value, $this->toString());
throw CoercionException::withValue($value, $this->toString(), $element);
}

if ($this->allow_unknown_fields) {
Expand Down Expand Up @@ -159,7 +159,7 @@ public function assert(mixed $value): array
$result = [];
foreach ($this->elements_types as $element => $type) {
if (Iter\contains_key($value, $element)) {
$result[$element] = $type->assert($value[$element]);
$result[$element] = $this->assertChildType($type, $value[$element], $element);

continue;
}
Expand All @@ -183,6 +183,7 @@ public function assert(mixed $value): array
throw AssertException::withValue(
$value,
$this->toString(),
$k
);
}
}
Expand Down
14 changes: 4 additions & 10 deletions src/Psl/Type/Internal/VecType.php
Original file line number Diff line number Diff line change
Expand Up @@ -58,17 +58,14 @@ public function coerce(mixed $value): iterable
throw CoercionException::withValue($value, $this->toString());
}

/** @var Type\Type<Tv> $value_type */
$value_type = $this->value_type;

/**
* @var list<Tv> $entries
*/
$result = [];

/** @var Tv $v */
foreach ($value as $v) {
$result[] = $value_type->coerce($v);
foreach ($value as $i => $v) {
$result[] = $this->coerceChildType($this->value_type, $v, (string) $i);
}

return $result;
Expand All @@ -87,16 +84,13 @@ public function assert(mixed $value): array
throw AssertException::withValue($value, $this->toString());
}

/** @var Type\Type<Tv> $value_type */
$value_type = $this->value_type;

$result = [];

/**
* @var Tv $v
*/
foreach ($value as $v) {
$result[] = $value_type->assert($v);
foreach ($value as $i => $v) {
$result[] = $this->assertChildType($this->value_type, $v, (string) $i);
}

return $result;
Expand Down
46 changes: 46 additions & 0 deletions src/Psl/Type/Type.php
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,7 @@
namespace Psl\Type;

use Psl\Type\Exception\AssertException;
use Psl\Type\Exception\CoercionException;

/**
* @template-covariant T
Expand All @@ -31,4 +32,49 @@ public function isOptional(): bool
{
return false;
}

/**
* @template Tc
* @param TypeInterface<Tc> $childType
*
* @return Tc
*
* @throws CoercionException
*/
protected function coerceChildType(TypeInterface $childType, mixed $value, ?string $path = null): mixed
{
try {
return $childType->coerce($value);
} catch (Exception\CoercionException $e) {
throw CoercionException::withValue(
$value,
$this->toString(),
$path,
$e,
);
}
}

/**
* @template Tc
* @param TypeInterface<Tc> $childType
*
* @psalm-assert Tc $value
* @return Tc
*
* @throws AssertException
*/
protected function assertChildType(TypeInterface $childType, mixed $value, ?string $path = null): mixed
{
try {
return $childType->assert($value);
} catch (Exception\AssertException $e) {
throw AssertException::withValue(
$value,
$this->toString(),
$path,
$e,
);
}
}
}

0 comments on commit 474de7e

Please sign in to comment.