Query and search API under the hood
Digital Factory 7 Search and Query API under the hood
#jahiaone
Benjamin Papež, QA Architect
Search and Query API under the hood
Overview on used search engine frameworks and API
Jahia's extensions to the Jackrabbit search engine
Maintenance tools and utilities
What's new in Digital Factory 7
Query tuning and general tips
Questions?
Overview on used search engine frameworks and API
Digital Factory: Search&query API under the hood
Jackrabbit extends Lucene
Improve performance for frequent index updates
Parse SQL-2, Xpath, QOM to create Lucene syntax queries
Support hierarchical queries
Support joins
Configure index rules, aggregates, analyzers…
Search and query tags / API in Digital Factory
Search tags / API
Site, document repository and subsection search using full text and simple metadata constraints
Returns list of hit objects (URL, excerpt, content node, score, references) sorted by score
HTML form creation via JSP tags. Predefined search result snippet.
OpenSearch and spellcheck support
Query tags / API
More flexible, supports all JCR 2.0 query functions (SQL-2, QOM, Xpath)
Returns iterator through found content nodes or row objects
No form creation. Provides some edit mode components.
Query tags and API
SQL-2, QOM based
Automatically handles multilingual content specifics
Supports faceting
No support for retrieving excerpts and spellcheck suggestions
Xpath based
No automatic multilingual content handling
No support for faceting
Support for retrieving excerpts (with result highlighting) and spellcheck suggestions
Jahia‘s extensions to the Jackrabbit search engine
Digital Factory: Search&query API under the hood
Handle multilingual content
Jahia‘s datamodel stores i18n properties in j:translation_* subnodes
Handle multilingual content
Indexing modifications
Non-i18n properties are copied down to fields of all translation node‘s Lucene documents
Some properties are excluded (j:locktoken, jcr:lockIsDeep, jcr:lockOwner, jcr:language, jcr:created, jcr:createdBy, jcr:lastModified, jcr:lastModifiedBy, j:lastPublished, j:lastPublishedBy, j:published)
Primary nodetype and mixin types of main node are also copied
Special fields in document: _:TRANSLATION_LANGUAGE and _:TRANSLATED_PARENT (_:PARENT field of translated node document points to grand-parent)
A change in the main node state automatically triggers re-indexing of translation nodes
Handle multilingual content
Query modifications
Adding current language constraint into queries (SQL-2, QOM)
Excluding translation nodes from child- or descendant-node queries
In SameNode or ChildNode join comparisons main and translation nodes are treated as they would be the same nodes (UUID)
In search results we filter out multiple results per node (due to translation node)
When sorting results, the position of the translation node overrules the position of the main node
Faceting support (SQL-2,QOM)
Integrate Apache Solr
SimpleJahiaJcrFacets based on Solr‘s SimpleFacets class
Reuse solr-common module and use exactly same facet query syntax and result objects as Solr does
facetable fields (set in CND) are prefixed with FACET: in Lucene document
Analyzed with KeywordAnalyzer
Date values are indexed with a Solr internal representation
For hierarchical fields (set in CND) we additionally add a _:FACET_HIERARCHY field with setting all the parent UUID nodes (implemented for categories, could be used for other hierarchies)
Faceting support
Facets can be added in Studio or Edit mode or even used programmatically
Count support
JCR specification does not support „select count“ 
In JCR there is just RangeIterator.getSize()
returns -1 when precise information is not available
Performance/memory overhead as all result objects are created and access rights checked
Digital Factory introduces a rep:count() function:
Query performance improvements
Access control checks
Visibility rule checks
Publication status check
Added special fields to index (_:ACL_UUID, _:CHECK_VISIBILITY, _:PUBLISHED)
Possible lazy loading of JCR nodes 
Same read rights and visibility rules are resolved only once 
Improved text extraction
Text extraction from documents is done asynchronously triggered by Quartz scheduler
Extracted text is stored as property in jnt:content node
Prevent recurring text extraction on re-indexing, publication, mark for deletion,…
Extract text from richtext field using Apache Tika (remove HTML markup)
Indexing related keywords in content definition (CND)    
Digital Factory Query JSP tags
jcr:sql
jcr:xpath
Digital Factory Query JSP tags
jcr:qom
Digital Factory Query REST API
Find servlet
http://www.jahia.com/community/documentation/jahia-modules/rest.html#Find
OpenSearch integration
http://www.jahia.com/community/documentation/jahia-modules/search/openSearch.html
Digital Factory Search JSP tags
<%@ taglib prefix="s" uri="http://www.jahia.org/tags/search" %>
…
<c:url value='${url.base}${currentNode.properties.result.node.path}.html'       var="searchUrl"/>
<s:form method="post" class="simplesearchform" action="${searchUrl}">
    <s:site value="${renderContext.site.name}" display="false"/>
    <s:pagePath value="${searchPath}" display="false" includeChildren="true"/>
    <s:nodeType value="${searchType}" display="false" />
    <s:language value="${renderContext.mainResource.locale}" display="false" />
    <s:term id="searchCustomTerm“ class="text-input“ value=“Search…”         match="all_words" searchIn="siteContent,tags"         onfocus="if(this.value==this.defaultValue)this.value='';"         onblur="if(this.value=='')this.value=this.defaultValue;"/>
    <s:nodeProperty nodeType="jmix:categorized" name="j:defaultCategory"                    id="searchCategories"/> 
    <input type="submit“ title=“Submit" value=""/>
