Project

General

Profile

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

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

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

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

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

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

    
65

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

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

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

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

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

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

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

    
95

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

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

    
124
	/*
125
	 * TODO - Migrated from CommonServiceBase
126
	 */
127
	@Transactional(readOnly = true)
128
	@Override
129
	public ISourceable getSourcedObjectByIdInSource(Class clazz, String idInSource, String idNamespace) {
130
		ISourceable result = null;
131

    
132
		List<T> list = dao.findOriginalSourceByIdInSource(idInSource, idNamespace);
133
		if (! list.isEmpty()){
134
			result = list.get(0);
135
		}
136
		return result;
137
	}
138

    
139
	@Transactional(readOnly = true)
140
	@Override
141
	public List<UuidAndTitleCache<T>> getUuidAndTitleCache(Integer limit, String pattern) {
142
		return dao.getUuidAndTitleCache(limit, pattern);
143
	}
144

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

    
151
    @Transactional(readOnly = true)
152
    @Override
153
    public String getTitleCache(UUID uuid, boolean refresh){
154
        return dao.getTitleCache(uuid, refresh);
155
    }
156

    
157
	@Transactional(readOnly = true)
158
	@Override
159
	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) {
160
		 long numberOfResults = dao.countByTitle(clazz, queryString, matchmode, criteria);
161

    
162
		 List<T> results = new ArrayList<>();
163
		 if(numberOfResults > 0) { // no point checking again //TODO use AbstractPagerImpl.hasResultsInRange(numberOfResults, pageNumber, pageSize)
164
				results = dao.findByTitle(clazz, queryString, matchmode, criteria, pageSize, pageNumber, orderHints, propertyPaths);
165
		 }
166

    
167
		  return new DefaultPagerImpl<>(pageNumber, numberOfResults, pageSize, results);
168
	}
169

    
170
	@Transactional(readOnly = true)
171
	@Override
172
	public Pager<T> findByTitle(IIdentifiableEntityServiceConfigurator<T> config){
173
		return findByTitle(config.getClazz(), config.getTitleSearchStringSqlized(), config.getMatchMode(), config.getCriteria(), config.getPageSize(), config.getPageNumber(), config.getOrderHints(), config.getPropertyPaths());
174
	}
175

    
176
	@Transactional(readOnly = true)
177
	@Override
