/*
* Hibernate Search, full-text search for your domain model
*
* License: GNU Lesser General Public License (LGPL), version 2.1 or later
* See the lgpl.txt file in the root directory or <http://www.gnu.org/licenses/lgpl-2.1.html>.
*/
package org.hibernate.search.test.embedded.depth;
import static org.junit.Assert.assertEquals;
import static org.junit.Assert.assertFalse;
import static org.junit.Assert.assertTrue;
import static org.junit.Assert.fail;
import java.io.IOException;
import java.io.Serializable;
import java.util.List;
import java.util.Map;
import org.apache.lucene.index.Term;
import org.apache.lucene.search.Query;
import org.apache.lucene.search.WildcardQuery;
import org.hibernate.Session;
import org.hibernate.Transaction;
import org.hibernate.search.FullTextSession;
import org.hibernate.search.Search;
import org.hibernate.search.backend.LuceneWork;
import org.hibernate.search.exception.SearchException;
import org.hibernate.search.query.dsl.QueryBuilder;
import org.hibernate.search.test.SearchTestBase;
import org.hibernate.search.testsupport.backend.LeakingLocalBackend;
import org.hibernate.testing.SkipForDialect;
import org.junit.After;
import org.junit.Before;
import org.junit.Test;
/**
* <pre>
* FAMILY: {@link org.hibernate.search.annotations.IndexedEmbedded#depth()} = 2
*
* 13 Philippa of Toulouse __________
* 12 William IX of Aquitaine _______|- 6 William X of Aquitaine __
* |
* 14 Aimery I of Châttellerault ____ |- 3 Eleanor of Aquitaine _________________________
* 15 Dangereuse de L'Isle Bouchard _|- 7 Aenor de Châtellerault __| |
* |
* |
* |-1 John of England
* 16 Fulk IV of Anjou ______________ |
* 17 Bertrade de Montfort __________|- 8 Fulk V of Anjou ________ |
* 9 Ermengarde of Maine ____|- 4 Geoffrey V of Anjou _ |
* 5 Empress Matilda _____|- 2 Henry II of England _|
* </pre>
*
* <pre>
* WORK: {@link org.hibernate.search.annotations.IndexedEmbedded#depth()#depth} = 1
*
* 20 Technical Manager____________
* 21 Leasing Manager _____________|-18 Real estate director _
* |
* 22 Financial Analyst ___________ |-1 John of England
* 23 Internal Audit Manager ______|-19 Financial Director ___|
* </pre>
* @author Davide D'Alto
* @author Sanne Grinovero
*/
@SkipForDialect(comment = "looks like a database deadlock", value = org.hibernate.dialect.SybaseASE15Dialect.class, jiraKey = "HSEARCH-1107")
public class WorkDoneOnEntitiesTest extends SearchTestBase {
@Test
public void testEmployeesIndexingInDepth() throws Exception {
List<WorkingPerson> result = search( "employees.name", "Technical Manager" );
assertEquals( "Unexpected number of results", 1, result.size() );
assertEquals(
"Should be able to index field inside depth and in path",
"Real estate director", result.get( 0 ).name
);
checkRawIndexFields();
}
@Test
public void testParentsIndexingInDepth() throws Exception {
List<WorkingPerson> result = search( "parents.parents.name", "Empress Matilda" );
assertEquals( "Unexpected number of results", 1, result.size() );
assertEquals(
"Should be able to index field inside depth and in path",
"John of England", result.get( 0 ).name
);
result = search( "parents.parents.parents.name", "Ermengarde of Maine" );
assertEquals( "Unexpected number of results", 1, result.size() );
checkRawIndexFields();
}
@Test
public void testNoWorkShouldBeExecutedOnPerson() throws Exception {
renamePerson( 17, "Montford" );
checkRawIndexFields();
assertEquals( 0, countWorksDoneOnPersonId( 1 ) );
}
@Test
public void testWorkShouldBeExecutedOnPerson() throws Exception {
renamePerson( 6, "William" );
checkRawIndexFields();
assertEquals( 1, countWorksDoneOnPersonId( 1 ) );
}
@Test
public void testNoWorkShouldBeExecutedOnEmployee() throws Exception {
renamePerson( 23, "LM" );
checkRawIndexFields();
assertEquals( 0, countWorksDoneOnPersonId( 1 ) );
}
@Test
public void testWorkShouldBeExecutedOnEmployee() throws Exception {
renamePerson( 19, "FM" );
checkRawIndexFields();
assertEquals( 1, countWorksDoneOnPersonId( 1 ) );
}
@Test
public void testShouldNotIndexParentsBeyondDepth() throws Exception {
try {
// fails only if DSL fails:
search( "parents.parents.parents.parents.name", "Bertrade de Montfort" );
fail( "Should not index a field if it is beyond the depth threshold" );
}
catch (SearchException e) {
}
checkRawIndexFields();
}
@Test
public void testShouldNotIndexBeyondMixedPathDepth() throws Exception {
try {
// fails only if DSL fails:
search( "parents.employees.employees.name", "Techincal Manager" );
fail( "Should not index a field if it is beyond the depth threshold, considering minimum depth along paths" );
}
catch (SearchException e) {
}
checkRawIndexFields();
}
@Test
public void testShouldNotIndexEmployeesBeyondDepth() throws Exception {
try {
search( "employees.employees.name", "Techincal Manager" );
fail( "Should not index a field if it is beyond the depth threshold" );
}
catch (SearchException e) {
}
checkRawIndexFields();
}
private void checkRawIndexFields() throws IOException {
// check raw index as well:
assertTrue( indexContainsField( "name" ) );
assertTrue( indexContainsField( "employees.name" ) );
assertTrue( indexContainsField( "parents.name" ) );
assertTrue( indexContainsField( "parents.parents.name" ) );
assertTrue( indexContainsField( "parents.parents.parents.name" ) );
assertTrue( indexContainsField( "parents.employees.name" ) );
assertTrue( indexContainsField( "parents.parents.employees.name" ) );
assertFalse( indexContainsField( "employees.employees.name" ) );
assertFalse( indexContainsField( "employees.parents.name" ) );
assertFalse( indexContainsField( "parents.employees.employees.name" ) );
assertFalse( indexContainsField( "parents.parents.parents.employees.name" ) );
}
private boolean indexContainsField(String fieldName) throws IOException {
return countDocuments( fieldName ) > 0;
}
@Override
@Before
public void setUp() throws Exception {
super.setUp();
Session session = openSession();
Transaction transaction = session.beginTransaction();
WorkingPerson[] ps = new WorkingPerson[27];
// array index starting from 1 to match ids of picture at http://en.wikipedia.org/wiki/John,_King_of_England
ps[1] = new WorkingPerson( 1, "John of England" );
ps[2] = new WorkingPerson( 2, "Henry II of England" );
ps[3] = new WorkingPerson( 3, "Eleanor of Aquitaine" );
ps[4] = new WorkingPerson( 4, "Geoffrey V of Anjou" );
ps[5] = new WorkingPerson( 5, "Empress Matilda" );
ps[6] = new WorkingPerson( 6, "William X of Aquitaine" );
ps[7] = new WorkingPerson( 7, "Aenor de Châtellerault" );
ps[8] = new WorkingPerson( 8, "Fulk V of Anjou" );
ps[9] = new WorkingPerson( 9, "Ermengarde of Maine" );
ps[10] = new WorkingPerson( 10, "Henry I of England" );
ps[11] = new WorkingPerson( 11, "Matilda of Scotland" );
ps[12] = new WorkingPerson( 12, "William IX of Aquitaine" );
ps[13] = new WorkingPerson( 13, "Philippa of Toulouse" );
ps[14] = new WorkingPerson( 14, "Aimery I of Châttellerault" );
ps[15] = new WorkingPerson( 15, "Dangereuse de L'Isle Bouchard" );
ps[16] = new WorkingPerson( 16, "Fulk IV of Anjou" );
ps[17] = new WorkingPerson( 17, "Bertrade de Montfort" );
ps[18] = new WorkingPerson( 18, "Real estate director" );
ps[19] = new WorkingPerson( 19, "Financial Director" );
ps[20] = new WorkingPerson( 20, "Technical Manager" );
ps[21] = new WorkingPerson( 21, "Leasing Manager" );
ps[22] = new WorkingPerson( 22, "Financial Analyst" );
ps[23] = new WorkingPerson( 23, "Internal Audit Manager" );
ps[24] = new WorkingPerson( 24, "Slave of Henry II" );
ps[25] = new WorkingPerson( 25, "Slave of Geoffrey V" );
ps[26] = new WorkingPerson( 26, "Assistant of Slave of Geoffrey V" );
ps[1].addParents( ps[2], ps[3] );
ps[2].addParents( ps[4], ps[5] );
ps[4].addParents( ps[8], ps[9] );
ps[8].addParents( ps[16], ps[17] );
ps[5].addParents( ps[10], ps[11] );
ps[3].addParents( ps[6], ps[7] );
ps[6].addParents( ps[12], ps[13] );
ps[7].addParents( ps[14], ps[15] );
ps[1].addEmployees( ps[18], ps[19] );
ps[2].addEmployees( ps[24] );
ps[5].addEmployees( ps[25] );
ps[25].addEmployees( ps[26] );
ps[18].addEmployees( ps[20], ps[21] );
ps[19].addEmployees( ps[22], ps[23] );
for ( int i = 1; i < ps.length; i++ ) {
session.save( ps[i] );
}
transaction.commit();
LeakingLocalBackend.reset();
}
@Override
@After
public void tearDown() throws Exception {
LeakingLocalBackend.reset();
super.tearDown();
}
private List<WorkingPerson> search(String field, String value) {
FullTextSession session = Search.getFullTextSession( getSession() );
@SuppressWarnings("unchecked")
List<WorkingPerson> result = session
.createFullTextQuery( searchQueryForValue( field, value, session ) )
.list();
return result;
}
private Query searchQueryForValue(String field, String value, FullTextSession session) {
QueryBuilder queryBuilder = session.getSearchFactory().buildQueryBuilder()
.forEntity( WorkingPerson.class ).get();
return queryBuilder.keyword().onField( field ).matching( value ).createQuery();
}
private int countDocuments(String field) {
FullTextSession session = Search.getFullTextSession( getSession() );
return session
/**
* We must use a raw Lucene query, and not use the QueryBuilder,
* because the field may not exist.
*/
.createFullTextQuery( new WildcardQuery( new Term( field, "*" ) ) )
.getResultSize();
}
private Query searchQueryWithWildcard(String field, FullTextSession session) {
QueryBuilder queryBuilder = session.getSearchFactory().buildQueryBuilder()
.forEntity( WorkingPerson.class ).get();
return queryBuilder.keyword().wildcard().onField( field ).matching( "*" ).createQuery();
}
private void renamePerson(Integer id, String newName) {
Transaction transaction = getSession().beginTransaction();
WorkingPerson person = getSession().load( WorkingPerson.class, id );
person.name = newName;
transaction.commit();
}
private int countWorksDoneOnPersonId(Integer pk) {
List<LuceneWork> processedQueue = LeakingLocalBackend.getLastProcessedQueue();
int count = 0;
for ( LuceneWork luceneWork : processedQueue ) {
Serializable id = luceneWork.getId();
if ( pk.equals( id ) ) {
count++;
}
}
return count;
}
@Override
public Class<?>[] getAnnotatedClasses() {
return new Class<?>[] { WorkingPerson.class };
}
@Override
public void configure(Map<String,Object> cfg) {
cfg.put( "hibernate.search.default.worker.backend", LeakingLocalBackend.class.getName() );
}
}