mirror of
https://gitlab.com/Chill-Projet/chill-bundles.git
synced 2025-06-07 18:44:08 +00:00
The documents are now generated in a queue, using symfony messenger. This queue should be configured: ```yaml # app/config/messenger.yaml framework: messenger: # reset services after consuming messages # reset_on_message: true failure_transport: failed transports: # https://symfony.com/doc/current/messenger.html#transport-configuration async: '%env(MESSENGER_TRANSPORT_DSN)%' priority: dsn: '%env(MESSENGER_TRANSPORT_DSN)%' failed: 'doctrine://default?queue_name=failed' routing: # ... other messages 'Chill\DocGeneratorBundle\Service\Messenger\RequestGenerationMessage': priority ``` `StoredObject`s now have additionnal properties: * status (pending, failure, ready (by default) ), which explain if the document is generated; * a generationTrialCounter, which is incremented on each generation trial, which prevent each generation more than 5 times; The generator computation is moved from the `DocGenTemplateController` to a `Generator` (implementing `GeneratorInterface`. There are new methods to `Context` which allow to normalize/denormalize context data to/from a messenger's `Message`.
144 lines
5.4 KiB
PHP
144 lines
5.4 KiB
PHP
<?php
|
|
|
|
namespace Chill\DocGeneratorBundle\Service\Generator;
|
|
|
|
use Chill\DocGeneratorBundle\Context\ContextManagerInterface;
|
|
use Chill\DocGeneratorBundle\Context\DocGeneratorContextWithPublicFormInterface;
|
|
use Chill\DocGeneratorBundle\Entity\DocGeneratorTemplate;
|
|
use Chill\DocGeneratorBundle\GeneratorDriver\DriverInterface;
|
|
use Chill\DocGeneratorBundle\GeneratorDriver\Exception\TemplateException;
|
|
use Chill\DocStoreBundle\Entity\StoredObject;
|
|
use Chill\DocStoreBundle\Service\StoredObjectManagerInterface;
|
|
use Doctrine\ORM\EntityManagerInterface;
|
|
use Psr\Log\LoggerInterface;
|
|
use Symfony\Component\HttpFoundation\File\File;
|
|
|
|
class Generator implements GeneratorInterface
|
|
{
|
|
private ContextManagerInterface $contextManager;
|
|
|
|
private DriverInterface $driver;
|
|
|
|
private EntityManagerInterface $entityManager;
|
|
|
|
private LoggerInterface $logger;
|
|
|
|
private StoredObjectManagerInterface $storedObjectManager;
|
|
|
|
private const LOG_PREFIX = '[docgen generator] ';
|
|
|
|
public function __construct(
|
|
ContextManagerInterface $contextManager,
|
|
DriverInterface $driver,
|
|
EntityManagerInterface $entityManager,
|
|
LoggerInterface $logger,
|
|
StoredObjectManagerInterface $storedObjectManager
|
|
) {
|
|
$this->contextManager = $contextManager;
|
|
$this->driver = $driver;
|
|
$this->entityManager = $entityManager;
|
|
$this->logger = $logger;
|
|
$this->storedObjectManager = $storedObjectManager;
|
|
}
|
|
|
|
/**
|
|
* @template T of File|null
|
|
* @template B of bool
|
|
* @param B $isTest
|
|
* @param (B is true ? T : null) $testFile
|
|
* @psalm-return (B is true ? string : null)
|
|
* @throws \Symfony\Component\Serializer\Exception\ExceptionInterface|\Throwable
|
|
*/
|
|
public function generateDocFromTemplate(
|
|
DocGeneratorTemplate $template,
|
|
int $entityId,
|
|
array $contextGenerationDataNormalized,
|
|
?StoredObject $destinationStoredObject = null,
|
|
bool $isTest = false,
|
|
?File $testFile = null
|
|
): ?string {
|
|
if ($destinationStoredObject instanceof StoredObject && StoredObject::STATUS_PENDING !== $destinationStoredObject->getStatus()) {
|
|
$this->logger->info(self::LOG_PREFIX.'Aborting generation of an already generated document');
|
|
throw new ObjectReadyException();
|
|
}
|
|
|
|
$this->logger->info(self::LOG_PREFIX.'Starting generation of a document', [
|
|
'entity_id' => $entityId,
|
|
'destination_stored_object' => $destinationStoredObject === null ? null : $destinationStoredObject->getId()
|
|
]);
|
|
|
|
$context = $this->contextManager->getContextByDocGeneratorTemplate($template);
|
|
|
|
$entity = $this
|
|
->entityManager
|
|
->find($context->getEntityClass(), $entityId)
|
|
;
|
|
|
|
if (null === $entity) {
|
|
throw new RelatedEntityNotFoundException($template->getEntity(), $entityId);
|
|
}
|
|
|
|
$contextGenerationDataNormalized = array_merge(
|
|
$contextGenerationDataNormalized,
|
|
$context instanceof DocGeneratorContextWithPublicFormInterface ?
|
|
$context->contextGenerationDataDenormalize($template, $entity, $contextGenerationDataNormalized)
|
|
: []
|
|
);
|
|
|
|
$data = $context->getData($template, $entity, $contextGenerationDataNormalized);
|
|
|
|
$destinationStoredObjectId = $destinationStoredObject instanceof StoredObject ? $destinationStoredObject->getId() : null;
|
|
$this->entityManager->clear();
|
|
gc_collect_cycles();
|
|
if (null !== $destinationStoredObjectId) {
|
|
$destinationStoredObject = $this->entityManager->find(StoredObject::class, $destinationStoredObjectId);
|
|
}
|
|
|
|
if ($isTest && ($testFile instanceof File)) {
|
|
$templateDecrypted = file_get_contents($testFile->getPathname());
|
|
} else {
|
|
$templateDecrypted = $this->storedObjectManager->read($template->getFile());
|
|
}
|
|
|
|
try {
|
|
$generatedResource = $this
|
|
->driver
|
|
->generateFromString(
|
|
$templateDecrypted,
|
|
$template->getFile()->getType(),
|
|
$data,
|
|
$template->getFile()->getFilename()
|
|
);
|
|
} catch (TemplateException $e) {
|
|
throw new GeneratorException($e->getErrors(), $e);
|
|
}
|
|
|
|
if ($isTest) {
|
|
$this->logger->info(self::LOG_PREFIX.'Finished generation of a document', [
|
|
'is_test' => true,
|
|
'entity_id' => $entityId,
|
|
'destination_stored_object' => $destinationStoredObject === null ? null : $destinationStoredObject->getId()
|
|
]);
|
|
return $generatedResource;
|
|
}
|
|
|
|
/** @var StoredObject $storedObject */
|
|
$destinationStoredObject
|
|
->setType($template->getFile()->getType())
|
|
->setFilename(sprintf('%s_odt', uniqid('doc_', true)))
|
|
->setStatus(StoredObject::STATUS_READY)
|
|
;
|
|
|
|
$this->storedObjectManager->write($destinationStoredObject, $generatedResource);
|
|
|
|
$this->entityManager->flush();
|
|
|
|
$this->logger->info(self::LOG_PREFIX.'Finished generation of a document', [
|
|
'entity_id' => $entityId,
|
|
'destination_stored_object' => $destinationStoredObject->getId(),
|
|
]);
|
|
|
|
return null;
|
|
}
|
|
}
|