</s:form>
Default search results (can be overridden):
modules\search\src\main\resources\jnt_searchResults\html\searchResults.jsp
Digital Factory Search API
RenderContext context = getContext();
CommaSeparatedMultipleValue oneSite = 
    new CommaSeparatedMultipleValue();
oneSite.setValue(“mySite”);
CommaSeparatedMultipleValue frenchLang = 
    new CommaSeparatedMultipleValue();
frenchLang.setValue("fr");
SearchCriteria criteria = new SearchCriteria();
criteria.setSites(oneSite);
criteria.setLanguages(frenchLang);
criteria.getTerms().get(0).setTerm("ACME");
criteria.getTerms().get(0).getFields().setSiteContent(true);
List<Hit<?>> hits = searchService.search(criteria, context).getResults();
Digital Factory Search API
Custom search result excerpts
Custom display (CSS class)
Custom excerpt if search term found in tags, category
Spellchecker (Did you mean)
Creates dictionary per site and language 
Used to suggest terms if current search produces too few results. 
Spellchecker is able to return more than one suggestion (configurable)
Limitation if multiple search terms are used and multiple have more than one suggestion, then only one suggestion is returned
Maintenance tools and utilities
Digital Factory: Search&query API under the hood
Maintenance tools
In the Digital Factory tools area there are some query/search related tools
JCR Query tool
Very useful to test queries and measure there performance
Possibility to run actions on the results (deletion, purge unused versions)
Search engine management
Check index consistency or trigger re-indexing
Document text extractor
Extract and show content from uploaded documents
Other query/search utilities
Auto-completion
Service offers auto-completion suggestions for typing in fields with short property values (users, categories, tags,…)
Auto-tagging
Simple solution used in Wise (parse document and automatically assign tags from an uploaded dictionary of tags)
Soon on the forge 
Apache Stanbol integration (for entity extraction, auto-tagging)
Digital Factory Crawler (using Nutch)
Similarity search, MoreLikeThis module
Synonym search module 
What‘s new in Digital Factory 7
Digital Factory: Search&query API under the hood
Framework version upgrades
Apache Jackrabbit (from 2.2.4 to 2.4.5)
Query related performance improvements and bug fixes 
Lucene native sort in SQL-2
Apache Lucene (from 2.4.1 to 3.0.3)
Performance improvement
Less memory usage (no huge byte arrays for norms)
Apache Solr (from 1.3.0 to 3.1.0)
Performance improvement 
New range facet type (not only date, also other kind of ranges)
Language dependent analyzers
Indexing of i18n full text properties is now done with the corresponding language specific Lucene analyzer (org.apache.lucene.analysis.*)
Default analyzers can be overridden or new ones added
Same language dependent analyzers are used to analyze full text search terms in queries
External data provider
Same query syntax and API is used to run queries on mapped external data provider
Results from different data providers are merged
Search also in extended external data
Other new features
Limit Spellchecker dictionary creation to content of particular properties
Store extracted text as binary instead of String
Before: huge String properties were always loaded to memory, but only used for indexing
JCR Query statistics in tools area
List slowest and most popular queries
Query tuning and general tips
Digital Factory: Search&query API under the hood
Ensure that queries perform
Test duration of queries with the „JCR Query tool“ using production data or a similar test environment
On Jahia xCM 6.6.x avoid SQL-2 queries with order by a property and a large result-set
Xpath sometimes perform better than SQL-2
Avoid that a query is executed too often (e.g. when using top-pager and bottom-pager we store query result in request)
Avoid cache.expiration=0 on components with queries
Make use of the „JCR Query statistic“ tool to identify slow or often used queries
Use available features
Use rep:count() instead of iterating through results
If you need a „select distinct“ feature (which is not supported in JCR), you could use faceting programmatically:
How to enable multiple search hits per page
Create a template for the nodetype, which should show as distinct search result
Or trigger URL modification rules
Escaping path and fulltext search terms
SQL-2
Xpath
Queries on multiple nodetypes
Rather (define and) use a common mixin
Use joins in SQL-2 / QOM
Add constraints on jcr:primaryType property
Questions?
Digital Factory: Search&query API under the hood
Stop searching, you found it Thank you for your attention #jahiaone
Digital Factory: Search&query API under the hood
 
     
     
     
     
     
     
     
     
     
     
     
     
     
     
     
     
     
     
     
     
     
     
     
     
     
     
     
     
     
     
     
     
     
     
     
     
     
    