| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143 | <?php/* * This file is part of the Symfony package. * * (c) Fabien Potencier <fabien@symfony.com> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */namespace Symfony\Component\String;/** * A string whose value is computed lazily by a callback. * * @author Nicolas Grekas <p@tchwork.com> */class LazyString implements \Stringable, \JsonSerializable{    private \Closure|string $value;    /**     * @param callable|array $callback A callable or a [Closure, method] lazy-callable     */    public static function fromCallable(callable|array $callback, mixed ...$arguments): static    {        if (\is_array($callback) && !\is_callable($callback) && !(($callback[0] ?? null) instanceof \Closure || 2 < \count($callback))) {            throw new \TypeError(sprintf('Argument 1 passed to "%s()" must be a callable or a [Closure, method] lazy-callable, "%s" given.', __METHOD__, '['.implode(', ', array_map('get_debug_type', $callback)).']'));        }        $lazyString = new static();        $lazyString->value = static function () use (&$callback, &$arguments, &$value): string {            if (null !== $arguments) {                if (!\is_callable($callback)) {                    $callback[0] = $callback[0]();                    $callback[1] ??= '__invoke';                }                $value = $callback(...$arguments);                $callback = self::getPrettyName($callback);                $arguments = null;            }            return $value ?? '';        };        return $lazyString;    }    public static function fromStringable(string|int|float|bool|\Stringable $value): static    {        if (\is_object($value)) {            return static::fromCallable($value->__toString(...));        }        $lazyString = new static();        $lazyString->value = (string) $value;        return $lazyString;    }    /**     * Tells whether the provided value can be cast to string.     */    final public static function isStringable(mixed $value): bool    {        return \is_string($value) || $value instanceof \Stringable || \is_scalar($value);    }    /**     * Casts scalars and stringable objects to strings.     *     * @throws \TypeError When the provided value is not stringable     */    final public static function resolve(\Stringable|string|int|float|bool $value): string    {        return $value;    }    public function __toString(): string    {        if (\is_string($this->value)) {            return $this->value;        }        try {            return $this->value = ($this->value)();        } catch (\Throwable $e) {            if (\TypeError::class === $e::class && __FILE__ === $e->getFile()) {                $type = explode(', ', $e->getMessage());                $type = substr(array_pop($type), 0, -\strlen(' returned'));                $r = new \ReflectionFunction($this->value);                $callback = $r->getStaticVariables()['callback'];                $e = new \TypeError(sprintf('Return value of %s() passed to %s::fromCallable() must be of the type string, %s returned.', $callback, static::class, $type));            }            throw $e;        }    }    public function __sleep(): array    {        $this->__toString();        return ['value'];    }    public function jsonSerialize(): string    {        return $this->__toString();    }    private function __construct()    {    }    private static function getPrettyName(callable $callback): string    {        if (\is_string($callback)) {            return $callback;        }        if (\is_array($callback)) {            $class = \is_object($callback[0]) ? get_debug_type($callback[0]) : $callback[0];            $method = $callback[1];        } elseif ($callback instanceof \Closure) {            $r = new \ReflectionFunction($callback);            if (str_contains($r->name, '{closure}') || !$class = \PHP_VERSION_ID >= 80111 ? $r->getClosureCalledClass() : $r->getClosureScopeClass()) {                return $r->name;            }            $class = $class->name;            $method = $r->name;        } else {            $class = get_debug_type($callback);            $method = '__invoke';        }        return $class.'::'.$method;    }}
 |