123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220 |
- <?php
- /**
- * Copyright © Magento, Inc. All rights reserved.
- * See COPYING.txt for license details.
- */
- namespace Magento\AsynchronousOperations\Model;
- use Magento\Framework\App\ObjectManager;
- use Magento\Framework\App\ResourceConnection;
- use Magento\AsynchronousOperations\Api\Data\BulkSummaryInterface;
- use Magento\AsynchronousOperations\Api\Data\BulkSummaryInterfaceFactory;
- use Magento\AsynchronousOperations\Api\Data\OperationInterface;
- use Magento\Framework\MessageQueue\BulkPublisherInterface;
- use Magento\Framework\EntityManager\EntityManager;
- use Magento\Framework\EntityManager\MetadataPool;
- use Magento\AsynchronousOperations\Model\ResourceModel\Operation\CollectionFactory;
- use Magento\Authorization\Model\UserContextInterface;
- /**
- * Class BulkManagement
- *
- * @SuppressWarnings(PHPMD.CouplingBetweenObjects)
- */
- class BulkManagement implements \Magento\Framework\Bulk\BulkManagementInterface
- {
- /**
- * @var EntityManager
- */
- private $entityManager;
- /**
- * @var BulkSummaryInterfaceFactory
- */
- private $bulkSummaryFactory;
- /**
- * @var CollectionFactory
- */
- private $operationCollectionFactory;
- /**
- * @var BulkPublisherInterface
- */
- private $publisher;
- /**
- * @var MetadataPool
- */
- private $metadataPool;
- /**
- * @var ResourceConnection
- */
- private $resourceConnection;
- /**
- * @var \Magento\Authorization\Model\UserContextInterface
- */
- private $userContext;
- /**
- * @var \Psr\Log\LoggerInterface
- */
- private $logger;
- /**
- * BulkManagement constructor.
- * @param EntityManager $entityManager
- * @param BulkSummaryInterfaceFactory $bulkSummaryFactory
- * @param CollectionFactory $operationCollectionFactory
- * @param BulkPublisherInterface $publisher
- * @param MetadataPool $metadataPool
- * @param ResourceConnection $resourceConnection
- * @param \Psr\Log\LoggerInterface $logger
- * @param UserContextInterface $userContext
- */
- public function __construct(
- EntityManager $entityManager,
- BulkSummaryInterfaceFactory $bulkSummaryFactory,
- CollectionFactory $operationCollectionFactory,
- BulkPublisherInterface $publisher,
- MetadataPool $metadataPool,
- ResourceConnection $resourceConnection,
- \Psr\Log\LoggerInterface $logger,
- UserContextInterface $userContext = null
- ) {
- $this->entityManager = $entityManager;
- $this->bulkSummaryFactory= $bulkSummaryFactory;
- $this->operationCollectionFactory = $operationCollectionFactory;
- $this->metadataPool = $metadataPool;
- $this->resourceConnection = $resourceConnection;
- $this->publisher = $publisher;
- $this->logger = $logger;
- $this->userContext = $userContext ?: ObjectManager::getInstance()->get(UserContextInterface::class);
- }
- /**
- * @inheritDoc
- */
- public function scheduleBulk($bulkUuid, array $operations, $description, $userId = null)
- {
- $metadata = $this->metadataPool->getMetadata(BulkSummaryInterface::class);
- $connection = $this->resourceConnection->getConnectionByName($metadata->getEntityConnectionName());
- // save bulk summary and related operations
- $connection->beginTransaction();
- $userType = $this->userContext->getUserType();
- if ($userType === null) {
- $userType = UserContextInterface::USER_TYPE_ADMIN;
- }
- try {
- /** @var \Magento\AsynchronousOperations\Api\Data\BulkSummaryInterface $bulkSummary */
- $bulkSummary = $this->bulkSummaryFactory->create();
- $this->entityManager->load($bulkSummary, $bulkUuid);
- $bulkSummary->setBulkId($bulkUuid);
- $bulkSummary->setDescription($description);
- $bulkSummary->setUserId($userId);
- $bulkSummary->setUserType($userType);
- $bulkSummary->setOperationCount((int)$bulkSummary->getOperationCount() + count($operations));
- $this->entityManager->save($bulkSummary);
- $connection->commit();
- } catch (\Exception $exception) {
- $connection->rollBack();
- $this->logger->critical($exception->getMessage());
- return false;
- }
- $this->publishOperations($operations);
- return true;
- }
- /**
- * Retry bulk operations that failed due to given errors.
- *
- * @param string $bulkUuid target bulk UUID
- * @param array $errorCodes list of corresponding error codes
- * @return int number of affected bulk operations
- */
- public function retryBulk($bulkUuid, array $errorCodes)
- {
- $metadata = $this->metadataPool->getMetadata(BulkSummaryInterface::class);
- $connection = $this->resourceConnection->getConnectionByName($metadata->getEntityConnectionName());
- /** @var \Magento\AsynchronousOperations\Model\ResourceModel\Operation[] $retriablyFailedOperations */
- $retriablyFailedOperations = $this->operationCollectionFactory->create()
- ->addFieldToFilter('error_code', ['in' => $errorCodes])
- ->addFieldToFilter('bulk_uuid', ['eq' => $bulkUuid])
- ->getItems();
- // remove corresponding operations from database (i.e. move them to 'open' status)
- $connection->beginTransaction();
- try {
- $operationIds = [];
- $currentBatchSize = 0;
- $maxBatchSize = 10000;
- /** @var OperationInterface $operation */
- foreach ($retriablyFailedOperations as $operation) {
- if ($currentBatchSize === $maxBatchSize) {
- $connection->delete(
- $this->resourceConnection->getTableName('magento_operation'),
- $connection->quoteInto('id IN (?)', $operationIds)
- );
- $operationIds = [];
- $currentBatchSize = 0;
- }
- $currentBatchSize++;
- $operationIds[] = $operation->getId();
- // Rescheduled operations must be put in queue in 'open' state (i.e. without ID)
- $operation->setId(null);
- }
- // remove operations from the last batch
- if (!empty($operationIds)) {
- $connection->delete(
- $this->resourceConnection->getTableName('magento_operation'),
- $connection->quoteInto('id IN (?)', $operationIds)
- );
- }
- $connection->commit();
- } catch (\Exception $exception) {
- $connection->rollBack();
- $this->logger->critical($exception->getMessage());
- return 0;
- }
- $this->publishOperations($retriablyFailedOperations);
- return count($retriablyFailedOperations);
- }
- /**
- * Publish list of operations to the corresponding message queues.
- *
- * @param array $operations
- * @return void
- */
- private function publishOperations(array $operations)
- {
- $operationsByTopics = [];
- foreach ($operations as $operation) {
- $operationsByTopics[$operation->getTopicName()][] = $operation;
- }
- foreach ($operationsByTopics as $topicName => $operations) {
- $this->publisher->publish($topicName, $operations);
- }
- }
- /**
- * @inheritDoc
- */
- public function deleteBulk($bulkId)
- {
- return $this->entityManager->delete(
- $this->entityManager->load(
- $this->bulkSummaryFactory->create(),
- $bulkId
- )
- );
- }
- }
|