diff --git a/app/code/Magento/CatalogPriceDataExporter/Model/Indexer/PriceBuilder.php b/app/code/Magento/CatalogPriceDataExporter/Model/Indexer/PriceBuilder.php new file mode 100644 index 00000000..30934485 --- /dev/null +++ b/app/code/Magento/CatalogPriceDataExporter/Model/Indexer/PriceBuilder.php @@ -0,0 +1,161 @@ +optionValueUid = $optionValueUid; + $this->downloadableLinksOptionUid = $downloadableLinksOptionUid; + } + + /** + * Build event data for product custom option price + * + * @param array $data + * + * @return array + * + * @throws \InvalidArgumentException + */ + public function buildCustomOptionPriceEventData(array $data): array + { + $id = $this->optionValueUid->resolve([CustomizableEnteredOptionValueUid::OPTION_ID => $data['option_id']]); + + return [ + 'id' => $id, + 'value' => $data['price'], + 'price_type' => $data['price_type'], + ]; + } + + /** + * Build complex product event data. + * + * @param string $parentId + * @param string $variationId + * @param string $linkType + * + * @return array + */ + public function buildComplexProductEventData(string $parentId, string $variationId, string $linkType): array + { + return [ + 'id' => $parentId, + 'variation_id' => $variationId, + 'price_type' => $linkType, + ]; + } + + /** + * Build custom option type price event data + * + * @param array $data + * + * @return array + * + * @throws \InvalidArgumentException + */ + public function buildCustomOptionTypePriceEventData(array $data): array + { + $id = $this->optionValueUid->resolve( + [ + CustomizableSelectedOptionValueUid::OPTION_ID => $data['option_id'], + CustomizableSelectedOptionValueUid::OPTION_VALUE_ID => $data['option_type_id'], + ] + ); + + return [ + 'id' => $id, + 'value' => $data['price'], + 'price_type' => $data['price_type'], + ]; + } + + /** + * Build downloadable link price event data. + * + * @param string $entityId + * @param string $value + * + * @return array + * + * @throws \InvalidArgumentException + */ + public function buildDownloadableLinkPriceEventData(string $entityId, string $value): array + { + $id = $this->downloadableLinksOptionUid->resolve([DownloadableLinksOptionUid::OPTION_ID => $entityId]); + return [ + 'id' => $id, + 'value' => $value, + ]; + } + + /** + * Build product price event data. + * + * @param string $entityId + * @param string $attributeCode + * @param string|null $attributeValue + * + * @return array + */ + public function buildProductPriceEventData(string $entityId, string $attributeCode, ?string $attributeValue): array + { + return [ + 'id' => $entityId, + 'attribute_code' => $attributeCode, + 'value' => $attributeValue, + ]; + } + + /** + * Build tier price event data. + * + * @param string $entityId + * @param string $qty + * @param string|null $priceType + * @param string|null $value + * @return array + */ + public function buildTierPriceEventData(string $entityId, string $qty, ?string $priceType, ?string $value): array + { + return [ + 'id' => $entityId, + 'attribute_code' => 'tier_price', + 'qty' => $qty, + 'price_type' => $priceType, + 'value' => $value, + ]; + } +} diff --git a/app/code/Magento/CatalogPriceDataExporter/Model/Provider/FullReindex/ComplexProductEvent.php b/app/code/Magento/CatalogPriceDataExporter/Model/Provider/FullReindex/ComplexProductEvent.php new file mode 100644 index 00000000..90a5f7ac --- /dev/null +++ b/app/code/Magento/CatalogPriceDataExporter/Model/Provider/FullReindex/ComplexProductEvent.php @@ -0,0 +1,142 @@ +resourceConnection = $resourceConnection; + $this->complexProductLink = $complexProductLink; + $this->eventKeyGenerator = $eventKeyGenerator; + $this->storeManager = $storeManager; + $this->priceBuilder = $priceBuilder; + $this->logger = $logger; + $this->linkType = $linkType; + } + + /** + * @inheritdoc + */ + public function retrieve(): \Generator + { + $continue = true; + $lastKnownId = 0; + try { + while ($continue === true) { + $result = []; + $select = $this->complexProductLink->getQuery(null, null, (int)$lastKnownId, self::BATCH_SIZE); + $cursor = $this->resourceConnection->getConnection()->query($select); + while ($row = $cursor->fetch()) { + $result[$row['parent_id']][] = $row['variation_id']; + $lastKnownId = $row['link_id']; + } + if (empty($result)) { + $continue = false; + } else { + yield $this->getEventsData($result); + } + } + } catch (\Throwable $exception) { + $this->logger->error($exception->getMessage()); + throw new UnableRetrieveData('Unable to retrieve complex product link data for full sync.'); + } + } + + /** + * Retrieve prices event data + * + * @param array $data + * + * @return array + * + * @throws LocalizedException + */ + private function getEventsData(array $data): array + { + $events = []; + $websiteId = (string)$this->storeManager->getWebsite(WebsiteInterface::ADMIN_CODE)->getWebsiteId(); + $key = $this->eventKeyGenerator->generate(self::EVENT_VARIATION_CHANGED, $websiteId, null); + $linkType = $this->linkType; + foreach ($data as $parentId => $children) { + foreach ($children as $variationId) { + $events[$key][] = $this->priceBuilder->buildComplexProductEventData( + (string)$parentId, + $variationId, + $linkType + ); + } + } + return $events; + } +} diff --git a/app/code/Magento/CatalogPriceDataExporter/Model/Provider/FullReindex/CustomOptionPriceEvent.php b/app/code/Magento/CatalogPriceDataExporter/Model/Provider/FullReindex/CustomOptionPriceEvent.php new file mode 100644 index 00000000..d8142353 --- /dev/null +++ b/app/code/Magento/CatalogPriceDataExporter/Model/Provider/FullReindex/CustomOptionPriceEvent.php @@ -0,0 +1,134 @@ +resourceConnection = $resourceConnection; + $this->customOptionPrice = $customOptionPrice; + $this->storeManager = $storeManager; + $this->priceBuilder = $priceBuilder; + $this->eventKeyGenerator = $eventKeyGenerator; + $this->logger = $logger; + } + + /** + * @inheritdoc + */ + public function retrieve(): \Generator + { + try { + foreach ($this->storeManager->getStores(true) as $store) { + $storeId = (int)$store->getId(); + $continue = true; + $lastKnownId = 0; + while ($continue === true) { + $result = []; + $select = $this->customOptionPrice->getQuery([], $storeId, $lastKnownId, self::BATCH_SIZE); + $cursor = $this->resourceConnection->getConnection()->query($select); + while ($row = $cursor->fetch()) { + $result[$row['option_id']] = [ + 'option_id' => $row['option_id'], + 'price' => $row['price'], + 'price_type' => $row['price_type'], + ]; + } + if (empty($result)) { + $continue = false; + } else { + yield $this->getEventData($result, $storeId); + $lastKnownId = array_key_last($result); + } + } + } + } catch (\Throwable $exception) { + $this->logger->error($exception->getMessage()); + throw new UnableRetrieveData('Unable to retrieve product custom options price data.'); + } + } + + /** + * Retrieve prices event data + * + * @param array $resultData + * @param int $storeId + * + * @return array + * + * @throws NoSuchEntityException + */ + private function getEventData(array $resultData, int $storeId): array + { + $events = []; + $websiteId = (string)$this->storeManager->getStore($storeId)->getWebsiteId(); + $key = $this->eventKeyGenerator->generate(self::EVENT_CUSTOM_OPTION_PRICE_CHANGED, $websiteId, null); + foreach ($resultData as $priceData) { + $events[$key][] = $this->priceBuilder->buildCustomOptionPriceEventData($priceData); + } + return $events; + } +} diff --git a/app/code/Magento/CatalogPriceDataExporter/Model/Provider/FullReindex/CustomOptionTypePriceEvent.php b/app/code/Magento/CatalogPriceDataExporter/Model/Provider/FullReindex/CustomOptionTypePriceEvent.php new file mode 100644 index 00000000..855969b2 --- /dev/null +++ b/app/code/Magento/CatalogPriceDataExporter/Model/Provider/FullReindex/CustomOptionTypePriceEvent.php @@ -0,0 +1,144 @@ +resourceConnection = $resourceConnection; + $this->customOptionTypePrice = $customOptionTypePrice; + $this->storeManager = $storeManager; + $this->priceBuilder = $priceBuilder; + $this->eventKeyGenerator = $eventKeyGenerator; + $this->logger = $logger; + } + + /** + * @inheritdoc + */ + public function retrieve(): \Generator + { + try { + foreach ($this->storeManager->getStores(true) as $store) { + $storeId = (int)$store->getId(); + $continue = true; + $lastKnownId = 0; + while ($continue === true) { + $select = $this->customOptionTypePrice->getQuery( + [], + $storeId, + $lastKnownId, + self::BATCH_SIZE + ); + $cursor = $this->resourceConnection->getConnection()->query($select); + $result = []; + while ($row = $cursor->fetch()) { + $result[$row['option_type_id']] = [ + 'option_id' => $row['option_id'], + 'option_type_id' => $row['option_type_id'], + 'price' => $row['price'], + 'price_type' => $row['price_type'], + ]; + } + if (empty($result)) { + $continue = false; + } else { + yield $this->getEventsData($result, $storeId); + $lastKnownId = array_key_last($result); + } + } + } + } catch (\Throwable $exception) { + $this->logger->error($exception->getMessage()); + throw new UnableRetrieveData('Unable to retrieve product custom option types price data for full sync.'); + } + } + + /** + * Retrieve prices event data + * + * @param array $resultData + * @param int $storeId + * + * @return array + * + * @throws NoSuchEntityException + */ + private function getEventsData(array $resultData, int $storeId): array + { + $events = []; + $websiteId = (string)$this->storeManager->getStore($storeId)->getWebsiteId(); + $key = $this->eventKeyGenerator->generate( + self::EVENT_CUSTOM_OPTION_TYPE_PRICE_CHANGED, + $websiteId, + null + ); + foreach ($resultData as $priceData) { + $events[$key][] = $this->priceBuilder->buildCustomOptionTypePriceEventData($priceData); + } + return $events; + } +} diff --git a/app/code/Magento/CatalogPriceDataExporter/Model/Provider/FullReindex/DownloadableLinkPriceEvent.php b/app/code/Magento/CatalogPriceDataExporter/Model/Provider/FullReindex/DownloadableLinkPriceEvent.php new file mode 100644 index 00000000..9a7cb19f --- /dev/null +++ b/app/code/Magento/CatalogPriceDataExporter/Model/Provider/FullReindex/DownloadableLinkPriceEvent.php @@ -0,0 +1,134 @@ +resourceConnection = $resourceConnection; + $this->downloadableLinkPrice = $downloadableLinkPrice; + $this->storeManager = $storeManager; + $this->priceBuilder = $priceBuilder; + $this->eventKeyGenerator = $eventKeyGenerator; + $this->logger = $logger; + } + + /** + * @inheritdoc + */ + public function retrieve(): \Generator + { + try { + foreach ($this->storeManager->getStores(true) as $store) { + $storeId = (int)$store->getId(); + $continue = true; + $lastKnownId = 0; + while ($continue === true) { + $result = []; + $select = $this->downloadableLinkPrice->getQuery([], $storeId, (int)$lastKnownId, self::BATCH_SIZE); + $cursor = $this->resourceConnection->getConnection()->query($select); + while ($row = $cursor->fetch()) { + $result[$row['entity_id']] = $row['value']; + $lastKnownId = $row['link_id']; + } + if (empty($result)) { + $continue = false; + } else { + yield $this->getEventsData($result, $storeId); + } + } + } + } catch (\Throwable $exception) { + $this->logger->error($exception->getMessage()); + throw new UnableRetrieveData('Unable to retrieve downloadable link price data for full sync.'); + } + } + + /** + * Retrieve prices event data + * + * @param array $actualData + * @param int $storeId + * + * @return array + * + * @throws NoSuchEntityException + */ + private function getEventsData(array $actualData, int $storeId): array + { + $events = []; + $websiteId = (string)$this->storeManager->getStore($storeId)->getWebsiteId(); + $key = $this->eventKeyGenerator->generate( + self::EVENT_DOWNLOADABLE_LINK_PRICE_CHANGED, + $websiteId, + null + ); + foreach ($actualData as $entityId => $value) { + $events[$key][] = $this->priceBuilder->buildDownloadableLinkPriceEventData((string)$entityId, $value); + } + return $events; + } +} diff --git a/app/code/Magento/CatalogPriceDataExporter/Model/Provider/FullReindex/ProductPriceEvent.php b/app/code/Magento/CatalogPriceDataExporter/Model/Provider/FullReindex/ProductPriceEvent.php new file mode 100644 index 00000000..12c1a901 --- /dev/null +++ b/app/code/Magento/CatalogPriceDataExporter/Model/Provider/FullReindex/ProductPriceEvent.php @@ -0,0 +1,146 @@ +resourceConnection = $resourceConnection; + $this->productPrice = $productPrice; + $this->storeManager = $storeManager; + $this->eventKeyGenerator = $eventKeyGenerator; + $this->priceBuilder = $priceBuilder; + $this->logger = $logger; + } + + /** + * @inheritdoc + */ + public function retrieve(): \Generator + { + try { + foreach ($this->storeManager->getStores(true) as $store) { + $storeId = (int)$store->getId(); + $continue = true; + $lastKnownId = 0; + while ($continue === true) { + $select = $this->productPrice->getQuery( + [], + $storeId, + self::PRICE_ATTRIBUTES, + $lastKnownId, + self::BATCH_SIZE + ); + $cursor = $this->resourceConnection->getConnection()->query($select); + $result = []; + while ($row = $cursor->fetch()) { + $result[$row['entity_id']][$row['attribute_code']] = $row['value']; + } + if (empty($result)) { + $continue = false; + } else { + yield $this->getEventsData($result, $storeId); + $lastKnownId = array_key_last($result); + } + } + } + } catch (\Throwable $exception) { + $this->logger->error($exception->getMessage()); + throw new UnableRetrieveData('Unable to retrieve product price data for full sync.'); + } + } + + /** + * Retrieve prices event data. + * + * @param array $data + * @param int $storeId + * + * @return array + * + * @throws NoSuchEntityException + */ + private function getEventsData(array $data, int $storeId): array + { + $events = []; + $websiteId = (string)$this->storeManager->getStore($storeId)->getWebsiteId(); + $key = $this->eventKeyGenerator->generate(self::EVENT_PRICE_CHANGED, $websiteId, null); + foreach ($data as $entityId => $priceData) { + foreach ($priceData as $attributeCode => $value) { + $events[$key][] = $this->priceBuilder->buildProductPriceEventData( + (string)$entityId, + $attributeCode, + $value + ); + } + } + return $events; + } +} diff --git a/app/code/Magento/CatalogPriceDataExporter/Model/Provider/FullReindex/TierPriceEvent.php b/app/code/Magento/CatalogPriceDataExporter/Model/Provider/FullReindex/TierPriceEvent.php new file mode 100644 index 00000000..8edf49fd --- /dev/null +++ b/app/code/Magento/CatalogPriceDataExporter/Model/Provider/FullReindex/TierPriceEvent.php @@ -0,0 +1,159 @@ +resourceConnection = $resourceConnection; + $this->tierPrice = $tierPrice; + $this->storeManager = $storeManager; + $this->eventKeyGenerator = $eventKeyGenerator; + $this->priceBuilder = $priceBuilder; + $this->logger = $logger; + } + + /** + * @inheritdoc + */ + public function retrieve(): \Generator + { + try { + foreach ($this->storeManager->getStores(true) as $store) { + $storeId = (int)$store->getId(); + $continue = true; + $lastKnownId = 0; + while ($continue === true) { + $result = []; + $select = $this->tierPrice->getQuery([], $storeId, $lastKnownId, self::BATCH_SIZE); + $cursor = $this->resourceConnection->getConnection()->query($select); + while ($row = $cursor->fetch()) { + $result[$row['entity_id']][$row['customer_group_id']][$row['qty']] = $row; + } + if (empty($result)) { + $continue = false; + } else { + yield $this->getEventsData($result, $storeId); + $lastKnownId = array_key_last($result); + } + } + } + } catch (\Throwable $exception) { + $this->logger->error($exception->getMessage()); + throw new UnableRetrieveData('Unable to retrieve product tier price data for full sync.'); + } + } + + /** + * Build query arguments + * + * @param array $indexData + * + * @return array + */ + private function buildQueryArguments(): array + { + $queryArguments = []; + foreach ($this->storeManager->getStores(true) as $store) { + $queryArguments[$store->getId()] = []; + } + return $queryArguments; + } + + /** + * Form prices event data. + * + * @param array $actualData + * @param int $storeId + * @return array + * + * @throws NoSuchEntityException + */ + private function getEventsData(array $actualData, int $storeId): array + { + $events = []; + $websiteId = (string)$this->storeManager->getStore($storeId)->getWebsiteId(); + foreach ($actualData as $entityId => $entityData) { + foreach ($entityData as $customerGroup => $groupData) { + foreach ($groupData as $qty => $priceData) { + $eventType = $qty > 1 ? self::EVENT_TIER_PRICE_CHANGED : self::EVENT_PRICE_CHANGED; + $key = $this->eventKeyGenerator->generate( + $eventType, + $websiteId, + (string)$customerGroup + ); + $events[$key][] = $this->priceBuilder->buildTierPriceEventData( + (string)$entityId, + $qty, + $priceData['group_price_type'], + $priceData['value'] + ); + } + } + } + return $events; + } +} diff --git a/app/code/Magento/CatalogPriceDataExporter/Model/Provider/PartialReindex/ComplexProductEvent.php b/app/code/Magento/CatalogPriceDataExporter/Model/Provider/PartialReindex/ComplexProductEvent.php new file mode 100644 index 00000000..1600fcec --- /dev/null +++ b/app/code/Magento/CatalogPriceDataExporter/Model/Provider/PartialReindex/ComplexProductEvent.php @@ -0,0 +1,153 @@ +resourceConnection = $resourceConnection; + $this->complexProductLink = $complexProductLink; + $this->eventKeyGenerator = $eventKeyGenerator; + $this->storeManager = $storeManager; + $this->priceBuilder = $priceBuilder; + $this->logger = $logger; + $this->linkType = $linkType; + } + + /** + * @inheritdoc + */ + public function retrieve(array $indexData): \Generator + { + try { + foreach (\array_chunk($indexData, self::BATCH_SIZE) as $indexDataChunk) { + $result = []; + $parentIds = []; + $variationIds = []; + + foreach ($indexDataChunk as $key => $data) { + if (null === $data['parent_id']) { + unset($indexDataChunk[$key]); + continue; + } + + $parentIds[] = $data['parent_id']; + $variationIds[] = $data['entity_id']; + } + + $select = $this->complexProductLink->getQuery($parentIds, $variationIds); + $cursor = $this->resourceConnection->getConnection()->query($select); + + while ($row = $cursor->fetch()) { + $result[$row['parent_id']][] = $row['variation_id']; + } + + yield $this->getEventData($indexDataChunk, $result); + } + } catch (\Throwable $exception) { + $this->logger->error($exception->getMessage()); + throw new UnableRetrieveData('Unable to retrieve complex product link data.'); + } + } + + /** + * Retrieve prices event data + * + * @param array $indexData + * @param array $actualData + * + * @return array + * + * @throws LocalizedException + */ + private function getEventData(array $indexData, array $actualData): array + { + $events = []; + $linkType = $this->linkType; + foreach ($indexData as $data) { + $actualVariations = $actualData[$data['parent_id']] ?? []; + $eventType = \in_array($data['entity_id'], $actualVariations) ? self::EVENT_VARIATION_CHANGED + : self::EVENT_VARIATION_DELETED; + $websiteId = (string)$this->storeManager->getWebsite(WebsiteInterface::ADMIN_CODE)->getWebsiteId(); + $key = $this->eventKeyGenerator->generate($eventType, $websiteId, null); + $events[$key][] = $this->priceBuilder->buildComplexProductEventData( + $data['parent_id'], + $data['entity_id'], + $linkType + ); + } + + return $events; + } +} diff --git a/app/code/Magento/CatalogPriceDataExporter/Model/Provider/PartialReindex/CustomOptionPriceEvent.php b/app/code/Magento/CatalogPriceDataExporter/Model/Provider/PartialReindex/CustomOptionPriceEvent.php new file mode 100644 index 00000000..ae06d81a --- /dev/null +++ b/app/code/Magento/CatalogPriceDataExporter/Model/Provider/PartialReindex/CustomOptionPriceEvent.php @@ -0,0 +1,149 @@ +resourceConnection = $resourceConnection; + $this->customOptionPrice = $customOptionPrice; + $this->storeManager = $storeManager; + $this->priceBuilder = $priceBuilder; + $this->eventKeyGenerator = $eventKeyGenerator; + $this->logger = $logger; + } + + /** + * @inheritdoc + */ + public function retrieve(array $indexData): \Generator + { + try { + foreach (\array_chunk($indexData, self::BATCH_SIZE) as $indexDataChunk) { + $result = []; + $queryArguments = $this->buildQueryArguments($indexDataChunk); + foreach ($queryArguments as $scopeId => $optionIds) { + $select = $this->customOptionPrice->getQuery( + $optionIds, + $scopeId + ); + $cursor = $this->resourceConnection->getConnection()->query($select); + while ($row = $cursor->fetch()) { + $result[$scopeId][$row['option_id']] = [ + 'option_id' => $row['option_id'], + 'price' => $row['price'], + 'price_type' => $row['price_type'], + ]; + } + } + yield $this->getEventData($result); + } + } catch (\Throwable $exception) { + $this->logger->error($exception->getMessage()); + throw new UnableRetrieveData('Unable to retrieve product custom options price data.'); + } + } + + /** + * Build query arguments from index data or no data in case of full sync + * + * @param array $indexData + * + * @return array + */ + private function buildQueryArguments(array $indexData): array + { + $queryArguments = []; + foreach ($indexData as $data) { + $queryArguments[$data['scope_id']][] = $data['entity_id']; + } + return $queryArguments; + } + + /** + * Retrieve prices event data + * + * @param array $resultData + * + * @return array + * + * @throws NoSuchEntityException + * @throws \InvalidArgumentException + */ + private function getEventData(array $resultData): array + { + $events = []; + foreach ($resultData as $scopeId => $pricesData) { + foreach ($pricesData as $priceData) { + $websiteId = (string)$this->storeManager->getStore($scopeId)->getWebsiteId(); + $key = $this->eventKeyGenerator->generate(self::EVENT_CUSTOM_OPTION_PRICE_CHANGED, $websiteId, null); + $events[$key][] = $this->indexerPriceBuilder->buildCustomOptionPriceEventData($priceData); + } + } + + return $events; + } +} diff --git a/app/code/Magento/CatalogPriceDataExporter/Model/Provider/PartialReindex/CustomOptionTypePriceEvent.php b/app/code/Magento/CatalogPriceDataExporter/Model/Provider/PartialReindex/CustomOptionTypePriceEvent.php new file mode 100644 index 00000000..a8a3c3d4 --- /dev/null +++ b/app/code/Magento/CatalogPriceDataExporter/Model/Provider/PartialReindex/CustomOptionTypePriceEvent.php @@ -0,0 +1,152 @@ +resourceConnection = $resourceConnection; + $this->customOptionTypePrice = $customOptionTypePrice; + $this->storeManager = $storeManager; + $this->priceBuilder = $priceBuilder; + $this->eventKeyGenerator = $eventKeyGenerator; + $this->logger = $logger; + } + + /** + * @inheritdoc + */ + public function retrieve(array $indexData): \Generator + { + try { + foreach (\array_chunk($indexData, self::BATCH_SIZE) as $indexDataChunk) { + $result = []; + $queryArguments = $this->buildQueryArguments($indexDataChunk); + foreach ($queryArguments as $scopeId => $optionTypeIds) { + $select = $this->customOptionTypePrice->getQuery($optionTypeIds, $scopeId); + $cursor = $this->resourceConnection->getConnection()->query($select); + while ($row = $cursor->fetch()) { + $result[$scopeId][$row['option_type_id']] = [ + 'option_id' => $row['option_id'], + 'option_type_id' => $row['option_type_id'], + 'price' => $row['price'], + 'price_type' => $row['price_type'], + ]; + } + } + yield $this->getEventsData($result); + } + } catch (\Throwable $exception) { + $this->logger->error($exception->getMessage()); + throw new UnableRetrieveData('Unable to retrieve product custom option types price data.'); + } + } + + /** + * Build query arguments from index data or no data in case of full sync + * + * @param array $indexData + * + * @return array + */ + private function buildQueryArguments(array $indexData): array + { + $queryArguments = []; + foreach ($indexData as $data) { + $queryArguments[$data['scope_id']][] = $data['entity_id']; + } + return $queryArguments; + } + + /** + * Retrieve prices event data + * + * @param array $resultData + * + * @return array + * + * @throws NoSuchEntityException + * @throws \InvalidArgumentException + */ + private function getEventsData(array $resultData): array + { + $events = []; + + foreach ($resultData as $scopeId => $data) { + foreach ($data as $priceData) { + $websiteId = (string)$this->storeManager->getStore($scopeId)->getWebsiteId(); + $key = $this->eventKeyGenerator->generate( + self::EVENT_CUSTOM_OPTION_TYPE_PRICE_CHANGED, + $websiteId, + null + ); + $events[$key][] = $this->priceBuilder->buildCustomOptionTypePriceEventData($priceData); + } + } + + return $events; + } +} diff --git a/app/code/Magento/CatalogPriceDataExporter/Model/Provider/PartialReindex/DownloadableLinkPriceEvent.php b/app/code/Magento/CatalogPriceDataExporter/Model/Provider/PartialReindex/DownloadableLinkPriceEvent.php new file mode 100644 index 00000000..351bdfbf --- /dev/null +++ b/app/code/Magento/CatalogPriceDataExporter/Model/Provider/PartialReindex/DownloadableLinkPriceEvent.php @@ -0,0 +1,136 @@ +resourceConnection = $resourceConnection; + $this->downloadableLinkPrice = $downloadableLinkPrice; + $this->storeManager = $storeManager; + $this->priceBuilder = $priceBuilder; + $this->eventKeyGenerator = $eventKeyGenerator; + $this->logger = $logger; + } + + /** + * @inheritdoc + */ + public function retrieve(array $indexData): \Generator + { + try { + foreach (\array_chunk($indexData, self::BATCH_SIZE) as $indexDataChunk) { + $result = []; + $queryArguments = []; + + foreach ($indexDataChunk as $data) { + $queryArguments[$data['scope_id']][] = $data['entity_id']; + } + + foreach ($queryArguments as $scopeId => $ids) { + $select = $this->downloadableLinkPrice->getQuery($ids, $scopeId); + $cursor = $this->resourceConnection->getConnection()->query($select); + + while ($row = $cursor->fetch()) { + $result[$row['entity_id']][$scopeId] = $row['value']; + } + } + + yield $this->getEventsData($indexDataChunk, $result); + } + } catch (\Throwable $exception) { + $this->logger->error($exception->getMessage()); + throw new UnableRetrieveData('Unable to retrieve downloadable link price data.'); + } + } + + /** + * Retrieve prices event data + * + * @param array $indexData + * @param array $actualData + * + * @return array + * + * @throws LocalizedException + * @throws \InvalidArgumentException + */ + private function getEventsData(array $indexData, array $actualData): array + { + $events = []; + + foreach ($indexData as $data) { + $value = $actualData[$data['entity_id']][$data['scope_id']] ?? null; + $eventType = null === $value ? self::EVENT_DOWNLOADABLE_LINK_PRICE_DELETED : + self::EVENT_DOWNLOADABLE_LINK_PRICE_CHANGED; + $websiteId = (string)$this->storeManager->getStore($data['scope_id'])->getWebsiteId(); + $key = $this->eventKeyGenerator->generate($eventType, $websiteId, null); + $events[$key][] = $this->priceBuilder->buildDownloadableLinkPriceEventData($data['entity_id'], $value); + } + + return $events; + } +} diff --git a/app/code/Magento/CatalogPriceDataExporter/Model/Provider/PartialReindex/PartialReindexPriceProviderInterface.php b/app/code/Magento/CatalogPriceDataExporter/Model/Provider/PartialReindex/PartialReindexPriceProviderInterface.php new file mode 100644 index 00000000..0434b96d --- /dev/null +++ b/app/code/Magento/CatalogPriceDataExporter/Model/Provider/PartialReindex/PartialReindexPriceProviderInterface.php @@ -0,0 +1,30 @@ +resourceConnection = $resourceConnection; + $this->tierPrice = $tierPrice; + $this->eventKeyGenerator = $eventKeyGenerator; + $this->priceBuilder = $priceBuilder; + $this->logger = $logger; + } + + /** + * @inheritdoc + */ + public function retrieve(array $indexData): \Generator + { + try { + foreach (\array_chunk($indexData, self::BATCH_SIZE) as $indexDataChunk) { + $result = []; + $queryArguments = []; + foreach ($indexDataChunk as $data) { + $queryArguments[$data['scope_id']][] = $data['entity_id']; + } + foreach ($queryArguments as $scopeId => $entityIds) { + $select = $this->tierPrice->getQuery($entityIds, $scopeId); + $cursor = $this->resourceConnection->getConnection()->query($select); + + while ($row = $cursor->fetch()) { + $result[$row['scope_id']][$row['customer_group_id']][$row['entity_id']][$row['qty']] = $row; + } + } + yield $this->getEventData($indexDataChunk, $result); + } + } catch (\Throwable $exception) { + $this->logger->error($exception->getMessage()); + throw new UnableRetrieveData('Unable to retrieve product tier price data.'); + } + } + + /** + * Retrieve prices event data + * + * @param array $indexData + * @param array $actualData + * + * @return array + */ + private function getEventData(array $indexData, array $actualData): array + { + $events = []; + + foreach ($indexData as $data) { + $row = $actualData[$data['scope_id']][$data['customer_group']][$data['entity_id']][$data['qty']] ?? null; + $eventType = $this->resolveEventType($data['qty'], $row); + + $key = $this->eventKeyGenerator->generate($eventType, $data['scope_id'], $data['customer_group']); + $priceType = $data['group_price_type'] ?? null; + $priceValue = $data['value'] ?? null; + $events[$key][] = $this->priceBuilder->buildTierPriceEventData( + (string)$indexData['entity_id'], + $indexData['qty'], + $priceType, + $priceValue + ); + } + + return $events; + } + + /** + * Resolve event type + * + * @param string $qty + * @param array|null $data + * + * @return string + */ + private function resolveEventType(string $qty, ?array $data): string + { + if ($qty > 1) { + return $data === null ? self::EVENT_TIER_PRICE_DELETED : self::EVENT_TIER_PRICE_CHANGED; + } + return $data === null ? self::EVENT_PRICE_DELETED : self::EVENT_PRICE_CHANGED; + } +}