pimcore / personalized-product-search
Installs: 21
Dependents: 0
Suggesters: 0
Security: 0
Stars: 2
Watchers: 10
Forks: 1
Open Issues: 1
Type:pimcore-bundle
Requires
- elasticsearch/elasticsearch: ^6 | ^7
Requires (Dev)
- phpstan/phpstan: ^0.12.19
- symfony/phpunit-bridge: ^5.0
This package is auto-updated.
Last update: 2025-01-16 12:20:29 UTC
README
This bundle adds the ability to customize Elasticsearch queries searching for products using various personalization criteria, such as browsing history, purchase history of a customer and the customer's assignment to customer groups. The bundle is written in an extensible way so that adding personalization using a custom criteria can easily be achieved. Furthermore, the bundle is designed to be integrated into existing solutions as easily as possible.
For details about the architecture of this bundle please see the corresponding architecture wiki page.
Provided Functionality in a Nutshell
- Personalization of Elasticsearch queries using various criterias
- Predefined decorators for adapter execution and weighting
EqualWeightDecorator
: Each adapter is executed with the same weightPerformanceMeasurementDecorator
: The time it takes for an adapter to modify the query is measured and written to an Elasticsearch index
- Predefined adapters
SegmentAdapter
: Boosts queries based on the users browsing behaviourPurchaseHistoryAdapter
: Boosts queries based on the order history of an customerRelevantProductAdapter
: Boosts queries based on the assigned customer group
- Written in an extensible way so that custom adapters and decorators can easily be added
Usage
Install the bundle
- Install the ecommerce bundle
- Install the personalized search bundle like any other pimcore bundle
Define parameters
Following two parameters need to be defined in your application (default values are provided, but most probably not suitable):
personalized_search_index_prefix
- prefix for index names of ETL ES indices (default set topersonalized_search_
)es_hosts
- array of ES hosts (default set to['localhost']
)
Implement necessary interfaces
The personalized search bundles needs some application dependent data to work. Therefore some interfaces are provided which must be implemented. Additionally, those implementations must be available for dependency injection.
In service.yaml
following entries must be added (keep in mind that the implementations can vary).
ProductSegmentExtractorInterface
The implementation of ProductSegmentExtractorInterface
is responsible for extracting segments from a product.
A default implementation always returns an empty array, so this is an interface that you need to implement and overwrite:
Pimcore\Bundle\PersonalizedSearchBundle\ExtractTransformLoad\SegmentExtractor\ProductSegmentExtractorInterface: '@AppBundle\Ecommerce\IndexService\SegmentGetter' AppBundle\Personalization\FactoryOrderManagerProvider: ~ Pimcore\Bundle\PersonalizedSearchBundle\ExtractTransformLoad\PersonalizationOrderManagerProvider: '@AppBundle\Personalization\FactoryOrderManagerProvider'
PersonalizationOrderManagerProvider
Provides access to the order manager (part of the ecommerce bundle) as well as the customer class id. A default implementation is provided, you might need to overwrite it.
AppBundle\Personalization\FactoryOrderManagerProvider: ~ Pimcore\Bundle\PersonalizedSearchBundle\ExtractTransformLoad\PersonalizationOrderManagerProvider: '@AppBundle\Personalization\FactoryOrderManagerProvider'
namespace AppBundle\Personalization; use Pimcore\Bundle\EcommerceFrameworkBundle\Factory; use Pimcore\Bundle\EcommerceFrameworkBundle\OrderManager\OrderManagerInterface; use Pimcore\Bundle\PersonalizedSearchBundle\ExtractTransformLoad\PersonalizationOrderManagerProvider; use Pimcore\Model\DataObject\Customer; class FactoryOrderManagerProvider implements PersonalizationOrderManagerProvider { function getOrderManager(): OrderManagerInterface { return Factory::getInstance()->getOrderManager(); } function getCustomerClassId(): string { return Customer::classId(); } }
PersonalizationAdapterCustomerIdProvider
Used to retrieve the id for the currently logged in user. A default implementation is provided that reads the currently logged in user from the symfony security.
In the following example functionality provided by the ecommerce framework is used.
AppBundle\Personalization\FactoryPersonalizationAdapterCustomerIdProvider: ~ Pimcore\Bundle\PersonalizedSearchBundle\Customer\PersonalizationAdapterCustomerIdProvider: '@AppBundle\Personalization\FactoryPersonalizationAdapterCustomerIdProvider'
namespace AppBundle\Personalization; use Pimcore\Bundle\PersonalizedSearchBundle\Customer\PersonalizationAdapterCustomerIdProvider; use Pimcore\Bundle\EcommerceFrameworkBundle\Factory; class FactoryPersonalizationAdapterCustomerIdProvider implements PersonalizationAdapterCustomerIdProvider { public function getCustomerId(): int { return Factory::getInstance()->getEnvironment()->getCurrentUserId(); } }
Injecting adapters/decorators
Default adapters (SegmentAdapter
, PurchaseHistoryAdapter
and RelevantProductsAdapter
) and default decorators (EqualWeightDecorator
, PerformanceMeasurementDecorator
) that are shipped with the Personalization-Bundle are already available via dependency injection. Just make sure all necessary interfaces are implemented and available as explained above.
Here is an example of how the injection of adapters may look like:
public function searchAction(SegmentAdapter $segmentAdapter, PurchaseHistoryAdapter $purchaseHistoryAdapter, RelevantProductsAdapter $relevantProductAdapter) { // use injected adapters here (see next section) }
Personalizing queries
Adapters modify existing queries. The following example should give an overview of how decorators and adapters can be used:
$queryKey = 'searchTerm'; $personalizationDecorator = new EqualWeightDecorator(); $personalizationDecorator ->addAdapter($relevantProductAdapter) ->addAdapter($purchaseHistoryAdapter) ->addAdapter($segmentAdapter); $query = $personalizationDecorator->addPersonalization($query); $productListing->addQueryCondition($query, $queryKey); echo '<pre>'; print_r($personalizationDecorator->getDebugInfo()); echo '</pre>';
First, the used decorators and adapters need to be injected, which is not shown in the code snippet. Then a decorator is defined to manage the underlying adapters. Those can easily be added to the decorator using the addAdapter
method. After adding adapters addPersonalization
can be called for the decorator. It takes the Elasticsearch-query as a parameter. This method returns the modified, personalized, query which can then be executed.
For debug purposes, it might be interesting what the modifications (segments and boosting value for each adapter) looks like. For this case the method getDebugInfo
exists.
More detailed information can be found here (adapters) and here (decorators)
Using the ETL mechanism
For a detailed documentation of the ETL mechanism including detailed setup instructions please see the corresponding documentation page. The ETL mechanism can be invoked in 3 ways.
Programatically through interfaces which can be injected
Extract purchase history for all customers:
purchaseHistoryProvider->updateOrderIndexFromOrderDb()
Or just for a single customer:
purchaseHistoryProvider->fillOrderIndex(customer)
Updating the customer-group assignments:
customerGroupProvider->updateCustomerGroupAndSegmentsIndicesFromOrderDb()
Via executing a command on the command line
./bin/console personalizedsearch:start-etl ExampleArgument
The argument ExampleArgument
is optional. When no argument is given, the ETL is executed for purchase history and relevant products, otherwise only for the given one. The value for the argument can be PurchaseHistory
or CustomerGroup
.
Via creating cron job entries to automate the invocation
For running the whole ETL every hour, following entry has to be made in the cron tab:
* */1 * * * /home/pimcoredemo/www/bin/console personalizedsearch:start-etl >> /tmp/personalizedsearch-etl.log
For more details about the ETL and its usage see ETL.
Implementing a custom adapter
A custom adapter should extend AbstractAdapter
which implements the AdapterInterface
. There are two methods to implement: addPersonalization
and getDebugInfo
. The goal of both methods is to create information on how certain segments are boosted. In addPersonalization
this information is used to create so called function scores that are added to the query whereas in getDebugInfo
the boosting information is returned. For detailed information on what such an implementation could look like take a look at the default adapters and the more detailed adapter documentation.
Finally, the adapter should be added to services.yml
to make it available via dependency injection, just like the default ones. The new entry in the services.yml
should look something like this:
AppBundle\Personalization\Adapter\YourAdapter: ~
Detailed documentation
Load test and architecture
A detailed documentation for the architecture is available under the following link. As well you can find details for the load tests under this link.