feat: add CachedQRCodeProvider
This commit is contained in:
parent
574dba85ad
commit
9bd3b5efff
@ -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.
|
||||||
|
@ -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
2
composer.lock
generated
@ -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",
|
||||||
|
@ -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.
|
||||||
|
@ -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
|
||||||
|
12
src/Service/CacheableQRCodeGeneratorInterface.php
Normal file
12
src/Service/CacheableQRCodeGeneratorInterface.php
Normal 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;
|
||||||
|
}
|
39
src/Service/CachedQRCodeGenerator.php
Normal file
39
src/Service/CachedQRCodeGenerator.php
Normal 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);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
@ -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);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user