Project

General

Profile

Download (24 KB) Statistics
| Branch: | Tag: | Revision:
1
// $Id$
2
/**
3
* Copyright (C) 2007 EDIT
4
* European Distributed Institute of Taxonomy
5
* http://www.e-taxonomy.eu
6
*
7
* The contents of this file are subject to the Mozilla Public License Version 1.1
8
* See LICENSE.TXT at the top of this package for the full license terms.
9
*/
10

    
11
package eu.etaxonomy.cdm.api.service;
12

    
13
import java.util.ArrayList;
14
import java.util.Arrays;
15
import java.util.HashMap;
16
import java.util.HashSet;
17
import java.util.List;
18
import java.util.Map;
19
import java.util.Set;
20
import java.util.UUID;
21

    
22
import org.apache.log4j.Logger;
23
import org.hibernate.criterion.Criterion;
24
import org.springframework.transaction.annotation.Transactional;
25

    
26
import eu.etaxonomy.cdm.api.service.config.IIdentifiableEntityServiceConfigurator;
27
import eu.etaxonomy.cdm.api.service.dto.FindByIdentifierDTO;
28
import eu.etaxonomy.cdm.api.service.dto.FindByMarkerDTO;
29
import eu.etaxonomy.cdm.api.service.pager.Pager;
30
import eu.etaxonomy.cdm.api.service.pager.impl.DefaultPagerImpl;
31
import eu.etaxonomy.cdm.common.monitor.DefaultProgressMonitor;
32
import eu.etaxonomy.cdm.common.monitor.IProgressMonitor;
33
import eu.etaxonomy.cdm.hibernate.HibernateProxyHelper;
34
import eu.etaxonomy.cdm.model.common.CdmBase;
35
import eu.etaxonomy.cdm.model.common.DefinedTerm;
36
import eu.etaxonomy.cdm.model.common.ISourceable;
37
import eu.etaxonomy.cdm.model.common.IdentifiableEntity;
38
import eu.etaxonomy.cdm.model.common.IdentifiableSource;
39
import eu.etaxonomy.cdm.model.common.LSID;
40
import eu.etaxonomy.cdm.model.common.MarkerType;
41
import eu.etaxonomy.cdm.model.media.Rights;
42
import eu.etaxonomy.cdm.model.name.NonViralName;
43
import eu.etaxonomy.cdm.model.reference.Reference;
44
import eu.etaxonomy.cdm.model.reference.ReferenceFactory;
45
import eu.etaxonomy.cdm.persistence.dao.common.IIdentifiableDao;
46
import eu.etaxonomy.cdm.persistence.dao.hibernate.HibernateBeanInitializer;
47
import eu.etaxonomy.cdm.persistence.dao.initializer.AutoPropertyInitializer;
48
import eu.etaxonomy.cdm.persistence.dto.UuidAndTitleCache;
49
import eu.etaxonomy.cdm.persistence.query.MatchMode;
50
import eu.etaxonomy.cdm.persistence.query.OrderHint;
51
import eu.etaxonomy.cdm.persistence.query.OrderHint.SortOrder;
52
import eu.etaxonomy.cdm.strategy.cache.common.IIdentifiableEntityCacheStrategy;
53
import eu.etaxonomy.cdm.strategy.match.DefaultMatchStrategy;
54
import eu.etaxonomy.cdm.strategy.match.IMatchStrategy;
55
import eu.etaxonomy.cdm.strategy.match.IMatchable;
56
import eu.etaxonomy.cdm.strategy.match.MatchException;
57
import eu.etaxonomy.cdm.strategy.merge.IMergable;
58
import eu.etaxonomy.cdm.strategy.merge.IMergeStrategy;
59
import eu.etaxonomy.cdm.strategy.merge.MergeException;
60

    
61
public abstract class IdentifiableServiceBase<T extends IdentifiableEntity, DAO extends IIdentifiableDao<T>> extends AnnotatableServiceBase<T,DAO>
62
						implements IIdentifiableEntityService<T>{
63

    
64

    
65
	protected static final int UPDATE_TITLE_CACHE_DEFAULT_STEP_SIZE = 1000;
66
	protected static final  Logger logger = Logger.getLogger(IdentifiableServiceBase.class);
67

    
68
	@Override
69
	@Transactional(readOnly = true)
70
	public Pager<Rights> getRights(T t, Integer pageSize, Integer pageNumber, List<String> propertyPaths) {
71
        Integer numberOfResults = dao.countRights(t);
72

    
73
		List<Rights> results = new ArrayList<Rights>();
74
		if(numberOfResults > 0) { // no point checking again //TODO use AbstractPagerImpl.hasResultsInRange(numberOfResults, pageNumber, pageSize)
75
			results = dao.getRights(t, pageSize, pageNumber,propertyPaths);
76
		}
77

    
78
		return new DefaultPagerImpl<Rights>(pageNumber, numberOfResults, pageSize, results);
79
	}
80

    
81
	@Override
82
	@Transactional(readOnly = true)
83
	public Pager<IdentifiableSource> getSources(T t, Integer pageSize, Integer pageNumber, List<String> propertyPaths) {
84
		 Integer numberOfResults = dao.countSources(t);
85

    
86
		 List<IdentifiableSource> results = new ArrayList<IdentifiableSource>();
87
		 if(numberOfResults > 0) { // no point checking again //TODO use AbstractPagerImpl.hasResultsInRange(numberOfResults, pageNumber, pageSize)
88
			 results = dao.getSources(t, pageSize, pageNumber,propertyPaths);
89
		 }
90

    
91
		 return new DefaultPagerImpl<IdentifiableSource>(pageNumber, numberOfResults, pageSize, results);
92
	}
93

    
94

    
95
	@Transactional(readOnly = false)
96
	@Override
97
	public T replace(T x, T y) {
98
		return dao.replace(x, y);
99
	}
100
	/**
101
	 * FIXME Candidate for harmonization
102
	 * Given that this method is strongly typed, and generic, could we not simply expose it as
103
	 * List<T> findByTitle(String title) as it is somewhat less cumbersome. Admittedly, I don't
104
	 * understand what is going on with the configurators etc. so maybe there is a good reason for
105
	 * the design of this method.
106
	 * @param title
107
	 * @return
108
	 */
109
	@Transactional(readOnly = true)
110
	protected List<T> findCdmObjectsByTitle(String title){
111
		return ((IIdentifiableDao)dao).findByTitle(title);
112
	}
113

    
114
	@Transactional(readOnly = true)
115
	protected List<T> findCdmObjectsByTitle(String title, Class<T> clazz){
116
		return ((IIdentifiableDao)dao).findByTitleAndClass(title, clazz);
117
	}
118
	@Transactional(readOnly = true)
119
	protected List<T> findCdmObjectsByTitle(String title, CdmBase sessionObject){
120
		return ((IIdentifiableDao)dao).findByTitle(title, sessionObject);
121
	}
122

    
123
	/*
124
	 * TODO - Migrated from CommonServiceBase
125
	 *  (non-Javadoc)
126
	 * @see eu.etaxonomy.cdm.api.service.ICommonService#getSourcedObjectById(java.lang.String, java.lang.String)
127
	 */
128
	@Transactional(readOnly = true)
129
	@Override
130
	public ISourceable getSourcedObjectByIdInSource(Class clazz, String idInSource, String idNamespace) {
131
		ISourceable result = null;
132

    
133
		List<T> list = dao.findOriginalSourceByIdInSource(idInSource, idNamespace);
134
		if (! list.isEmpty()){
135
			result = list.get(0);
136
		}
137
		return result;
138
	}
139
//
140
//	@Override
141
//    public List<UuidAndTitleCache<T>> getUuidAndTitleCache() {
142
//	    return getUuidAndTitleCache(null, null);
143
//	}
144

    
145

    
146
	@Transactional(readOnly = true)
147
	@Override
148
	public List<UuidAndTitleCache<T>> getUuidAndTitleCache(Integer limit, String pattern) {
149
		return dao.getUuidAndTitleCache(limit, pattern);
150
	}
151

    
152
	@Transactional(readOnly = true)
153
	@Override
154
	public Pager<T> findByTitle(Class<? extends T> clazz, String queryString,MatchMode matchmode, List<Criterion> criteria, Integer pageSize, Integer pageNumber, List<OrderHint> orderHints, List<String> propertyPaths) {
155
		 Integer numberOfResults = dao.countByTitle(clazz, queryString, matchmode, criteria);
156

    
157
		 List<T> results = new ArrayList<T>();
158
		 if(numberOfResults > 0) { // no point checking again //TODO use AbstractPagerImpl.hasResultsInRange(numberOfResults, pageNumber, pageSize)
159
				results = dao.findByTitle(clazz, queryString, matchmode, criteria, pageSize, pageNumber, orderHints, propertyPaths);
160
		 }
161

    
162
		  return new DefaultPagerImpl<T>(pageNumber, numberOfResults, pageSize, results);
163
	}
164

    
165
	@Transactional(readOnly = true)
166
	@Override
167
	public Pager<T> findByTitle(IIdentifiableEntityServiceConfigurator<T> config){
168
		return findByTitle(config.getClazz(), config.getTitleSearchStringSqlized(), config.getMatchMode(), config.getCriteria(), config.getPageSize(), config.getPageNumber(), config.getOrderHints(), config.getPropertyPaths());
169
	}
170

    
171
	@Transactional(readOnly = true)
172
	@Override
173
	public List<T> listByTitle(Class<? extends T> clazz, String queryString,MatchMode matchmode, List<Criterion> criteria, Integer pageSize, Integer pageNumber, List<OrderHint> orderHints, List<String> propertyPaths) {
174
		 Integer numberOfResults = dao.countByTitle(clazz, queryString, matchmode, criteria);
175

    
176
		 List<T> results = new ArrayList<T>();
177
		 if(numberOfResults > 0) { // no point checking again //TODO use AbstractPagerImpl.hasResultsInRange(numberOfResults, pageNumber, pageSize)
178
				results = dao.findByTitle(clazz, queryString, matchmode, criteria, pageSize, pageNumber, orderHints, propertyPaths);
179
		 }
180
		 return results;
181
	}
182

    
183
	@Transactional(readOnly = true)
184
	@Override
185
	public Pager<T> findTitleCache(Class<? extends T> clazz, String queryString, Integer pageSize, Integer pageNumber, List<OrderHint> orderHints, MatchMode matchMode){
186
		long numberOfResults = dao.countTitleCache(clazz, queryString, matchMode);
187

    
188
		 List<T> results = new ArrayList<T>();
189
		 if(numberOfResults > 0) { // no point checking again //TODO use AbstractPagerImpl.hasResultsInRange(numberOfResults, pageNumber, pageSize)
190
				results = dao.findTitleCache(clazz, queryString, pageSize, pageNumber, orderHints, matchMode);
191
		 }
192
		 int r = 0;
193
		 r += numberOfResults;
194

    
195
		  return new DefaultPagerImpl<T>(pageNumber, r , pageSize, results);
196
	}
197

    
198
	@Transactional(readOnly = true)
199
	@Override
200
	public List<T> listByReferenceTitle(Class<? extends T> clazz, String queryString,MatchMode matchmode, List<Criterion> criteria, Integer pageSize, Integer pageNumber, List<OrderHint> orderHints, List<String> propertyPaths) {
201
		 Integer numberOfResults = dao.countByReferenceTitle(clazz, queryString, matchmode, criteria);
202

    
203
		 List<T> results = new ArrayList<T>();
204
		 if(numberOfResults > 0) { // no point checking again //TODO use AbstractPagerImpl.hasResultsInRange(numberOfResults, pageNumber, pageSize)
205
				results = dao.findByReferenceTitle(clazz, queryString, matchmode, criteria, pageSize, pageNumber, orderHints, propertyPaths);
206
		 }
207
		 return results;
208
	}
209

    
210
	@Transactional(readOnly = true)
211
	@Override
212
	public T find(LSID lsid) {
213
		return dao.find(lsid);
214
	}
215

    
216
	@Transactional(readOnly = true)
217
	@Override
218
	public Pager<T> search(Class<? extends T> clazz, String queryString, Integer pageSize, Integer pageNumber, List<OrderHint> orderHints, List<String> propertyPaths) {
219
        Integer numberOfResults = dao.count(clazz,queryString);
220

    
221
		List<T> results = new ArrayList<T>();
222
		if(numberOfResults > 0) { // no point checking again //TODO use AbstractPagerImpl.hasResultsInRange(numberOfResults, pageNumber, pageSize)
223
			results = dao.search(clazz,queryString, pageSize, pageNumber, orderHints, propertyPaths);
224
		}
225

    
226
		return new DefaultPagerImpl<T>(pageNumber, numberOfResults, pageSize, results);
227
	}
228

    
229
	@Override
230
	@Transactional(readOnly = false)
231
	public void updateTitleCache() {
232
		updateTitleCache(null, null, null, null);
233
	}
234

    
235
	@Transactional(readOnly = false)  //TODO check transactional behaviour, e.g. what happens with the session if count is very large
236
	protected <S extends T > void updateTitleCacheImpl(Class<S> clazz, Integer stepSize, IIdentifiableEntityCacheStrategy<T> cacheStrategy, IProgressMonitor monitor) {
237
		if (stepSize == null){
238
			stepSize = UPDATE_TITLE_CACHE_DEFAULT_STEP_SIZE;
239
		}
240
		if (monitor == null){
241
			monitor = DefaultProgressMonitor.NewInstance();
242
		}
243

    
244
		int count = dao.count(clazz);
245
		monitor.beginTask("update titles", count);
246
		int worked = 0;
247
		for(int i = 0 ; i < count ; i = i + stepSize){
248
			// not sure if such strict ordering is necessary here, but for safety reasons I do it
249
			ArrayList<OrderHint> orderHints = new ArrayList<OrderHint>();
250
			orderHints.add( new OrderHint("id", OrderHint.SortOrder.ASCENDING));
251

    
252

    
253
			Map<Class<? extends CdmBase>, AutoPropertyInitializer<CdmBase>> oldAutoInit = switchOfAutoinitializer();
254
			List<S> list = this.list(clazz, stepSize, i, orderHints, null);
255
			switchOnOldAutoInitializer(oldAutoInit);
256

    
257
			List<T> entitiesToUpdate = new ArrayList<T>();
258
			for (T entity : list){
259
				HibernateProxyHelper.deproxy(entity, clazz);
260
				if (entity.isProtectedTitleCache() == false){
261
					updateTitleCacheForSingleEntity(cacheStrategy, entitiesToUpdate, entity);
262
				}
263
				worked++;
264
			}
265
			for (T entity: entitiesToUpdate){
266
				if (entity.getTitleCache() != null){
267
					//System.err.println(entity.getTitleCache());
268
				}else{
269
					//System.err.println("no titleCache" + ((NonViralName)entity).getNameCache());
270
				}
271
			}
272
			saveOrUpdate(entitiesToUpdate);
273
			monitor.worked(list.size());
274
			if (monitor.isCanceled()){
275
				monitor.done();
276
				return;
277
			}
278
		}
279
		monitor.done();
280
	}
281

    
282
	/**
283
	 * Brings back all auto initializers to the bean initializer
284
	 * @see #switchOfAutoinitializer()
285
	 * @param oldAutoInit
286
	 */
287
	protected void switchOnOldAutoInitializer(
288
			Map<Class<? extends CdmBase>, AutoPropertyInitializer<CdmBase>> oldAutoInit) {
289
		HibernateBeanInitializer initializer = (HibernateBeanInitializer)this.appContext.getBean("defaultBeanInitializer");
290
		initializer.setBeanAutoInitializers(oldAutoInit);
291
	}
292

    
293
	/**
294
	 * Removes all auto initializers from the bean initializer
295
	 *
296
	 * @see #switchOnOldAutoInitializer(Map)
297
	 * @return
298
	 */
299
	protected Map<Class<? extends CdmBase>, AutoPropertyInitializer<CdmBase>> switchOfAutoinitializer() {
300
		HibernateBeanInitializer initializer = (HibernateBeanInitializer)this.appContext.getBean("defaultBeanInitializer");
301
		Map<Class<? extends CdmBase>, AutoPropertyInitializer<CdmBase>> oldAutoInitializers = initializer.getBeanAutoInitializers();
302
		Map<Class<? extends CdmBase>, AutoPropertyInitializer<CdmBase>> map = new HashMap<Class<? extends CdmBase>, AutoPropertyInitializer<CdmBase>>();
303
		initializer.setBeanAutoInitializers(map);
304
		return oldAutoInitializers;
305
	}
306

    
307
	/**
308
	 * @param cacheStrategy
309
	 * @param entitiesToUpdate
310
	 * @param entity
311
	 */
312
	/**
313
	 * @param cacheStrategy
314
	 * @param entitiesToUpdate
315
	 * @param entity
316
	 */
317
	@SuppressWarnings("unchecked")
318
	private void updateTitleCacheForSingleEntity(
319
			IIdentifiableEntityCacheStrategy<T> cacheStrategy,
320
			List<T> entitiesToUpdate, T entity) {
321

    
322
		//assert (entity.isProtectedTitleCache() == false );
323

    
324
		//exclude recursive inreferences
325
		if (entity.isInstanceOf(Reference.class)){
326
			Reference ref = CdmBase.deproxy(entity, Reference.class);
327
			if (ref.getInReference() != null && ref.getInReference().equals(ref)){
328
				return;
329
			}
330
		}
331

    
332

    
333
		//define the correct cache strategy
334
		IIdentifiableEntityCacheStrategy entityCacheStrategy = cacheStrategy;
335
		if (entityCacheStrategy == null){
336
			entityCacheStrategy = entity.getCacheStrategy();
337
			//FIXME find out why the wrong cache strategy is loaded here, see #1876
338
			if (entity instanceof Reference){
339
				entityCacheStrategy = ReferenceFactory.newReference(((Reference)entity).getType()).getCacheStrategy();
340
			}
341
		}
342

    
343

    
344

    
345
		//old titleCache
346
		entity.setProtectedTitleCache(true);
347

    
348
		String oldTitleCache = entity.getTitleCache();
349
		entity.setTitleCache(oldTitleCache, false);   //before we had entity.setProtectedTitleCache(false) but this deleted the titleCache itself
350
		entity.setCacheStrategy(entityCacheStrategy);
351
		//NonViralNames and Reference have more caches //TODO handle in NameService
352
		String oldNameCache = null;
353
		String oldFullTitleCache = null;
354
		String oldAbbrevTitleCache = null;
355
		if (entity instanceof NonViralName ){
356

    
357
			try{
358
				NonViralName<?> nvn = (NonViralName) entity;
359
				if (!nvn.isProtectedNameCache()){
360
					nvn.setProtectedNameCache(true);
361
					oldNameCache = nvn.getNameCache();
362
					nvn.setProtectedNameCache(false);
363
				}
364
				if (!nvn.isProtectedFullTitleCache()){
365
					nvn.setProtectedFullTitleCache(true);
366
					oldFullTitleCache = nvn.getFullTitleCache();
367
					nvn.setProtectedFullTitleCache(false);
368
				}
369
			}catch(ClassCastException e){
370
				System.out.println("entity: " + entity.getTitleCache());
371
			}
372

    
373
		}else if (entity instanceof Reference){
374
			Reference ref = (Reference) entity;
375
			if (!ref.isProtectedAbbrevTitleCache()){
376
				ref.setProtectedAbbrevTitleCache(true);
377
				oldAbbrevTitleCache = ref.getAbbrevTitleCache();
378
				ref.setProtectedAbbrevTitleCache(false);
379
			}
380
		}
381
		setOtherCachesNull(entity);
382
		String newTitleCache= null;
383
		NonViralName<?> nvn = null;//TODO find better solution
384
		try{
385
			if (entity instanceof NonViralName){
386
				nvn = (NonViralName) entity;
387
				newTitleCache = entityCacheStrategy.getTitleCache(nvn);
388
			} else{
389
				 newTitleCache = entityCacheStrategy.getTitleCache(entity);
390
			}
391
		}catch (ClassCastException e){
392
			nvn = HibernateProxyHelper.deproxy(entity, NonViralName.class);
393
			newTitleCache = entityCacheStrategy.getTitleCache(nvn);
394
			//System.out.println("titleCache: " +entity.getTitleCache());
395
		}
396

    
397
		if ( oldTitleCache == null   || oldTitleCache != null && ! oldTitleCache.equals(newTitleCache) ){
398
			entity.setTitleCache(null, false);
399
			String newCache = entity.getTitleCache();
400

    
401
			if (newCache == null){
402
				logger.warn("newCache should never be null");
403
			}
404
			if (oldTitleCache == null){
405
				logger.info("oldTitleCache should never be null");
406
			}
407
			if (nvn != null){
408
				//NonViralName<?> nvn = (NonViralName) entity;
409
				nvn.getNameCache();
410
				nvn.getFullTitleCache();
411
			}
412
			if (entity instanceof Reference){
413
				Reference ref = (Reference) entity;
414
				ref.getAbbrevTitleCache();
415
			}
416
			entitiesToUpdate.add(entity);
417
		}else if (nvn != null){
418
			//NonViralName<?> nvn = (NonViralName) entity;
419
			String newNameCache = nvn.getNameCache();
420
			String newFullTitleCache = nvn.getFullTitleCache();
421
			if ((oldNameCache == null && !nvn.isProtectedNameCache()) || (oldNameCache != null && !oldNameCache.equals(newNameCache))){
422
				entitiesToUpdate.add(entity);
423
			}else if ((oldFullTitleCache == null && !nvn.isProtectedFullTitleCache()) || (oldFullTitleCache != null && !oldFullTitleCache.equals(newFullTitleCache))){
424
				entitiesToUpdate.add(entity);
425
			}
426
		}else if (entity instanceof Reference){
427
			Reference ref = (Reference) entity;
428
			String newAbbrevTitleCache = ref.getAbbrevTitleCache();
429
			if ( (oldAbbrevTitleCache == null && !ref.isProtectedAbbrevTitleCache() ) || (oldAbbrevTitleCache != null && !oldAbbrevTitleCache.equals(newAbbrevTitleCache))){
430
				entitiesToUpdate.add(entity);
431
			}
432
		}
433

    
434

    
435
	}
436

    
437

    
438

    
439
	/**
440
	 * Needs override if not only the title cache should be set to null to
441
	 * generate the correct new title cache
442
	 */
443
	protected void setOtherCachesNull(T entity) {
444
		return;
445
	}
446

    
447

    
448

    
449
	private class DeduplicateState{
450
		String lastTitleCache;
451
		Integer pageSize = 50;
452
		int nPages = 3;
453
		int startPage = 0;
454
		boolean isCompleted = false;
455
		int result;
456
	}
457

    
458
	@Override
459
	@Transactional(readOnly = false)
460
	public int deduplicate(Class<? extends T> clazz, IMatchStrategy matchStrategy, IMergeStrategy mergeStrategy) {
461
		DeduplicateState dedupState = new DeduplicateState();
462

    
463
		if (clazz == null){
464
			logger.warn("Deduplication clazz must not be null!");
465
			return 0;
466
		}
467
		if (! ( IMatchable.class.isAssignableFrom(clazz) && IMergable.class.isAssignableFrom(clazz) )  ){
468
			logger.warn("Deduplication implemented only for classes implementing IMatchable and IMergeable. No deduplication performed!");
469
			return 0;
470
		}
471
		Class matchableClass = clazz;
472
		if (matchStrategy == null){
473
			matchStrategy = DefaultMatchStrategy.NewInstance(matchableClass);
474
		}
475
		List<T> nextGroup = new ArrayList<T>();
476

    
477
		int result = 0;
478
//		double countTotal = count(clazz);
479
//
480
//		Number countPagesN = Math.ceil(countTotal/dedupState.pageSize.doubleValue()) ;
481
//		int countPages = countPagesN.intValue();
482
//
483

    
484
		List<OrderHint> orderHints = Arrays.asList(new OrderHint[]{new OrderHint("titleCache", SortOrder.ASCENDING)});
485

    
486
		while (! dedupState.isCompleted){
487
			//get x page sizes
488
			List<T> objectList = getPages(clazz, dedupState, orderHints);
489
			//after each page check if any changes took place
490
			int nUnEqualPages = handleAllPages(objectList, dedupState, nextGroup, matchStrategy, mergeStrategy);
491
			nUnEqualPages = nUnEqualPages + dedupState.pageSize * dedupState.startPage;
492
			//refresh start page counter
493
			int finishedPages = nUnEqualPages / dedupState.pageSize;
494
			dedupState.startPage = finishedPages;
495
		}
496

    
497
		result += handleLastGroup(nextGroup, matchStrategy, mergeStrategy);
498
		return result;
499
	}
500

    
501

    
502
	private int handleAllPages(List<T> objectList, DeduplicateState dedupState, List<T> nextGroup, IMatchStrategy matchStrategy, IMergeStrategy mergeStrategy) {
503
		int nUnEqual = 0;
504
		for (T object : objectList){
505
			String currentTitleCache = object.getTitleCache();
506
			if (currentTitleCache != null && currentTitleCache.equals(dedupState.lastTitleCache)){
507
				//=titleCache
508
				nextGroup.add(object);
509
			}else{
510
				//<> titleCache
511
				dedupState.result += handleLastGroup(nextGroup, matchStrategy, mergeStrategy);
512
				nextGroup = new ArrayList<T>();
513
				nextGroup.add(object);
514
				nUnEqual++;
515
			}
516
			dedupState.lastTitleCache = currentTitleCache;
517
		}
518
		handleLastGroup(nextGroup, matchStrategy, mergeStrategy);
519
		return nUnEqual;
520
	}
521

    
522
	private List<T> getPages(Class<? extends T> clazz, DeduplicateState dedupState, List<OrderHint> orderHints) {
523
		List<T> result = new ArrayList<T>();
524
		for (int pageNo = dedupState.startPage; pageNo < dedupState.startPage + dedupState.nPages; pageNo++){
525
			List<T> objectList = listByTitle(clazz, null, null, null, dedupState.pageSize, pageNo, orderHints, null);
526
			result.addAll(objectList);
527
		}
528
		if (result.size()< dedupState.nPages * dedupState.pageSize ){
529
			dedupState.isCompleted = true;
530
		}
531
		return result;
532
	}
533

    
534
	private int handleLastGroup(List<T> group, IMatchStrategy matchStrategy, IMergeStrategy mergeStrategy) {
535
		int result = 0;
536
		int size = group.size();
537
		Set<Integer> exclude = new HashSet<Integer>();  //set to collect all objects, that have been merged already
538
		for (int i = 0; i < size - 1; i++){
539
			if (exclude.contains(i)){
540
				continue;
541
			}
542
			for (int j = i + 1; j < size; j++){
543
				if (exclude.contains(j)){
544
					continue;
545
				}
546
				T firstObject = group.get(i);
547
				T secondObject = group.get(j);
548

    
549
				try {
550
					if (matchStrategy.invoke((IMatchable)firstObject, (IMatchable)secondObject)){
551
						commonService.merge((IMergable)firstObject, (IMergable)secondObject, mergeStrategy);
552
						exclude.add(j);
553
						result++;
554
					}
555
				} catch (MatchException e) {
556
					logger.warn("MatchException when trying to match " + firstObject.getTitleCache());
557
					e.printStackTrace();
558
				} catch (MergeException e) {
559
					logger.warn("MergeException when trying to merge " + firstObject.getTitleCache());
560
					e.printStackTrace();
561
				}
562
			}
563
		}
564
		return result;
565
	}
566

    
567
	@Transactional(readOnly = true)
568
	@Override
569
	public Integer countByTitle(Class<? extends T> clazz, String queryString,MatchMode matchmode, List<Criterion> criteria){
570
		 Integer numberOfResults = dao.countByTitle(clazz, queryString, matchmode, criteria);
571

    
572
		 return numberOfResults;
573
	}
574

    
575
	@Transactional(readOnly = true)
576
	@Override
577
	public Integer countByTitle(IIdentifiableEntityServiceConfigurator<T> config){
578
		return countByTitle(config.getClazz(), config.getTitleSearchStringSqlized(),
579
				config.getMatchMode(), config.getCriteria());
580

    
581
	}
582

    
583
	@Override
584
	@Transactional(readOnly = true)
585
	public <S extends T> Pager<FindByIdentifierDTO<S>> findByIdentifier(
586
			Class<S> clazz, String identifier, DefinedTerm identifierType, MatchMode matchmode,
587
			boolean includeEntity, Integer pageSize,
588
			Integer pageNumber,	List<String> propertyPaths) {
589

    
590
		Integer numberOfResults = dao.countByIdentifier(clazz, identifier, identifierType, matchmode);
591
        List<Object[]> daoResults = new ArrayList<Object[]>();
592
        if(numberOfResults > 0) { // no point checking again
593
        	daoResults = dao.findByIdentifier(clazz, identifier, identifierType,
594
    				matchmode, includeEntity, pageSize, pageNumber, propertyPaths);
595
        }
596

    
597
        List<FindByIdentifierDTO<S>> result = new ArrayList<FindByIdentifierDTO<S>>();
598
        for (Object[] daoObj : daoResults){
599
        	if (includeEntity){
600
        		result.add(new FindByIdentifierDTO<S>((DefinedTerm)daoObj[0], (String)daoObj[1], (S)daoObj[2]));
601
        	}else{
602
        		result.add(new FindByIdentifierDTO<S>((DefinedTerm)daoObj[0], (String)daoObj[1], (UUID)daoObj[2], (String)daoObj[3]));
603
        	}
604
        }
605
		return new DefaultPagerImpl<FindByIdentifierDTO<S>>(pageNumber, numberOfResults, pageSize, result);
606
	}
607

    
608
    @Override
609
    @Transactional(readOnly = true)
610
    public <S extends T> Pager<FindByMarkerDTO<S>> findByMarker(
611
            Class<S> clazz, MarkerType markerType, Boolean markerValue,
612
            boolean includeEntity, Integer pageSize,
613
            Integer pageNumber, List<String> propertyPaths) {
614

    
615
        Long numberOfResults = dao.countByMarker(clazz, markerType, markerValue);
616
        List<Object[]> daoResults = new ArrayList<>();
617
        if(numberOfResults > 0) { // no point checking again
618
            daoResults = dao.findByMarker(clazz, markerType, markerValue, includeEntity,
619
                    pageSize, pageNumber, propertyPaths);
620
        }
621

    
622
        List<FindByMarkerDTO<S>> result = new ArrayList<>();
623
        for (Object[] daoObj : daoResults){
624
            if (includeEntity){
625
                result.add(new FindByMarkerDTO<S>((MarkerType)daoObj[0], (Boolean)daoObj[1], (S)daoObj[2]));
626
            }else{
627
                result.add(new FindByMarkerDTO<S>((MarkerType)daoObj[0], (Boolean)daoObj[1], (UUID)daoObj[2], (String)daoObj[3]));
628
            }
629
        }
630
        return new DefaultPagerImpl<FindByMarkerDTO<S>>(pageNumber, numberOfResults, pageSize, result);
631
    }
632
}
633

    
(67-67/98)