178
	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) {
179
		 long numberOfResults = dao.countByTitle(clazz, queryString, matchmode, criteria);
180

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

    
188
	@Transactional(readOnly = true)
189
	@Override
190
	public Pager<T> findTitleCache(Class<? extends T> clazz, String queryString, Integer pageSize, Integer pageNumber, List<OrderHint> orderHints, MatchMode matchMode){
191
		long numberOfResults = dao.countTitleCache(clazz, queryString, matchMode);
192

    
193
		 List<T> results = new ArrayList<>();
194
		 if(numberOfResults > 0) { // no point checking again //TODO use AbstractPagerImpl.hasResultsInRange(numberOfResults, pageNumber, pageSize)
195
				results = dao.findTitleCache(clazz, queryString, pageSize, pageNumber, orderHints, matchMode);
196
		 }
197
		 long r = 0;
198
		 r += numberOfResults;
199

    
200
		  return new DefaultPagerImpl<>(pageNumber, r , pageSize, results);
201
	}
202

    
203
	@Transactional(readOnly = true)
204
	@Override
205
	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) {
206
		 long numberOfResults = dao.countByReferenceTitle(clazz, queryString, matchmode, criteria);
207

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

    
215
	@Transactional(readOnly = true)
216
	@Override
217
	public T find(LSID lsid) {
218
		return dao.find(lsid);
219
	}
220

    
221
	@Transactional(readOnly = true)
222
	@Override
223
	public Pager<T> search(Class<? extends T> clazz, String queryString, Integer pageSize, Integer pageNumber, List<OrderHint> orderHints, List<String> propertyPaths) {
224
        long numberOfResults = dao.count(clazz,queryString);
225

    
226
		List<T> results = new ArrayList<>();
227
		if(numberOfResults > 0) { // no point checking again //TODO use AbstractPagerImpl.hasResultsInRange(numberOfResults, pageNumber, pageSize)
228
			results = dao.search(clazz,queryString, pageSize, pageNumber, orderHints, propertyPaths);
229
		}
230

    
231
		return new DefaultPagerImpl<>(pageNumber, numberOfResults, pageSize, results);
232
	}
233

    
234
	@Override
235
	@Transactional(readOnly = false)
236
	public void updateTitleCache() {
237
		updateTitleCache(null, null, null, null);
238
	}
239

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

    
249
		long count = dao.count(clazz);
250
		monitor.beginTask("update titles", Long.valueOf(count).intValue());
251
		int worked = 0;
252
		for(int i = 0 ; i < count ; i = i + stepSize){
253
			// not sure if such strict ordering is necessary here, but for safety reasons I do it
254
			ArrayList<OrderHint> orderHints = new ArrayList<>();
255
			orderHints.add( new OrderHint("id", OrderHint.SortOrder.ASCENDING));
256

    
257

    
258
			Map<Class<? extends CdmBase>, AutoPropertyInitializer<CdmBase>> oldAutoInit = switchOfAutoinitializer();
259
			List<S> list = this.list(clazz, stepSize, i, orderHints, null);
260
			switchOnOldAutoInitializer(oldAutoInit);
261

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

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

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

    
312
	/**
313
	 * @param cacheStrategy
314
	 * @param entitiesToUpdate
315
	 * @param entity
316
	 */
317
	/**
318
	 * @param cacheStrategy
319
	 * @param entitiesToUpdate
320
	 * @param entity
321
	 */
322
	@SuppressWarnings("unchecked")
323
	private void updateTitleCacheForSingleEntity(
324
			IIdentifiableEntityCacheStrategy<T> cacheStrategy,
325
			List<T> entitiesToUpdate,
326
			T entity) {
327

    
328
		//assert (entity.isProtectedTitleCache() == false );
329
	    entity = CdmBase.deproxy(entity);
330

    
331
		//exclude recursive inreferences
332
		if (entity.isInstanceOf(Reference.class)){
333
			Reference ref = (Reference)entity;
334
			if (ref.getInReference() != null && ref.getInReference().equals(ref)){
335
				return;
336
			}
337
		}
338

    
339
		//define the correct cache strategy
340
		IIdentifiableEntityCacheStrategy entityCacheStrategy = cacheStrategy;
341
		if (entityCacheStrategy == null){
342
			entityCacheStrategy = entity.getCacheStrategy();
343
			//FIXME find out why the wrong cache strategy is loaded here, see #1876
344
			if (entity instanceof Reference){
345
				entityCacheStrategy = ReferenceFactory.newReference(((Reference)entity).getType()).getCacheStrategy();
346
			}
347
		}
348

    
349

    
350
		//old titleCache
351
		entity.setProtectedTitleCache(true);
352

    
353
		String oldTitleCache = entity.getTitleCache();
354
		entity.setTitleCache(oldTitleCache, false);   //before we had entity.setProtectedTitleCache(false) but this deleted the titleCache itself
355
		entity.setCacheStrategy(entityCacheStrategy);
356
		//NonViralNames and Reference have more caches //TODO handle in NameService
357
		String oldNameCache = null;
358
		String oldFullTitleCache = null;
359
		String oldAbbrevTitleCache = null;
360
		if (entity instanceof TaxonName ){
361
		    if (((TaxonName) entity).isNonViral()) {
362
                try{
363
                	INonViralName nvn = (INonViralName) entity;
364
                	if (!nvn.isProtectedNameCache()){
365
                	    nvn.setProtectedNameCache(true);
366
                		oldNameCache = nvn.getNameCache();
367
                		nvn.setProtectedNameCache(false);
368
                	}
369
                	if (!nvn.isProtectedFullTitleCache()){
370
                	    nvn.setProtectedFullTitleCache(true);
371
                		oldFullTitleCache = nvn.getFullTitleCache();
372
                		nvn.setProtectedFullTitleCache(false);
373
                	}
374
                }catch(ClassCastException e){
375
                	System.out.println("entity: " + entity.getTitleCache());
376
                }
377
            }
378
		}else if (entity instanceof Reference){
379
			Reference ref = (Reference) entity;
380
			if (!ref.isProtectedAbbrevTitleCache()){
381
				ref.setProtectedAbbrevTitleCache(true);
382
				oldAbbrevTitleCache = ref.getAbbrevTitleCache();
383
				ref.setProtectedAbbrevTitleCache(false);
384
			}
385
		}
386
		setOtherCachesNull(entity);
387
		String newTitleCache= null;
388
		INonViralName nvn = null; //TODO find better solution
389
		try{
390
			if (entity instanceof TaxonName){
391
				nvn = (ITaxonName) entity;
392
				newTitleCache = entityCacheStrategy.getTitleCache(nvn);
393
			} else{
394
				 newTitleCache = entityCacheStrategy.getTitleCache(entity);
395
			}
396
		}catch (ClassCastException e){
397
			nvn = HibernateProxyHelper.deproxy(entity, TaxonName.class);
398
			newTitleCache = entityCacheStrategy.getTitleCache(nvn);
399
		}
400

    
401
		if ( oldTitleCache == null   || ! oldTitleCache.equals(newTitleCache) ){
402
			entity.setTitleCache(null, false);
403
			String newCache = entity.getTitleCache();
404

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

    
437

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

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

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

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

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

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

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

    
494
		result += handleLastGroup(nextGroup, matchStrategy, mergeStrategy);
495
		return result;
496
	}
497

    
498

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

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

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

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

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

    
569
		 return numberOfResults;
570
	}
571

    
572
	@Transactional(readOnly = true)
573
	@Override
574
	public long countByTitle(IIdentifiableEntityServiceConfigurator<T> config){
575
		return countByTitle(config.getClazz(), config.getTitleSearchStringSqlized(),
576
				config.getMatchMode(), config.getCriteria());
577

    
578
	}
579

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

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

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

    
605
	@Override
606
    @Transactional(readOnly = true)
607
    public <S extends T> List<IdentifiedEntityDTO<S>> listByIdentifier(
608
            Class<S> clazz, String identifier, DefinedTerm identifierType, MatchMode matchmode,
609
            boolean includeEntity, List<String> propertyPaths, Integer limit) {
610

    
611
        Integer numberOfResults = dao.countByIdentifier(clazz, identifier, identifierType, matchmode);
612
        List<Object[]> daoResults = new ArrayList<>();
613
        if(numberOfResults > 0) { // no point checking again
614
            daoResults = dao.findByIdentifier(clazz, identifier, identifierType,
615
                    matchmode, includeEntity, limit, 0, propertyPaths);
616
        }
617

    
618
        List<IdentifiedEntityDTO<S>> result = new ArrayList<>();
619
        for (Object[] daoObj : daoResults){
620
            if (includeEntity){
621
                result.add(new IdentifiedEntityDTO<>((DefinedTerm)daoObj[0], (String)daoObj[1], (S)daoObj[2]));
622
            }else{
623
                result.add(new IdentifiedEntityDTO<>((DefinedTerm)daoObj[0], (String)daoObj[1], (UUID)daoObj[2], (String)daoObj[3], null));
624
            }
625
        }
626
        return result;
627
    }
628

    
629

    
630
    @Override
631
    @Transactional(readOnly = true)
632
    public <S extends T> Pager<MarkedEntityDTO<S>> findByMarker(
633
            Class<S> clazz, MarkerType markerType, Boolean markerValue,
634
            boolean includeEntity, Integer pageSize,
635
            Integer pageNumber, List<String> propertyPaths) {
636

    
637
        Long numberOfResults = dao.countByMarker(clazz, markerType, markerValue);
638
        List<Object[]> daoResults = new ArrayList<>();
639
        if(numberOfResults > 0) { // no point checking again
640
            daoResults = dao.findByMarker(clazz, markerType, markerValue, includeEntity,
641
                    pageSize, pageNumber, propertyPaths);
642
        }
643

    
644
        List<MarkedEntityDTO<S>> result = new ArrayList<>();
645
        for (Object[] daoObj : daoResults){
646
            if (includeEntity){
647
                result.add(new MarkedEntityDTO<>((MarkerType)daoObj[0], (Boolean)daoObj[1], (S)daoObj[2]));
648
            }else{
649
                result.add(new MarkedEntityDTO<>((MarkerType)daoObj[0], (Boolean)daoObj[1], (UUID)daoObj[2], (String)daoObj[3]));
650
            }
651
        }
652
        return new DefaultPagerImpl<>(pageNumber, numberOfResults, pageSize, result);
653
    }
654

    
655

    
656
}
657

    
(71-71/106)