Project

General

Profile

Download (30.6 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.Objects;
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.IdentifiedEntityDTO;
28
import eu.etaxonomy.cdm.api.service.dto.MarkedEntityDTO;
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.agent.Person;
35
import eu.etaxonomy.cdm.model.agent.Team;
36
import eu.etaxonomy.cdm.model.common.CdmBase;
37
import eu.etaxonomy.cdm.model.common.DefinedTerm;
38
import eu.etaxonomy.cdm.model.common.ISourceable;
39
import eu.etaxonomy.cdm.model.common.IdentifiableEntity;
40
import eu.etaxonomy.cdm.model.common.IdentifiableSource;
41
import eu.etaxonomy.cdm.model.common.LSID;
42
import eu.etaxonomy.cdm.model.common.MarkerType;
43
import eu.etaxonomy.cdm.model.media.Rights;
44
import eu.etaxonomy.cdm.model.name.INonViralName;
45
import eu.etaxonomy.cdm.model.name.ITaxonName;
46
import eu.etaxonomy.cdm.model.name.TaxonName;
47
import eu.etaxonomy.cdm.model.reference.Reference;
48
import eu.etaxonomy.cdm.model.reference.ReferenceFactory;
49
import eu.etaxonomy.cdm.persistence.dao.common.IIdentifiableDao;
50
import eu.etaxonomy.cdm.persistence.dao.common.Restriction;
51
import eu.etaxonomy.cdm.persistence.dao.hibernate.HibernateBeanInitializer;
52
import eu.etaxonomy.cdm.persistence.dao.initializer.AutoPropertyInitializer;
53
import eu.etaxonomy.cdm.persistence.dto.UuidAndTitleCache;
54
import eu.etaxonomy.cdm.persistence.query.MatchMode;
55
import eu.etaxonomy.cdm.persistence.query.OrderHint;
56
import eu.etaxonomy.cdm.persistence.query.OrderHint.SortOrder;
57
import eu.etaxonomy.cdm.strategy.cache.common.IIdentifiableEntityCacheStrategy;
58
import eu.etaxonomy.cdm.strategy.match.DefaultMatchStrategy;
59
import eu.etaxonomy.cdm.strategy.match.IMatchStrategyEqual;
60
import eu.etaxonomy.cdm.strategy.match.IMatchable;
61
import eu.etaxonomy.cdm.strategy.match.MatchException;
62
import eu.etaxonomy.cdm.strategy.merge.IMergable;
63
import eu.etaxonomy.cdm.strategy.merge.IMergeStrategy;
64
import eu.etaxonomy.cdm.strategy.merge.MergeException;
65

    
66
public abstract class IdentifiableServiceBase<T extends IdentifiableEntity, DAO extends IIdentifiableDao<T>> extends AnnotatableServiceBase<T,DAO>
67
						implements IIdentifiableEntityService<T>{
68

    
69

    
70
	protected static final int UPDATE_TITLE_CACHE_DEFAULT_STEP_SIZE = 1000;
71
	protected static final  Logger logger = Logger.getLogger(IdentifiableServiceBase.class);
72

    
73
	@Override
74
	@Transactional(readOnly = true)
75
	public Pager<Rights> getRights(T t, Integer pageSize, Integer pageNumber, List<String> propertyPaths) {
76
        long numberOfResults = dao.countRights(t);
77

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

    
83
		return new DefaultPagerImpl<>(pageNumber, numberOfResults, pageSize, results);
84
	}
85

    
86
	@Override
87
	@Transactional(readOnly = true)
88
	public Pager<IdentifiableSource> getSources(T t, Integer pageSize, Integer pageNumber, List<String> propertyPaths) {
89
		 long numberOfResults = dao.countSources(t);
90

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

    
96
		 return new DefaultPagerImpl<>(pageNumber, numberOfResults, pageSize, results);
97
	}
98

    
99

    
100
	@Transactional(readOnly = false)
101
	@Override
102
	public T replace(T x, T y) {
103
		return dao.replace(x, y);
104
	}
105

    
106
	/*
107
	 * TODO - Migrated from CommonServiceBase
108
	 */
109
	@Transactional(readOnly = true)
110
	@Override
111
	public ISourceable getSourcedObjectByIdInSource(Class clazz, String idInSource, String idNamespace) {
112
		ISourceable<?> result = null;
113

    
114
		List<T> list = dao.findOriginalSourceByIdInSource(idInSource, idNamespace);
115
		if (! list.isEmpty()){
116
			result = list.get(0);
117
		}
118
		return result;
119
	}
120

    
121
	@Transactional(readOnly = true)
122
	@Override
123
	public List<UuidAndTitleCache<T>> getUuidAndTitleCache(Integer limit, String pattern) {
124
		return dao.getUuidAndTitleCache(limit, pattern);
125
	}
126

    
127
    @Transactional(readOnly = true)
128
    @Override
129
    public <S extends T> List<UuidAndTitleCache<S>> getUuidAndTitleCache(Class<S> clazz,Integer limit, String pattern) {
130
        return dao.getUuidAndTitleCache(clazz, limit, pattern);
131
    }
132

    
133
    @Transactional(readOnly = true)
134
    @Override
135
    public String getTitleCache(UUID uuid, boolean refresh){
136
        return dao.getTitleCache(uuid, refresh);
137
    }
138

    
139
    @Transactional(readOnly = true)
140
    @Override
141
    public <S extends T> Pager<S> findByTitle(Class<S> clazz, String queryString,MatchMode matchmode, List<Criterion> criteria, Integer pageSize, Integer pageNumber, List<OrderHint> orderHints, List<String> propertyPaths) {
142
         long numberOfResults = dao.countByTitle(clazz, queryString, matchmode, criteria);
143

    
144
         List<S> results = new ArrayList<>();
145
         if(numberOfResults > 0) { // no point checking again //TODO use AbstractPagerImpl.hasResultsInRange(numberOfResults, pageNumber, pageSize)
146
                results = dao.findByTitle(clazz, queryString, matchmode, criteria, pageSize, pageNumber, orderHints, propertyPaths);
147
         }
148

    
149
         return new DefaultPagerImpl<>(pageNumber, numberOfResults, pageSize, results);
150
    }
151

    
152
	@Transactional(readOnly = true)
153
	@Override
154
	public <S extends T> Pager<S> findByTitleWithRestrictions(Class<S> clazz, String queryString, MatchMode matchmode, List<Restriction<?>> restrictions, Integer pageSize, Integer pageNumber, List<OrderHint> orderHints, List<String> propertyPaths) {
155
		 long numberOfResults = dao.countByTitleWithRestrictions(clazz, queryString, matchmode, restrictions);
156

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

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

    
165

    
166
	@Transactional(readOnly = true)
167
	@Override
168
	public <S extends T> Pager<S> findByTitle(IIdentifiableEntityServiceConfigurator<S> config){
169

    
170
	    boolean withRestrictions = config.getRestrictions() != null && !config.getRestrictions().isEmpty();
171
	    boolean withCriteria = config.getCriteria() != null && !config.getCriteria().isEmpty();
172

    
173
	    if(withCriteria && withRestrictions){
174
	        throw new RuntimeException("Restrictions and Criteria can not be used at the same time");
175
	    } else if(withRestrictions){
176
	        return findByTitleWithRestrictions((Class<S>)config.getClazz(), config.getTitleSearchStringSqlized(), config.getMatchMode(), config.getRestrictions(), config.getPageSize(), config.getPageNumber(), config.getOrderHints(), config.getPropertyPaths());
177
	    } else {
178
	        return findByTitle((Class<S>) config.getClazz(), config.getTitleSearchStringSqlized(), config.getMatchMode(), config.getCriteria(), config.getPageSize(), config.getPageNumber(), config.getOrderHints(), config.getPropertyPaths());
179
	    }
180
	}
181

    
182
   @Transactional(readOnly = true)
183
    @Override
184
    public <S extends T> List<S> listByTitle(Class<S> clazz, String queryString,MatchMode matchmode, List<Criterion> criteria, Integer pageSize, Integer pageNumber, List<OrderHint> orderHints, List<String> propertyPaths) {
185
         long numberOfResults = dao.countByTitle(clazz, queryString, matchmode, criteria);
186

    
187
         List<S> results = new ArrayList<>();
188
         if(numberOfResults > 0) { // no point checking again //TODO use AbstractPagerImpl.hasResultsInRange(numberOfResults, pageNumber, pageSize)
189
                results = dao.findByTitle(clazz, queryString, matchmode, criteria, pageSize, pageNumber, orderHints, propertyPaths);
190
         }
191
         return results;
192
    }
193

    
194
	@Transactional(readOnly = true)
195
	@Override
196
	public <S extends T> List<S> listByTitleWithRestrictions(Class<S> clazz, String queryString,MatchMode matchmode, List<Restriction<?>> restrictions, Integer pageSize, Integer pageNumber, List<OrderHint> orderHints, List<String> propertyPaths) {
197
		 long numberOfResults = dao.countByTitleWithRestrictions(clazz, queryString, matchmode, restrictions);
198

    
199
		 List<S> results = new ArrayList<>();
200
		 if(numberOfResults > 0) { // no point checking again //TODO use AbstractPagerImpl.hasResultsInRange(numberOfResults, pageNumber, pageSize)
201
				results = dao.findByTitleWithRestrictions(clazz, queryString, matchmode, restrictions, pageSize, pageNumber, orderHints, propertyPaths);
202
		 }
203
		 return results;
204
	}
205

    
206
	@Transactional(readOnly = true)
207
	@Override
208
	public <S extends T> Pager<S> findTitleCache(Class<S> clazz, String queryString, Integer pageSize, Integer pageNumber, List<OrderHint> orderHints, MatchMode matchMode){
209
		long numberOfResults = dao.countTitleCache(clazz, queryString, matchMode);
210

    
211
		 List<S> results = new ArrayList<>();
212
		 if(numberOfResults > 0) { // no point checking again //TODO use AbstractPagerImpl.hasResultsInRange(numberOfResults, pageNumber, pageSize)
213
				results = dao.findTitleCache(clazz, queryString, pageSize, pageNumber, orderHints, matchMode);
214
		 }
215
		 long r = 0;
216
		 r += numberOfResults;
217

    
218
		 return new DefaultPagerImpl<>(pageNumber, r , pageSize, results);
219
	}
220

    
221
    @Transactional(readOnly = true)
222
    @Override
223
    public <S extends T> List<S> listByReferenceTitle(Class<S> clazz, String queryString,MatchMode matchmode, List<Criterion> criteria, Integer pageSize, Integer pageNumber, List<OrderHint> orderHints, List<String> propertyPaths) {
224
         long numberOfResults = dao.countByReferenceTitle(clazz, queryString, matchmode, criteria);
225

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

    
233
	@Transactional(readOnly = true)
234
	@Override
235
	public <S extends T> List<S> listByReferenceTitleWithRestrictions(Class<S> clazz, String queryString,MatchMode matchmode, List<Restriction<?>> restrictions, Integer pageSize, Integer pageNumber, List<OrderHint> orderHints, List<String> propertyPaths) {
236
		 long numberOfResults = dao.countByReferenceTitleWithRestrictions(clazz, queryString, matchmode, restrictions);
237

    
238
		 List<S> results = new ArrayList<>();
239
		 if(numberOfResults > 0) { // no point checking again //TODO use AbstractPagerImpl.hasResultsInRange(numberOfResults, pageNumber, pageSize)
240
		     results = dao.findByReferenceTitleWithRestrictions(clazz, queryString, matchmode, restrictions, pageSize, pageNumber, orderHints, propertyPaths);
241
		 }
242
		 return results;
243
	}
244

    
245
	@Transactional(readOnly = true)
246
	@Override
247
	public T find(LSID lsid) {
248
		return dao.find(lsid);
249
	}
250

    
251
	@Transactional(readOnly = true)
252
	@Override
253
	public Pager<T> search(Class<? extends T> clazz, String queryString, Integer pageSize, Integer pageNumber, List<OrderHint> orderHints, List<String> propertyPaths) {
254
//	public <S extends T> Pager<S> search(Class<S> clazz, String queryString, Integer pageSize, Integer pageNumber, List<OrderHint> orderHints, List<String> propertyPaths) {
255
        long numberOfResults = dao.count(clazz,queryString);
256

    
257
		List<T> results = new ArrayList<>();
258
		if(numberOfResults > 0) { // no point checking again //TODO use AbstractPagerImpl.hasResultsInRange(numberOfResults, pageNumber, pageSize)
259
			results = dao.search(clazz,queryString, pageSize, pageNumber, orderHints, propertyPaths);
260
		}
261

    
262
		return new DefaultPagerImpl<>(pageNumber, numberOfResults, pageSize, results);
263
	}
264

    
265
	@Override
266
	@Transactional(readOnly = false)
267
	public void updateTitleCache() {
268
		updateTitleCache(null, null, null, null);
269
	}
270

    
271
	@Transactional(readOnly = false)  //TODO check transactional behavior, e.g. what happens with the session if count is very large
272
	protected <S extends T > void updateTitleCacheImpl(Class<S> clazz, Integer stepSize, IIdentifiableEntityCacheStrategy<T> cacheStrategy, IProgressMonitor monitor) {
273
		if (stepSize == null){
274
			stepSize = UPDATE_TITLE_CACHE_DEFAULT_STEP_SIZE;
275
		}
276
		if (monitor == null){
277
			monitor = DefaultProgressMonitor.NewInstance();
278
		}
279

    
280
		long count = dao.count(clazz);
281
		long countUpdated = 0;
282
		monitor.beginTask("update titles for " + clazz.getSimpleName(), Long.valueOf(count).intValue());
283
		int worked = 0;
284
		for(int i = 0 ; i < count ; i = i + stepSize){
285
			// not sure if such strict ordering is necessary here, but for safety reasons I do it
286
			ArrayList<OrderHint> orderHints = new ArrayList<>();
287
			orderHints.add( new OrderHint("id", OrderHint.SortOrder.ASCENDING));
288

    
289

    
290
			Map<Class<? extends CdmBase>, AutoPropertyInitializer<CdmBase>> oldAutoInit = switchOfAutoinitializer();
291
			List<S> list = this.list(clazz, stepSize, i, orderHints, null);
292
			switchOnOldAutoInitializer(oldAutoInit);
293

    
294
			List<T> entitiesToUpdate = new ArrayList<>();
295
			for (T entity : list){
296
				HibernateProxyHelper.deproxy(entity, clazz);
297
				if (entity.hasUnprotectedCache()){
298
				    // always execute for TeamOrPersonBase to allow updating the nomenclaturalTitle
299
					updateTitleCacheForSingleEntity(cacheStrategy, entitiesToUpdate, entity);
300
				}
301
				worked++;
302
			}
303
			for (T entity: entitiesToUpdate){
304
				if (entity.getTitleCache() != null){
305
					//System.err.println(entity.getTitleCache());
306
				}else{
307
				    //System.err.println("no titleCache" + ((TaxonName)entity).getNameCache());
308
				}
309
			}
310
			saveOrUpdate(entitiesToUpdate);
311
			monitor.worked(list.size());
312
			if (monitor.isCanceled()){
313
				monitor.done();
314
				return;
315
			}
316
		}
317
		monitor.done();
318
	}
319

    
320
	/**
321
	 * Brings back all auto initializers to the bean initializer
322
	 * @see #switchOfAutoinitializer()
323
	 * @param oldAutoInit
324
	 */
325
	protected void switchOnOldAutoInitializer(
326
			Map<Class<? extends CdmBase>, AutoPropertyInitializer<CdmBase>> oldAutoInit) {
327
		HibernateBeanInitializer initializer = (HibernateBeanInitializer)this.appContext.getBean("defaultBeanInitializer");
328
		initializer.setBeanAutoInitializers(oldAutoInit);
329
	}
330

    
331
	/**
332
	 * Removes all auto initializers from the bean initializer
333
	 *
334
	 * @see #switchOnOldAutoInitializer(Map)
335
	 * @return
336
	 */
337
	protected Map<Class<? extends CdmBase>, AutoPropertyInitializer<CdmBase>> switchOfAutoinitializer() {
338
		HibernateBeanInitializer initializer = (HibernateBeanInitializer)this.appContext.getBean("defaultBeanInitializer");
339
		Map<Class<? extends CdmBase>, AutoPropertyInitializer<CdmBase>> oldAutoInitializers = initializer.getBeanAutoInitializers();
340
		Map<Class<? extends CdmBase>, AutoPropertyInitializer<CdmBase>> map = new HashMap<>();
341
		initializer.setBeanAutoInitializers(map);
342
		return oldAutoInitializers;
343
	}
344

    
345
	/**
346
	 * @param cacheStrategy
347
	 * @param entitiesToUpdate
348
	 * @param entity
349
	 */
350
	@SuppressWarnings("unchecked")
351
	private void updateTitleCacheForSingleEntity(
352
			IIdentifiableEntityCacheStrategy<T> cacheStrategy,
353
			List<T> entitiesToUpdate,
354
			T entity) {
355

    
356
		//assert (entity.isProtectedTitleCache() == false );
357
	    entity = CdmBase.deproxy(entity);
358

    
359
		//exclude recursive inreferences
360
		if (entity.isInstanceOf(Reference.class)){
361
			Reference ref = (Reference)entity;
362
			if (ref.getInReference() != null && ref.getInReference().equals(ref)){
363
				return;
364
			}
365
		}
366

    
367
		//define the correct cache strategy
368
		IIdentifiableEntityCacheStrategy entityCacheStrategy = cacheStrategy;
369
		if (entityCacheStrategy == null){
370
			entityCacheStrategy = entity.getCacheStrategy();
371
			//FIXME find out why the wrong cache strategy is loaded here, see #1876
372
			if (entity instanceof Reference){
373
				entityCacheStrategy = ReferenceFactory.newReference(((Reference)entity).getType()).getCacheStrategy();
374
			}
375
		}
376

    
377

    
378
		//old titleCache
379
		entity.setProtectedTitleCache(true);
380

    
381
		String oldTitleCache = entity.getTitleCache();
382
		entity.setTitleCache(oldTitleCache, false);   //before we had entity.setProtectedTitleCache(false) but this deleted the titleCache itself
383
		entity.setCacheStrategy(entityCacheStrategy);
384
		//NonViralNames and Reference have more caches //TODO handle in NameService
385
		String oldNameCache = null;
386
		String oldFullTitleCache = null;
387
		String oldAbbrevTitleCache = null;
388
        String oldNomenclaturalTitle = null;
389
		if (entity instanceof TaxonName ){
390
		    if (((TaxonName) entity).isNonViral()) {
391
                try{
392
                	INonViralName nvn = (INonViralName) entity;
393
                	if (!nvn.isProtectedNameCache()){
394
                	    nvn.setProtectedNameCache(true);
395
                		oldNameCache = nvn.getNameCache();
396
                		nvn.setProtectedNameCache(false);
397
                	}
398
                	if (!nvn.isProtectedFullTitleCache()){
399
                	    nvn.setProtectedFullTitleCache(true);
400
                		oldFullTitleCache = nvn.getFullTitleCache();
401
                		nvn.setProtectedFullTitleCache(false);
402
                	}
403
                }catch(ClassCastException e){
404
                	System.out.println("entity: " + entity.getTitleCache());
405
                }
406
            }
407
		}else if (entity instanceof Reference){
408
			Reference ref = (Reference) entity;
409
			if (!ref.isProtectedAbbrevTitleCache()){
410
				ref.setProtectedAbbrevTitleCache(true);
411
				oldAbbrevTitleCache = ref.getAbbrevTitleCache();
412
				ref.setProtectedAbbrevTitleCache(false);
413
			}
414
		} else if (entity instanceof Team){
415
            Team team = (Team) entity;
416
            if(!team.isProtectedNomenclaturalTitleCache()){
417
                team.setProtectedNomenclaturalTitleCache(true);
418
                oldNomenclaturalTitle = team.getNomenclaturalTitle();
419
                team.setProtectedNomenclaturalTitleCache(false);
420
            }
421
        } else if (entity instanceof Person){
422
            // Person has no flag for protecting the nomenclaturalTitle
423
            Person person = (Person) entity;
424
            oldNomenclaturalTitle = person.getNomenclaturalTitle();
425
        }
426
		setOtherCachesNull(entity);
427
		String newTitleCache= null;
428
		INonViralName nvn = null; //TODO find better solution
429
		try{
430
			if (entity instanceof TaxonName){
431
				nvn = (ITaxonName) entity;
432
				newTitleCache = entityCacheStrategy.getTitleCache(nvn);
433
			} else{
434
				 newTitleCache = entityCacheStrategy.getTitleCache(entity);
435
			}
436
		}catch (ClassCastException e){
437
			nvn = HibernateProxyHelper.deproxy(entity, TaxonName.class);
438
			newTitleCache = entityCacheStrategy.getTitleCache(nvn);
439
		}
440

    
441
		if ( oldTitleCache == null   || ! oldTitleCache.equals(newTitleCache) ){
442
			entity.setTitleCache(null, false);
443
			String newCache = entity.getTitleCache();
444

    
445
			if (newCache == null){
446
				logger.warn("newCache should never be null");
447
			}
448
			if (oldTitleCache == null){
449
				logger.info("oldTitleCache should never be null");
450
			}
451
			if (nvn != null){
452
				nvn.getNameCache();
453
				nvn.getFullTitleCache();
454
			}
455
			if (entity instanceof Reference){
456
				Reference ref = (Reference) entity;
457
				ref.getAbbrevTitleCache();
458
			}
459
			entitiesToUpdate.add(entity);
460
		}else if (nvn != null){
461
			String newNameCache = nvn.getNameCache();
462
			String newFullTitleCache = nvn.getFullTitleCache();
463
			if ((oldNameCache == null && !nvn.isProtectedNameCache()) || (oldNameCache != null && !oldNameCache.equals(newNameCache))){
464
				entitiesToUpdate.add(entity);
465
			}else if ((oldFullTitleCache == null && !nvn.isProtectedFullTitleCache()) || (oldFullTitleCache != null && !oldFullTitleCache.equals(newFullTitleCache))){
466
				entitiesToUpdate.add(entity);
467
			}
468
		}else if (entity instanceof Reference){
469
			Reference ref = (Reference) entity;
470
			String newAbbrevTitleCache = ref.getAbbrevTitleCache();
471
			if ( (oldAbbrevTitleCache == null && !ref.isProtectedAbbrevTitleCache() ) || (oldAbbrevTitleCache != null && !oldAbbrevTitleCache.equals(newAbbrevTitleCache))){
472
				entitiesToUpdate.add(entity);
473
			}
474
		}else if (entity instanceof Team){
475
		    Team team = (Team) entity;
476
            String newNomenclaturalTitle = team.getNomenclaturalTitle();
477
            if ( (oldNomenclaturalTitle == null && !team.isProtectedNomenclaturalTitleCache() ) || !Objects.equals(oldNomenclaturalTitle, newNomenclaturalTitle)){
478
                entitiesToUpdate.add(entity);
479
            }
480
        }else if (entity instanceof Person){
481
            Person person = (Person) entity;
482
            String newNomenclaturalTitle = person.getNomenclaturalTitle();
483
            if (!Objects.equals(oldNomenclaturalTitle, newNomenclaturalTitle)){
484
                entitiesToUpdate.add(entity);
485
            }
486
        }
487
	}
488

    
489

    
490
	/**
491
	 * Needs override if not only the title cache should be set to null to
492
	 * generate the correct new title cache
493
	 */
494
	protected void setOtherCachesNull(T entity) {
495
		return;
496
	}
497

    
498
	private class DeduplicateState{
499
		String lastTitleCache;
500
		Integer pageSize = 50;
501
		int nPages = 3;
502
		int startPage = 0;
503
		boolean isCompleted = false;
504
		int result;
505
	}
506

    
507
	@Override
508
	@Transactional(readOnly = false)
509
	public int deduplicate(Class<? extends T> clazz, IMatchStrategyEqual matchStrategy, IMergeStrategy mergeStrategy) {
510
		DeduplicateState dedupState = new DeduplicateState();
511

    
512
		if (clazz == null){
513
			logger.warn("Deduplication clazz must not be null!");
514
			return 0;
515
		}
516
		if (! ( IMatchable.class.isAssignableFrom(clazz) && IMergable.class.isAssignableFrom(clazz) )  ){
517
			logger.warn("Deduplication implemented only for classes implementing IMatchable and IMergeable. No deduplication performed!");
518
			return 0;
519
		}
520
		Class matchableClass = clazz;
521
		if (matchStrategy == null){
522
			matchStrategy = DefaultMatchStrategy.NewInstance(matchableClass);
523
		}
524
		List<T> nextGroup = new ArrayList<>();
525

    
526
		int result = 0;
527
//		double countTotal = count(clazz);
528
//
529
//		Number countPagesN = Math.ceil(countTotal/dedupState.pageSize.doubleValue()) ;
530
//		int countPages = countPagesN.intValue();
531
//
532

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

    
535
		while (! dedupState.isCompleted){
536
			//get x page sizes
537
			List<? extends T> objectList = getPages(clazz, dedupState, orderHints);
538
			//after each page check if any changes took place
539
			int nUnEqualPages = handleAllPages(objectList, dedupState, nextGroup, matchStrategy, mergeStrategy);
540
			nUnEqualPages = nUnEqualPages + dedupState.pageSize * dedupState.startPage;
541
			//refresh start page counter
542
			int finishedPages = nUnEqualPages / dedupState.pageSize;
543
			dedupState.startPage = finishedPages;
544
		}
545

    
546
		result += handleLastGroup(nextGroup, matchStrategy, mergeStrategy);
547
		return result;
548
	}
549

    
550

    
551
	private int handleAllPages(List<? extends T> objectList, DeduplicateState dedupState, List<T> nextGroup, IMatchStrategyEqual matchStrategy, IMergeStrategy mergeStrategy) {
552
		int nUnEqual = 0;
553
		for (T object : objectList){
554
			String currentTitleCache = object.getTitleCache();
555
			if (currentTitleCache != null && currentTitleCache.equals(dedupState.lastTitleCache)){
556
				//=titleCache
557
				nextGroup.add(object);
558
			}else{
559
				//<> titleCache
560
				dedupState.result += handleLastGroup(nextGroup, matchStrategy, mergeStrategy);
561
				nextGroup = new ArrayList<>();
562
				nextGroup.add(object);
563
				nUnEqual++;
564
			}
565
			dedupState.lastTitleCache = currentTitleCache;
566
		}
567
		handleLastGroup(nextGroup, matchStrategy, mergeStrategy);
568
		return nUnEqual;
569
	}
570

    
571
	private <S extends T> List<S> getPages(Class<S> clazz, DeduplicateState dedupState, List<OrderHint> orderHints) {
572
		List<S> result = new ArrayList<>();
573
		for (int pageNo = dedupState.startPage; pageNo < dedupState.startPage + dedupState.nPages; pageNo++){
574
			List<S> objectList = listByTitleWithRestrictions(clazz, null, null, null, dedupState.pageSize, pageNo, orderHints, null);
575
			result.addAll(objectList);
576
		}
577
		if (result.size()< dedupState.nPages * dedupState.pageSize ){
578
			dedupState.isCompleted = true;
579
		}
580
		return result;
581
	}
582

    
583
	private int handleLastGroup(List<T> group, IMatchStrategyEqual matchStrategy, IMergeStrategy mergeStrategy) {
584
		int result = 0;
585
		int size = group.size();
586
		Set<Integer> exclude = new HashSet<>();  //set to collect all objects, that have been merged already
587
		for (int i = 0; i < size - 1; i++){
588
			if (exclude.contains(i)){
589
				continue;
590
			}
591
			for (int j = i + 1; j < size; j++){
592
				if (exclude.contains(j)){
593
					continue;
594
				}
595
				T firstObject = group.get(i);
596
				T secondObject = group.get(j);
597

    
598
				try {
599
					if (matchStrategy.invoke((IMatchable)firstObject, (IMatchable)secondObject).isSuccessful()){
600
						commonService.merge((IMergable)firstObject, (IMergable)secondObject, mergeStrategy);
601
						exclude.add(j);
602
						result++;
603
					}
604
				} catch (MatchException e) {
605
					logger.warn("MatchException when trying to match " + firstObject.getTitleCache());
606
					e.printStackTrace();
607
				} catch (MergeException e) {
608
					logger.warn("MergeException when trying to merge " + firstObject.getTitleCache());
609
					e.printStackTrace();
610
				}
611
			}
612
		}
613
		return result;
614
	}
615

    
616
    @Transactional(readOnly = true)
617
    @Override
618
    public long countByTitle(Class<? extends T> clazz, String queryString,MatchMode matchmode, List<Criterion> criteria){
619
         long numberOfResults = dao.countByTitle(clazz, queryString, matchmode, criteria);
620

    
621
         return numberOfResults;
622
    }
623

    
624
	@Transactional(readOnly = true)
625
	@Override
626
	public long countByTitleWithRestrictions(Class<? extends T> clazz, String queryString, MatchMode matchmode,  List<Restriction<?>> restrictions){
627
		 long numberOfResults = dao.countByTitleWithRestrictions(clazz, queryString, matchmode, restrictions);
628

    
629
		 return numberOfResults;
630
	}
631

    
632
	@Transactional(readOnly = true)
633
	@Override
634
	public long countByTitle(IIdentifiableEntityServiceConfigurator<T> config){
635

    
636
        boolean withRestrictions = config.getRestrictions() != null && !config.getRestrictions().isEmpty();
637
        boolean withCriteria = config.getCriteria() != null && !config.getCriteria().isEmpty();
638

    
639
        if(withCriteria && withRestrictions){
640
            throw new RuntimeException("Restrictions and Criteria can not be used at the same time");
641
        } else if(withRestrictions){
642
            return countByTitleWithRestrictions(config.getClazz(), config.getTitleSearchStringSqlized(), config.getMatchMode(), config.getRestrictions());
643
        } else {
644
            return countByTitle(config.getClazz(), config.getTitleSearchStringSqlized(), config.getMatchMode(), config.getCriteria());
645
        }
646

    
647
	}
648

    
649
	@Override
650
	@Transactional(readOnly = true)
651
	public <S extends T> Pager<IdentifiedEntityDTO<S>> findByIdentifier(
652
			Class<S> clazz, String identifier, DefinedTerm identifierType, MatchMode matchmode,
653
			boolean includeEntity, Integer pageSize,
654
			Integer pageNumber,	List<String> propertyPaths) {
655

    
656
		long numberOfResults = dao.countByIdentifier(clazz, identifier, identifierType, matchmode);
657
        List<Object[]> daoResults = new ArrayList<>();
658
        if(numberOfResults > 0) { // no point checking again
659
        	daoResults = dao.findByIdentifier(clazz, identifier, identifierType,
660
    				matchmode, includeEntity, pageSize, pageNumber, propertyPaths);
661
        }
662

    
663
        List<IdentifiedEntityDTO<S>> result = new ArrayList<>();
664
        for (Object[] daoObj : daoResults){
665
        	if (includeEntity){
666
        		result.add(new IdentifiedEntityDTO<>((DefinedTerm)daoObj[0], (String)daoObj[1], (S)daoObj[2]));
667
        	}else{
668
        		result.add(new IdentifiedEntityDTO<>((DefinedTerm)daoObj[0], (String)daoObj[1], (UUID)daoObj[2], (String)daoObj[3], null));
669
        	}
670
        }
671
		return new DefaultPagerImpl<>(pageNumber, numberOfResults, pageSize, result);
672
	}
673

    
674
	@Override
675
    @Transactional(readOnly = true)
676
    public <S extends T> List<IdentifiedEntityDTO<S>> listByIdentifier(
677
            Class<S> clazz, String identifier, DefinedTerm identifierType, MatchMode matchmode,
678
            boolean includeEntity, List<String> propertyPaths, Integer limit) {
679

    
680
        long numberOfResults = dao.countByIdentifier(clazz, identifier, identifierType, matchmode);
681
        List<Object[]> daoResults = new ArrayList<>();
682
        if(numberOfResults > 0) { // no point checking again
683
            daoResults = dao.findByIdentifier(clazz, identifier, identifierType,
684
                    matchmode, includeEntity, limit, 0, propertyPaths);
685
        }
686

    
687
        List<IdentifiedEntityDTO<S>> result = new ArrayList<>();
688
        for (Object[] daoObj : daoResults){
689
            if (includeEntity){
690
                result.add(new IdentifiedEntityDTO<>((DefinedTerm)daoObj[0], (String)daoObj[1], (S)daoObj[2]));
691
            }else{
692
                result.add(new IdentifiedEntityDTO<>((DefinedTerm)daoObj[0], (String)daoObj[1], (UUID)daoObj[2], (String)daoObj[3], null));
693
            }
694
        }
695
        return result;
696
    }
697

    
698

    
699
    @Override
700
    @Transactional(readOnly = true)
701
    public <S extends T> Pager<MarkedEntityDTO<S>> findByMarker(
702
            Class<S> clazz, MarkerType markerType, Boolean markerValue,
703
            boolean includeEntity, Integer pageSize,
704
            Integer pageNumber, List<String> propertyPaths) {
705

    
706
        Long numberOfResults = dao.countByMarker(clazz, markerType, markerValue);
707
        List<Object[]> daoResults = new ArrayList<>();
708
        if(numberOfResults > 0) { // no point checking again
709
            daoResults = dao.findByMarker(clazz, markerType, markerValue, includeEntity,
710
                    pageSize, pageNumber, propertyPaths);
711
        }
712

    
713
        List<MarkedEntityDTO<S>> result = new ArrayList<>();
714
        for (Object[] daoObj : daoResults){
715
            if (includeEntity){
716
                result.add(new MarkedEntityDTO<>((MarkerType)daoObj[0], (Boolean)daoObj[1], (S)daoObj[2]));
717
            }else{
718
                result.add(new MarkedEntityDTO<>((MarkerType)daoObj[0], (Boolean)daoObj[1], (UUID)daoObj[2], (String)daoObj[3]));
719
            }
720
        }
721
        return new DefaultPagerImpl<>(pageNumber, numberOfResults, pageSize, result);
722
    }
723

    
724
    @Override
725
    @Transactional(readOnly = true)
726
    public List<UuidAndTitleCache<T>> findUuidAndTitleCacheByMarker(Integer limit, String pattern,
727
            MarkerType markerType){
728

    
729
        Long numberOfResults = dao.countByMarker(null, markerType, null);
730
        List<UuidAndTitleCache<T>> daoResults = new ArrayList<>();
731
        if(numberOfResults > 0) { // no point checking again
732
            daoResults = dao.getUuidAndTitleCacheByMarker(limit, pattern, markerType);
733
        }
734

    
735

    
736
        return daoResults;
737
    }
738

    
739

    
740

    
741
}
742

    
(71-71/103)