<?php

declare(strict_types=1);

namespace ShlinkioTest\Shlink\Rest\Exception;

use Exception;
use Mezzio\ProblemDetails\Exception\ProblemDetailsExceptionInterface;
use PHPUnit\Framework\TestCase;
use Shlinkio\Shlink\Core\Exception\DeleteShortUrlException;
use Shlinkio\Shlink\Core\Exception\DomainNotFoundException;
use Shlinkio\Shlink\Core\Exception\ForbiddenTagOperationException;
use Shlinkio\Shlink\Core\Exception\InvalidUrlException;
use Shlinkio\Shlink\Core\Exception\NonUniqueSlugException;
use Shlinkio\Shlink\Core\Exception\ShortUrlNotFoundException;
use Shlinkio\Shlink\Core\Exception\TagConflictException;
use Shlinkio\Shlink\Core\Exception\TagNotFoundException;
use Shlinkio\Shlink\Core\Exception\ValidationException;
use Shlinkio\Shlink\Rest\Exception\BackwardsCompatibleProblemDetailsException;
use Shlinkio\Shlink\Rest\Exception\MercureException;
use Shlinkio\Shlink\Rest\Exception\MissingAuthenticationException;
use Shlinkio\Shlink\Rest\Exception\VerifyAuthenticationException;

class BackwardsCompatibleProblemDetailsExceptionTest extends TestCase
{
    /**
     * @test
     * @dataProvider provideTypes
     */
    public function typeIsRemappedOnWrappedException(
        string $wrappedType,
        string $expectedType,
        bool $expectSameType = false,
    ): void {
        $original = new class ($wrappedType) extends Exception implements ProblemDetailsExceptionInterface {
            public function __construct(private readonly string $type)
            {
                parent::__construct('');
            }

            public function getStatus(): int
            {
                return 123;
            }

            public function getType(): string
            {
                return $this->type;
            }

            public function getTitle(): string
            {
                return 'title';
            }

            public function getDetail(): string
            {
                return 'detail';
            }

            public function getAdditionalData(): array
            {
                return [];
            }

            public function toArray(): array
            {
                return ['type' => $this->type];
            }

            public function jsonSerialize(): array
            {
                return ['type' => $this->type];
            }
        };
        $e = BackwardsCompatibleProblemDetailsException::fromProblemDetails($original);

        self::assertEquals($e->getType(), $expectedType);
        self::assertEquals($e->toArray(), ['type' => $expectedType]);
        self::assertEquals($e->jsonSerialize(), ['type' => $expectedType]);

        self::assertEquals($original->getTitle(), $e->getTitle());
        self::assertEquals($original->getDetail(), $e->getDetail());
        self::assertEquals($original->getAdditionalData(), $e->getAdditionalData());

        if ($expectSameType) {
            self::assertEquals($original->getType(), $e->getType());
            self::assertEquals($original->toArray(), $e->toArray());
            self::assertEquals($original->jsonSerialize(), $e->jsonSerialize());
        } else {
            self::assertNotEquals($original->getType(), $e->getType());
            self::assertNotEquals($original->toArray(), $e->toArray());
            self::assertNotEquals($original->jsonSerialize(), $e->jsonSerialize());
        }
    }

    public function provideTypes(): iterable
    {
        yield ['foo', 'foo', true];
        yield ['bar', 'bar', true];
        yield [ValidationException::ERROR_CODE, 'INVALID_ARGUMENT'];
        yield [DeleteShortUrlException::ERROR_CODE, 'INVALID_SHORT_URL_DELETION'];
        yield [DomainNotFoundException::ERROR_CODE, 'DOMAIN_NOT_FOUND'];
        yield [ForbiddenTagOperationException::ERROR_CODE, 'FORBIDDEN_OPERATION'];
        yield [InvalidUrlException::ERROR_CODE, 'INVALID_URL'];
        yield [NonUniqueSlugException::ERROR_CODE, 'INVALID_SLUG'];
        yield [ShortUrlNotFoundException::ERROR_CODE, 'INVALID_SHORTCODE'];
        yield [TagConflictException::ERROR_CODE, 'TAG_CONFLICT'];
        yield [TagNotFoundException::ERROR_CODE, 'TAG_NOT_FOUND'];
        yield [MercureException::ERROR_CODE, 'MERCURE_NOT_CONFIGURED'];
        yield [MissingAuthenticationException::ERROR_CODE, 'INVALID_AUTHORIZATION'];
        yield [VerifyAuthenticationException::ERROR_CODE, 'INVALID_API_KEY'];
    }
}