Symfony PHPUnit - Внедрение зависимости

Я хочу протестировать этот TokenProvider

<?php

declare(strict_types=1);

namespace App\Services\Provider;

use App\Repository\UserRepository;
use Symfony\Component\HttpKernel\Exception\NotFoundHttpException;
use Symfony\Component\Security\Core\Encoder\UserPasswordEncoderInterface;
use Lexik\Bundle\JWTAuthenticationBundle\Encoder\JWTEncoderInterface;
use Symfony\Component\Security\Core\Exception\BadCredentialsException;

/**
 * Class TokenProvider
 * @package App\Services\Provider
 */
class TokenProvider
{
    /** @var JWTEncoderInterface */
    private $JWTEncoder;
    /** @var UserPasswordEncoderInterface */
    private $passwordEncoder;
    /** @var UserRepository */
    private $userRepository;

    /**
     * TokenProvider constructor.
     *
     * @param JWTEncoderInterface          $JWTEncoder
     * @param UserPasswordEncoderInterface $passwordEncoder
     * @param UserRepository               $userRepository
     */
    public function __construct(JWTEncoderInterface $JWTEncoder, UserPasswordEncoderInterface $passwordEncoder, UserRepository $userRepository)
    {
        $this->JWTEncoder = $JWTEncoder;
        $this->passwordEncoder = $passwordEncoder;
        $this->userRepository = $userRepository;
    }

    /**
     * @param string $email
     * @param string $password
     *
     * @return string
     * @throws \Lexik\Bundle\JWTAuthenticationBundle\Exception\JWTEncodeFailureException
     */
    public function getToken(string $email, string $password): string
    {
        $user = $this->userRepository->findOneBy([
            'email' => $email,
        ]);

        if (!$user) {
            throw new NotFoundHttpException('User Not Found');
        }

        $isValid = $this->passwordEncoder->isPasswordValid($user, $password);
        if (!$isValid) {
            throw new BadCredentialsException();
        }

        return $this->JWTEncoder->encode([
            'email' => $user->getEmail(),
            'exp' => time() + 3600 // 1 hour expiration
        ]);
    }
}

Вот мой тест. Это еще не конец.

Я хочу вставить JWTEncoderInterface $encoder и UserPasswordEncoder $passwordEncoder в свой testGetToken().

class TokenProviderTest extends TestCase
{
    /**
     * @throws \Lexik\Bundle\JWTAuthenticationBundle\Exception\JWTEncodeFailureException
     */
    public function testGetToken()
    {
        $this->markTestSkipped();

        $JWTEncoder = //TODO;
        $passwordEncoder = //TODO;

        $tokenProvider = new TokenProvider(
            $JWTEncoder,
            $passwordEncoder,
            new class extends UserRepository{
                public function findOneBy(array $criteria, array $orderBy = null)
                {
                    return (new User())
                        ->setEmail('[email protected]')
                        ->setPassword('password')
                    ;
                }
            }
        );

        $token = $tokenProvider->getToken('[email protected]', 'password');
        $this->assertEquals(true, $token);
    }
}

Как лучше всего это сделать в TestCase?

Я не хочу издеваться над этими двумя сервисами, потому что хочу проверить, действителен ли мой токен, с помощью LexikJWTAuthenticationBundle.

Стоит ли изучать PHP в 2026-2027 годах?
Стоит ли изучать PHP в 2026-2027 годах?
Привет всем, сегодня я хочу высказать свои соображения по поводу вопроса, который я уже много раз получал в своем сообществе: "Стоит ли изучать PHP в...
Symfony Station Communiqué - 7 июля 2023 г
Symfony Station Communiqué - 7 июля 2023 г
Это коммюнике первоначально появилось на Symfony Station .
Symfony Station Communiqué - 17 февраля 2023 г
Symfony Station Communiqué - 17 февраля 2023 г
Это коммюнике первоначально появилось на Symfony Station , вашем источнике передовых новостей Symfony, PHP и кибербезопасности.
Управление ответами api для исключений на Symfony с помощью KernelEvents
Управление ответами api для исключений на Symfony с помощью KernelEvents
Много раз при создании api нам нужно возвращать клиентам разные ответы в зависимости от возникшего исключения.
9
0
8 418
2
Перейти к ответу Данный вопрос помечен как решенный

Ответы 2

Ответ принят как подходящий

Я рекомендую вам расширить KernelTestCase и функцию setUp () и получить вашу зависимость следующим образом:

use Symfony\Bundle\FrameworkBundle\Test\KernelTestCase;

class TokenProviderTest extends KernelTestCase
{
private $jWTEncoder

protected function setUp()
{
   self::bootKernel();
   $this->jWTEncoder = self::$container->get('App\Services\TokenProvider');
}

public function testGetToken()
{
  //Your code
}
}

Пусть это вам поможет: https://symfony.com/doc/current/testing/doctrine.html#functional-testing

Я столкнулся с этим ответом, когда продолжал получать следующее предупреждение об устаревании:

  1x: Since symfony/twig-bundle 5.2: Accessing the "twig" service directly from the container is deprecated, use dependency injection instead.
    1x in ExtensionTest::testIconsExtension from App\Tests\Templates\Icons

Я обошел это, используя:

static::bootKernel();
$container = self::$kernel->getContainer()->get('test.service_container');
$twig      = $container->get('twig');

Как описано в эта документация Symfony

Другие вопросы по теме