Opticon Stockholm is on Tuesday September 10th, hope to see you there!
Opticon Stockholm is on Tuesday September 10th, hope to see you there!
Hi Anders,
I am not sure how dynamic attributes are built in your PIM but I think that there would be some of fixed properties and some of dynamic properties. I suggest that you can consider dynamic properties in PIM as a list of key and value and implement dynamic search based on that list.
Hi Binh
Thank you for your reply. I think probably Optimizely already have this dynamic list you are mentioning. But how do we index this list in Optimizely Search & Navigation?
Our attributes are normal attributes in PIM. We want our PIM system to be master for product information. So in PIM attributes are managed as they usually are.
Since Optimizely is not the master of product information, we would like Optimizely to reflect how data is modelled in PIM.
We import data into Optimizely using the ServiceAPI and import data using the Catalog.xml file format. This creates products and their attributes nicely in the Commerce subsystem. We also create the MetaFields and MetaClasses in the Commerce subsystem just fine. As far as I can see, this is also reflected in the Commerce Manager user interface. So Optimizely Commerce is workning fine with these attributes being managed in PIM.
I also think CMS is working well with our data. We can create the C# properties that are "fixed" (our webshop functionality depends on these properties being there), and we can access the rest of the attributes using the PropertyDataCollection through the Properties property on the ContentData base class. Using this PropertyDataCollection, we have a dynamic collection with all our attributes and corresponding metadata from the MetaFields in Commerce. We can use this dynamically on the webshop pages, where we need the dynamic functionality.
So I guess Optimizely provides all our attributes as a dynamic collection. How do we ensure that they are indexed and available for search? All the documentation I have found points towards inclusion or exclusion of C# properties. Do you know of any documentation or example that shows how we can define search fields without having them as C# attributes?
Thank you for your help.
Regards
Anders
Hi Manoj
We want to store the properties as regular Optimizely Commerce properties. We import using Catalog.xml, where we can specify both MetaClass and MetaFields and also set properties as specified from PIM. Using Catalog.xml we can add new MetaFields as we need them (we can't remove them again using Catalog.xml - that is OK).
So in the "Commerce" part, it will be real properties, and we can also see each property in the Commerce Manager UI.
But they are not directly available in the Content part, because we haven't created C# Properties. We can still access the information through the PropertyDataCollection, where they are also available as properties based on the MetaFields specified in Commerce.
But they are not indexed automatically, since indexing only automatically indexes C# Properties. And the Convention functionality also only works with C# properties, as far as I can see. The issue is probably, that C# Properties are defined on compile-time. Conventions are loaded on Commerce startup time (initialization) based on compile-time properties. But we want to index based on runtime information that is specified from PIM.
Regards
Anders
Hi Manoj
No, unfortunately our properties are not indexed. Optimizely only indexes fields that are avaiilable as C# Properties (or otherwise handled through the Conventions mechanizm.
We can add a Convention to include the GetProperties property in the index. But then it will be indexed as a 1 key in Search&Navigation with a serialized dictionary value. And we can't use that for facetting.
We are starting to reach the conclusion, that what we are trying to achieve is not possible in Optimizely. But it would be nice, if somebody can point me in the direction that it can somehow be done anyway :)
Hi Anders,
By default, properties without managed by code are not indexed because indexing job loads contents via IContentRepository as class object of Content Proxy Type Class. This object is serialized as json to send to indexing service. That is why these propeties are indexed by default.
The most simple way that I can think in your case is:
public class DynamicProperty
{
public string Name { get; set; }
public string StringValue { get; set; }
public int? IntValue { get; set; }
public double? DoubleValue { get; set; }
public DateTime? DateTimeValue { get; set; }
public bool? BoolValue { get; set;}
}
public static IList<DynamicProperty> SearchableProperties(this EntryContentBase content)
{
var searchableProperties = new List<DynamicProperty>();
foreach (var property in content.Property)
{
searchableProperties.Add(new DynamicProperty()
{
Name = property.Name,
IntValue = property.Value is int intVal ? intVal : null,
StringValue = property.Value is string stringVal ? stringVal : null,
DoubleValue = property.Value is double doubleVal ? doubleVal : null,
BoolValue = property.Value is bool boolVal ? boolVal : null,
DateTimeValue = property.Value is DateTime date ? date : null
});
}
return searchableProperties;
}
[ModuleDependency(typeof(InitializationModule))]
[ModuleDependency(typeof(ShellInitialization))]
[ModuleDependency(typeof(IndexingModule))]
public class FindInitialization : IConfigurableModule
{
public void ConfigureContainer(ServiceConfigurationContext context)
{
}
public void Initialize(InitializationEngine context)
{
if (context.HostType == HostType.WebApplication || context.HostType == HostType.TestFramework)
{
var client = ServiceProviderExtensions.GetInstance<IClient>(context.Locate.Advanced);
client.Conventions.ForInstancesOf<EntryContentBase>().IncludeField(x => x.SearchableProperties());
client.Conventions.NestedConventions.Add<EntryContentBase>(x => x.SearchableProperties());
}
}
public void Uninitialize(InitializationEngine context)
{
}
}
public static ITypeSearch<TSource> FilterByDynamicProperties<TSource>(this ITypeSearch<TSource> search,
string propertyName, object propertyValue) where TSource : EntryContentBase
{
if (string.IsNullOrWhiteSpace(propertyName))
return search;
if (propertyValue is int intVal)
{
search = search.Filter(x => x.SearchableProperties().MatchItem(y => y.Name.Match(propertyName) & y.IntValue.Match(intVal)));
}
else if (propertyValue is double doubleVal)
{
search = search.Filter(x => x.SearchableProperties().MatchItem(y => y.Name.Match(propertyName) & y.DoubleValue.Match(doubleVal)));
}
else if (propertyValue is bool boolVal)
{
search = search.Filter(x => x.SearchableProperties().MatchItem(y => y.Name.Match(propertyName) & y.BoolValue.Match(boolVal)));
}
else if (propertyValue is DateTime dateVal)
{
search = search.Filter(x => x.SearchableProperties().MatchItem(y => y.Name.Match(propertyName) & y.DateTimeValue.Match(dateVal)));
}
else if (propertyValue != null)
{
search = search.Filter(x => x.SearchableProperties().MatchItem(y => y.Name.Match(propertyName) & y.StringValue.Match(propertyValue.ToString())));
}
return search;
}
Hi Binh
Thank you very much for your suggestion. We will take a look at your code, and see if we can make it work. It looks promising.
Regards
Anders
Hi,
You can use terms facet for nested object like this:
search.TermsFacetFor(prod => prod.SearchableProperties(), item => item.StringValue, item => item.Name.Match(propertyName), action => action.Name = propertyName);
Hi Binh
Perfect. I think that was the last ting we needed to fix.
Thank you very much for your help.
Regards
Anders
Hi Commerce forum
In our new webshop, we are maintaining product data and structure in an external PIM system. Even though product structure can be dynamic, we still want to achieve a great search and navigation functionality, so we need to dynamically decide which product attributes to index.
I have created this topic in the Search & Navigation forum: https://world.optimizely.com/forum/developers-add-ons-forum/Search/Thread-Container/2024/8/dynamic-indexing-of-commerce-properties/. I'm also posting my question here, since it is also related to Commerce. I hope that is OK, and I suggest that questions, suggestions and help is given on the thread that I have linked to in the Search & Navigation forum.
Let me know if I have broken any rules by posting my question in this way - it was not my intention.
Regards
Anders