apimatic/core-lib-php

View on GitHub
src/Response/Types/ErrorType.php

Summary

Maintainability
A
0 mins
Test Coverage
A
100%
<?php

declare(strict_types=1);

namespace Core\Response\Types;

use Core\Response\Context;
use Core\Utils\CoreHelper;
use CoreInterfaces\Core\Response\ResponseInterface;
use Rs\Json\Pointer;

class ErrorType
{
    /**
     * Initializes a new object with the description and class name provided.
     */
    public static function init(string $description, ?string $className = null): self
    {
        return new self($description, $className, false);
    }

    /**
     * Initializes a new object with error template and class name provided.
     */
    public static function initWithErrorTemplate(string $errorTemplate, ?string $className = null): self
    {
        return new self($errorTemplate, $className, true);
    }

    private $description;
    private $className;
    private $hasErrorTemplate;

    private function __construct(string $description, ?string $className, bool $hasErrorTemplate)
    {
        $this->description = $description;
        $this->className = $className;
        $this->hasErrorTemplate = $hasErrorTemplate;
    }

    /**
     * Throws an Api exception from the context provided.
     */
    public function throwable(Context $context)
    {
        $this->updateErrorDescriptionTemplate($context->getResponse());

        return $context->toApiException($this->description, $this->className);
    }

    private function updateErrorDescriptionTemplate($response): void
    {
        if (!$this->hasErrorTemplate) {
            return;
        }

        $errorDescriptionTemplate = $this->description;

        $jsonPointersInTemplate = $this->getJsonPointersFromTemplate($errorDescriptionTemplate);

        $errorDescription = $this->updateResponsePlaceholderValues(
            $errorDescriptionTemplate,
            $jsonPointersInTemplate,
            $response
        );

        $errorDescription = $this->updateHeaderPlaceHolderValues($errorDescription, $response);

        $errorDescription = $this->addPlaceHolderValue(
            $errorDescription,
            '{$statusCode}',
            $response->getStatusCode()
        );

        $this->description = $errorDescription;
    }

    private function updateHeaderPlaceHolderValues(string $errorDescription, ResponseInterface $response): string
    {
        $headers = $response->getHeaders();
        $headerKeys = array_keys($headers);

        for ($x = 0; $x < count($headerKeys); $x++) {
            $errorDescription = $this->addPlaceHolderValue(
                $errorDescription,
                '{$response.header.' . $headerKeys[$x] . '}',
                $headers[$headerKeys[$x]],
                true
            );
        }

        return $errorDescription;
    }

    /**
     * @param $errorDescription string
     * @param $jsonPointersInTemplate string[]
     * @param $response ResponseInterface
     * @return string Updated error string template.
     */
    private function updateResponsePlaceholderValues(
        string $errorDescription,
        array $jsonPointersInTemplate,
        ResponseInterface $response
    ): string {
        if (count($jsonPointersInTemplate[0]) < 1) {
            return $this->addPlaceHolderValue(
                $errorDescription,
                '{$response.body}',
                $response->getRawBody()
            );
        }

        $jsonResponsePointer = $this->initializeJsonPointer($response);

        $jsonPointers = $jsonPointersInTemplate[0];

        for ($x = 0; $x < count($jsonPointers); $x++) {
            $placeHolderValue = $this->getJsonPointerValue($jsonResponsePointer, ltrim($jsonPointers[$x], '#'));

            $errorDescription = $this->addPlaceHolderValue(
                $errorDescription,
                '{$response.body' . $jsonPointers[$x] . '}',
                $placeHolderValue
            );
        }

        return $errorDescription;
    }

    private function getJsonPointersFromTemplate(string $template): array
    {
        $pointerPattern = '/#[\w\/]*/i';

        preg_match_all($pointerPattern, $template, $matches);

        return $matches;
    }

    private function addPlaceHolderValue(
        string $template,
        string $placeHolder,
        $value,
        bool $searchCaseInsensitive = false
    ): string {
        if (!is_string($value)) {
            $value = var_export($value, true);
        }

        if ($searchCaseInsensitive) {
            return str_ireplace($placeHolder, $value, $template);
        }

        return str_replace($placeHolder, $value, $template);
    }

    /**
     * @param $jsonPointer ?Pointer
     * @param $pointer string
     * @return mixed Json pointer value from the JSON provided.
     */
    private function getJsonPointerValue(?Pointer $jsonPointer, string $pointer)
    {
        if ($jsonPointer == null || trim($pointer) === '') {
            return "";
        }

        try {
            $pointerValue = $jsonPointer->get($pointer);

            if (is_object($pointerValue)) {
                return CoreHelper::serialize($pointerValue);
            }

            return $pointerValue;
        } catch (\Exception $ex) {
            return "";
        }
    }

    private function initializeJsonPointer(ResponseInterface $response): ?Pointer
    {
        try {
            return new Pointer($response->getRawBody());
        } catch (\Exception $ex) {
            return null;
        }
    }
}