Upgrade the Search module
Edit on GitHubUpgrading from version 8.9.* to version 8.10.*
This migration guide is a part of the Search migration effort.
To upgrade the module, do the following:
- Install and/or update the modules using Composer:
composer update spryker/search --with-dependencies
composer require spryker/search-elasticsearch
- Regenerate transfer classes:
console transfer:generate
- Adjust all project-level implementations of
Spryker\Client\Search\Dependency\Plugin\QueryInterface
. First, changeSpryker\Client\Search\Dependency\Plugin\QueryInterface
toSpryker\Client\SearchExtension\Dependency\Plugin\QueryInterface
. This does not require changing any implementation details. After that implement\Spryker\Client\SearchExtension\Dependency\Plugin\SearchContextAwareQueryInterface
as described in the Search Migration Concept. - Remove
Pyz\Client\Search\SearchDependencyProvider::createSearchConfigBuilderPlugin()
. - Remove
Pyz\Client\Search\SearchDependencyProvider::createSearchConfigExpanderPlugins()
. - Enable
ElasticsearchSearchAdapterPlugin
andElasticsearchSearchContextExpanderPlugin
inPyz\Client\Search\SearchDependencyProvider
:
Pyz\Client\Search
...
use Spryker\Client\SearchElasticsearch\Plugin\ElasticsearchSearchAdapterPlugin;
use Spryker\Client\SearchElasticsearch\Plugin\ElasticsearchSearchContextExpanderPlugin;
class SearchDependencyProvider extends SprykerSearchDependencyProvider
{
...
/**
* @return \Spryker\Client\SearchExtension\Dependency\Plugin\SearchAdapterPluginInterface[]
*/
protected function getClientAdapterPlugins(): array
{
return [
new ElasticsearchSearchAdapterPlugin(),
];
}
/**
* @return \Spryker\Client\SearchExtension\Dependency\Plugin\SearchContextExpanderPluginInterface[]
*/
protected function getSearchContextExpanderPlugins(): array
{
return [
new ElasticsearchSearchContextExpanderPlugin(),
];
}
...
}
- Remove
Pyz\Zed\Search\SearchDependencyProvider::getSearchPageMapPlugins()
. - Enable
ElasticsearchIndexInstallerPlugin
andElasticsearchIndexMapInstallerPlugin
inPyz\Zed\Search\SearchDependencyProvider
:
Pyz\Zed\Search
<?php
namespace Pyz\Zed\Search;
...
use Spryker\Zed\Search\SearchDependencyProvider as SprykerSearchDependencyProvider;
use Spryker\Zed\SearchElasticsearch\Communication\Plugin\Search\ElasticsearchIndexInstallerPlugin;
use Spryker\Zed\SearchElasticsearch\Communication\Plugin\Search\ElasticsearchIndexMapInstallerPlugin;
class SearchDependencyProvider extends SprykerSearchDependencyProvider
{
/**
* @return \Spryker\Zed\SearchExtension\Dependency\Plugin\InstallPluginInterface[]
*/
protected function getSearchSourceInstallerPlugins(): array
{
return [
new ElasticsearchIndexInstallerPlugin(),
];
}
/**
* @return \Spryker\Zed\SearchExtension\Dependency\Plugin\InstallPluginInterface[]
*/
protected function getSearchMapInstallerPlugins(): array
{
return [
new ElasticsearchIndexMapInstallerPlugin(),
];
}
}
- Configure the list of sources, which are being handled by Elasticsearch, on the project level.
<?php
namespace Pyz\Shared\SearchElasticsearch;
use Spryker\Shared\SearchElasticsearch\SearchElasticsearchConfig as SprykerSearchElasticsearchConfig;
class SearchElasticsearchConfig extends SprykerSearchElasticsearchConfig
{
protected const SUPPORTED_SOURCE_IDENTIFIERS = [
'page',
'product-review',
];
}
-
Adjust all project-level Elasticsearch index definition JSON files (if any) as follows:
- Each JSON file should be renamed, so it would have one of the source identifiers (see above) as its name. The name of the definition file matters and will later be translated into index name.
- Each JSON file should provide a definition only for one mapping type, suitable for that index. By default, index’s only mapping type should have the same name as the JSON file it’s described by. For example,
page.json
should only contain a definition for thepage
mapping type. - Each JSON file should be placed inside of the directory, which matches a path pattern defined by
SearchElasticsearchConfig::getJsonSchemaDefinitionDirectories()
.
Upgrading from version 7.* to version 8.*
With this version of the Search module we have migrated to Elasticsearch 5.6. Please read the Elasticsearch Breaking Changes in 5.0 official documentation to adjust your custom implementation accordingly.
Your development environment needs to be updated with Elasticsearch 5.6.x.
Elasticsearch 5 related breaking change highlights
string fields
replaced by text/keyword field: mapping changed for all string fields in the indexes.index
property: the index property now only acceptstrue/false
instead ofnot_analyzed/no
.size
: 0 on Terms, Significant Terms and Geohash Grid Aggregations: the Demoshop used this feature to aggregate infinite number of categories. Size should be set to a fixed number instead.missing
query was removed, use a negated exists query instead.
Other breaking changes
Previously the vendor/bin/console setup:search
command installed indexes for all stores.
Now it only installs the index for the current store.
Upgrading from version 6.* to version 7.*
Zed changes: With version 7 we have fixed a bug with incorrect mapping of a filter name with request parameters. If you have modified/extended:
\Spryker\Client\Search\Model\Elasticsearch\AggregationExtractor\CategoryExtractor
\Spryker\Client\Search\Model\Elasticsearch\AggregationExtractor\FacetExtractor
\Spryker\Client\Search\Model\Elasticsearch\AggregationExtractor\RangeExtractor
you have to merge the latest changes with the core. Especially this is important for extractDataFromAggregations
method.
\Spryker\Client\Search\Plugin\Config\FacetConfigBuilder
now looks for facet from the request parameters.\Spryker\Client\Search\Plugin\Config\SortConfigBuilder
now looks for configuration by configuration field name.
Yves changes:
The UrlGenerator
was incorrectly setting the request parameters, therefore now it’s necessary to change processFacetSearchResultTransfer and processRangeSearchResultTransfer as shown in the code sample below.
Code sample:
namespace Pyz\Yves\Catalog\ActiveSearchFilter;
class UrlGenerator implements UrlGeneratorInterface
{
/**
* @param array $params
* @param \Generated\Shared\Transfer\FacetSearchResultTransfer $searchResultTransfer
* @param string|null $filterValue
*
* @return array
*/
protected function processFacetSearchResultTransfer(array $params, FacetSearchResultTransfer $searchResultTransfer, $filterValue = null)
{
$param = $params[$searchResultTransfer->getName()];
if (is_array($param) && $filterValue !== null) {
$index = array_search($filterValue, $param);
unset($params[$searchResultTransfer->getName()][$index]);
return $params;
}
unset($params[$searchResultTransfer->getName()]);
return $params;
}
/**
* @param array $params
* @param \Generated\Shared\Transfer\RangeSearchResultTransfer $searchResultTransfer
*
* @return array
*/
protected function processRangeSearchResultTransfer(array $params, RangeSearchResultTransfer $searchResultTransfer)
{
unset($params[$searchResultTransfer->getName()]);
return $params;
}
You have to change the way filters are configured in twig templates. Previously there was an incorrect setting on using a name, instead of a request parameter. The filters are under Pyz/Yves/Catalog/Theme/default/catalog/partials/filters
directory.
Twig templates also require changes:
- “multi-select.twig”
<input type="checkbox" name="{{ filter.name }}[]" ...
should be
<input type="checkbox" name="{{ filter.config.parameterName }}[]" ...
- “price-range.twig”
<input type="number" name="{{ filter.name }}[min]" ... ... <input type="number" name="{{ filter.name }}[max]"
should be
<input type="number" name="{{ filter.config.parameterName }}[min]" ... ... <input type="number" name="{{ filter.config.parameterName }}[max]" ...
- “range.twig”
<input type="number" name="{{ filter.name }}[min]" ... ... <input type="number" name="{{ filter.name }}[max]" ...
should be
<input type="number" name="{{ filter.config.parameterName }}[min]" ... ... <input type="number" name="{{ filter.config.parameterName }}[max]" ...
- “rating.twig”
<input type="hidden" name="{{ filter.name }}[min]" ...
should be
<input type="hidden" name="{{ filter.config.parameterName }}[min]" ...
- “single-select.twig”
<input type="radio" name="{{ filter.name }}" ...
should be
<input type="radio" name="{{ filter.config.parameterName }}" ...
- “Pyz/Yves/Catalog/Theme/default/catalog/partials/filters.twig”
{{ ('product.filter.' ~ filter.name) | trans }}
should be {{ ('product.filter.' ~ filter.name | lower) | trans }}
Upgrading from version 4.* to version 5.*
We changed the way dynamic search configuration was cached and then used. This feature caused the following non-backward compatible changes:
- The
Spryker\Shared\Search\SearchConstants::SEARCH_CONFIG_CACHE_KEY
config was removed, but previously it was required to be filled with a key that was used to save the search config cache into Redis. - Removed
SearchFacade::saveSearchConfigCache()
method which stored the given search cache configuration into Redis. - In the new version, instead of the removed code mentioned above, you’ll need to provide a list of
Spryker\Client\Search\Dependency\Plugin\SearchConfigExpanderPluginInterface
inPyz\Client\Search\SearchDependencyProvider::createSearchConfigExpanderPlugins()
instead.
We moved the possible facet type option constants from Spryker\Client\Search\Plugin\Config\FacetConfigBuilder
to \Spryker\Shared\Search\SearchConstants
:
FacetConfigBuilder::TYPE_ENUMERATION
->SearchConstants::FACET_TYPE_ENUMERATION
FacetConfigBuilder::TYPE_RANGE
-> SearchConstants::FACET_TYPE_RANGEFacetConfigBuilder::TYPE_PRICE_RANGE
->SearchConstants::FACET_TYPE_PRICE_RANGE
FacetConfigBuilder::TYPE_CATEGORY
->SearchConstants::FACET_TYPE_CATEGORY
FacetConfigBuilder::TYPE_BOOL
-> not supported
We have added a type field to the default “page” index type defined by Search/src/Spryker/Shared/Search/IndexMap/search.json
. With this field it’s possible to differentiate multiple item types (e.g. products, cms pages, categories, etc.). Additionally, we also fixed the indexing strategy of store and locale
field, they are set to “not_analyzed”. These changes require a repeated indexation of your existing data. In a non-production environment this means that you need to delete your index and then install the new one by running vendor/bin/console setup:search
.
In production environments, follow the official Elasticsearch guide about Index Aliases and Zero Downtime.
Upgrading from version 3.* to version 4.*
With the version 4 of the Search module, the logic and configuration of how the results are sorted has been changed. Previously there were two request parameters that controlled what field we are sorting by as well as the direction of the sorting (e.g /?sort=price&sort_order=desc).
The new version now works with one parameter only (e.g. /?sort=price_asc
). To migrate to the new version, you’ll need to change your configurations in your classes that implement \Spryker\Client\Search\Dependency\Plugin\SearchConfigBuilderInterface
. Instead of providing one SortConfigTransfer
per sorted attribute, now you need to provide two if you wish to sort by both ascending and descending order. To do this, use the SortConfigTransfer::setIsDescending()
method, and make sure that the values in SortConfigTransfer::setParameterName()
are unique.
Thank you!
For submitting the form