| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427 | <?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\ErrorHandler\Exception;use Symfony\Component\HttpFoundation\Exception\RequestExceptionInterface;use Symfony\Component\HttpFoundation\Response;use Symfony\Component\HttpKernel\Exception\HttpExceptionInterface;/** * FlattenException wraps a PHP Error or Exception to be able to serialize it. * * Basically, this class removes all objects from the trace. * * @author Fabien Potencier <fabien@symfony.com> */class FlattenException{    /** @var string */    private $message;    /** @var int|string */    private $code;    /** @var self|null */    private $previous;    /** @var array */    private $trace;    /** @var string */    private $traceAsString;    /** @var string */    private $class;    /** @var int */    private $statusCode;    /** @var string */    private $statusText;    /** @var array */    private $headers;    /** @var string */    private $file;    /** @var int */    private $line;    /** @var string|null */    private $asString;    /**     * @return static     */    public static function create(\Exception $exception, ?int $statusCode = null, array $headers = []): self    {        return static::createFromThrowable($exception, $statusCode, $headers);    }    /**     * @return static     */    public static function createFromThrowable(\Throwable $exception, ?int $statusCode = null, array $headers = []): self    {        $e = new static();        $e->setMessage($exception->getMessage());        $e->setCode($exception->getCode());        if ($exception instanceof HttpExceptionInterface) {            $statusCode = $exception->getStatusCode();            $headers = array_merge($headers, $exception->getHeaders());        } elseif ($exception instanceof RequestExceptionInterface) {            $statusCode = 400;        }        if (null === $statusCode) {            $statusCode = 500;        }        if (class_exists(Response::class) && isset(Response::$statusTexts[$statusCode])) {            $statusText = Response::$statusTexts[$statusCode];        } else {            $statusText = 'Whoops, looks like something went wrong.';        }        $e->setStatusText($statusText);        $e->setStatusCode($statusCode);        $e->setHeaders($headers);        $e->setTraceFromThrowable($exception);        $e->setClass(\get_class($exception));        $e->setFile($exception->getFile());        $e->setLine($exception->getLine());        $previous = $exception->getPrevious();        if ($previous instanceof \Throwable) {            $e->setPrevious(static::createFromThrowable($previous));        }        return $e;    }    public function toArray(): array    {        $exceptions = [];        foreach (array_merge([$this], $this->getAllPrevious()) as $exception) {            $exceptions[] = [                'message' => $exception->getMessage(),                'class' => $exception->getClass(),                'trace' => $exception->getTrace(),            ];        }        return $exceptions;    }    public function getStatusCode(): int    {        return $this->statusCode;    }    /**     * @return $this     */    public function setStatusCode(int $code): self    {        $this->statusCode = $code;        return $this;    }    public function getHeaders(): array    {        return $this->headers;    }    /**     * @return $this     */    public function setHeaders(array $headers): self    {        $this->headers = $headers;        return $this;    }    public function getClass(): string    {        return $this->class;    }    /**     * @return $this     */    public function setClass(string $class): self    {        $this->class = false !== strpos($class, "@anonymous\0") ? (get_parent_class($class) ?: key(class_implements($class)) ?: 'class').'@anonymous' : $class;        return $this;    }    public function getFile(): string    {        return $this->file;    }    /**     * @return $this     */    public function setFile(string $file): self    {        $this->file = $file;        return $this;    }    public function getLine(): int    {        return $this->line;    }    /**     * @return $this     */    public function setLine(int $line): self    {        $this->line = $line;        return $this;    }    public function getStatusText(): string    {        return $this->statusText;    }    /**     * @return $this     */    public function setStatusText(string $statusText): self    {        $this->statusText = $statusText;        return $this;    }    public function getMessage(): string    {        return $this->message;    }    /**     * @return $this     */    public function setMessage(string $message): self    {        if (false !== strpos($message, "@anonymous\0")) {            $message = preg_replace_callback('/[a-zA-Z_\x7f-\xff][\\\\a-zA-Z0-9_\x7f-\xff]*+@anonymous\x00.*?\.php(?:0x?|:[0-9]++\$)[0-9a-fA-F]++/', function ($m) {                return class_exists($m[0], false) ? (get_parent_class($m[0]) ?: key(class_implements($m[0])) ?: 'class').'@anonymous' : $m[0];            }, $message);        }        $this->message = $message;        return $this;    }    /**     * @return int|string int most of the time (might be a string with PDOException)     */    public function getCode()    {        return $this->code;    }    /**     * @param int|string $code     *     * @return $this     */    public function setCode($code): self    {        $this->code = $code;        return $this;    }    public function getPrevious(): ?self    {        return $this->previous;    }    /**     * @return $this     */    public function setPrevious(?self $previous): self    {        $this->previous = $previous;        return $this;    }    /**     * @return self[]     */    public function getAllPrevious(): array    {        $exceptions = [];        $e = $this;        while ($e = $e->getPrevious()) {            $exceptions[] = $e;        }        return $exceptions;    }    public function getTrace(): array    {        return $this->trace;    }    /**     * @return $this     */    public function setTraceFromThrowable(\Throwable $throwable): self    {        $this->traceAsString = $throwable->getTraceAsString();        return $this->setTrace($throwable->getTrace(), $throwable->getFile(), $throwable->getLine());    }    /**     * @return $this     */    public function setTrace(array $trace, ?string $file, ?int $line): self    {        $this->trace = [];        $this->trace[] = [            'namespace' => '',            'short_class' => '',            'class' => '',            'type' => '',            'function' => '',            'file' => $file,            'line' => $line,            'args' => [],        ];        foreach ($trace as $entry) {            $class = '';            $namespace = '';            if (isset($entry['class'])) {                $parts = explode('\\', $entry['class']);                $class = array_pop($parts);                $namespace = implode('\\', $parts);            }            $this->trace[] = [                'namespace' => $namespace,                'short_class' => $class,                'class' => $entry['class'] ?? '',                'type' => $entry['type'] ?? '',                'function' => $entry['function'] ?? null,                'file' => $entry['file'] ?? null,                'line' => $entry['line'] ?? null,                'args' => isset($entry['args']) ? $this->flattenArgs($entry['args']) : [],            ];        }        return $this;    }    private function flattenArgs(array $args, int $level = 0, int &$count = 0): array    {        $result = [];        foreach ($args as $key => $value) {            if (++$count > 1e4) {                return ['array', '*SKIPPED over 10000 entries*'];            }            if ($value instanceof \__PHP_Incomplete_Class) {                $result[$key] = ['incomplete-object', $this->getClassNameFromIncomplete($value)];            } elseif (\is_object($value)) {                $result[$key] = ['object', \get_class($value)];            } elseif (\is_array($value)) {                if ($level > 10) {                    $result[$key] = ['array', '*DEEP NESTED ARRAY*'];                } else {                    $result[$key] = ['array', $this->flattenArgs($value, $level + 1, $count)];                }            } elseif (null === $value) {                $result[$key] = ['null', null];            } elseif (\is_bool($value)) {                $result[$key] = ['boolean', $value];            } elseif (\is_int($value)) {                $result[$key] = ['integer', $value];            } elseif (\is_float($value)) {                $result[$key] = ['float', $value];            } elseif (\is_resource($value)) {                $result[$key] = ['resource', get_resource_type($value)];            } else {                $result[$key] = ['string', (string) $value];            }        }        return $result;    }    private function getClassNameFromIncomplete(\__PHP_Incomplete_Class $value): string    {        $array = new \ArrayObject($value);        return $array['__PHP_Incomplete_Class_Name'];    }    public function getTraceAsString(): string    {        return $this->traceAsString;    }    /**     * @return $this     */    public function setAsString(?string $asString): self    {        $this->asString = $asString;        return $this;    }    public function getAsString(): string    {        if (null !== $this->asString) {            return $this->asString;        }        $message = '';        $next = false;        foreach (array_reverse(array_merge([$this], $this->getAllPrevious())) as $exception) {            if ($next) {                $message .= 'Next ';            } else {                $next = true;            }            $message .= $exception->getClass();            if ('' != $exception->getMessage()) {                $message .= ': '.$exception->getMessage();            }            $message .= ' in '.$exception->getFile().':'.$exception->getLine().                "\nStack trace:\n".$exception->getTraceAsString()."\n\n";        }        return rtrim($message);    }}
 |