feat: add CachedQRCodeProvider

This commit is contained in:
Ondrej Vlach 2024-01-18 15:21:41 +01:00
parent 574dba85ad
commit 9bd3b5efff
Signed by: ovlach
GPG Key ID: 4FF1A23B4914DE70
8 changed files with 73 additions and 6 deletions

View File

@ -2,3 +2,4 @@ TODO:
----- -----
- [ ] Chybí speciální slovník nebo vypnutí slovníku pro testy - [ ] Chybí speciální slovník nebo vypnutí slovníku pro testy
- [ ] V reálný aplikaci bych použil Mockery, nicméně tady mě to přijde zbytečný - [ ] V reálný aplikaci bych použil Mockery, nicméně tady mě to přijde zbytečný
- [ ] Nastavení cache ideálně v memcached/redis etc.

View File

@ -20,6 +20,7 @@
"php-http/httplug": "*", "php-http/httplug": "*",
"symfony/asset": "7.0.*", "symfony/asset": "7.0.*",
"symfony/asset-mapper": "7.0.*", "symfony/asset-mapper": "7.0.*",
"symfony/cache": "7.0.*",
"symfony/console": "7.0.*", "symfony/console": "7.0.*",
"symfony/dotenv": "7.0.*", "symfony/dotenv": "7.0.*",
"symfony/flex": "^2", "symfony/flex": "^2",

2
composer.lock generated
View File

@ -4,7 +4,7 @@
"Read more about it at https://getcomposer.org/doc/01-basic-usage.md#installing-dependencies", "Read more about it at https://getcomposer.org/doc/01-basic-usage.md#installing-dependencies",
"This file is @generated automatically" "This file is @generated automatically"
], ],
"content-hash": "dda7db8ccac14db47a72f26e4769dac8", "content-hash": "15e05e23293f0f945b50f450fe8213c3",
"packages": [ "packages": [
{ {
"name": "composer/semver", "name": "composer/semver",

View File

@ -1,7 +1,7 @@
framework: framework:
cache: cache:
# Unique name of your app: used to compute stable namespaces for cache keys. # Unique name of your app: used to compute stable namespaces for cache keys.
#prefix_seed: your_vendor_name/app_name prefix_seed: ovlach/symfony_example_app
# The "app" cache stores to the filesystem by default. # The "app" cache stores to the filesystem by default.
# The data in this cache should persist between deploys. # The data in this cache should persist between deploys.

View File

@ -40,9 +40,14 @@ services:
$username: '%app.usetreno.username%' $username: '%app.usetreno.username%'
$password: '%app.usetreno.password%' $password: '%app.usetreno.password%'
App\Service\CachedQRCodeGenerator:
arguments:
$innerGenerator: '@App\Service\Remote\UsetrenoQRCodeProvider'
$cacheDuration: 'PT60S'
App\Service\CurrencyListerInterface: '@App\Service\StaticCurrencyLister' App\Service\CurrencyListerInterface: '@App\Service\StaticCurrencyLister'
App\Service\QRCodeQROptionsProviderInterface: '@App\Service\QRCodeQROptionsDefaultProvider' App\Service\QRCodeQROptionsProviderInterface: '@App\Service\QRCodeQROptionsDefaultProvider'
App\Service\QRCodeGeneratorInterface: '@App\Service\Remote\UsetrenoQRCodeProvider' App\Service\QRCodeGeneratorInterface: '@App\Service\CachedQRCodeGenerator'
# add more service definitions when explicit configuration is needed # add more service definitions when explicit configuration is needed
# please note that last definitions always *replace* previous ones # please note that last definitions always *replace* previous ones

View File

@ -0,0 +1,12 @@
<?php
declare(strict_types=1);
namespace App\Service;
use App\Entity\QRCode\QRCode;
interface CacheableQRCodeGeneratorInterface extends QRCodeGeneratorInterface
{
// We can't calculate cache key directly from QRCode
public function getCacheKey(QRCode $entity): string;
}

View File

@ -0,0 +1,39 @@
<?php
declare(strict_types=1);
namespace App\Service;
use App\Entity\QRCode\QRCode;
use Exception;
use Psr\Cache\InvalidArgumentException;
use Psr\Log\LoggerInterface;
use Symfony\Contracts\Cache\CacheInterface;
use Symfony\Contracts\Cache\ItemInterface;
final readonly class CachedQRCodeGenerator implements QRCodeGeneratorInterface
{
private \DateInterval $cacheDuration;
/**
* @throws Exception
*/
public function __construct(private CacheableQRCodeGeneratorInterface $innerGenerator, private CacheInterface $cache, private LoggerInterface $logger, string $cacheDuration)
{
$this->cacheDuration = new \DateInterval($cacheDuration);
}
public function generateQRCodeFromEntity(QRCode $entity): string
{
$key = $this->innerGenerator->getCacheKey($entity);
return $this->cache->get($key, function(ItemInterface $c) use ($entity) {
$this->logger->debug("cache miss for key " . $c->getKey(), [
'entity' => $entity,
]);
$c->expiresAfter($this->cacheDuration);
return $this->innerGenerator->generateQRCodeFromEntity($entity);
});
}
}

View File

@ -4,13 +4,12 @@ declare(strict_types=1);
namespace App\Service\Remote; namespace App\Service\Remote;
use App\Entity\QRCode\QRCode; use App\Entity\QRCode\QRCode;
use App\Service\QRCodeGeneratorInterface; use App\Service\CacheableQRCodeGeneratorInterface;
use App\Service\Remote\Edge\QRCodeEntityConverter; use App\Service\Remote\Edge\QRCodeEntityConverter;
use App\Service\Remote\Exception\AuthorizeException;
use App\Service\Remote\Exception\UsetrenoQRCodeException; use App\Service\Remote\Exception\UsetrenoQRCodeException;
use Psr\Log\LoggerInterface; use Psr\Log\LoggerInterface;
class UsetrenoQRCodeProvider implements QRCodeGeneratorInterface class UsetrenoQRCodeProvider implements CacheableQRCodeGeneratorInterface
{ {
const QRCODE_API = 'https://topapi.top-test.cz/chameleon/api/v1/qr-code/create-for-bank-account-payment'; const QRCODE_API = 'https://topapi.top-test.cz/chameleon/api/v1/qr-code/create-for-bank-account-payment';
const QRCODE_METHOD = 'POST'; const QRCODE_METHOD = 'POST';
@ -86,4 +85,14 @@ class UsetrenoQRCodeProvider implements QRCodeGeneratorInterface
return $data->data->base64Data; return $data->data->base64Data;
} }
public function getCacheKey(QRCode $entity): string
{
$edgeEntity = $this->codeEntityConverter->convert($entity);
$encodedEntity = json_encode($edgeEntity);
if ($encodedEntity === false) {
throw new \RuntimeException("Can't serialize edge entity");
}
return base64_encode($encodedEntity);
}
} }