Integrate Algolia
Edit on GitHubThis document describes how to integrate Algolia into a Spryker shop.
Prerequisites
Before integrating Algolia, ensure the following prerequisites are met:
-
Make sure your project is ACP-enabled.
-
The Algolia app catalog page lists specific packages which must be installed or upgraded before you can use the Algolia app. To check the list of the necessary packages, in the Back Office, go to Apps-> Algolia.
Ensure that your installation meets these requirements.
Integrate Algolia
Follow these steps to integrate Algolia.
1. Configure shared configs
Add the following config to config/Shared/config_default.php
:
//...
use Generated\Shared\Transfer\InitializeProductExportTransfer;
use Generated\Shared\Transfer\ProductCreatedTransfer;
use Generated\Shared\Transfer\ProductDeletedTransfer;
use Generated\Shared\Transfer\ProductExportedTransfer;
use Generated\Shared\Transfer\ProductUpdatedTransfer;
use Generated\Shared\Transfer\SearchEndpointAvailableTransfer;
use Generated\Shared\Transfer\SearchEndpointRemovedTransfer;
//...
$config[MessageBrokerConstants::MESSAGE_TO_CHANNEL_MAP] =
$config[MessageBrokerAwsConstants::MESSAGE_TO_CHANNEL_MAP] = [
//...
ProductExportedTransfer::class => 'product-events',
ProductCreatedTransfer::class => 'product-events',
ProductUpdatedTransfer::class => 'product-events',
ProductDeletedTransfer::class => 'product-events',
InitializeProductExportTransfer::class => 'product-commands',
SearchEndpointAvailableTransfer::class => 'search-commands',
SearchEndpointRemovedTransfer::class => 'search-commands',
];
$config[MessageBrokerConstants::CHANNEL_TO_TRANSPORT_MAP] = [
//...
'product-commands' => MessageBrokerAwsConfig::SQS_TRANSPORT,
'search-commands' => MessageBrokerAwsConfig::SQS_TRANSPORT,
'product-events' => 'http',
];
$config[MessageBrokerAwsConstants::CHANNEL_TO_RECEIVER_TRANSPORT_MAP] = [
//...
'product-commands' => MessageBrokerAwsConfig::SQS_TRANSPORT,
'search-commands' => MessageBrokerAwsConfig::SQS_TRANSPORT,
];
$config[MessageBrokerAwsConstants::CHANNEL_TO_SENDER_TRANSPORT_MAP] = [
//...
'product-events' => 'http',
];
2. Configure modules and dependencies
Configure modules and add the necessary dependencies according to these guidelines.
Configure Catalog dependencies in Client
Add the following code to src/Pyz/Client/Catalog/CatalogDependencyProvider.php
:
//...
use Generated\Shared\Transfer\SearchContextTransfer;
use Generated\Shared\Transfer\SearchHttpSearchContextTransfer;
use Spryker\Client\Catalog\Plugin\SearchHttp\ResultFormatter\ProductConcreteCatalogSearchHttpResultFormatterPlugin;
use Spryker\Client\CatalogPriceProductConnector\Plugin\Catalog\QueryExpander\ProductPriceSearchHttpQueryExpanderPlugin;
use Spryker\Client\CatalogPriceProductConnector\Plugin\Catalog\ResultFormatter\CurrencyAwareCatalogSearchHttpResultFormatterPlugin;
use Spryker\Client\CatalogPriceProductConnector\Plugin\Catalog\ResultFormatter\CurrencyAwareCatalogSearchHttpResultFormatterPlugin;
use Spryker\Client\CategoryStorage\Plugin\Catalog\ResultFormatter\CategorySuggestionsSearchHttpResultFormatterPlugin;
use Spryker\Client\CategoryStorage\Plugin\Catalog\ResultFormatter\CategoryTreeFilterSearchHttpResultFormatterPlugin;
use Spryker\Client\MerchantProductOfferSearch\Plugin\Catalog\MerchantReferenceSearchHttpQueryExpanderPlugin;
use Spryker\Client\ProductLabelStorage\Plugin\Catalog\ProductLabelSearchHttpFacetConfigTransferBuilderPlugin;
use Spryker\Client\SearchHttp\Plugin\Catalog\Query\ProductConcreteSearchHttpQueryPlugin;
use Spryker\Client\SearchHttp\Plugin\Catalog\Query\SearchHttpQueryPlugin;
use Spryker\Client\SearchHttp\Plugin\Catalog\Query\SuggestionSearchHttpQueryPlugin;
use Spryker\Client\SearchHttp\Plugin\Catalog\QueryExpander\BasicSearchHttpQueryExpanderPlugin;
use Spryker\Client\SearchHttp\Plugin\Catalog\QueryExpander\FacetSearchHttpQueryExpanderPlugin;
use Spryker\Client\SearchHttp\Plugin\Catalog\ResultFormatter\CompletionSearchHttpResultFormatterPlugin;
use Spryker\Client\SearchHttp\Plugin\Catalog\ResultFormatter\FacetSearchHttpResultFormatterPlugin;
use Spryker\Client\SearchHttp\Plugin\Catalog\ResultFormatter\PaginationSearchHttpResultFormatterPlugin;
use Spryker\Client\SearchHttp\Plugin\Catalog\ResultFormatter\ProductSearchHttpResultFormatterPlugin;
use Spryker\Client\SearchHttp\Plugin\Catalog\ResultFormatter\ProductSuggestionSearchHttpResultFormatterPlugin;
use Spryker\Client\SearchHttp\Plugin\Catalog\ResultFormatter\SortSearchHttpResultFormatterPlugin;
use Spryker\Client\SearchHttp\Plugin\Catalog\ResultFormatter\SpellingSuggestionSearchHttpResultFormatterPlugin;
use Spryker\Shared\SearchHttp\SearchHttpConfig;
//...
class CatalogDependencyProvider extends SprykerCatalogDependencyProvider
//...
/**
* @return array<string, array<\Spryker\Client\Catalog\Dependency\Plugin\FacetConfigTransferBuilderPluginInterface>>
*/
protected function getFacetConfigTransferBuilderPluginVariants(): array
{
return [
SearchHttpConfig::TYPE_SEARCH_HTTP => [
new CategoryFacetConfigTransferBuilderPlugin(),
new PriceFacetConfigTransferBuilderPlugin(),
new RatingFacetConfigTransferBuilderPlugin(),
new ProductLabelSearchHttpFacetConfigTransferBuilderPlugin(),
],
];
}
/**
* @phpstan-return array<\Spryker\Client\SearchExtension\Dependency\Plugin\QueryInterface>
*
* @return array<\Spryker\Client\Search\Dependency\Plugin\QueryInterface>
*/
protected function createCatalogSearchQueryPluginVariants(): array
{
return [
new SearchHttpQueryPlugin(),
];
}
/**
* @phpstan-return array<\Spryker\Client\SearchExtension\Dependency\Plugin\QueryInterface>
*
* @return array<\Spryker\Client\Search\Dependency\Plugin\QueryInterface>
*/
protected function createSuggestionQueryPluginVariants(): array
{
return [
new SuggestionSearchHttpQueryPlugin(),
];
}
/**
* @return array<string, array<\Spryker\Client\SearchExtension\Dependency\Plugin\ResultFormatterPluginInterface>>
*/
protected function createProductConcreteCatalogSearchResultFormatterPluginVariants(): array
{
return [
SearchHttpConfig::TYPE_PRODUCT_CONCRETE_SEARCH_HTTP => [
new ProductConcreteCatalogSearchHttpResultFormatterPlugin(),
],
];
}
/**
* @return array<string, array<\Spryker\Client\SearchExtension\Dependency\Plugin\ResultFormatterPluginInterface>>
*/
protected function createSuggestionResultFormatterPluginVariants(): array
{
return [
SearchHttpConfig::TYPE_SUGGESTION_SEARCH_HTTP => [
new CompletionSearchHttpResultFormatterPlugin(),
new CurrencyAwareCatalogSearchHttpResultFormatterPlugin(
new ProductSuggestionSearchHttpResultFormatterPlugin(),
),
new CategorySuggestionsSearchHttpResultFormatterPlugin(),
],
];
}
/**
* @phpstan-return array<\Spryker\Client\SearchExtension\Dependency\Plugin\QueryInterface>
*
* @return array<\Spryker\Client\Search\Dependency\Plugin\QueryInterface>
*/
protected function createProductConcreteCatalogSearchQueryPluginVariants(): array
{
return [
new ProductConcreteSearchHttpQueryPlugin(),
];
}
/**
* @return array<string, array<\Spryker\Client\SearchExtension\Dependency\Plugin\QueryExpanderPluginInterface>>
*/
protected function createCatalogSearchQueryExpanderPluginVariants(): array
{
return [
SearchHttpConfig::TYPE_SEARCH_HTTP => [
new BasicSearchHttpQueryExpanderPlugin(),
new ProductPriceSearchHttpQueryExpanderPlugin(),
new FacetSearchHttpQueryExpanderPlugin(),
],
];
}
/**
* @return array<string, array<\Spryker\Client\SearchExtension\Dependency\Plugin\ResultFormatterPluginInterface>>
*/
protected function createCatalogSearchResultFormatterPluginVariants(): array
{
return [
SearchHttpConfig::TYPE_SEARCH_HTTP => [
new PaginationSearchHttpResultFormatterPlugin(),
new SortSearchHttpResultFormatterPlugin(),
new CurrencyAwareCatalogSearchHttpResultFormatterPlugin(
new ProductSearchHttpResultFormatterPlugin(),
),
new SpellingSuggestionSearchHttpResultFormatterPlugin(),
new FacetSearchHttpResultFormatterPlugin(),
new CategoryTreeFilterSearchHttpResultFormatterPlugin(),
],
];
}
//...
}
Adjust RabbitMq configuration in Client
Add the following code to src/Pyz/Client/RabbitMq/RabbitMqConfig.php
:
//...
use Spryker\Shared\SearchHttp\SearchHttpConfig;
//...
class RabbitMqConfig extends SprykerRabbitMqConfig
{
//...
/**
* @return array<mixed>
*/
protected function getSynchronizationQueueConfiguration(): array
{
return [
//...
SearchHttpConfig::SEARCH_HTTP_CONFIG_SYNC_QUEUE,
];
}
//...
}
Configure Search dependencies in Client
Add the following code to src/Pyz/Client/Search/SearchDependencyProvider.php
:
//...
use Spryker\Client\SearchHttp\Plugin\Search\SearchHttpSearchAdapterPlugin;
use Spryker\Client\SearchHttp\Plugin\Search\SearchHttpSearchContextExpanderPlugin;
//...
class SearchDependencyProvider extends SprykerSearchDependencyProvider
{
//...
/**
* @return array<\Spryker\Client\SearchExtension\Dependency\Plugin\SearchAdapterPluginInterface>
*/
protected function getClientAdapterPlugins(): array
{
return [
new SearchHttpSearchAdapterPlugin(), # It is very important to put this plugin at the top of the list.
//...
];
}
/**
* @return array<\Spryker\Client\SearchExtension\Dependency\Plugin\SearchContextExpanderPluginInterface>
*/
protected function getSearchContextExpanderPlugins(): array
{
return [
new SearchHttpSearchContextExpanderPlugin(), # It is very important to put this plugin at the top of the list.
//...
];
//...
}
Configure SearchHttp dependencies in Client
Add the following code to src/Pyz/Client/SearchHttp/SearchHttpDependencyProvider.php
:
<?php
namespace Pyz\Client\SearchHttp;
use Spryker\Client\Catalog\Plugin\ConfigTransferBuilder\CategoryFacetConfigTransferBuilderPlugin;
use Spryker\Client\Catalog\Plugin\SearchHttp\CatalogSearchHttpConfigBuilderPlugin;
use Spryker\Client\CatalogPriceProductConnector\Plugin\ConfigTransferBuilder\PriceFacetConfigTransferBuilderPlugin;
use Spryker\Client\ProductLabelStorage\Plugin\ProductLabelFacetConfigTransferBuilderPlugin;
use Spryker\Client\ProductReview\Plugin\RatingFacetConfigTransferBuilderPlugin;
use Spryker\Client\ProductSearchConfigStorage\Plugin\Config\ProductSearchConfigExpanderPlugin;
use Spryker\Client\SearchHttp\SearchHttpDependencyProvider as SprykerSearchHttpDependencyProvider;
class SearchHttpDependencyProvider extends SprykerSearchHttpDependencyProvider
{
/**
* @return array<\Spryker\Client\Catalog\Dependency\Plugin\FacetConfigTransferBuilderPluginInterface>
*/
protected function getFacetConfigTransferBuilders(): array
{
return [
new CategoryFacetConfigTransferBuilderPlugin(),
new PriceFacetConfigTransferBuilderPlugin(),
new RatingFacetConfigTransferBuilderPlugin(),
new ProductLabelFacetConfigTransferBuilderPlugin(),
];
}
/**
* @return array<\Spryker\Client\SearchExtension\Dependency\Plugin\SearchConfigBuilderPluginInterface>
*/
protected function getSearchConfigBuilderPlugins(): array
{
return [
new CatalogSearchHttpConfigBuilderPlugin(),
];
}
/**
* @return array<\Spryker\Client\SearchExtension\Dependency\Plugin\SearchConfigExpanderPluginInterface>
*/
protected function getSearchConfigExpanderPlugins(): array
{
return [
new ProductSearchConfigExpanderPlugin(),
];
}
}
Configure MessageBroker dependencies in Zed
Add the following code to src/Pyz/Zed/MessageBroker/MessageBrokerDependencyProvider.php
:
//...
use Spryker\Zed\Product\Communication\Plugin\MessageBroker\InitializeProductExportMessageHandlerPlugin;
use Spryker\Zed\SearchHttp\Communication\Plugin\MessageBroker\SearchEndpointAvailableMessageHandlerPlugin;
use Spryker\Zed\SearchHttp\Communication\Plugin\MessageBroker\SearchEndpointRemovedMessageHandlerPlugin;
//...
class MessageBrokerDependencyProvider extends SprykerMessageBrokerDependencyProvider
{
//...
/**
* @return array<\Spryker\Zed\MessageBrokerExtension\Dependency\Plugin\MessageHandlerPluginInterface>
*/
public function getMessageHandlerPlugins(): array
{
return [
//...
new ProductExportMessageHandlerPlugin(),
new SearchEndpointMessageHandlerPlugin(),
];
}
//...
}
Adjust MessageBroker configuration in Zed
Add the following code to src/Pyz/Zed/MessageBroker/MessageBrokerConfig.php
:
//...
class MessageBrokerConfig extends SprykerMessageBrokerConfig
{
//...
/**
* @return array<string>
*/
public function getDefaultWorkerChannels(): array
{
return [
//...
'product-commands',
'search-commands',
];
}
//...
}
Adjust Product configuration in Zed
Add the following code to src/Pyz/Zed/Product/ProductConfig.php
:
//...
use Spryker\Shared\ProductBundleStorage\ProductBundleStorageConfig;
use Spryker\Zed\PriceProduct\Dependency\PriceProductEvents;
use Spryker\Zed\Product\Dependency\ProductEvents;
use Spryker\Zed\Product\ProductConfig as SprykerProductConfig;
use Spryker\Zed\ProductCategory\Dependency\ProductCategoryEvents;
use Spryker\Zed\ProductImage\Dependency\ProductImageEvents;
use Spryker\Zed\ProductReview\Dependency\ProductReviewEvents;
//...
class ProductConfig extends SprykerProductConfig
{
//...
/**
* @api
*
* @return array<string>
*/
public function getProductAbstractUpdateMessageBrokerPublisherSubscribedEvents(): array
{
return [
ProductEvents::PRODUCT_ABSTRACT_PUBLISH,
ProductCategoryEvents::PRODUCT_CATEGORY_PUBLISH,
ProductImageEvents::PRODUCT_IMAGE_PRODUCT_ABSTRACT_PUBLISH,
PriceProductEvents::PRICE_ABSTRACT_PUBLISH,
ProductReviewEvents::PRODUCT_ABSTRACT_REVIEW_PUBLISH,
];
}
/**
* @api
*
* @return array<string>
*/
public function getProductUpdateMessageBrokerPublisherSubscribedEvents(): array
{
return [
ProductEvents::ENTITY_SPY_PRODUCT_UPDATE,
ProductEvents::PRODUCT_CONCRETE_UPDATE,
ProductEvents::PRODUCT_CONCRETE_PUBLISH,
ProductBundleStorageConfig::PRODUCT_BUNDLE_PUBLISH,
ProductImageEvents::PRODUCT_IMAGE_PRODUCT_CONCRETE_PUBLISH,
];
}
//...
}
Configure product dependencies in Zed
Add the following code to src/Pyz/Zed/Product/ProductDependencyProvider.php
:
//...
# MerchantProductOffer is used only for Marketplace
use Spryker\Zed\MerchantProductOffer\Communication\Plugin\Product\MerchantProductOfferProductConcreteExpanderPlugin;
//...
use Spryker\Zed\PriceProduct\Communication\Plugin\Product\PriceProductConcreteMergerPlugin;
use Spryker\Zed\ProductCategory\Communication\Plugin\Product\ProductConcreteCategoriesExpanderPlugin;
use Spryker\Zed\ProductImage\Communication\Plugin\Product\ImageSetProductConcreteMergerPlugin;
use Spryker\Zed\ProductReview\Communication\Plugin\Product\ProductReviewProductConcreteExpanderPlugin;
use Spryker\Zed\ProductApproval\Communication\Plugin\Product\ApprovalStatusProductConcreteMergerPlugin;
use Spryker\Zed\ProductLabel\Communication\Plugin\Product\ProductLabelProductConcreteExpanderPlugin;
//...
class ProductDependencyProvider extends SprykerProductDependencyProvider
{
//...
/**
* @return array<\Spryker\Zed\ProductExtension\Dependency\Plugin\ProductConcreteExpanderPluginInterface>
*/
protected function getProductConcreteExpanderPlugins(): array
{
return [
...
new ProductReviewProductConcreteExpanderPlugin(),
new MerchantProductOfferProductConcreteExpanderPlugin(), # Marketplace only
new ProductConcreteCategoriesExpanderPlugin(),
new ProductLabelProductConcreteExpanderPlugin(),
];
}
/**
* @return array<\Spryker\Zed\ProductExtension\Dependency\Plugin\ProductConcreteMergerPluginInterface>
*/
protected function getProductConcreteMergerPlugins(): array
{
return [
new ImageSetProductConcreteMergerPlugin(),
new PriceProductConcreteMergerPlugin(),
new ApprovalStatusProductConcreteMergerPlugin(), # Add this plugin if you are using the spryker/product-approval module
];
}
//...
}
Configure queue dependencies in Zed
Add the following code to src/Pyz/Zed/Queue/QueueDependencyProvider.php
:
//...
use Spryker\Shared\SearchHttp\SearchHttpConfig;
//...
class QueueDependencyProvider extends SprykerDependencyProvider
{
//...
/**
* @param \Spryker\Zed\Kernel\Container $container
*
* @return array<\Spryker\Zed\Queue\Dependency\Plugin\QueueMessageProcessorPluginInterface>
*/
protected function getProcessorMessagePlugins(Container $container): array
{
return [
//...
SearchHttpConfig::SEARCH_HTTP_CONFIG_SYNC_QUEUE => new SynchronizationStorageQueueMessageProcessorPlugin(),
];
}
//...
}
Adjust Publisher configuration in Zed
Add the following code to src/Pyz/Zed/Publisher/PublisherDependencyProvider.php
:
//...
use Spryker\Zed\Product\Communication\Plugin\Publisher\ProductAbstractUpdatedMessageBrokerPublisherPlugin;
use Spryker\Zed\Product\Communication\Plugin\Publisher\ProductConcreteCreatedMessageBrokerPublisherPlugin;
use Spryker\Zed\Product\Communication\Plugin\Publisher\ProductConcreteDeletedMessageBrokerPublisherPlugin;
use Spryker\Zed\Product\Communication\Plugin\Publisher\ProductConcreteExportedMessageBrokerPublisherPlugin;
use Spryker\Zed\Product\Communication\Plugin\Publisher\ProductConcreteUpdatedMessageBrokerPublisherPlugin;
//...
class PublisherDependencyProvider extends SprykerPublisherDependencyProvider
{
//...
/**
* @return \Spryker\Zed\PublisherExtension\Dependency\Plugin\PublisherPluginInterface[]
*/
protected function getPublisherPlugins(): array
{
return array_merge(
//...
$this->getProductExportPlugins(),
);
}
/**
* @return array<\Spryker\Zed\PublisherExtension\Dependency\Plugin\PublisherPluginInterface>
*/
protected function getProductExportPlugins(): array
{
return [
new ProductConcreteExportedMessageBrokerPublisherPlugin(),
new ProductConcreteCreatedMessageBrokerPublisherPlugin(),
new ProductConcreteUpdatedMessageBrokerPublisherPlugin(),
new ProductConcreteDeletedMessageBrokerPublisherPlugin(),
new ProductAbstractUpdatedMessageBrokerPublisherPlugin(),
];
}
//...
}
Add SearchHttp configuration in Zed
Add the following code to src/Pyz/Zed/SearchHttp/SearchHttpConfig.php
:
<?php
namespace Pyz\Zed\SearchHttp;
use Pyz\Zed\Synchronization\SynchronizationConfig;
use Spryker\Zed\SearchHttp\SearchHttpConfig as SprykerSearchHttpConfig;
class SearchHttpConfig extends SprykerSearchHttpConfig
{
/**
* @return string|null
*/
public function getSearchHttpSynchronizationPoolName(): ?string
{
return SynchronizationConfig::DEFAULT_SYNCHRONIZATION_POOL_NAME;
}
}
Configure Synchronization dependencies in Zed
Add the following code to src/Pyz/Zed/Synchronization/SynchronizationDependencyProvider.php
:
//...
use Spryker\Zed\SearchHttp\Communication\Plugin\Synchronization\SearchHttpSynchronizationDataPlugin;
//...
class SynchronizationDependencyProvider extends SprykerSynchronizationDependencyProvider
{
//...
/**
* @return array<\Spryker\Zed\SynchronizationExtension\Dependency\Plugin\SynchronizationDataPluginInterface>
*/
protected function getSynchronizationDataPlugins(): array
{
return [
//...
new SearchHttpSynchronizationDataPlugin(),
];
}
//...
}
Console command for receiving messages
Receive messages from the channel:
console message-broker:consume
This command must be executed periodically. To achieve this, configure Jenkins in config/Zed/cronjobs/jenkins.php
:
$jobs[] = [
'name' => 'message-broker-consume-channels',
'command' => '$PHP_BIN vendor/bin/console message-broker:consume --time-limit=15 --sleep=5',
'schedule' => '* * * * *',
'enable' => true,
'stores' => $allStores,
];
Additional information on Algolia integration
When integrating Algolia, you should keep in mind some peculiarities of the SearchHTTP plugins setup and differences of the default facets.
SearchHTTP plugins setup
Spryker’s SearchHTTP
module transfers Glue search requests to external search providers, one of which is Algolia. The SearchHTTP
query is built using the QueryExpanderPlugin
classes. Their order is defined in the CatalogDependencyProvider::createCatalogSearchQueryExpanderPluginVariants()
method. The order of execution of those plugins might be customized on the project level. By default, all module-specific query builder plugins will be executed before parsing GET
query parameters, so any GET
query parameters may overwrite search query parameters set before.
Default facets differences
There is a difference in how default facets behave on Algolia and on the default Spryker installation using Elasticsearch. Some default Spryker facets like brand
only accept one value as a filter, so it is impossible to specify multiple brands to filter on in one search request. This is not the case with Algolia, as multiple brands can be specified in the same search requests. This also applies to other configured facets.
Next steps
Configure the Algolia app for your store.
Thank you!
For submitting the form