|null * * @readonly */ public $arguments; /** * An optional condition that will determine if the message will be executed. * * @var callable(): bool|null */ public $condition = null; /** * Creates a new higher order message. * * @param array|null $arguments */ public function __construct(string $filename, int $line, string $methodName, $arguments) { $this->filename = $filename; $this->line = $line; $this->name = $methodName; $this->arguments = $arguments; } /** * Re-throws the given `$throwable` with the good line and filename. * * @return mixed */ public function call(object $target) { /* @phpstan-ignore-next-line */ if (is_callable($this->condition) && call_user_func(Closure::bind($this->condition, $target)) === false) { return $target; } if ($this->hasHigherOrderCallable()) { /* @phpstan-ignore-next-line */ return (new HigherOrderCallables($target))->{$this->name}(...$this->arguments); } try { return is_array($this->arguments) ? Reflection::call($target, $this->name, $this->arguments) : $target->{$this->name}; /* @phpstan-ignore-line */ } catch (Throwable $throwable) { Reflection::setPropertyValue($throwable, 'file', $this->filename); Reflection::setPropertyValue($throwable, 'line', $this->line); if ($throwable->getMessage() === self::getUndefinedMethodMessage($target, $this->name)) { /** @var ReflectionClass $reflection */ $reflection = new ReflectionClass($target); /* @phpstan-ignore-next-line */ $reflection = $reflection->getParentClass() ?: $reflection; Reflection::setPropertyValue($throwable, 'message', sprintf('Call to undefined method %s::%s()', $reflection->getName(), $this->name)); } throw $throwable; } } /** * Indicates that this message should only be called when the given condition is true. * * @param callable(): bool $condition */ public function when(callable $condition): self { $this->condition = $condition; return $this; } /** * Determines whether or not there exists a higher order callable with the message name. * * @return bool */ private function hasHigherOrderCallable() { return in_array($this->name, get_class_methods(HigherOrderCallables::class), true); } private static function getUndefinedMethodMessage(object $target, string $methodName): string { if (\PHP_MAJOR_VERSION >= 8) { return sprintf(sprintf(self::UNDEFINED_METHOD, sprintf('%s::%s()', get_class($target), $methodName))); } return sprintf(self::UNDEFINED_METHOD, $methodName); } }