The first part of this tutorial described the low-level mechanisms of eZ Find, and the way eZ Publish attributes ( names, types ) and Solr fields are mapped. This post describes how eZ Find can dramatically ease development of some functionalities ( avoiding complex template operators and heavy SQL queries ) by automatically indexing additional fields in Solr when an object is published, which can in turn be leveraged for facetting or advanced filtering.
This tutorial requires to know how to set up eZ Find. The online documentation describes the required operation in details, there : http://ez.no/doc/extensions/ez_find/2_2.
You should also read and understand the first part of this tutorial :
http://share.ez.no/tutorials/ez-publish/advanced-development-with-ez-find-part-1-datatypes-in-solr-and-ez-find
The presented case has been chosen for its high educational value. It is a frequently requested feature, is relatively simple to implement and properly illustrates the underlying concept.
News websites, or blog post lists on blogs often propose to filter the content by year, or by a combination of year and month. Usually in this case, a template operator is developed which builds the appropriate SQL queries. This can quickly become complicated, and often has sharp limitations. Taking a look at the eZArchive operator helps understanding what I mean here.
Frequently, the complex SQL machinery required suffers from functional lacks, like access control propagation, language handling, unability to leverage some subtleties of MySQL or PostGreSql. The developer necessarily has to deal with these problematics, since the API is circumvented to directly write SQL statements. Another limitation of this approach is illustrated by the eZArchive operator : only the 'publication_date' parameter is taken into account, and no room is left for using a content-class-specific date attribute.
Let us see how we can develop a year and year & month filter with eZ Find.
eZ Find settings ( ezfind.ini, to be overridden in the ezfind.ini.append.php file of your extension ) allow for delegating the indexing process of an eZ Publish datatype to a given PHP class :
[SolrFieldMapSettings] CustomMap[ezdate]=ezfSolrDocumentFieldDate
Our PHP class, randomly named , ezfSolrDocumentFieldDate inherits from the ezfSolrDocumentFieldBase class, must be added to the /extension/myextension/classes folder and must inherit the following skeleton :
<?php class ezfSolrDocumentFieldDate extends ezfSolrDocumentFieldBase { public static function getFieldName( eZContentClassAttribute $classAttribute, $subAttribute = null, $context = 'search' ) { // return the fieldname like : attr_mydate_d } public function getData() { // return the array keys (fieldname => value), like : array('attr_mydate_dt' => '2010-04-30T00:00:00Z') } } ?>
This method is invoked the attributes names (within eZ Find) to Solr field names. For instance, when building a facet using the following syntax : 'mycontentclass/mydateattribute', this method receives 'mydateattribute' and should return 'attr_mydateattribute_dt'. Here is how we are going to implement the body of this method :
const DEFAULT_SUBATTRIBUTE_TYPE = 'date'; public static function getFieldName( eZContentClassAttribute $classAttribute, $subAttribute = null, $context = 'search' ) { switch ( $classAttribute->attribute( 'data_type_string' ) ) { case 'ezdate' : { if ( $subAttribute and $subAttribute !== '' ) { // A subattribute was passed return parent::generateSubattributeFieldName( $classAttribute, $subAttribute, self::DEFAULT_SUBATTRIBUTE_TYPE ); } else { // return the default field name here. return parent::generateAttributeFieldName( $classAttribute, self::getClassAttributeType( $classAttribute, null, $context ) ); } } break; default: {} break; } }
This method is invoked to extract data from eZ Publish, and prepare it prior to indexing in Solr. This method is the place to add additional fields like 'year' et 'yearmonth'. Once these new fields (subattributes) are indexed, we will leverage them by facetting or filtering on them, using the following, classical syntax:
public function getData() { $contentClassAttribute = $this->ContentObjectAttribute->attribute( 'contentclass_attribute' ); switch ( $contentClassAttribute->attribute( 'data_type_string' ) ) { case 'ezdate' : { $returnArray = array(); // Get timestamp attribute value $value = $this->ContentObjectAttribute->metaData(); // Generate the main filedName attr_XXX_dt $fieldName = parent::generateAttributeFieldName( $contentClassAttribute, self::DEFAULT_ATTRIBUTE_TYPE ); $returnArray[$fieldName] = parent::convertTimestampToDate( $value ); // Generate the yearmonth subattribute filedName subattr_year_dt $fieldName = parent::generateSubattributeFieldName( $contentClassAttribute, 'year', self::DEFAULT_SUBATTRIBUTE_TYPE ); $year = date("Y", $value); // Get Year value : 2010 $returnArray[$fieldName] = parent::convertTimestampToDate( strtotime($year.'-01-01') ); // Generate the yearmonth subattribute filedName subattr_yearmonth_dt $fieldName = parent::generateSubattributeFieldName( $contentClassAttribute, 'yearmonth', self::DEFAULT_SUBATTRIBUTE_TYPE ); $month = date("n", $value); // Get Month value : 3 $returnArray[$fieldName] = parent::convertTimestampToDate( strtotime($year.'-'.$month.'-01') ); return $returnArray; } break; default: {} break; } } }
Note:
$returnArray contains an associative array looking like this ( using var_dump ) :
array(3) { ["attr_date_dt"]=> string(24) "2008-12-28T00:00:00.000Z" ["subattr_date-year_dt"]=> string(24) "2008-01-01T00:00:00.000Z" ["subattr_date-yearmonth_dt"]=> string(24) "2008-12-01T00:00:00.000Z" }
Note :
Solr uses the ISO 8601 date format, of the form : '2010-04-30T00:00:00Z'. The parent class ( ezfSolrDocumentFieldBase ) exposes the convertTimestampToDate() methodm used to convert a timestamp to an ISO 8601 formatted date.
Our per-year and per year/month data are now available, indexed in Solr. Left is only to build the facet navigation, the usual way :
{def $search_yearmonth=fetch( ezfind, search, hash( 'query' , '', 'facet', array( hash('field', 'billet/date/year', 'sort', 'alpha', 'limit', 20 ), hash('field', 'billet/date/yearmonth', 'sort', 'alpha', 'limit', 20 ) ), 'class_id', array('billet'), 'subtree_array', array(2) ))} {def $search_extras_year=$search_yearmonth['SearchExtras'].facet_fields[0].nameList|reverse} {def $search_extras_yearmonth=$search_yearmonth['SearchExtras'].facet_fields[1].nameList|reverse} {def $date_count = 0 $date_ts = 0} <li id="blog_block_10" class="colonne_block"> <h1>Archives par années :</h1> <ul class="list {$current_css}"> {foreach $search_extras_year as $facetID => $datevalue} {set $date_count = $search_yearmonth['SearchExtras'].facet_fields[0].countList[$facetID]} {set $date_ts = $datevalue|strtotime} <li><a href={concat('/Blogs/(year)/',$date_ts|datetime( 'custom', '%Y' ))|ezurl} title="Archives : {$date_ts|datetime( 'custom', '%Y' )} // {$date_count} Billet(s)">{$date_ts|datetime( 'custom', '%Y' )}</a></li> {/foreach} </ul> </li> <li id="blog_block_11" class="colonne_block"> <h1>Archives par mois / années :</h1> <ul class="list {$current_css}"> {foreach $search_extras_yearmonth as $facetID => $datevalue} {set $date_count = $search_yearmonth['SearchExtras'].facet_fields[1].countList[$facetID]} {set $date_ts = $datevalue|strtotime} <li><a href={concat('/Blogs/(year)/',$date_ts|datetime( 'custom', '%Y' ),'/(month)/',$date_ts|datetime( 'custom', '%n' ))|ezurl} title="Archives : {$date_ts|datetime( 'custom', '%F %Y' )} //{$date_count} Billet(s)">{$date_ts|datetime( 'custom', '%F %Y' )|upfirst}</a></li> {/foreach} </ul> </li> {undef $date_ts $date_count $search_yearmonth $search_extras_yearmonth}
Note :
The presented fetch is quite basic, for an easier understanding of the mechanism. This code needs to be elaborated to achieve the expected functional result (be able to use filter for instance). This is however not this tutorial's purpose, the official documentation already giving all hints for doing this.
2nd Note :
The 'sort', 'alpha' statement does not actually specify an alphabetical sort. It rather helps specifying that no 'count' sort should occur (number of items matching a given facet). In this case, Solr automatically uses an 'increasing' sort, based on its index and the datatype of the concerned field (this explains the usage of the reverse operator to get an 'increasing' list).
This second post out of three describes how to add additional fields in Solr/Lucene's index when indexing content objects. This allows for :
I would like to thank Nicolas Pastorino for translating this tutorial to english, and Paul Borgermans for his availability.
This article is available for offline reading :
Gilles Guirand - Advanced development with eZ Find - part 2 - Indexing additional fields in Solr - PDF version
Gilles Guirand is a certified eZ Publish Developer. He is widely acknowledged by the community to be one of the national experts on highly technical and complex eZ Publish issues. With over 12 years experience in designing complex web architectures, he has been the driving force behind some of the most ambitious eZ Publish Projects: Web Site Generators, HighAvailability, Widgets, SOA, eZ Find, SSO, Web Accessibility and IT systems Integrations.
This work is licensed under the Creative Commons – Share Alike license ( http://creativecommons.org/licenses/by-sa/3.0 ).
Timing: | Jan 18 2025 00:06:48 |
Script start | |
Timing: | Jan 18 2025 00:06:48 |
Module start 'layout' | |
Timing: | Jan 18 2025 00:06:48 |
Module start 'content' | |
Warning: XML output handler: link | Jan 18 2025 00:06:48 |
Node #95436 doesn't exist | |
Warning: XML output handler: link | Jan 18 2025 00:06:48 |
Node #95436 doesn't exist | |
Timing: | Jan 18 2025 00:06:48 |
Module end 'content' | |
Timing: | Jan 18 2025 00:06:48 |
Script end |
Total runtime | 0.1658 sec |
Peak memory usage | 4,096.0000 KB |
Database Queries | 53 |
Checkpoint | Start (sec) | Duration (sec) | Memory at start (KB) | Memory used (KB) |
---|---|---|---|---|
Script start | 0.0000 | 0.0057 | 588.4063 | 152.6875 |
Module start 'layout' | 0.0057 | 0.0040 | 741.0938 | 39.5313 |
Module start 'content' | 0.0097 | 0.1548 | 780.6250 | 835.8906 |
Module end 'content' | 0.1644 | 0.0013 | 1,616.5156 | 24.4297 |
Script end | 0.1658 | 1,640.9453 |
Accumulator | Duration (sec) | Duration (%) | Count | Average (sec) |
---|---|---|---|---|
Ini load | ||||
Load cache | 0.0032 | 1.9187 | 16 | 0.0002 |
Check MTime | 0.0013 | 0.8019 | 16 | 0.0001 |
Mysql Total | ||||
Database connection | 0.0008 | 0.4854 | 1 | 0.0008 |
Mysqli_queries | 0.0483 | 29.1575 | 53 | 0.0009 |
Looping result | 0.0006 | 0.3763 | 50 | 0.0000 |
Template Total | 0.1357 | 81.8 | 2 | 0.0679 |
Template load | 0.0020 | 1.1965 | 2 | 0.0010 |
Template processing | 0.1337 | 80.6456 | 2 | 0.0669 |
Template load and register function | 0.0001 | 0.0687 | 1 | 0.0001 |
states | ||||
state_id_array | 0.0040 | 2.4364 | 6 | 0.0007 |
state_identifier_array | 0.0034 | 2.0719 | 7 | 0.0005 |
Override | ||||
Cache load | 0.0032 | 1.9183 | 177 | 0.0000 |
Sytem overhead | ||||
Fetch class attribute name | 0.0033 | 1.9649 | 6 | 0.0005 |
Fetch class attribute can translate value | 0.0000 | 0.0270 | 4 | 0.0000 |
class_abstraction | ||||
Instantiating content class attribute | 0.0000 | 0.0092 | 6 | 0.0000 |
XML | ||||
Image XML parsing | 0.0073 | 4.3923 | 4 | 0.0018 |
General | ||||
dbfile | 0.0068 | 4.1285 | 22 | 0.0003 |
String conversion | 0.0000 | 0.0059 | 4 | 0.0000 |
Note: percentages do not add up to 100% because some accumulators overlap |
Usage | Requested template | Template | Template loaded | Edit | Override |
---|---|---|---|---|---|
1 | node/view/full.tpl | full/article.tpl | extension/sevenx/design/simple/override/templates/full/article.tpl | ||
2 | content/datatype/view/ezxmltext.tpl | <No override> | extension/community_design/design/suncana/templates/content/datatype/view/ezxmltext.tpl | ||
11 | content/datatype/view/ezxmltags/header.tpl | <No override> | design/standard/templates/content/datatype/view/ezxmltags/header.tpl | ||
4 | content/datatype/view/ezxmltags/embed.tpl | <No override> | design/standard/templates/content/datatype/view/ezxmltags/embed.tpl | ||
4 | content/view/embed.tpl | embed/image.tpl | extension/sevenx/design/simple/override/templates/embed/image.tpl | ||
4 | content/datatype/view/ezimage.tpl | <No override> | extension/sevenx/design/simple/templates/content/datatype/view/ezimage.tpl | ||
20 | content/datatype/view/ezxmltags/link.tpl | <No override> | design/standard/templates/content/datatype/view/ezxmltags/link.tpl | ||
33 | content/datatype/view/ezxmltags/paragraph.tpl | <No override> | extension/ezwebin/design/ezwebin/templates/content/datatype/view/ezxmltags/paragraph.tpl | ||
13 | content/datatype/view/ezxmltags/separator.tpl | <No override> | extension/community_design/design/suncana/templates/content/datatype/view/ezxmltags/separator.tpl | ||
8 | content/datatype/view/ezxmltags/line.tpl | <No override> | design/standard/templates/content/datatype/view/ezxmltags/line.tpl | ||
6 | content/datatype/view/ezxmltags/newpage.tpl | <No override> | extension/community/design/standard/templates/content/datatype/view/ezxmltags/newpage.tpl | ||
6 | content/datatype/view/ezxmltags/literal.tpl | <No override> | extension/community/design/standard/templates/content/datatype/view/ezxmltags/literal.tpl | ||
16 | content/datatype/view/ezxmltags/strong.tpl | <No override> | design/standard/templates/content/datatype/view/ezxmltags/strong.tpl | ||
12 | content/datatype/view/ezxmltags/li.tpl | <No override> | design/standard/templates/content/datatype/view/ezxmltags/li.tpl | ||
4 | content/datatype/view/ezxmltags/ul.tpl | <No override> | design/standard/templates/content/datatype/view/ezxmltags/ul.tpl | ||
1 | content/datatype/view/ezxmltags/emphasize.tpl | <No override> | design/standard/templates/content/datatype/view/ezxmltags/emphasize.tpl | ||
1 | content/datatype/view/ezxmltags/embed-inline.tpl | <No override> | design/standard/templates/content/datatype/view/ezxmltags/embed-inline.tpl | ||
1 | content/view/embed-inline.tpl | <No override> | design/standard/templates/content/view/embed-inline.tpl | ||
1 | print_pagelayout.tpl | <No override> | extension/community/design/community/templates/print_pagelayout.tpl | ||
Number of times templates used: 148 Number of unique templates used: 19 |
Time used to render debug report: 0.0001 secs