From 6d98e2a57059ef75d9d0dd0585b84bb33953e107 Mon Sep 17 00:00:00 2001 From: Maksym Aposov Date: Mon, 17 Oct 2016 09:18:00 +0300 Subject: [PATCH 01/15] MAGETWO-58701: memory_limit issue after running customer_grid indexer --- .../Magento/Customer/Model/Indexer/Source.php | 108 ++++++++++++++++++ .../Customer/Indexer/Collection.php | 20 ++++ app/code/Magento/Customer/etc/indexer.xml | 2 +- .../Magento/Framework/Indexer/Action/Base.php | 12 ++ .../Framework/Indexer/Action/Entity.php | 2 +- .../Framework/Indexer/SaveHandler/Batch.php | 16 +-- .../Framework/Indexer/Test/Unit/BatchTest.php | 3 +- 7 files changed, 150 insertions(+), 13 deletions(-) create mode 100644 app/code/Magento/Customer/Model/Indexer/Source.php create mode 100644 app/code/Magento/Customer/Model/ResourceModel/Customer/Indexer/Collection.php diff --git a/app/code/Magento/Customer/Model/Indexer/Source.php b/app/code/Magento/Customer/Model/Indexer/Source.php new file mode 100644 index 0000000000000..60522227eb222 --- /dev/null +++ b/app/code/Magento/Customer/Model/Indexer/Source.php @@ -0,0 +1,108 @@ +customerCollection = $collectionFactory->create(); + $this->batchSize = $batchSize; + } + + /** + * {@inheritdoc} + */ + public function getMainTable() + { + return $this->customerCollection->getMainTable(); + } + + /** + * {@inheritdoc} + */ + public function getIdFieldName() + { + return $this->customerCollection->getIdFieldName(); + } + + /** + * {@inheritdoc} + */ + public function addFieldToSelect($fieldName, $alias = null) + { + $this->customerCollection->addFieldToSelect($fieldName, $alias); + return $this; + } + + /** + * {@inheritdoc} + */ + public function getSelect() + { + return $this->customerCollection->getSelect(); + } + + /** + * {@inheritdoc} + */ + public function addFieldToFilter($attribute, $condition = null) + { + $this->customerCollection->addFieldToFilter($attribute, $condition); + return $this; + } + + /** + * @return int + */ + public function count() + { + return $this->customerCollection->getSize(); + } + + /** + * Retrieve an iterator + * + * @return Traversable + */ + public function getIterator() + { + $this->customerCollection->setPageSize($this->batchSize); + $lastPage = $this->customerCollection->getLastPageNumber(); + $pageNumber = 0; + do { + $this->customerCollection->clear(); + $this->customerCollection->setCurPage($pageNumber); + foreach ($this->customerCollection->getItems() as $key => $value) { + yield $key => $value; + } + $pageNumber++; + } while ($pageNumber <= $lastPage); + } +} diff --git a/app/code/Magento/Customer/Model/ResourceModel/Customer/Indexer/Collection.php b/app/code/Magento/Customer/Model/ResourceModel/Customer/Indexer/Collection.php new file mode 100644 index 0000000000000..5b9716af5393a --- /dev/null +++ b/app/code/Magento/Customer/Model/ResourceModel/Customer/Indexer/Collection.php @@ -0,0 +1,20 @@ +Customer Grid Rebuild Customer grid index -
diff --git a/lib/internal/Magento/Framework/Indexer/Action/Base.php b/lib/internal/Magento/Framework/Indexer/Action/Base.php index e7edcae5d4391..d761f388456b6 100644 --- a/lib/internal/Magento/Framework/Indexer/Action/Base.php +++ b/lib/internal/Magento/Framework/Indexer/Action/Base.php @@ -37,11 +37,13 @@ class Base implements ActionInterface /** * @var AdapterInterface + * @deprecated */ protected $connection; /** * @var SourceProviderInterface[] + * @deprecated */ protected $sources; @@ -52,6 +54,7 @@ class Base implements ActionInterface /** * @var HandlerInterface[] + * @deprecated */ protected $handlers; @@ -62,6 +65,7 @@ class Base implements ActionInterface /** * @var array + * @deprecated */ protected $columnTypesMap = [ 'varchar' => ['type' => Table::TYPE_TEXT, 'size' => 255], @@ -71,11 +75,13 @@ class Base implements ActionInterface /** * @var array + * @deprecated */ protected $filterColumns; /** * @var array + * @deprecated */ protected $searchColumns; @@ -96,6 +102,7 @@ class Base implements ActionInterface /** * @var String + * @deprecated */ protected $string; @@ -106,11 +113,13 @@ class Base implements ActionInterface /** * @var array + * @deprecated */ protected $filterable = []; /** * @var array + * @deprecated */ protected $searchable = []; @@ -272,6 +281,7 @@ protected function getPrimaryFieldset() protected function createResultCollection() { $select = $this->getPrimaryResource()->getSelect(); + $select->reset(\Magento\Framework\DB\Select::COLUMNS); $select->columns($this->getPrimaryResource()->getIdFieldName()); foreach ($this->data['fieldsets'] as $fieldset) { if (isset($fieldset['references'])) { @@ -342,6 +352,8 @@ protected function prepareFields() * * @param array $field * @return void + * + * @deprecated */ protected function saveFieldByType($field) { diff --git a/lib/internal/Magento/Framework/Indexer/Action/Entity.php b/lib/internal/Magento/Framework/Indexer/Action/Entity.php index e88d281fd4b81..b8390b9d5a002 100644 --- a/lib/internal/Magento/Framework/Indexer/Action/Entity.php +++ b/lib/internal/Magento/Framework/Indexer/Action/Entity.php @@ -24,6 +24,6 @@ protected function prepareDataSource(array $ids = []) { return !count($ids) ? $this->createResultCollection() - : $this->createResultCollection()->addFieldToFilter($this->getPrimaryResource()->getRowIdFieldName(), $ids); + : $this->createResultCollection()->addFieldToFilter($this->getPrimaryResource()->getIdFieldName(), $ids); } } diff --git a/lib/internal/Magento/Framework/Indexer/SaveHandler/Batch.php b/lib/internal/Magento/Framework/Indexer/SaveHandler/Batch.php index 31883fa8516ea..7d53e558287bd 100644 --- a/lib/internal/Magento/Framework/Indexer/SaveHandler/Batch.php +++ b/lib/internal/Magento/Framework/Indexer/SaveHandler/Batch.php @@ -10,27 +10,23 @@ class Batch /** * @param \Traversable $documents * @param int $size - * @return array + * @return \Generator */ public function getItems(\Traversable $documents, $size) { - if (count($documents) == 0) { - return []; - } - $i = 0; - $batch = $items = []; + $batch = []; + foreach ($documents as $documentName => $documentValue) { $batch[$documentName] = $documentValue; - if (++$i >= $size) { - $items[] = $batch; + if (++$i == $size) { + yield $batch; $i = 0; $batch = []; } } if (count($batch) > 0) { - $items[] = $batch; + yield $batch; } - return $items; } } diff --git a/lib/internal/Magento/Framework/Indexer/Test/Unit/BatchTest.php b/lib/internal/Magento/Framework/Indexer/Test/Unit/BatchTest.php index 0df98125f9a08..1571bba0a7d23 100644 --- a/lib/internal/Magento/Framework/Indexer/Test/Unit/BatchTest.php +++ b/lib/internal/Magento/Framework/Indexer/Test/Unit/BatchTest.php @@ -27,7 +27,8 @@ protected function setUp() public function testGetItems(array $itemsData, $size, array $expected) { $items = new \ArrayObject($itemsData); - $this->assertSame($expected, $this->object->getItems($items, $size)); + $data = $this->object->getItems($items, $size); + $this->assertSame($expected, iterator_to_array($data)); } /** From e0ccf4cb2df38277e670b4a2274e5765e3b7a5ab Mon Sep 17 00:00:00 2001 From: Illia Grybkov Date: Wed, 23 Nov 2016 14:49:08 +0200 Subject: [PATCH 02/15] MAGETWO-61159: Configurable Product visible on the Category when options are Out of Stock and Disabled --- .../Indexer/Stock/Configurable.php | 9 ++++- .../Search/Adapter/Mysql/AdapterTest.php | 35 +++++++++++++++++++ .../Search/_files/product_configurable.php | 32 ++--------------- .../_files/product_configurable_rollback.php | 2 +- ...oduct_configurable_with_disabled_child.php | 18 ++++++++++ ...figurable_with_disabled_child_rollback.php | 7 ++++ 6 files changed, 72 insertions(+), 31 deletions(-) create mode 100644 dev/tests/integration/testsuite/Magento/Framework/Search/_files/product_configurable_with_disabled_child.php create mode 100644 dev/tests/integration/testsuite/Magento/Framework/Search/_files/product_configurable_with_disabled_child_rollback.php diff --git a/app/code/Magento/ConfigurableProduct/Model/ResourceModel/Indexer/Stock/Configurable.php b/app/code/Magento/ConfigurableProduct/Model/ResourceModel/Indexer/Stock/Configurable.php index 6f2b989e5e9bf..794034b446f72 100644 --- a/app/code/Magento/ConfigurableProduct/Model/ResourceModel/Indexer/Stock/Configurable.php +++ b/app/code/Magento/ConfigurableProduct/Model/ResourceModel/Indexer/Stock/Configurable.php @@ -29,18 +29,25 @@ protected function _getStockStatusSelect($entityIds = null, $usePrimaryTable = f $connection = $this->getConnection(); $idxTable = $usePrimaryTable ? $this->getMainTable() : $this->getIdxTable(); $select = parent::_getStockStatusSelect($entityIds, $usePrimaryTable); + $linkField = $metadata->getLinkField(); $select->reset( \Magento\Framework\DB\Select::COLUMNS )->columns( ['e.entity_id', 'cis.website_id', 'cis.stock_id'] )->joinLeft( ['l' => $this->getTable('catalog_product_super_link')], - 'l.parent_id = e.' . $metadata->getLinkField(), + 'l.parent_id = e.' . $linkField, [] )->join( ['le' => $this->getTable('catalog_product_entity')], 'le.entity_id = l.product_id', [] + )->joinInner( + ['cpei' => $this->getTable('catalog_product_entity_int')], + 'le.' . $linkField . ' = cpei.' . $linkField + . ' AND cpei.attribute_id = ' . $this->_getAttribute('status')->getId() + . ' AND cpei.value = ' . ProductStatus::STATUS_ENABLED, + [] )->joinLeft( ['i' => $idxTable], 'i.product_id = l.product_id AND cis.website_id = i.website_id AND cis.stock_id = i.stock_id', diff --git a/dev/tests/integration/testsuite/Magento/Framework/Search/Adapter/Mysql/AdapterTest.php b/dev/tests/integration/testsuite/Magento/Framework/Search/Adapter/Mysql/AdapterTest.php index 67a6e41697349..7f50638f469de 100644 --- a/dev/tests/integration/testsuite/Magento/Framework/Search/Adapter/Mysql/AdapterTest.php +++ b/dev/tests/integration/testsuite/Magento/Framework/Search/Adapter/Mysql/AdapterTest.php @@ -331,6 +331,10 @@ public function testBoolFilterWithNestedRangeInNegativeBoolFilter() * * @magentoConfigFixture current_store catalog/search/engine mysql * @dataProvider advancedSearchDataProvider + * @param string $nameQuery + * @param string $descriptionQuery + * @param array $rangeFilter + * @param int $expectedRecordsCount */ public function testSimpleAdvancedSearch( $nameQuery, @@ -445,6 +449,37 @@ public function testAdvancedSearchCompositeProductWithOutOfStockOption() $this->assertEquals(1, $queryResponse->count()); } + /** + * @magentoDataFixture Magento/Framework/Search/_files/product_configurable_with_disabled_child.php + * @magentoConfigFixture current_store catalog/search/engine mysql + */ + public function testAdvancedSearchCompositeProductWithDisabledChild() + { + /** @var \Magento\Catalog\Model\ResourceModel\Eav\Attribute $attribute */ + $attribute = $this->objectManager->get(\Magento\Catalog\Model\ResourceModel\Eav\Attribute::class) + ->loadByCode(\Magento\Catalog\Model\Product::ENTITY, 'test_configurable'); + /** @var \Magento\Eav\Model\ResourceModel\Entity\Attribute\Option\Collection $selectOptions */ + $selectOptions = $this->objectManager + ->create(\Magento\Eav\Model\ResourceModel\Entity\Attribute\Option\Collection::class) + ->setAttributeFilter($attribute->getId()); + + $firstOption = $selectOptions->getFirstItem(); + $firstOptionId = $firstOption->getId(); + $this->requestBuilder->bind('test_configurable', $firstOptionId); + $this->requestBuilder->setRequestName('filter_out_of_stock_child'); + + $queryResponse = $this->executeQuery(); + $this->assertEquals(0, $queryResponse->count()); + + $secondOption = $selectOptions->getLastItem(); + $secondOptionId = $secondOption->getId(); + $this->requestBuilder->bind('test_configurable', $secondOptionId); + $this->requestBuilder->setRequestName('filter_out_of_stock_child'); + + $queryResponse = $this->executeQuery(); + $this->assertEquals(0, $queryResponse->count()); + } + public function dateDataProvider() { return [ diff --git a/dev/tests/integration/testsuite/Magento/Framework/Search/_files/product_configurable.php b/dev/tests/integration/testsuite/Magento/Framework/Search/_files/product_configurable.php index 590f3c8041c6d..37b8731fc1d4c 100644 --- a/dev/tests/integration/testsuite/Magento/Framework/Search/_files/product_configurable.php +++ b/dev/tests/integration/testsuite/Magento/Framework/Search/_files/product_configurable.php @@ -15,7 +15,7 @@ use Magento\Eav\Api\Data\AttributeOptionInterface; use Magento\TestFramework\Helper\Bootstrap; -\Magento\TestFramework\Helper\Bootstrap::getInstance()->reinitialize(); +Bootstrap::getInstance()->reinitialize(); require __DIR__ . '/configurable_attribute.php'; @@ -33,7 +33,7 @@ $attributeValues = []; $attributeSetId = $installer->getAttributeSetId('catalog_product', 'Default'); $associatedProductIds = []; -$productIds = [10, 20]; +$productIds = [1010, 1020]; array_shift($options); //remove the first option which is empty $isFirstOption = true; @@ -108,29 +108,8 @@ $product->setExtensionAttributes($extensionConfigurableAttributes); -// Remove any previously created product with the same id. -/** @var \Magento\Framework\Registry $registry */ -$registry = Bootstrap::getObjectManager()->get(\Magento\Framework\Registry::class); -$registry->unregister('isSecureArea'); -$registry->register('isSecureArea', true); -try { - $productToDelete = $productRepository->getById(1); - $productRepository->delete($productToDelete); - - /** @var \Magento\Quote\Model\ResourceModel\Quote\Item $itemResource */ - $itemResource = Bootstrap::getObjectManager()->get(\Magento\Quote\Model\ResourceModel\Quote\Item::class); - $itemResource->getConnection()->delete( - $itemResource->getMainTable(), - 'product_id = ' . $productToDelete->getId() - ); -} catch (\Exception $e) { - // Nothing to remove -} -$registry->unregister('isSecureArea'); -$registry->register('isSecureArea', false); - $product->setTypeId(Configurable::TYPE_CODE) - ->setId(1) + ->setId(1001) ->setAttributeSetId($attributeSetId) ->setWebsiteIds([1]) ->setName('Configurable Product') @@ -140,8 +119,3 @@ ->setStockData(['use_config_manage_stock' => 1, 'is_in_stock' => 1]); $productRepository->save($product); -// -///** @var \Magento\Catalog\Model\Indexer\Product\Eav\Processor $eavIndexer */ -//$eavIndexer = Bootstrap::getObjectManager() -// ->get(\Magento\Catalog\Model\Indexer\Product\Eav\Processor::class); -//$eavIndexer->reindexAll(); diff --git a/dev/tests/integration/testsuite/Magento/Framework/Search/_files/product_configurable_rollback.php b/dev/tests/integration/testsuite/Magento/Framework/Search/_files/product_configurable_rollback.php index 8ba4e3abe21cc..cc585d177aedb 100644 --- a/dev/tests/integration/testsuite/Magento/Framework/Search/_files/product_configurable_rollback.php +++ b/dev/tests/integration/testsuite/Magento/Framework/Search/_files/product_configurable_rollback.php @@ -16,7 +16,7 @@ $productRepository = \Magento\TestFramework\Helper\Bootstrap::getObjectManager() ->get(\Magento\Catalog\Api\ProductRepositoryInterface::class); -foreach (['simple_10', 'simple_20', 'configurable'] as $sku) { +foreach (['simple_1010', 'simple_1020', 'configurable'] as $sku) { try { $product = $productRepository->get($sku, false, null, true); diff --git a/dev/tests/integration/testsuite/Magento/Framework/Search/_files/product_configurable_with_disabled_child.php b/dev/tests/integration/testsuite/Magento/Framework/Search/_files/product_configurable_with_disabled_child.php new file mode 100644 index 0000000000000..39c9782ba712a --- /dev/null +++ b/dev/tests/integration/testsuite/Magento/Framework/Search/_files/product_configurable_with_disabled_child.php @@ -0,0 +1,18 @@ +create(ProductRepositoryInterface::class); + +$product = $productRepository->get('simple_1020'); +$product->setStatus(\Magento\Catalog\Model\Product\Attribute\Source\Status::STATUS_DISABLED); +$productRepository->save($product); diff --git a/dev/tests/integration/testsuite/Magento/Framework/Search/_files/product_configurable_with_disabled_child_rollback.php b/dev/tests/integration/testsuite/Magento/Framework/Search/_files/product_configurable_with_disabled_child_rollback.php new file mode 100644 index 0000000000000..58d1dcd79acde --- /dev/null +++ b/dev/tests/integration/testsuite/Magento/Framework/Search/_files/product_configurable_with_disabled_child_rollback.php @@ -0,0 +1,7 @@ + Date: Wed, 16 Nov 2016 12:39:46 +0200 Subject: [PATCH 03/15] MAGETWO-59990: Disable child product doesn't work in Configurable variations grid --- .../web/js/components/dynamic-rows-configurable.js | 6 ++++-- 1 file changed, 4 insertions(+), 2 deletions(-) diff --git a/app/code/Magento/ConfigurableProduct/view/adminhtml/web/js/components/dynamic-rows-configurable.js b/app/code/Magento/ConfigurableProduct/view/adminhtml/web/js/components/dynamic-rows-configurable.js index c182d9f8216c0..be44c110c2b62 100644 --- a/app/code/Magento/ConfigurableProduct/view/adminhtml/web/js/components/dynamic-rows-configurable.js +++ b/app/code/Magento/ConfigurableProduct/view/adminhtml/web/js/components/dynamic-rows-configurable.js @@ -32,6 +32,7 @@ define([ identificationProperty: 'id', 'attribute_set_id': '', attributesTmp: [], + changedFlag: 'was_changed', listens: { 'insertDataFromGrid': 'processingInsertDataFromGrid', 'insertDataFromWizard': 'processingInsertDataFromWizard', @@ -391,9 +392,9 @@ define([ 'small_image': row['small_image'], image: row.image, 'thumbnail': row.thumbnail, - 'attributes': attributesText, - 'was_changed': true + 'attributes': attributesText }; + product[this.changedFlag] = true; product[this.canEditField] = row.editable; product[this.newProductField] = row.newProduct; tmpArray.push(product); @@ -515,6 +516,7 @@ define([ tmpArray[rowIndex].status = 1; } + tmpArray[rowIndex][this.changedFlag] = true; this.unionInsertData(tmpArray); } }); From 767944cd522f729452744aaac4a4488dadc70d75 Mon Sep 17 00:00:00 2001 From: Illia Grybkov Date: Wed, 21 Dec 2016 16:48:24 +0200 Subject: [PATCH 04/15] MAGETWO-58174: When catalog is being indexed it should index in place or leverage an index alias so store can still function during a long index run --- .../CatalogSearch/Model/Indexer/Fulltext.php | 49 +++- .../Model/Indexer/IndexStructure.php | 5 +- .../Model/Indexer/IndexerHandler.php | 6 +- .../Model/Indexer/Scope/IndexSwitcher.php | 81 +++++++ .../Scope/IndexTableNotExistException.php | 18 ++ .../Model/Indexer/Scope/ScopeProxy.php | 76 +++++++ .../Model/Indexer/Scope/State.php | 66 ++++++ .../Model/Indexer/Scope/TemporaryResolver.php | 43 ++++ .../Indexer/Scope/UnknownStateException.php | 18 ++ .../Test/Unit/Model/Indexer/FulltextTest.php | 89 +++++--- .../Model/Indexer/Scope/IndexSwitcherTest.php | 215 ++++++++++++++++++ .../ResourceModel/Advanced/CollectionTest.php | 5 +- .../ResourceModel/Fulltext/CollectionTest.php | 5 +- app/code/Magento/CatalogSearch/etc/di.xml | 23 ++ .../ScopeResolver/IndexScopeResolver.php | 11 +- 15 files changed, 665 insertions(+), 45 deletions(-) create mode 100644 app/code/Magento/CatalogSearch/Model/Indexer/Scope/IndexSwitcher.php create mode 100644 app/code/Magento/CatalogSearch/Model/Indexer/Scope/IndexTableNotExistException.php create mode 100644 app/code/Magento/CatalogSearch/Model/Indexer/Scope/ScopeProxy.php create mode 100644 app/code/Magento/CatalogSearch/Model/Indexer/Scope/State.php create mode 100644 app/code/Magento/CatalogSearch/Model/Indexer/Scope/TemporaryResolver.php create mode 100644 app/code/Magento/CatalogSearch/Model/Indexer/Scope/UnknownStateException.php create mode 100644 app/code/Magento/CatalogSearch/Test/Unit/Model/Indexer/Scope/IndexSwitcherTest.php diff --git a/app/code/Magento/CatalogSearch/Model/Indexer/Fulltext.php b/app/code/Magento/CatalogSearch/Model/Indexer/Fulltext.php index 08e75ceebecc6..f21cedfb2094f 100644 --- a/app/code/Magento/CatalogSearch/Model/Indexer/Fulltext.php +++ b/app/code/Magento/CatalogSearch/Model/Indexer/Fulltext.php @@ -6,11 +6,17 @@ namespace Magento\CatalogSearch\Model\Indexer; use Magento\CatalogSearch\Model\Indexer\Fulltext\Action\FullFactory; +use Magento\CatalogSearch\Model\Indexer\Scope\IndexSwitcher; +use Magento\CatalogSearch\Model\Indexer\Scope\State; use Magento\CatalogSearch\Model\ResourceModel\Fulltext as FulltextResource; -use \Magento\Framework\Search\Request\Config as SearchRequestConfig; +use Magento\Framework\App\ObjectManager; +use Magento\Framework\Search\Request\Config as SearchRequestConfig; use Magento\Framework\Search\Request\DimensionFactory; use Magento\Store\Model\StoreManagerInterface; +/** + * Provide functionality for Fulltext Search indexing + */ class Fulltext implements \Magento\Framework\Indexer\ActionInterface, \Magento\Framework\Mview\ActionInterface { /** @@ -25,27 +31,42 @@ class Fulltext implements \Magento\Framework\Indexer\ActionInterface, \Magento\F * @var IndexerHandlerFactory */ private $indexerHandlerFactory; + /** * @var StoreManagerInterface */ private $storeManager; + /** * @var DimensionFactory */ private $dimensionFactory; + /** - * @var Full + * @var \Magento\CatalogSearch\Model\Indexer\Fulltext\Action\Full */ private $fullAction; + /** * @var FulltextResource */ private $fulltextResource; + /** * @var SearchRequestConfig */ private $searchRequestConfig; + /** + * @var \Magento\CatalogSearch\Model\Indexer\Scope\IndexSwitcher + */ + private $indexSwitcher; + + /** + * @var State + */ + private $indexScopeState; + /** * @param FullFactory $fullActionFactory * @param IndexerHandlerFactory $indexerHandlerFactory @@ -54,6 +75,8 @@ class Fulltext implements \Magento\Framework\Indexer\ActionInterface, \Magento\F * @param FulltextResource $fulltextResource * @param SearchRequestConfig $searchRequestConfig * @param array $data + * @param IndexSwitcher $indexSwitcher + * @param Scope\State $indexScopeState */ public function __construct( FullFactory $fullActionFactory, @@ -62,7 +85,9 @@ public function __construct( DimensionFactory $dimensionFactory, FulltextResource $fulltextResource, SearchRequestConfig $searchRequestConfig, - array $data + array $data, + IndexSwitcher $indexSwitcher = null, + State $indexScopeState = null ) { $this->fullAction = $fullActionFactory->create(['data' => $data]); $this->indexerHandlerFactory = $indexerHandlerFactory; @@ -71,6 +96,14 @@ public function __construct( $this->fulltextResource = $fulltextResource; $this->searchRequestConfig = $searchRequestConfig; $this->data = $data; + if (null === $indexSwitcher) { + $indexSwitcher = ObjectManager::getInstance()->get(IndexSwitcher::class); + } + if (null === $indexScopeState) { + $indexScopeState = ObjectManager::getInstance()->get(State::class); + } + $this->indexSwitcher = $indexSwitcher; + $this->indexScopeState = $indexScopeState; } /** @@ -106,10 +139,14 @@ public function executeFull() 'data' => $this->data ]); foreach ($storeIds as $storeId) { - $dimension = $this->dimensionFactory->create(['name' => 'scope', 'value' => $storeId]); - $saveHandler->cleanIndex([$dimension]); - $saveHandler->saveIndex([$dimension], $this->fullAction->rebuildStoreIndex($storeId)); + $dimensions = [$this->dimensionFactory->create(['name' => 'scope', 'value' => $storeId])]; + $this->indexScopeState->useTemporaryIndex(); + + $saveHandler->cleanIndex($dimensions); + $saveHandler->saveIndex($dimensions, $this->fullAction->rebuildStoreIndex($storeId)); + $this->indexSwitcher->switchIndex($dimensions); + $this->indexScopeState->useRegularIndex(); } $this->fulltextResource->resetSearchResults(); $this->searchRequestConfig->reset(); diff --git a/app/code/Magento/CatalogSearch/Model/Indexer/IndexStructure.php b/app/code/Magento/CatalogSearch/Model/Indexer/IndexStructure.php index 9a2bb76c04ce7..aa2584955253e 100644 --- a/app/code/Magento/CatalogSearch/Model/Indexer/IndexStructure.php +++ b/app/code/Magento/CatalogSearch/Model/Indexer/IndexStructure.php @@ -12,6 +12,7 @@ use Magento\Framework\Search\Request\Dimension; use Magento\Framework\Indexer\IndexStructureInterface; use Magento\Framework\Indexer\ScopeResolver\IndexScopeResolver; +use Magento\Framework\Search\Request\IndexScopeResolverInterface; class IndexStructure implements IndexStructureInterface { @@ -26,11 +27,11 @@ class IndexStructure implements IndexStructureInterface /** * @param ResourceConnection $resource - * @param IndexScopeResolver $indexScopeResolver + * @param IndexScopeResolverInterface $indexScopeResolver */ public function __construct( ResourceConnection $resource, - IndexScopeResolver $indexScopeResolver + IndexScopeResolverInterface $indexScopeResolver ) { $this->resource = $resource; $this->indexScopeResolver = $indexScopeResolver; diff --git a/app/code/Magento/CatalogSearch/Model/Indexer/IndexerHandler.php b/app/code/Magento/CatalogSearch/Model/Indexer/IndexerHandler.php index 8d99cd4492bf5..2aa9a418bd725 100644 --- a/app/code/Magento/CatalogSearch/Model/Indexer/IndexerHandler.php +++ b/app/code/Magento/CatalogSearch/Model/Indexer/IndexerHandler.php @@ -7,13 +7,11 @@ use Magento\Eav\Model\Config; use Magento\Framework\App\ResourceConnection; -use Magento\Framework\DB\Adapter\AdapterInterface; use Magento\Framework\Indexer\SaveHandler\IndexerInterface; use Magento\Framework\Indexer\IndexStructureInterface; use Magento\Framework\Search\Request\Dimension; use Magento\Framework\Search\Request\IndexScopeResolverInterface; use Magento\Framework\Indexer\SaveHandler\Batch; -use Magento\Framework\Indexer\ScopeResolver\IndexScopeResolver; class IndexerHandler implements IndexerInterface { @@ -62,7 +60,7 @@ class IndexerHandler implements IndexerInterface * @param ResourceConnection $resource * @param Config $eavConfig * @param Batch $batch - * @param \Magento\Framework\Indexer\ScopeResolver\IndexScopeResolver $indexScopeResolver + * @param IndexScopeResolverInterface $indexScopeResolver * @param array $data * @param int $batchSize */ @@ -71,7 +69,7 @@ public function __construct( ResourceConnection $resource, Config $eavConfig, Batch $batch, - IndexScopeResolver $indexScopeResolver, + IndexScopeResolverInterface $indexScopeResolver, array $data, $batchSize = 100 ) { diff --git a/app/code/Magento/CatalogSearch/Model/Indexer/Scope/IndexSwitcher.php b/app/code/Magento/CatalogSearch/Model/Indexer/Scope/IndexSwitcher.php new file mode 100644 index 0000000000000..6f16dde916073 --- /dev/null +++ b/app/code/Magento/CatalogSearch/Model/Indexer/Scope/IndexSwitcher.php @@ -0,0 +1,81 @@ +resource = $resource; + $this->resolver = $indexScopeResolver; + $this->state = $state; + } + + /** + * Switch current index with temporary index + * + * It will drop current index table and rename temporary index table to the current index table. + * + * @param array $dimensions + * @return void + * @throws IndexTableNotExistException + */ + public function switchIndex(array $dimensions) + { + if (State::USE_TEMPORARY_INDEX === $this->state->getState()) { + $index = \Magento\CatalogSearch\Model\Indexer\Fulltext::INDEXER_ID; + + $temporalIndexTable = $this->resolver->resolve($index, $dimensions); + if (!$this->resource->getConnection()->isTableExists($temporalIndexTable)) { + throw new IndexTableNotExistException( + __( + "Temporary table for index $index doesn't exist," + . " which is inconsistent with state of scope resolver" + ) + ); + } + + $this->state->useRegularIndex(); + $tableName = $this->resolver->resolve($index, $dimensions); + if ($this->resource->getConnection()->isTableExists($tableName)) { + $this->resource->getConnection()->dropTable($tableName); + } + + $this->resource->getConnection()->renameTable($temporalIndexTable, $tableName); + $this->state->useTemporaryIndex(); + } + } +} diff --git a/app/code/Magento/CatalogSearch/Model/Indexer/Scope/IndexTableNotExistException.php b/app/code/Magento/CatalogSearch/Model/Indexer/Scope/IndexTableNotExistException.php new file mode 100644 index 0000000000000..6974f8c278ab5 --- /dev/null +++ b/app/code/Magento/CatalogSearch/Model/Indexer/Scope/IndexTableNotExistException.php @@ -0,0 +1,18 @@ +objectManager = $objectManager; + $this->scopeState = $scopeState; + $this->states = $states; + } + + /** + * Creates class instance with specified parameters + * + * @param string $state + * @return \Magento\Framework\Search\Request\IndexScopeResolverInterface + * @throws UnknownStateException + */ + private function create($state) + { + if (!array_key_exists($state, $this->states)) { + throw new UnknownStateException(__("Requested resolver for unknown indexer state: $state")); + } + return $this->objectManager->create($this->states[$state]); + } + + /** + * @param string $index + * @param Dimension[] $dimensions + * @return string + */ + public function resolve($index, array $dimensions) + { + return $this->create($this->scopeState->getState())->resolve($index, $dimensions); + } +} diff --git a/app/code/Magento/CatalogSearch/Model/Indexer/Scope/State.php b/app/code/Magento/CatalogSearch/Model/Indexer/Scope/State.php new file mode 100644 index 0000000000000..f44dbb430db78 --- /dev/null +++ b/app/code/Magento/CatalogSearch/Model/Indexer/Scope/State.php @@ -0,0 +1,66 @@ +state = self::USE_TEMPORARY_INDEX; + } + + /** + * Set the state to use regular Index + * @return void + */ + public function useRegularIndex() + { + $this->state = self::USE_REGULAR_INDEX; + } + + /** + * @return string + */ + public function getState() + { + return $this->state; + } + +} diff --git a/app/code/Magento/CatalogSearch/Model/Indexer/Scope/TemporaryResolver.php b/app/code/Magento/CatalogSearch/Model/Indexer/Scope/TemporaryResolver.php new file mode 100644 index 0000000000000..51037eb637cf3 --- /dev/null +++ b/app/code/Magento/CatalogSearch/Model/Indexer/Scope/TemporaryResolver.php @@ -0,0 +1,43 @@ +indexScopeResolver = $indexScopeResolver; + } + + /** + * @param string $index + * @param Dimension[] $dimensions + * @return string + */ + public function resolve($index, array $dimensions) + { + $tableName = $this->indexScopeResolver->resolve($index, $dimensions); + $tableName .= \Magento\Framework\Indexer\Table\StrategyInterface::TMP_SUFFIX; + + return $tableName; + } +} diff --git a/app/code/Magento/CatalogSearch/Model/Indexer/Scope/UnknownStateException.php b/app/code/Magento/CatalogSearch/Model/Indexer/Scope/UnknownStateException.php new file mode 100644 index 0000000000000..04803ef27480e --- /dev/null +++ b/app/code/Magento/CatalogSearch/Model/Indexer/Scope/UnknownStateException.php @@ -0,0 +1,18 @@ +fullAction = $this->getClassMock(\Magento\CatalogSearch\Model\Indexer\Fulltext\Action\Full::class); @@ -80,27 +88,29 @@ protected function setUp() [] ); - $this->dimension = $this->getClassMock(\Magento\Framework\Search\Request\Dimension::class); - $dimensionFactory = $this->getMock( - \Magento\Framework\Search\Request\DimensionFactory::class, - ['create'], - [], - '', - false - ); - $dimensionFactory->expects($this->any())->method('create')->willReturn($this->dimension); + $this->dimensionFactory = $this->getMock(DimensionFactory::class, ['create'], [], '', false); $this->fulltextResource = $this->getClassMock(\Magento\CatalogSearch\Model\ResourceModel\Fulltext::class); $this->searchRequestConfig = $this->getClassMock(\Magento\Framework\Search\Request\Config::class); - $this->model = new \Magento\CatalogSearch\Model\Indexer\Fulltext( - $fullActionFactory, - $indexerHandlerFactory, - $this->storeManager, - $dimensionFactory, - $this->fulltextResource, - $this->searchRequestConfig, - [] + $this->indexSwitcher = $this->getMockBuilder(\Magento\CatalogSearch\Model\Indexer\Scope\IndexSwitcher::class) + ->disableOriginalConstructor() + ->setMethods(['switchIndex']) + ->getMock(); + + $objectManagerHelper = new ObjectManagerHelper($this); + $this->model = $objectManagerHelper->getObject( + \Magento\CatalogSearch\Model\Indexer\Fulltext::class, + [ + 'fullActionFactory' => $fullActionFactory, + 'indexerHandlerFactory' => $indexerHandlerFactory, + 'storeManager' => $this->storeManager, + 'dimensionFactory' => $this->dimensionFactory, + 'fulltextResource' => $this->fulltextResource, + 'searchRequestConfig' => $this->searchRequestConfig, + 'data' => [], + 'indexSwitcher' => $this->indexSwitcher, + ] ); } @@ -131,13 +141,38 @@ public function testExecute() public function testExecuteFull() { $stores = [0 => 'Store 1', 1 => 'Store 2']; - $indexData = new \ArrayObject([]); + $indexData = new \ArrayObject([new \ArrayObject([]), new \ArrayObject([])]); $this->storeManager->expects($this->once())->method('getStores')->willReturn($stores); - $this->saveHandler->expects($this->exactly(count($stores)))->method('cleanIndex'); - $this->saveHandler->expects($this->exactly(2))->method('saveIndex'); + + $dimensionScope1 = $this->getMock(Dimension::class, [], ['scope', '1']); + $dimensionScope2 = $this->getMock(Dimension::class, [], ['scope', '2']); + + $this->dimensionFactory->expects($this->any())->method('create')->willReturnOnConsecutiveCalls( + $dimensionScope1, + $dimensionScope2 + ); + $this->indexSwitcher->expects($this->exactly(2))->method('switchIndex') + ->withConsecutive( + [$this->equalTo([$dimensionScope1])], + [$this->equalTo([$dimensionScope2])] + ); + + $this->saveHandler->expects($this->exactly(count($stores)))->method('cleanIndex') + ->withConsecutive( + [$this->equalTo([$dimensionScope1])], + [$this->equalTo([$dimensionScope2])] + ); + + $this->saveHandler->expects($this->exactly(2))->method('saveIndex') + ->withConsecutive( + [$this->equalTo([$dimensionScope1]), $this->equalTo($indexData)], + [$this->equalTo([$dimensionScope2]), $this->equalTo($indexData)] + ); $this->fullAction->expects($this->exactly(2)) ->method('rebuildStoreIndex') - ->willReturn(new \ArrayObject([$indexData, $indexData])); + ->withConsecutive([0], [1]) + ->willReturn($indexData); + $this->fulltextResource->expects($this->once())->method('resetSearchResults'); $this->searchRequestConfig->expects($this->once())->method('reset'); diff --git a/app/code/Magento/CatalogSearch/Test/Unit/Model/Indexer/Scope/IndexSwitcherTest.php b/app/code/Magento/CatalogSearch/Test/Unit/Model/Indexer/Scope/IndexSwitcherTest.php new file mode 100644 index 0000000000000..6b96281cd09ea --- /dev/null +++ b/app/code/Magento/CatalogSearch/Test/Unit/Model/Indexer/Scope/IndexSwitcherTest.php @@ -0,0 +1,215 @@ +resource = $this->getMockBuilder(ResourceConnection::class) + ->setMethods(['getConnection']) + ->disableOriginalConstructor() + ->getMock(); + $this->connection = $this->getMockBuilder(\Magento\Framework\DB\Adapter\AdapterInterface::class) + ->setMethods(['isTableExists', 'dropTable', 'renameTable']) + ->getMockForAbstractClass(); + $this->resource->expects($this->any()) + ->method('getConnection') + ->willReturn($this->connection); + $this->indexScopeResolver = $this->getMockBuilder( + \Magento\Framework\Search\Request\IndexScopeResolverInterface::class + ) + ->setMethods(['resolve']) + ->getMockForAbstractClass(); + $this->scopeState = $this->getMockBuilder(State::class) + ->setMethods(['getState', 'useRegularIndex', 'useTemporaryIndex']) + ->disableOriginalConstructor() + ->getMock(); + + $objectManagerHelper = new ObjectManagerHelper($this); + $this->indexSwitcher = $objectManagerHelper->getObject( + \Magento\CatalogSearch\Model\Indexer\Scope\IndexSwitcher::class, + [ + 'resource' => $this->resource, + 'indexScopeResolver' => $this->indexScopeResolver, + 'state' => $this->scopeState, + ] + ); + } + + public function testSwitchRegularIndex() + { + $dimensions = [$this->getMockBuilder(Dimension::class)->setConstructorArgs(['scope', '1'])]; + + $this->scopeState->expects($this->once()) + ->method('getState') + ->willReturn(State::USE_REGULAR_INDEX); + + $this->indexScopeResolver->expects($this->never())->method('resolve'); + $this->connection->expects($this->never())->method('renameTable'); + + $this->indexSwitcher->switchIndex($dimensions); + } + + public function testSwitchIndexWithUnknownState() + { + $dimensions = [$this->getMockBuilder(Dimension::class)->setConstructorArgs(['scope', '1'])]; + + $this->scopeState->expects($this->once()) + ->method('getState') + ->willReturn('unknown_state'); + + $this->indexScopeResolver->expects($this->never())->method('resolve'); + $this->connection->expects($this->never())->method('renameTable'); + + $this->indexSwitcher->switchIndex($dimensions); + } + + public function testSwitchTemporaryIndexWhenRegularIndexExist() + { + $dimensions = [$this->getMockBuilder(Dimension::class)->setConstructorArgs(['scope', '1'])]; + + $this->scopeState->expects($this->once()) + ->method('getState') + ->willReturn(State::USE_TEMPORARY_INDEX); + + + $this->scopeState->expects($this->at(1))->method('useRegularIndex'); + $this->scopeState->expects($this->at(2))->method('useTemporaryIndex'); + + $this->indexScopeResolver->expects($this->exactly(2))->method('resolve') + ->withConsecutive( + [$this->equalTo(FulltextIndexer::INDEXER_ID), $this->equalTo($dimensions)], + [$this->equalTo(FulltextIndexer::INDEXER_ID), $this->equalTo($dimensions)] + ) + ->willReturnOnConsecutiveCalls( + 'catalogsearch_fulltext_scope1_tmp1', + 'catalogsearch_fulltext_scope1' + ); + + $this->connection->expects($this->exactly(2))->method('isTableExists') + ->withConsecutive( + [$this->equalTo('catalogsearch_fulltext_scope1_tmp1'), $this->equalTo(null)], + [$this->equalTo('catalogsearch_fulltext_scope1'), $this->equalTo(null)] + ) + ->willReturnOnConsecutiveCalls( + true, + true + ); + + $this->connection->expects($this->once())->method('dropTable')->with('catalogsearch_fulltext_scope1', null); + $this->connection->expects($this->once()) + ->method('renameTable') + ->with('catalogsearch_fulltext_scope1_tmp1', 'catalogsearch_fulltext_scope1'); + + $this->indexSwitcher->switchIndex($dimensions); + } + + public function testSwitchTemporaryIndexWhenRegularIndexNotExist() + { + $dimensions = [$this->getMockBuilder(Dimension::class)->setConstructorArgs(['scope', '1'])]; + + $this->scopeState->expects($this->once()) + ->method('getState') + ->willReturn(State::USE_TEMPORARY_INDEX); + + + $this->scopeState->expects($this->at(1))->method('useRegularIndex'); + $this->scopeState->expects($this->at(2))->method('useTemporaryIndex'); + + $this->indexScopeResolver->expects($this->exactly(2))->method('resolve') + ->withConsecutive( + [$this->equalTo(FulltextIndexer::INDEXER_ID), $this->equalTo($dimensions)], + [$this->equalTo(FulltextIndexer::INDEXER_ID), $this->equalTo($dimensions)] + ) + ->willReturnOnConsecutiveCalls( + 'catalogsearch_fulltext_scope1_tmp1', + 'catalogsearch_fulltext_scope1' + ); + + $this->connection->expects($this->exactly(2))->method('isTableExists') + ->withConsecutive( + [$this->equalTo('catalogsearch_fulltext_scope1_tmp1'), $this->equalTo(null)], + [$this->equalTo('catalogsearch_fulltext_scope1'), $this->equalTo(null)] + ) + ->willReturnOnConsecutiveCalls( + true, + false + ); + + $this->connection->expects($this->never())->method('dropTable')->with('catalogsearch_fulltext_scope1', null); + $this->connection->expects($this->once()) + ->method('renameTable') + ->with('catalogsearch_fulltext_scope1_tmp1', 'catalogsearch_fulltext_scope1'); + + $this->indexSwitcher->switchIndex($dimensions); + } + + /** + * @expectedException \Magento\CatalogSearch\Model\Indexer\Scope\IndexTableNotExistException + * @expectedExceptionMessage Temporary table for index catalogsearch_fulltext doesn't exist + */ + public function testSwitchWhenTemporaryIndexNotExist() + { + $dimensions = [$this->getMockBuilder(Dimension::class)->setConstructorArgs(['scope', '1'])]; + + $this->scopeState->expects($this->once()) + ->method('getState') + ->willReturn(State::USE_TEMPORARY_INDEX); + + + $this->scopeState->expects($this->never())->method('useRegularIndex'); + $this->scopeState->expects($this->never())->method('useTemporaryIndex'); + + $this->indexScopeResolver->expects($this->once())->method('resolve') + ->with(FulltextIndexer::INDEXER_ID, $dimensions) + ->willReturn('catalogsearch_fulltext_scope1_tmp1'); + + $this->connection->expects($this->once()) + ->method('isTableExists') + ->with('catalogsearch_fulltext_scope1_tmp1', null) + ->willReturn(false); + + $this->connection->expects($this->never())->method('dropTable')->with('catalogsearch_fulltext_scope1', null); + $this->connection->expects($this->never())->method('renameTable'); + + $this->indexSwitcher->switchIndex($dimensions); + } +} diff --git a/app/code/Magento/CatalogSearch/Test/Unit/Model/ResourceModel/Advanced/CollectionTest.php b/app/code/Magento/CatalogSearch/Test/Unit/Model/ResourceModel/Advanced/CollectionTest.php index 4b04e7d66ab56..d44fef54683fc 100644 --- a/app/code/Magento/CatalogSearch/Test/Unit/Model/ResourceModel/Advanced/CollectionTest.php +++ b/app/code/Magento/CatalogSearch/Test/Unit/Model/ResourceModel/Advanced/CollectionTest.php @@ -74,7 +74,10 @@ protected function setUp() $productLimitationMock = $this->getMock( \Magento\Catalog\Model\ResourceModel\Product\Collection\ProductLimitation::class ); - $productLimitationFactoryMock = $this->getMock(ProductLimitationFactory::class, ['create']); + $productLimitationFactoryMock = $this->getMockBuilder(ProductLimitationFactory::class) + ->disableOriginalConstructor() + ->setMethods(['create']) + ->getMock(); $productLimitationFactoryMock->method('create') ->willReturn($productLimitationMock); diff --git a/app/code/Magento/CatalogSearch/Test/Unit/Model/ResourceModel/Fulltext/CollectionTest.php b/app/code/Magento/CatalogSearch/Test/Unit/Model/ResourceModel/Fulltext/CollectionTest.php index 5e56a25356359..5732ed2c8aee1 100644 --- a/app/code/Magento/CatalogSearch/Test/Unit/Model/ResourceModel/Fulltext/CollectionTest.php +++ b/app/code/Magento/CatalogSearch/Test/Unit/Model/ResourceModel/Fulltext/CollectionTest.php @@ -80,7 +80,10 @@ protected function setUp() $productLimitationMock = $this->getMock( \Magento\Catalog\Model\ResourceModel\Product\Collection\ProductLimitation::class ); - $productLimitationFactoryMock = $this->getMock(ProductLimitationFactory::class, ['create']); + $productLimitationFactoryMock = $this->getMockBuilder(ProductLimitationFactory::class) + ->disableOriginalConstructor() + ->setMethods(['create']) + ->getMock(); $productLimitationFactoryMock->method('create') ->willReturn($productLimitationMock); diff --git a/app/code/Magento/CatalogSearch/etc/di.xml b/app/code/Magento/CatalogSearch/etc/di.xml index 7e9451f9b83e7..8b0bfa9d06775 100644 --- a/app/code/Magento/CatalogSearch/etc/di.xml +++ b/app/code/Magento/CatalogSearch/etc/di.xml @@ -246,4 +246,27 @@ + + + \Magento\CatalogSearch\Model\Indexer\Scope\ScopeProxy + + + + + \Magento\CatalogSearch\Model\Indexer\Scope\ScopeProxy + + + + + \Magento\CatalogSearch\Model\Indexer\Scope\ScopeProxy + + + + + + \Magento\CatalogSearch\Model\Indexer\Scope\TemporaryResolver + \Magento\Framework\Indexer\ScopeResolver\IndexScopeResolver + + + diff --git a/lib/internal/Magento/Framework/Indexer/ScopeResolver/IndexScopeResolver.php b/lib/internal/Magento/Framework/Indexer/ScopeResolver/IndexScopeResolver.php index eaab236bbca5b..8bcb4acaa11a3 100644 --- a/lib/internal/Magento/Framework/Indexer/ScopeResolver/IndexScopeResolver.php +++ b/lib/internal/Magento/Framework/Indexer/ScopeResolver/IndexScopeResolver.php @@ -52,6 +52,7 @@ public function resolve($index, array $dimensions) $tableNameParts[] = $dimension->getName() . $dimension->getValue(); } } + return $this->resource->getTableName(implode('_', $tableNameParts)); } @@ -63,10 +64,12 @@ public function resolve($index, array $dimensions) */ private function getScopeId($dimension) { - if (is_numeric($dimension->getValue())) { - return $dimension->getValue(); - } else { - return $this->scopeResolver->getScope($dimension->getValue())->getId(); + $scopeId = $dimension->getValue(); + + if (!is_numeric($scopeId)) { + $scopeId = $this->scopeResolver->getScope($scopeId)->getId(); } + + return $scopeId; } } From 4638e6febcdbd498886b5aa9b04141c43f68f697 Mon Sep 17 00:00:00 2001 From: Illia Grybkov Date: Thu, 22 Dec 2016 11:29:28 +0200 Subject: [PATCH 05/15] MAGETWO-58174: When catalog is being indexed it should index in place or leverage an index alias so store can still function during a long index run - Fix static tests --- app/code/Magento/CatalogSearch/Model/Indexer/Fulltext.php | 8 +++++--- .../CatalogSearch/Model/Indexer/IndexStructure.php | 1 + .../Magento/CatalogSearch/Model/Indexer/Scope/State.php | 1 - .../Test/Unit/Model/Indexer/Scope/IndexSwitcherTest.php | 3 --- 4 files changed, 6 insertions(+), 7 deletions(-) diff --git a/app/code/Magento/CatalogSearch/Model/Indexer/Fulltext.php b/app/code/Magento/CatalogSearch/Model/Indexer/Fulltext.php index f21cedfb2094f..46d928ff035ec 100644 --- a/app/code/Magento/CatalogSearch/Model/Indexer/Fulltext.php +++ b/app/code/Magento/CatalogSearch/Model/Indexer/Fulltext.php @@ -24,7 +24,9 @@ class Fulltext implements \Magento\Framework\Indexer\ActionInterface, \Magento\F */ const INDEXER_ID = 'catalogsearch_fulltext'; - /** @var array index structure */ + /** + * @var array index structure + */ protected $data; /** @@ -53,7 +55,7 @@ class Fulltext implements \Magento\Framework\Indexer\ActionInterface, \Magento\F private $fulltextResource; /** - * @var SearchRequestConfig + * @var \Magento\Framework\Search\Request\Config */ private $searchRequestConfig; @@ -63,7 +65,7 @@ class Fulltext implements \Magento\Framework\Indexer\ActionInterface, \Magento\F private $indexSwitcher; /** - * @var State + * @var \Magento\CatalogSearch\Model\Indexer\Scope\State */ private $indexScopeState; diff --git a/app/code/Magento/CatalogSearch/Model/Indexer/IndexStructure.php b/app/code/Magento/CatalogSearch/Model/Indexer/IndexStructure.php index aa2584955253e..d2d76bc71ccbf 100644 --- a/app/code/Magento/CatalogSearch/Model/Indexer/IndexStructure.php +++ b/app/code/Magento/CatalogSearch/Model/Indexer/IndexStructure.php @@ -20,6 +20,7 @@ class IndexStructure implements IndexStructureInterface * @var Resource */ private $resource; + /** * @var IndexScopeResolver */ diff --git a/app/code/Magento/CatalogSearch/Model/Indexer/Scope/State.php b/app/code/Magento/CatalogSearch/Model/Indexer/Scope/State.php index f44dbb430db78..2bba29ae8d842 100644 --- a/app/code/Magento/CatalogSearch/Model/Indexer/Scope/State.php +++ b/app/code/Magento/CatalogSearch/Model/Indexer/Scope/State.php @@ -62,5 +62,4 @@ public function getState() { return $this->state; } - } diff --git a/app/code/Magento/CatalogSearch/Test/Unit/Model/Indexer/Scope/IndexSwitcherTest.php b/app/code/Magento/CatalogSearch/Test/Unit/Model/Indexer/Scope/IndexSwitcherTest.php index 6b96281cd09ea..31bec5906ae5a 100644 --- a/app/code/Magento/CatalogSearch/Test/Unit/Model/Indexer/Scope/IndexSwitcherTest.php +++ b/app/code/Magento/CatalogSearch/Test/Unit/Model/Indexer/Scope/IndexSwitcherTest.php @@ -110,7 +110,6 @@ public function testSwitchTemporaryIndexWhenRegularIndexExist() ->method('getState') ->willReturn(State::USE_TEMPORARY_INDEX); - $this->scopeState->expects($this->at(1))->method('useRegularIndex'); $this->scopeState->expects($this->at(2))->method('useTemporaryIndex'); @@ -150,7 +149,6 @@ public function testSwitchTemporaryIndexWhenRegularIndexNotExist() ->method('getState') ->willReturn(State::USE_TEMPORARY_INDEX); - $this->scopeState->expects($this->at(1))->method('useRegularIndex'); $this->scopeState->expects($this->at(2))->method('useTemporaryIndex'); @@ -194,7 +192,6 @@ public function testSwitchWhenTemporaryIndexNotExist() ->method('getState') ->willReturn(State::USE_TEMPORARY_INDEX); - $this->scopeState->expects($this->never())->method('useRegularIndex'); $this->scopeState->expects($this->never())->method('useTemporaryIndex'); From dbbd83dad7e94fba4979f8e36ecaf8b15dd9594e Mon Sep 17 00:00:00 2001 From: Illia Grybkov Date: Thu, 22 Dec 2016 17:00:50 +0200 Subject: [PATCH 06/15] MAGETWO-58174: When catalog is being indexed it should index in place or leverage an index alias so store can still function during a long index run - Refactor to use adapter-specific index switchers --- .../CatalogSearch/Model/Indexer/Fulltext.php | 8 +- .../Model/Indexer/IndexSwitcherInterface.php | 22 +++++ .../Model/Indexer/IndexSwitcherProxy.php | 97 +++++++++++++++++++ .../Model/Indexer/Scope/IndexSwitcher.php | 13 +-- app/code/Magento/CatalogSearch/etc/di.xml | 9 ++ 5 files changed, 136 insertions(+), 13 deletions(-) create mode 100644 app/code/Magento/CatalogSearch/Model/Indexer/IndexSwitcherInterface.php create mode 100644 app/code/Magento/CatalogSearch/Model/Indexer/IndexSwitcherProxy.php diff --git a/app/code/Magento/CatalogSearch/Model/Indexer/Fulltext.php b/app/code/Magento/CatalogSearch/Model/Indexer/Fulltext.php index 46d928ff035ec..27f5fc92fd54d 100644 --- a/app/code/Magento/CatalogSearch/Model/Indexer/Fulltext.php +++ b/app/code/Magento/CatalogSearch/Model/Indexer/Fulltext.php @@ -40,7 +40,7 @@ class Fulltext implements \Magento\Framework\Indexer\ActionInterface, \Magento\F private $storeManager; /** - * @var DimensionFactory + * @var \Magento\Framework\Search\Request\DimensionFactory */ private $dimensionFactory; @@ -77,7 +77,7 @@ class Fulltext implements \Magento\Framework\Indexer\ActionInterface, \Magento\F * @param FulltextResource $fulltextResource * @param SearchRequestConfig $searchRequestConfig * @param array $data - * @param IndexSwitcher $indexSwitcher + * @param IndexSwitcherInterface $indexSwitcher * @param Scope\State $indexScopeState */ public function __construct( @@ -88,7 +88,7 @@ public function __construct( FulltextResource $fulltextResource, SearchRequestConfig $searchRequestConfig, array $data, - IndexSwitcher $indexSwitcher = null, + IndexSwitcherInterface $indexSwitcher = null, State $indexScopeState = null ) { $this->fullAction = $fullActionFactory->create(['data' => $data]); @@ -99,7 +99,7 @@ public function __construct( $this->searchRequestConfig = $searchRequestConfig; $this->data = $data; if (null === $indexSwitcher) { - $indexSwitcher = ObjectManager::getInstance()->get(IndexSwitcher::class); + $indexSwitcher = ObjectManager::getInstance()->get(IndexSwitcherInterface::class); } if (null === $indexScopeState) { $indexScopeState = ObjectManager::getInstance()->get(State::class); diff --git a/app/code/Magento/CatalogSearch/Model/Indexer/IndexSwitcherInterface.php b/app/code/Magento/CatalogSearch/Model/Indexer/IndexSwitcherInterface.php new file mode 100644 index 0000000000000..ba5fb461dde35 --- /dev/null +++ b/app/code/Magento/CatalogSearch/Model/Indexer/IndexSwitcherInterface.php @@ -0,0 +1,22 @@ +objectManager = $objectManager; + $this->scopeConfig = $scopeConfig; + $this->configPath = $configPath; + $this->handlers = $handlers; + } + + /** + * @inheritDoc + * + * As index switcher is an optional part of the search SPI, it may be not defined by a search engine. + * It is especially reasonable for search engines with pre-defined indexes declaration (like old SOLR and Sphinx) + * which cannot create temporary indexes on the fly. + * That's the reason why this method do nothing for the case + * when switcher is not defined for a specific search engine. + */ + public function switchIndex(array $dimensions) + { + $currentHandler = $this->scopeConfig->getValue($this->configPath, ScopeInterface::SCOPE_STORE); + if (!isset($this->handlers[$currentHandler])) { + return; + } + $this->create($currentHandler)->switchIndex($dimensions); + } + + /** + * Create indexer handler + * + * @param string $handler + * @return IndexSwitcherInterface + */ + private function create($handler) + { + $indexSwitcher = $this->objectManager->create($this->handlers[$handler]); + + if (!$indexSwitcher instanceof IndexSwitcherInterface) { + throw new \InvalidArgumentException( + $handler . ' index switcher doesn\'t implement ' . IndexSwitcherInterface::class + ); + } + + return $indexSwitcher; + } +} diff --git a/app/code/Magento/CatalogSearch/Model/Indexer/Scope/IndexSwitcher.php b/app/code/Magento/CatalogSearch/Model/Indexer/Scope/IndexSwitcher.php index 6f16dde916073..87a7b7110d3aa 100644 --- a/app/code/Magento/CatalogSearch/Model/Indexer/Scope/IndexSwitcher.php +++ b/app/code/Magento/CatalogSearch/Model/Indexer/Scope/IndexSwitcher.php @@ -5,14 +5,14 @@ */ namespace Magento\CatalogSearch\Model\Indexer\Scope; +use Magento\CatalogSearch\Model\Indexer\IndexSwitcherInterface; use Magento\Framework\App\ResourceConnection; -use Magento\Framework\Indexer\IndexStructureInterface; use Magento\Framework\Search\Request\IndexScopeResolverInterface; /** - * Provides a functionality to replace main index table with its temporary state + * Provides a functionality to replace main index with its temporary representation */ -class IndexSwitcher +class IndexSwitcher implements IndexSwitcherInterface { /** * @var Resource @@ -45,12 +45,7 @@ public function __construct( } /** - * Switch current index with temporary index - * - * It will drop current index table and rename temporary index table to the current index table. - * - * @param array $dimensions - * @return void + * {@inheritdoc} * @throws IndexTableNotExistException */ public function switchIndex(array $dimensions) diff --git a/app/code/Magento/CatalogSearch/etc/di.xml b/app/code/Magento/CatalogSearch/etc/di.xml index 8b0bfa9d06775..7116abf263b0d 100644 --- a/app/code/Magento/CatalogSearch/etc/di.xml +++ b/app/code/Magento/CatalogSearch/etc/di.xml @@ -12,6 +12,7 @@ + Magento\CatalogSearch\Model\ResourceModel\EngineInterface::CONFIG_ENGINE_PATH @@ -20,6 +21,14 @@ + + + Magento\CatalogSearch\Model\ResourceModel\EngineInterface::CONFIG_ENGINE_PATH + + \Magento\CatalogSearch\Model\Indexer\Scope\IndexSwitcher + + + Magento\CatalogSearch\Model\ResourceModel\EngineInterface::CONFIG_ENGINE_PATH From a0538725bd7854f9ce349cb8dc8a24d5ae1608aa Mon Sep 17 00:00:00 2001 From: Illia Grybkov Date: Fri, 23 Dec 2016 12:56:16 +0200 Subject: [PATCH 07/15] MAGETWO-58174: When catalog is being indexed it should index in place or leverage an index alias so store can still function during a long index run - Refactor to use adapter-specific index switchers --- .../Model/Indexer/IndexSwitcherProxy.php | 5 +- .../Model/Indexer/Scope/ScopeProxy.php | 2 +- .../Model/Indexer/IndexSwitcherMock.php | 52 ++++++ .../Indexer/SwitcherUsedInFulltextTest.php | 164 ++++++++++++++++++ .../Magento/CatalogSearch/_files/etc/di.xml | 14 ++ 5 files changed, 235 insertions(+), 2 deletions(-) create mode 100644 dev/tests/integration/testsuite/Magento/CatalogSearch/Model/Indexer/IndexSwitcherMock.php create mode 100644 dev/tests/integration/testsuite/Magento/CatalogSearch/Model/Indexer/SwitcherUsedInFulltextTest.php create mode 100644 dev/tests/integration/testsuite/Magento/CatalogSearch/_files/etc/di.xml diff --git a/app/code/Magento/CatalogSearch/Model/Indexer/IndexSwitcherProxy.php b/app/code/Magento/CatalogSearch/Model/Indexer/IndexSwitcherProxy.php index c78051260c547..2ce093ed99e3a 100644 --- a/app/code/Magento/CatalogSearch/Model/Indexer/IndexSwitcherProxy.php +++ b/app/code/Magento/CatalogSearch/Model/Indexer/IndexSwitcherProxy.php @@ -10,6 +10,9 @@ use Magento\Framework\ObjectManagerInterface; use Magento\Store\Model\ScopeInterface; +/** + * Proxy for adapter-specific index switcher + */ class IndexSwitcherProxy implements IndexSwitcherInterface { /** @@ -59,7 +62,7 @@ public function __construct( } /** - * @inheritDoc + * {@inheritDoc} * * As index switcher is an optional part of the search SPI, it may be not defined by a search engine. * It is especially reasonable for search engines with pre-defined indexes declaration (like old SOLR and Sphinx) diff --git a/app/code/Magento/CatalogSearch/Model/Indexer/Scope/ScopeProxy.php b/app/code/Magento/CatalogSearch/Model/Indexer/Scope/ScopeProxy.php index 535f51ed9de84..14832af303bf4 100644 --- a/app/code/Magento/CatalogSearch/Model/Indexer/Scope/ScopeProxy.php +++ b/app/code/Magento/CatalogSearch/Model/Indexer/Scope/ScopeProxy.php @@ -20,7 +20,7 @@ class ScopeProxy implements \Magento\Framework\Search\Request\IndexScopeResolver * * @var \Magento\Framework\ObjectManagerInterface */ - protected $objectManager = null; + private $objectManager; /** * @var array diff --git a/dev/tests/integration/testsuite/Magento/CatalogSearch/Model/Indexer/IndexSwitcherMock.php b/dev/tests/integration/testsuite/Magento/CatalogSearch/Model/Indexer/IndexSwitcherMock.php new file mode 100644 index 0000000000000..d3568a975b9ce --- /dev/null +++ b/dev/tests/integration/testsuite/Magento/CatalogSearch/Model/Indexer/IndexSwitcherMock.php @@ -0,0 +1,52 @@ +indexSwitcher = $indexSwitcher; + } + + + /** + * Switch current index with temporary index + * + * It will drop current index table and rename temporary index table to the current index table. + * + * @param array $dimensions + * @return void + */ + public function switchIndex(array $dimensions) + { + $this->isSwitched = true; + $this->indexSwitcher->switchIndex($dimensions); + } + + /** + * @return bool + */ + public function isSwitched() + { + return $this->isSwitched; + } +} diff --git a/dev/tests/integration/testsuite/Magento/CatalogSearch/Model/Indexer/SwitcherUsedInFulltextTest.php b/dev/tests/integration/testsuite/Magento/CatalogSearch/Model/Indexer/SwitcherUsedInFulltextTest.php new file mode 100644 index 0000000000000..81d93daf1198a --- /dev/null +++ b/dev/tests/integration/testsuite/Magento/CatalogSearch/Model/Indexer/SwitcherUsedInFulltextTest.php @@ -0,0 +1,164 @@ +indexer = Bootstrap::getObjectManager()->create( + \Magento\Indexer\Model\Indexer::class + ); + $this->indexer->load('catalogsearch_fulltext'); + + $objectManager = Bootstrap::getObjectManager(); + $this->engine = $objectManager->get( + \Magento\CatalogSearch\Model\ResourceModel\Engine::class + ); + + $this->resourceFulltext = Bootstrap::getObjectManager()->get( + \Magento\CatalogSearch\Model\ResourceModel\Fulltext::class + ); + + $this->queryFactory = Bootstrap::getObjectManager()->get( + \Magento\Search\Model\QueryFactory::class + ); + + $this->dimension = Bootstrap::getObjectManager()->create( + \Magento\Framework\Search\Request\Dimension::class, + ['name' => 'scope', 'value' => '1'] + ); + + $this->productApple = $this->getProductBySku('fulltext-1'); + $this->productBanana = $this->getProductBySku('fulltext-2'); + $this->productOrange = $this->getProductBySku('fulltext-3'); + $this->productPapaya = $this->getProductBySku('fulltext-4'); + $this->productCherry = $this->getProductBySku('fulltext-5'); + } + + public function testReindexAll() + { + $this->indexer->reindexAll(); + + $products = $this->search('Apple'); + $this->assertCount(1, $products); + $this->assertEquals($this->productApple->getId(), $products[0]->getId()); + + $products = $this->search('Simple Product'); + $this->assertCount(5, $products); + /** @var \Magento\CatalogSearch\Model\Indexer\IndexSwitcherMock $indexSwitcher */ + $indexSwitcher = Bootstrap::getObjectManager()->get( + \Magento\CatalogSearch\Model\Indexer\IndexSwitcherMock::class + ); + $this->assertTrue($indexSwitcher->isSwitched()); + } + + /** + * Search the text and return result collection + * + * @param string $text + * @return Product[] + */ + protected function search($text) + { + $this->resourceFulltext->resetSearchResults(); + $query = $this->queryFactory->get(); + $query->unsetData(); + $query->setQueryText($text); + $query->saveIncrementalPopularity(); + $products = []; + $collection = Bootstrap::getObjectManager()->create( + Collection::class, + [ + 'searchRequestName' => 'quick_search_container' + ] + ); + $collection->addSearchFilter($text); + foreach ($collection as $product) { + $products[] = $product; + } + return $products; + } + + /** + * Return product by SKU + * + * @param string $sku + * @return Product + */ + protected function getProductBySku($sku) + { + /** @var Product $product */ + $product = Bootstrap::getObjectManager()->get( + \Magento\Catalog\Model\Product::class + ); + return $product->loadByAttribute('sku', $sku); + } +} diff --git a/dev/tests/integration/testsuite/Magento/CatalogSearch/_files/etc/di.xml b/dev/tests/integration/testsuite/Magento/CatalogSearch/_files/etc/di.xml new file mode 100644 index 0000000000000..c8583132661be --- /dev/null +++ b/dev/tests/integration/testsuite/Magento/CatalogSearch/_files/etc/di.xml @@ -0,0 +1,14 @@ + + + + + + \Magento\CatalogSearch\Model\Indexer\IndexSwitcherMock + + + From 17dffbe3b9580b66e032b7efc0d1ad6c70b9af73 Mon Sep 17 00:00:00 2001 From: Illia Grybkov Date: Fri, 23 Dec 2016 15:18:37 +0200 Subject: [PATCH 08/15] MAGETWO-58174: When catalog is being indexed it should index in place or leverage an index alias so store can still function during a long index run - Add integration tests to ensure that Index Switcher was actually used in one type of indexation and not used in others --- .../Model/Indexer/IndexSwitcherMock.php | 8 ++- .../Indexer/SwitcherUsedInFulltextTest.php | 69 ++++++++++++++++--- .../Magento/CatalogSearch/_files/etc/di.xml | 14 ---- 3 files changed, 63 insertions(+), 28 deletions(-) delete mode 100644 dev/tests/integration/testsuite/Magento/CatalogSearch/_files/etc/di.xml diff --git a/dev/tests/integration/testsuite/Magento/CatalogSearch/Model/Indexer/IndexSwitcherMock.php b/dev/tests/integration/testsuite/Magento/CatalogSearch/Model/Indexer/IndexSwitcherMock.php index d3568a975b9ce..955ccce3bb59e 100644 --- a/dev/tests/integration/testsuite/Magento/CatalogSearch/Model/Indexer/IndexSwitcherMock.php +++ b/dev/tests/integration/testsuite/Magento/CatalogSearch/Model/Indexer/IndexSwitcherMock.php @@ -9,6 +9,9 @@ use Magento\CatalogSearch\Model\Indexer\IndexSwitcherInterface; +/** + * The proxy class around index switcher which allows to ensure that the IndexSwitcher was actually used + */ class IndexSwitcherMock extends \PHPUnit_Framework_Assert implements IndexSwitcherInterface { private $isSwitched = false; @@ -27,7 +30,6 @@ public function __construct( $this->indexSwitcher = $indexSwitcher; } - /** * Switch current index with temporary index * @@ -38,7 +40,7 @@ public function __construct( */ public function switchIndex(array $dimensions) { - $this->isSwitched = true; + $this->isSwitched |= true; $this->indexSwitcher->switchIndex($dimensions); } @@ -47,6 +49,6 @@ public function switchIndex(array $dimensions) */ public function isSwitched() { - return $this->isSwitched; + return (bool) $this->isSwitched; } } diff --git a/dev/tests/integration/testsuite/Magento/CatalogSearch/Model/Indexer/SwitcherUsedInFulltextTest.php b/dev/tests/integration/testsuite/Magento/CatalogSearch/Model/Indexer/SwitcherUsedInFulltextTest.php index 81d93daf1198a..9148e2a8f2599 100644 --- a/dev/tests/integration/testsuite/Magento/CatalogSearch/Model/Indexer/SwitcherUsedInFulltextTest.php +++ b/dev/tests/integration/testsuite/Magento/CatalogSearch/Model/Indexer/SwitcherUsedInFulltextTest.php @@ -15,6 +15,11 @@ */ class SwitcherUsedInFulltextTest extends \PHPUnit_Framework_TestCase { + /** + * @var IndexSwitcherInterface + */ + private $indexSwitcher; + /** * @var \Magento\Framework\Indexer\IndexerInterface */ @@ -72,30 +77,47 @@ class SwitcherUsedInFulltextTest extends \PHPUnit_Framework_TestCase protected function setUp() { + $objectManager = Bootstrap::getObjectManager(); + + $objectManager->configure( + [ + 'Magento\CatalogSearch\Model\Indexer\Fulltext' => [ + 'arguments' => [ + 'indexSwitcher' => [ + 'instance' => 'Magento\CatalogSearch\Model\Indexer\IndexSwitcherMock', + ], + ], + ], + ] + ); + /** @var \Magento\Framework\Indexer\IndexerInterface indexer */ - $this->indexer = Bootstrap::getObjectManager()->create( + $this->indexer = $objectManager->create( \Magento\Indexer\Model\Indexer::class ); $this->indexer->load('catalogsearch_fulltext'); - $objectManager = Bootstrap::getObjectManager(); $this->engine = $objectManager->get( \Magento\CatalogSearch\Model\ResourceModel\Engine::class ); - $this->resourceFulltext = Bootstrap::getObjectManager()->get( + $this->resourceFulltext = $objectManager->get( \Magento\CatalogSearch\Model\ResourceModel\Fulltext::class ); - $this->queryFactory = Bootstrap::getObjectManager()->get( + $this->queryFactory = $objectManager->get( \Magento\Search\Model\QueryFactory::class ); - $this->dimension = Bootstrap::getObjectManager()->create( + $this->dimension = $objectManager->create( \Magento\Framework\Search\Request\Dimension::class, ['name' => 'scope', 'value' => '1'] ); + $this->indexSwitcher = Bootstrap::getObjectManager()->get( + \Magento\CatalogSearch\Model\Indexer\IndexSwitcherMock::class + ); + $this->productApple = $this->getProductBySku('fulltext-1'); $this->productBanana = $this->getProductBySku('fulltext-2'); $this->productOrange = $this->getProductBySku('fulltext-3'); @@ -103,16 +125,13 @@ protected function setUp() $this->productCherry = $this->getProductBySku('fulltext-5'); } + /** + * @magentoAppIsolation enabled + */ public function testReindexAll() { $this->indexer->reindexAll(); - $products = $this->search('Apple'); - $this->assertCount(1, $products); - $this->assertEquals($this->productApple->getId(), $products[0]->getId()); - - $products = $this->search('Simple Product'); - $this->assertCount(5, $products); /** @var \Magento\CatalogSearch\Model\Indexer\IndexSwitcherMock $indexSwitcher */ $indexSwitcher = Bootstrap::getObjectManager()->get( \Magento\CatalogSearch\Model\Indexer\IndexSwitcherMock::class @@ -120,6 +139,34 @@ public function testReindexAll() $this->assertTrue($indexSwitcher->isSwitched()); } + /** + * @magentoAppIsolation enabled + */ + public function testReindexList() + { + $this->indexer->reindexList([$this->productApple->getId(), $this->productBanana->getId()]); + + /** @var \Magento\CatalogSearch\Model\Indexer\IndexSwitcherMock $indexSwitcher */ + $indexSwitcher = Bootstrap::getObjectManager()->get( + \Magento\CatalogSearch\Model\Indexer\IndexSwitcherMock::class + ); + $this->assertFalse($indexSwitcher->isSwitched()); + } + + /** + * @magentoAppIsolation enabled + */ + public function testReindexRow() + { + $this->indexer->reindexRow($this->productPapaya->getId()); + + /** @var \Magento\CatalogSearch\Model\Indexer\IndexSwitcherMock $indexSwitcher */ + $indexSwitcher = Bootstrap::getObjectManager()->get( + \Magento\CatalogSearch\Model\Indexer\IndexSwitcherMock::class + ); + $this->assertFalse($indexSwitcher->isSwitched()); + } + /** * Search the text and return result collection * diff --git a/dev/tests/integration/testsuite/Magento/CatalogSearch/_files/etc/di.xml b/dev/tests/integration/testsuite/Magento/CatalogSearch/_files/etc/di.xml deleted file mode 100644 index c8583132661be..0000000000000 --- a/dev/tests/integration/testsuite/Magento/CatalogSearch/_files/etc/di.xml +++ /dev/null @@ -1,14 +0,0 @@ - - - - - - \Magento\CatalogSearch\Model\Indexer\IndexSwitcherMock - - - From b04ff64fc288504b66c3137383dcfbd3bec700e2 Mon Sep 17 00:00:00 2001 From: Illia Grybkov Date: Fri, 23 Dec 2016 15:29:18 +0200 Subject: [PATCH 09/15] MAGETWO-58174: When catalog is being indexed it should index in place or leverage an index alias so store can still function during a long index run - Depend on abstraction instead of details --- app/code/Magento/CatalogSearch/Model/Indexer/Fulltext.php | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/app/code/Magento/CatalogSearch/Model/Indexer/Fulltext.php b/app/code/Magento/CatalogSearch/Model/Indexer/Fulltext.php index 27f5fc92fd54d..337134c9453c0 100644 --- a/app/code/Magento/CatalogSearch/Model/Indexer/Fulltext.php +++ b/app/code/Magento/CatalogSearch/Model/Indexer/Fulltext.php @@ -6,7 +6,6 @@ namespace Magento\CatalogSearch\Model\Indexer; use Magento\CatalogSearch\Model\Indexer\Fulltext\Action\FullFactory; -use Magento\CatalogSearch\Model\Indexer\Scope\IndexSwitcher; use Magento\CatalogSearch\Model\Indexer\Scope\State; use Magento\CatalogSearch\Model\ResourceModel\Fulltext as FulltextResource; use Magento\Framework\App\ObjectManager; @@ -60,7 +59,7 @@ class Fulltext implements \Magento\Framework\Indexer\ActionInterface, \Magento\F private $searchRequestConfig; /** - * @var \Magento\CatalogSearch\Model\Indexer\Scope\IndexSwitcher + * @var IndexSwitcherInterface */ private $indexSwitcher; From cc04f5abb53939d8678b925d4d7fc2e039b11f9b Mon Sep 17 00:00:00 2001 From: Illia Grybkov Date: Fri, 23 Dec 2016 15:50:11 +0200 Subject: [PATCH 10/15] MAGETWO-58174: When catalog is being indexed it should index in place or leverage an index alias so store can still function during a long index run - Use ::class instead of string --- .../Model/Indexer/SwitcherUsedInFulltextTest.php | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/dev/tests/integration/testsuite/Magento/CatalogSearch/Model/Indexer/SwitcherUsedInFulltextTest.php b/dev/tests/integration/testsuite/Magento/CatalogSearch/Model/Indexer/SwitcherUsedInFulltextTest.php index 9148e2a8f2599..b1eadcdb70237 100644 --- a/dev/tests/integration/testsuite/Magento/CatalogSearch/Model/Indexer/SwitcherUsedInFulltextTest.php +++ b/dev/tests/integration/testsuite/Magento/CatalogSearch/Model/Indexer/SwitcherUsedInFulltextTest.php @@ -81,10 +81,10 @@ protected function setUp() $objectManager->configure( [ - 'Magento\CatalogSearch\Model\Indexer\Fulltext' => [ + ltrim(\Magento\CatalogSearch\Model\Indexer\Fulltext::class, '\\') => [ 'arguments' => [ 'indexSwitcher' => [ - 'instance' => 'Magento\CatalogSearch\Model\Indexer\IndexSwitcherMock', + 'instance' => ltrim(\Magento\CatalogSearch\Model\Indexer\IndexSwitcherMock::class, '\\'), ], ], ], From a29406dc4a089dafd77e844cf24e5c42b31cbd8b Mon Sep 17 00:00:00 2001 From: Illia Grybkov Date: Fri, 23 Dec 2016 15:55:59 +0200 Subject: [PATCH 11/15] MAGETWO-58174: When catalog is being indexed it should index in place or leverage an index alias so store can still function during a long index run - Fix coupling between objects in test class --- .../Indexer/SwitcherUsedInFulltextTest.php | 30 ++++++++++--------- 1 file changed, 16 insertions(+), 14 deletions(-) diff --git a/dev/tests/integration/testsuite/Magento/CatalogSearch/Model/Indexer/SwitcherUsedInFulltextTest.php b/dev/tests/integration/testsuite/Magento/CatalogSearch/Model/Indexer/SwitcherUsedInFulltextTest.php index b1eadcdb70237..6efb4531a0360 100644 --- a/dev/tests/integration/testsuite/Magento/CatalogSearch/Model/Indexer/SwitcherUsedInFulltextTest.php +++ b/dev/tests/integration/testsuite/Magento/CatalogSearch/Model/Indexer/SwitcherUsedInFulltextTest.php @@ -6,6 +6,8 @@ namespace Magento\CatalogSearch\Model\Indexer; use Magento\Catalog\Model\Product; +use Magento\CatalogSearch\Model\Indexer\Fulltext; +use Magento\CatalogSearch\Model\Indexer\IndexSwitcherMock; use Magento\CatalogSearch\Model\ResourceModel\Fulltext\Collection; use Magento\TestFramework\Helper\Bootstrap; @@ -46,27 +48,27 @@ class SwitcherUsedInFulltextTest extends \PHPUnit_Framework_TestCase protected $queryFactory; /** - * @var Product + * @var \Magento\Catalog\Model\Product */ protected $productApple; /** - * @var Product + * @var \Magento\Catalog\Model\Product */ protected $productBanana; /** - * @var Product + * @var \Magento\Catalog\Model\Product */ protected $productOrange; /** - * @var Product + * @var \Magento\Catalog\Model\Product */ protected $productPapaya; /** - * @var Product + * @var \Magento\Catalog\Model\Product */ protected $productCherry; @@ -81,10 +83,10 @@ protected function setUp() $objectManager->configure( [ - ltrim(\Magento\CatalogSearch\Model\Indexer\Fulltext::class, '\\') => [ + ltrim(Fulltext::class, '\\') => [ 'arguments' => [ 'indexSwitcher' => [ - 'instance' => ltrim(\Magento\CatalogSearch\Model\Indexer\IndexSwitcherMock::class, '\\'), + 'instance' => ltrim(IndexSwitcherMock::class, '\\'), ], ], ], @@ -115,7 +117,7 @@ protected function setUp() ); $this->indexSwitcher = Bootstrap::getObjectManager()->get( - \Magento\CatalogSearch\Model\Indexer\IndexSwitcherMock::class + IndexSwitcherMock::class ); $this->productApple = $this->getProductBySku('fulltext-1'); @@ -132,9 +134,9 @@ public function testReindexAll() { $this->indexer->reindexAll(); - /** @var \Magento\CatalogSearch\Model\Indexer\IndexSwitcherMock $indexSwitcher */ + /** @var IndexSwitcherMock $indexSwitcher */ $indexSwitcher = Bootstrap::getObjectManager()->get( - \Magento\CatalogSearch\Model\Indexer\IndexSwitcherMock::class + IndexSwitcherMock::class ); $this->assertTrue($indexSwitcher->isSwitched()); } @@ -146,9 +148,9 @@ public function testReindexList() { $this->indexer->reindexList([$this->productApple->getId(), $this->productBanana->getId()]); - /** @var \Magento\CatalogSearch\Model\Indexer\IndexSwitcherMock $indexSwitcher */ + /** @var IndexSwitcherMock $indexSwitcher */ $indexSwitcher = Bootstrap::getObjectManager()->get( - \Magento\CatalogSearch\Model\Indexer\IndexSwitcherMock::class + IndexSwitcherMock::class ); $this->assertFalse($indexSwitcher->isSwitched()); } @@ -160,9 +162,9 @@ public function testReindexRow() { $this->indexer->reindexRow($this->productPapaya->getId()); - /** @var \Magento\CatalogSearch\Model\Indexer\IndexSwitcherMock $indexSwitcher */ + /** @var IndexSwitcherMock $indexSwitcher */ $indexSwitcher = Bootstrap::getObjectManager()->get( - \Magento\CatalogSearch\Model\Indexer\IndexSwitcherMock::class + IndexSwitcherMock::class ); $this->assertFalse($indexSwitcher->isSwitched()); } From 2ee18cd557ddf3175402c7e2904ee3182b1a558d Mon Sep 17 00:00:00 2001 From: Illia Grybkov Date: Fri, 23 Dec 2016 16:09:04 +0200 Subject: [PATCH 12/15] MAGETWO-58174: When catalog is being indexed it should index in place or leverage an index alias so store can still function during a long index run - Fix coupling between objects in test class --- .../CatalogSearch/Model/Indexer/SwitcherUsedInFulltextTest.php | 1 + 1 file changed, 1 insertion(+) diff --git a/dev/tests/integration/testsuite/Magento/CatalogSearch/Model/Indexer/SwitcherUsedInFulltextTest.php b/dev/tests/integration/testsuite/Magento/CatalogSearch/Model/Indexer/SwitcherUsedInFulltextTest.php index 6efb4531a0360..afcfef5fc4c8e 100644 --- a/dev/tests/integration/testsuite/Magento/CatalogSearch/Model/Indexer/SwitcherUsedInFulltextTest.php +++ b/dev/tests/integration/testsuite/Magento/CatalogSearch/Model/Indexer/SwitcherUsedInFulltextTest.php @@ -14,6 +14,7 @@ /** * @magentoDbIsolation disabled * @magentoDataFixture Magento/CatalogSearch/_files/indexer_fulltext.php + * @SuppressWarnings(PHPMD.CouplingBetweenObjects) */ class SwitcherUsedInFulltextTest extends \PHPUnit_Framework_TestCase { From e6f24bcdbe90ac05d9be9756afca10016511fbe2 Mon Sep 17 00:00:00 2001 From: Yaroslav Onischenko Date: Wed, 21 Dec 2016 18:28:52 +0200 Subject: [PATCH 13/15] MAGETWO-62387: Travis Build Fail - 2.2.0 --- .../framework/Magento/TestFramework/Bootstrap/DocBlock.php | 2 ++ .../framework/Magento/TestFramework/ObjectManager.php | 2 -- .../integration/testsuite/Magento/Catalog/Helper/DataTest.php | 1 + 3 files changed, 3 insertions(+), 2 deletions(-) diff --git a/dev/tests/integration/framework/Magento/TestFramework/Bootstrap/DocBlock.php b/dev/tests/integration/framework/Magento/TestFramework/Bootstrap/DocBlock.php index bf890b2448e23..d23985bc38ef9 100644 --- a/dev/tests/integration/framework/Magento/TestFramework/Bootstrap/DocBlock.php +++ b/dev/tests/integration/framework/Magento/TestFramework/Bootstrap/DocBlock.php @@ -40,6 +40,7 @@ public function registerAnnotations(\Magento\TestFramework\Application $applicat * * Note: order of registering (and applying) annotations is important. * To allow config fixtures to deal with fixture stores, data fixtures should be processed first. + * ConfigFixture applied twice because data fixtures could clean config and clean custom settings * * @param \Magento\TestFramework\Application $application * @return array @@ -68,6 +69,7 @@ protected function _getSubscribers(\Magento\TestFramework\Application $applicati new \Magento\TestFramework\Annotation\AppArea($application), new \Magento\TestFramework\Annotation\Cache($application), new \Magento\TestFramework\Annotation\AdminConfigFixture(), + new \Magento\TestFramework\Annotation\ConfigFixture(), ]; } } diff --git a/dev/tests/integration/framework/Magento/TestFramework/ObjectManager.php b/dev/tests/integration/framework/Magento/TestFramework/ObjectManager.php index ad685845cde00..03fb9aa9ddae6 100644 --- a/dev/tests/integration/framework/Magento/TestFramework/ObjectManager.php +++ b/dev/tests/integration/framework/Magento/TestFramework/ObjectManager.php @@ -24,8 +24,6 @@ class ObjectManager extends \Magento\Framework\App\ObjectManager * @var array */ protected $persistedInstances = [ - \Magento\TestFramework\App\Config::class, - \Magento\Framework\App\Config\ScopeConfigInterface::class, \Magento\Framework\App\ResourceConnection::class, \Magento\Framework\Config\Scope::class, \Magento\Framework\ObjectManager\RelationsInterface::class, diff --git a/dev/tests/integration/testsuite/Magento/Catalog/Helper/DataTest.php b/dev/tests/integration/testsuite/Magento/Catalog/Helper/DataTest.php index 071c2499bb332..c2db58728d33d 100644 --- a/dev/tests/integration/testsuite/Magento/Catalog/Helper/DataTest.php +++ b/dev/tests/integration/testsuite/Magento/Catalog/Helper/DataTest.php @@ -248,6 +248,7 @@ public function testGetPageTemplateProcessor() * @magentoDataFixture Magento/Customer/_files/customer.php * @magentoDataFixture Magento/Customer/_files/customer_address.php * @magentoDbIsolation enabled + * @magentoAppIsolation enabled * @dataProvider getTaxPriceDataProvider */ public function testGetTaxPrice( From a90893cee8e2217354139cb5ef7c2c0e3f3afa46 Mon Sep 17 00:00:00 2001 From: Sergii Kovalenko Date: Wed, 21 Dec 2016 17:43:03 +0200 Subject: [PATCH 14/15] MAGETWO-61829: SOAP/REST tests fail on develop branch but builds are green --- .../Api/ProductCustomOptionRepositoryTest.php | 12 ++----- .../Api/ProductRepositoryInterfaceTest.php | 20 +---------- .../_files/product_with_two_websites.php | 20 +++++------ .../product_with_two_websites_rollback.php | 35 +++++++++++++++++++ .../Magento/Catalog/_files/second_website.php | 2 ++ .../_files/second_website_rollback.php | 22 ++++++++++++ 6 files changed, 71 insertions(+), 40 deletions(-) create mode 100644 dev/tests/integration/testsuite/Magento/Catalog/_files/product_with_two_websites_rollback.php create mode 100644 dev/tests/integration/testsuite/Magento/Catalog/_files/second_website_rollback.php diff --git a/dev/tests/api-functional/testsuite/Magento/Catalog/Api/ProductCustomOptionRepositoryTest.php b/dev/tests/api-functional/testsuite/Magento/Catalog/Api/ProductCustomOptionRepositoryTest.php index 0b761d8c191ab..6a09bea591cb8 100644 --- a/dev/tests/api-functional/testsuite/Magento/Catalog/Api/ProductCustomOptionRepositoryTest.php +++ b/dev/tests/api-functional/testsuite/Magento/Catalog/Api/ProductCustomOptionRepositoryTest.php @@ -391,6 +391,7 @@ public function validOptionDataProvider() */ public function testUpdateNegative($optionData, $message) { + $this->_markTestAsRestOnly(); $productSku = 'simple'; /** @var ProductRepository $productRepository */ $productRepository = $this->objectManager->create(ProductRepository::class); @@ -403,18 +404,9 @@ public function testUpdateNegative($optionData, $message) 'resourcePath' => '/V1/products/options/' . $optionId, 'httpMethod' => \Magento\Framework\Webapi\Rest\Request::HTTP_METHOD_PUT, ], - 'soap' => [ - 'service' => self::SERVICE_NAME, - 'serviceVersion' => 'V1', - 'operation' => self::SERVICE_NAME . 'Save', - ], ]; - if (TESTS_WEB_API_ADAPTER == self::ADAPTER_SOAP) { - $this->setExpectedException('SoapFault'); - } else { - $this->setExpectedException('Exception', $message, 400); - } + $this->setExpectedException('Exception', $message, 400); $this->_webApiCall($serviceInfo, ['option' => $optionData]); } diff --git a/dev/tests/api-functional/testsuite/Magento/Catalog/Api/ProductRepositoryInterfaceTest.php b/dev/tests/api-functional/testsuite/Magento/Catalog/Api/ProductRepositoryInterfaceTest.php index cd02c6d4d7500..ed641ac0f85b8 100644 --- a/dev/tests/api-functional/testsuite/Magento/Catalog/Api/ProductRepositoryInterfaceTest.php +++ b/dev/tests/api-functional/testsuite/Magento/Catalog/Api/ProductRepositoryInterfaceTest.php @@ -134,15 +134,6 @@ public function productCreationProvider() ]; } - private function markAreaAsSecure() - { - /** @var \Magento\Framework\Registry $registry */ - $registry = \Magento\TestFramework\Helper\Bootstrap::getObjectManager() - ->get(\Magento\Framework\Registry::class); - $registry->unregister("isSecureArea"); - $registry->register("isSecureArea", true); - } - /** * Test removing association between product and website 1 * @magentoApiDataFixture Magento/Catalog/_files/product_with_two_websites.php @@ -169,9 +160,6 @@ public function testUpdateWithDeleteWebsites() $response[ProductInterface::EXTENSION_ATTRIBUTES_KEY]["website_ids"], $websitesData["website_ids"] ); - $this->deleteProduct($productBuilder[ProductInterface::SKU]); - $this->markAreaAsSecure(); - $website->delete(); } /** @@ -198,9 +186,6 @@ public function testDeleteAllWebsiteAssociations() $response[ProductInterface::EXTENSION_ATTRIBUTES_KEY]["website_ids"], $websitesData["website_ids"] ); - $this->deleteProduct($productBuilder[ProductInterface::SKU]); - $this->markAreaAsSecure(); - $website->delete(); } /** @@ -222,7 +207,7 @@ public function testCreateWithMultipleWebsites() $websitesData = [ 'website_ids' => [ 1, - $website->getId(), + (int) $website->getId(), ] ]; $productBuilder[ProductInterface::EXTENSION_ATTRIBUTES_KEY] = $websitesData; @@ -231,9 +216,6 @@ public function testCreateWithMultipleWebsites() $response[ProductInterface::EXTENSION_ATTRIBUTES_KEY]["website_ids"], $websitesData["website_ids"] ); - $this->deleteProduct($productBuilder[ProductInterface::SKU]); - $this->markAreaAsSecure(); - $website->delete(); } /** diff --git a/dev/tests/integration/testsuite/Magento/Catalog/_files/product_with_two_websites.php b/dev/tests/integration/testsuite/Magento/Catalog/_files/product_with_two_websites.php index fa64de38e4311..426ee922f2614 100644 --- a/dev/tests/integration/testsuite/Magento/Catalog/_files/product_with_two_websites.php +++ b/dev/tests/integration/testsuite/Magento/Catalog/_files/product_with_two_websites.php @@ -9,25 +9,23 @@ $objectManager = \Magento\TestFramework\Helper\Bootstrap::getObjectManager(); /** @var Magento\Store\Model\Website $website */ $website = $objectManager->get(Magento\Store\Model\Website::class); -$website->load('second_website', 'code'); -if (!$website->getId()) { - $website->setData( - [ - 'code' => 'second_website', - 'name' => 'Test Website', - ] - ); +$website->setData( + [ + 'code' => 'second_website', + 'name' => 'Test Website', + ] +); - $website->save(); -} +$website->save(); + +$objectManager->get(\Magento\Store\Model\StoreManagerInterface::class)->reinitStores(); /** @var $product \Magento\Catalog\Model\Product */ $product = \Magento\TestFramework\Helper\Bootstrap::getObjectManager() ->create(ProductInterface::class); $product ->setTypeId('simple') - ->setId(1) ->setAttributeSetId(4) ->setWebsiteIds([1, $website->getId()]) ->setName('Simple Product') diff --git a/dev/tests/integration/testsuite/Magento/Catalog/_files/product_with_two_websites_rollback.php b/dev/tests/integration/testsuite/Magento/Catalog/_files/product_with_two_websites_rollback.php new file mode 100644 index 0000000000000..68f1106dcfe30 --- /dev/null +++ b/dev/tests/integration/testsuite/Magento/Catalog/_files/product_with_two_websites_rollback.php @@ -0,0 +1,35 @@ +get(\Magento\Framework\Registry::class); +$registry->unregister("isSecureArea"); +$registry->register("isSecureArea", true); + +/** @var Magento\Store\Model\Website $website */ +$website = $objectManager->create(\Magento\Store\Model\Website::class); +$website->load('second_website'); +$website->delete(); + +/** @var \Magento\Catalog\Api\ProductRepositoryInterface $productRepository */ +$productRepository = $objectManager->create(\Magento\Catalog\Api\ProductRepositoryInterface::class); + +try { + $firstProduct = $productRepository->get('unique-simple-azaza'); + $firstProduct->delete(); +} catch (\Magento\Framework\Exception\NoSuchEntityException $exception) { + //Product already removed +} + +$registry->unregister('isSecureArea'); +$registry->register('isSecureArea', false); + +$objectManager->get(\Magento\Store\Model\StoreManagerInterface::class)->reinitStores(); diff --git a/dev/tests/integration/testsuite/Magento/Catalog/_files/second_website.php b/dev/tests/integration/testsuite/Magento/Catalog/_files/second_website.php index 6749d421e7998..4dc026d6fad4d 100644 --- a/dev/tests/integration/testsuite/Magento/Catalog/_files/second_website.php +++ b/dev/tests/integration/testsuite/Magento/Catalog/_files/second_website.php @@ -19,3 +19,5 @@ $website->save(); } + +$objectManager->get(\Magento\Store\Model\StoreManagerInterface::class)->reinitStores(); diff --git a/dev/tests/integration/testsuite/Magento/Catalog/_files/second_website_rollback.php b/dev/tests/integration/testsuite/Magento/Catalog/_files/second_website_rollback.php new file mode 100644 index 0000000000000..173793a1caa49 --- /dev/null +++ b/dev/tests/integration/testsuite/Magento/Catalog/_files/second_website_rollback.php @@ -0,0 +1,22 @@ +get('Magento\Framework\Registry'); +$registry->unregister('isSecureArea'); +$registry->register('isSecureArea', true); + +$objectManager = \Magento\TestFramework\Helper\Bootstrap::getObjectManager(); +$website = $objectManager->get(Magento\Store\Model\Website::class); +$website->load('test_website', 'code'); + +if ($website->getId()) { + $website->delete(); +} + +$registry->unregister('isSecureArea'); +$registry->register('isSecureArea', false); + +$objectManager->get(\Magento\Store\Model\StoreManagerInterface::class)->reinitStores(); From e4f0890d82368918b1e953fbb849c9cd08b7ff0a Mon Sep 17 00:00:00 2001 From: Illia Grybkov Date: Wed, 28 Dec 2016 16:23:21 +0200 Subject: [PATCH 15/15] MAGETWO-58174: When catalog is being indexed it should index in place or leverage an index alias so store can still function during a long index run - Update unit test to match class design changes in a code --- .../Test/Unit/Model/Search/Indexer/IndexStructureTest.php | 8 +------- 1 file changed, 1 insertion(+), 7 deletions(-) diff --git a/app/code/Magento/CatalogSearch/Test/Unit/Model/Search/Indexer/IndexStructureTest.php b/app/code/Magento/CatalogSearch/Test/Unit/Model/Search/Indexer/IndexStructureTest.php index 3bb270e990617..8f4bb7736cd37 100644 --- a/app/code/Magento/CatalogSearch/Test/Unit/Model/Search/Indexer/IndexStructureTest.php +++ b/app/code/Magento/CatalogSearch/Test/Unit/Model/Search/Indexer/IndexStructureTest.php @@ -48,9 +48,8 @@ protected function setUp() ->method('getConnection') ->willReturn($this->connection); $this->indexScopeResolver = $this->getMockBuilder( - \Magento\Framework\Indexer\ScopeResolver\IndexScopeResolver::class + \Magento\Framework\Search\Request\IndexScopeResolverInterface::class )->setMethods(['resolve']) - ->disableOriginalConstructor() ->getMock(); $objectManager = new ObjectManager($this); @@ -64,11 +63,6 @@ protected function setUp() ); } - /** - * @param string $table - * @param array $dimensions - * @param bool $isTableExist - */ public function testDelete() { $index = 'index_name';