markitosgv/JWTRefreshTokenBundle

Hash currently clear refresh tokens on database

ninsuo opened this issue · 2 comments

Hello mates,

I was wondering why refresh tokens are stored clear on the database. Refresh tokens should be considered as sensitive as passwords as they can lead to full accounts compromise, and we all know we should not store passwords clear.

In this context, we do not need to use a strong password hashing such as bcrypt because refresh tokens have a very strong entropy (they are currently generated using 64 random bytes), so using SHA256 looks fair enough, and allows to find() existing tokens on the database without prior knowledge of the requester's username.

For those interested, it is already possible to hash refresh tokens, but at the cost of performance because it involves a doctrine subscriber. It may be more performant by doing it natively on this bundle; I may develop it if core contributors of this bundle think it worth it.

Implementation as of today

First, create a hash tool:

<?php

namespace App\Tools;

class Hash
{
    static public function hash(string $secret) : string
    {
        return hash('sha256', $_ENV['APP_SECRET'].$secret);
    }
}

Then, create a service that will overwrite the get() method that seeks for a refresh token on the database

<?php

namespace App\Service;

use App\Tools\Hash;
use Gesdinet\JWTRefreshTokenBundle\Doctrine\RefreshTokenManager as BaseRefreshTokenManager;

class RefreshTokenManager extends BaseRefreshTokenManager
{
    public function get($refreshToken)
    {
        return parent::get(
            Hash::hash($refreshToken)
        );
    }
}

And its DI to overwrite the original one:

# config/services.yaml
services:
  gesdinet.jwtrefreshtoken.refresh_token_manager:
    class: App\Service\RefreshTokenManager
    arguments:
      - '@gesdinet.jwtrefreshtoken.object_manager'
      - '%gesdinet.jwtrefreshtoken.refresh_token.class%'

Now, create a Doctrine subscriber:

<?php

namespace App\EventSubscriber\Doctrine;

use App\Tools\Hash;
use Doctrine\Bundle\DoctrineBundle\EventSubscriber\EventSubscriberInterface;
use Doctrine\ORM\Event\LifecycleEventArgs;
use Doctrine\ORM\Events;
use Gesdinet\JWTRefreshTokenBundle\Entity\RefreshToken;

class RefreshTokenSubscriber implements EventSubscriberInterface
{
    private array $cleartexts = [];

    public function getSubscribedEvents() : array
    {
        return [
            Events::prePersist,
            Events::postPersist,
        ];
    }

    public function prePersist(LifecycleEventArgs $args)
    {
        $entity = $args->getObject();

        if (!$entity instanceof RefreshToken) {
            return;
        }

        $this->hashBeforeSaving($entity);
    }

    public function postPersist(LifecycleEventArgs $args)
    {
        $entity = $args->getObject();

        if (!$entity instanceof RefreshToken) {
            return;
        }

        $this->makeClearBeforeUsing($entity);
    }

    private function hashBeforeSaving(RefreshToken $refreshToken)
    {
        if (128 !== strlen($refreshToken->getRefreshToken())) {
            return;
        }

        $this->cleartexts[spl_object_hash($refreshToken)] = $refreshToken->getRefreshToken();

        $refreshToken->setRefreshToken(Hash::hash($refreshToken));
    }

    private function makeClearBeforeUsing(RefreshToken $refreshToken)
    {
        if (!($this->cleartexts[spl_object_hash($refreshToken)] ?? false)) {
            return;
        }

        $refreshToken->setRefreshToken($this->cleartexts[spl_object_hash($refreshToken)]);

        unset($this->cleartexts[spl_object_hash($refreshToken)]);
    }
}

Finally, enforce single use of refresh tokens (because for multiple uses, we would need to read the clear token - but multiple uses of a refresh token is not a good practice anyway).

gesdinet_jwt_refresh_token:
  single_use: true

Hello,

Has this been implemented in the last versions ?
Thanks a lot !

Thank you so much! Very useful