Symfony-使用LexikJWTAuthenticationBundle设置TokenController

时间:2018-10-26 02:05:29

标签: symfony symfony4 lexikjwtauthbundle

我正在使用LexikJWTAuthenticationBundle https://github.com/lexik/LexikJWTAuthenticationBundle/blob/master/Resources/doc/index.md#installation

我正在设置控制器以获取令牌:

class TokenController extends AbstractController
{
    /**
     * @Route("/api/token", name="token", methods={"POST"})
     * @param Request $request
     * @param JWTEncoderInterface $JWTEncoder
     * @return JsonResponse
     * @throws \Lexik\Bundle\JWTAuthenticationBundle\Exception\JWTEncodeFailureException
     */
    public function token(Request $request, JWTEncoderInterface $JWTEncoder)
    {
        $user = $this->getDoctrine()->getRepository(User::class)->findOneBy([
            'email' => $request->getUser(),
        ]);

        if (!$user) {
            throw $this->createNotFoundException('User Not Found');
        }

        $isValid = $this->get('security.password_encoder')
            ->isPasswordValid($user, $request->getPassword());
        if (!$isValid) {
            throw new BadCredentialsException();
        }
        $token = $JWTEncoder->encode([
                'email' => $user->getEmail(),
                'exp' => time() + 3600 // 1 hour expiration
            ]);

        return new JsonResponse(['token' => $token]);
    }
}

但是我有这个错误:

  

找不到服务“ security.password_encoder”:即使它存在   在应用程序的容器中,   “ App \ Controller \ TokenController”是一个较小的服务定位器,它   只知道“教义”,“ form.factory”,“ http_kernel”,   “ parameter_bag”,“ request_stack”,“ router”,   “ security.authorization_checker”,“ security.csrf.token_manager”,   “ security.token_storage”,“ serializer”,“ session”和“ twig”服务。   除非您需要额外的懒惰,否则请尝试使用依赖项注入   代替。否则,您需要使用声明   “ TokenController :: getSubscribedServices()”。

我已经使用了依赖注入,这是我的服务配置文件

services:
    # default configuration for services in *this* file
    _defaults:
        autowire: true      # Automatically injects dependencies in your services.
        autoconfigure: true # Automatically registers your services as commands, event subscribers, etc.
        public: false       # Allows optimizing the container by removing unused services; this also means
                            # fetching services directly from the container via $container->get() won't work.
                            # The best practice is to be explicit about your dependencies anyway.

    # makes classes in src/ available to be used as services
    # this creates a service per class whose id is the fully-qualified class name
    App\:
        resource: '../src/*'
        exclude: '../src/{DependencyInjection,Entity,Migrations,Tests,Kernel.php}'

    # controllers are imported separately to make sure services can be injected
    # as action arguments even if you don't extend any base controller class
    App\Controller\:
        resource: '../src/Controller'
        tags: ['controller.service_arguments']

问题出在哪里?

1 个答案:

答案 0 :(得分:1)

您正在从AbstractController扩展,使用此控制器,将限制您使用$this->get()访问的服务。要访问密码编码器服务,您可以将Symfony\Component\Security\Core\Encoder\UserPasswordEncoderInterface注入到控制器动作中或通过控制器类构造函数。

private $passwordEncoder;

public function __construct(UserPasswordEncoderInterface $passwordEncoder)
{
  $this->passwordEncoder = $passwordEncoder;
}
...

$this->passwordEncoder->isPasswordValid()
...

或者您可以从Symfony\Bundle\FrameworkBundle\Controller\Controller扩展为完整的容器访问权限。 $this->get('security.password_encoder')应该可以使用。