Project

General

Profile

Download (24.1 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.persistence.dao.common.IIdentifiableDao;
42
import eu.etaxonomy.cdm.persistence.dao.common.Restriction;
43
import eu.etaxonomy.cdm.persistence.dao.hibernate.HibernateBeanInitializer;
44
import eu.etaxonomy.cdm.persistence.dao.initializer.AutoPropertyInitializer;
45
import eu.etaxonomy.cdm.persistence.dto.UuidAndTitleCache;
46
import eu.etaxonomy.cdm.persistence.query.MatchMode;
47
import eu.etaxonomy.cdm.persistence.query.OrderHint;
48
import eu.etaxonomy.cdm.persistence.query.OrderHint.SortOrder;
49
import eu.etaxonomy.cdm.strategy.cache.common.IIdentifiableEntityCacheStrategy;
50
import eu.etaxonomy.cdm.strategy.match.DefaultMatchStrategy;
51
import eu.etaxonomy.cdm.strategy.match.IMatchStrategyEqual;
52
import eu.etaxonomy.cdm.strategy.match.IMatchable;
53
import eu.etaxonomy.cdm.strategy.match.MatchException;
54
import eu.etaxonomy.cdm.strategy.merge.IMergable;
55
import eu.etaxonomy.cdm.strategy.merge.IMergeStrategy;
56
import eu.etaxonomy.cdm.strategy.merge.MergeException;
57

    
58
public abstract class IdentifiableServiceBase<T extends IdentifiableEntity, DAO extends IIdentifiableDao<T>> extends AnnotatableServiceBase<T,DAO>
59
						implements IIdentifiableEntityService<T>{
60

    
61

    
62
	protected static final int UPDATE_TITLE_CACHE_DEFAULT_STEP_SIZE = 1000;
63
	protected static final  Logger logger = Logger.getLogger(IdentifiableServiceBase.class);
64

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

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

    
75
		return new DefaultPagerImpl<>(pageNumber, numberOfResults, pageSize, results);
76
	}
77

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

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

    
88
		 return new DefaultPagerImpl<>(pageNumber, numberOfResults, pageSize, results);
89
	}
90

    
91

    
92
	@Transactional(readOnly = false)
93
	@Override
94
	public T replace(T x, T y) {
95
		return dao.replace(x, y);
96
	}
97

    
98
	/*
99
	 * TODO - Migrated from CommonServiceBase
100
	 */
101
	@Transactional(readOnly = true)
102
	@Override
103
	public ISourceable getSourcedObjectByIdInSource(Class clazz, String idInSource, String idNamespace) {
104
		ISourceable<?> result = null;
105

    
106
		List<T> list = dao.findOriginalSourceByIdInSource(idInSource, idNamespace);
107
		if (! list.isEmpty()){
108
			result = list.get(0);
109
		}
110
		return result;
111
	}
112

    
113
	@Transactional(readOnly = true)
114
	@Override
115
	public List<UuidAndTitleCache<T>> getUuidAndTitleCache(Integer limit, String pattern) {
116
		return dao.getUuidAndTitleCache(limit, pattern);
117
	}
118

    
119
    @Transactional(readOnly = true)
120
    @Override
121
    public <S extends T> List<UuidAndTitleCache<S>> getUuidAndTitleCache(Class<S> clazz,Integer limit, String pattern) {
122
        return dao.getUuidAndTitleCache(clazz, limit, pattern);
123
    }
124

    
125
    @Transactional(readOnly = true)
126
    @Override
127
    public String getTitleCache(UUID uuid, boolean refresh){
128
        return dao.getTitleCache(uuid, refresh);
129
    }
130

    
131
    @Transactional(readOnly = true)
132
    @Override
133
    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) {
134
         long numberOfResults = dao.countByTitle(clazz, queryString, matchmode, criteria);
135

    
136
         List<S> results = new ArrayList<>();
137
         if(numberOfResults > 0) { // no point checking again //TODO use AbstractPagerImpl.hasResultsInRange(numberOfResults, pageNumber, pageSize)
138
                results = dao.findByTitle(clazz, queryString, matchmode, criteria, pageSize, pageNumber, orderHints, propertyPaths);
139
         }
140

    
141
         return new DefaultPagerImpl<>(pageNumber, numberOfResults, pageSize, results);
142
    }
143

    
144
	@Transactional(readOnly = true)
145
	@Override
146
	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) {
147
		 long numberOfResults = dao.countByTitleWithRestrictions(clazz, queryString, matchmode, restrictions);
148

    
149
		 List<S> results = new ArrayList<>();
150
		 if(numberOfResults > 0) { // no point checking again //TODO use AbstractPagerImpl.hasResultsInRange(numberOfResults, pageNumber, pageSize)
151
				results = dao.findByTitleWithRestrictions(clazz, queryString, matchmode, restrictions, pageSize, pageNumber, orderHints, propertyPaths);
152
		 }
153

    
154
		 return new DefaultPagerImpl<>(pageNumber, numberOfResults, pageSize, results);
155
	}
156

    
157

    
158
	@Transactional(readOnly = true)
159
	@Override
160
	public <S extends T> Pager<S> findByTitle(IIdentifiableEntityServiceConfigurator<S> config){
161

    
162
	    boolean withRestrictions = config.getRestrictions() != null && !config.getRestrictions().isEmpty();
163
	    boolean withCriteria = config.getCriteria() != null && !config.getCriteria().isEmpty();
164

    
165
	    if(withCriteria && withRestrictions){
166
	        throw new RuntimeException("Restrictions and Criteria can not be used at the same time");
167
	    } else if(withRestrictions){
168
	        return findByTitleWithRestrictions((Class<S>)config.getClazz(), config.getTitleSearchStringSqlized(), config.getMatchMode(), config.getRestrictions(), config.getPageSize(), config.getPageNumber(), config.getOrderHints(), config.getPropertyPaths());
169
	    } else {
170
	        return findByTitle((Class<S>) config.getClazz(), config.getTitleSearchStringSqlized(), config.getMatchMode(), config.getCriteria(), config.getPageSize(), config.getPageNumber(), config.getOrderHints(), config.getPropertyPaths());
171
	    }
172
	}
173

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

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

    
186
	@Transactional(readOnly = true)
187
	@Override
188
	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) {
189
		 long numberOfResults = dao.countByTitleWithRestrictions(clazz, queryString, matchmode, restrictions);
190

    
191
		 List<S> results = new ArrayList<>();
192
		 if(numberOfResults > 0) { // no point checking again //TODO use AbstractPagerImpl.hasResultsInRange(numberOfResults, pageNumber, pageSize)
193
				results = dao.findByTitleWithRestrictions(clazz, queryString, matchmode, restrictions, pageSize, pageNumber, orderHints, propertyPaths);
194
		 }
195
		 return results;
196
	}
197

    
198
	@Transactional(readOnly = true)
199
	@Override
200
	public <S extends T> Pager<S> findTitleCache(Class<S> clazz, String queryString, Integer pageSize, Integer pageNumber, List<OrderHint> orderHints, MatchMode matchMode){
201
		long numberOfResults = dao.countTitleCache(clazz, queryString, matchMode);
202

    
203
		 List<S> results = new ArrayList<>();
204
		 if(numberOfResults > 0) { // no point checking again //TODO use AbstractPagerImpl.hasResultsInRange(numberOfResults, pageNumber, pageSize)
205
				results = dao.findTitleCache(clazz, queryString, pageSize, pageNumber, orderHints, matchMode);
206
		 }
207
		 long r = 0;
208
		 r += numberOfResults;
209

    
210
		 return new DefaultPagerImpl<>(pageNumber, r , pageSize, results);
211
	}
212

    
213
    @Transactional(readOnly = true)
214
    @Override
215
    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) {
216
         long numberOfResults = dao.countByReferenceTitle(clazz, queryString, matchmode, criteria);
217

    
218
         List<S> results = new ArrayList<>();
219
         if(numberOfResults > 0) { // no point checking again //TODO use AbstractPagerImpl.hasResultsInRange(numberOfResults, pageNumber, pageSize)
220
             results = dao.findByReferenceTitle(clazz, queryString, matchmode, criteria, pageSize, pageNumber, orderHints, propertyPaths);
221
         }
222
         return results;
223
    }
224

    
225
	@Transactional(readOnly = true)
226
	@Override
227
	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) {
228
		 long numberOfResults = dao.countByReferenceTitleWithRestrictions(clazz, queryString, matchmode, restrictions);
229

    
230
		 List<S> results = new ArrayList<>();
231
		 if(numberOfResults > 0) { // no point checking again //TODO use AbstractPagerImpl.hasResultsInRange(numberOfResults, pageNumber, pageSize)
232
		     results = dao.findByReferenceTitleWithRestrictions(clazz, queryString, matchmode, restrictions, pageSize, pageNumber, orderHints, propertyPaths);
233
		 }
234
		 return results;
235
	}
236

    
237
	@Transactional(readOnly = true)
238
	@Override
239
	public T find(LSID lsid) {
240
		return dao.find(lsid);
241
	}
242

    
243
	@Transactional(readOnly = true)
244
	@Override
245
	public Pager<T> search(Class<? extends T> clazz, String queryString, Integer pageSize, Integer pageNumber, List<OrderHint> orderHints, List<String> propertyPaths) {
246
//	public <S extends T> Pager<S> search(Class<S> clazz, String queryString, Integer pageSize, Integer pageNumber, List<OrderHint> orderHints, List<String> propertyPaths) {
247
        long numberOfResults = dao.count(clazz,queryString);
248

    
249
		List<T> results = new ArrayList<>();
250
		if(numberOfResults > 0) { // no point checking again //TODO use AbstractPagerImpl.hasResultsInRange(numberOfResults, pageNumber, pageSize)
251
			results = dao.search(clazz,queryString, pageSize, pageNumber, orderHints, propertyPaths);
252
		}
253

    
254
		return new DefaultPagerImpl<>(pageNumber, numberOfResults, pageSize, results);
255
	}
256

    
257
	@Override
258
	@Transactional(readOnly = false)
259
	public void updateTitleCache() {
260
		updateTitleCache(null, null, null, null);
261
	}
262

    
263
	@Transactional(readOnly = false)  //TODO check transactional behavior, e.g. what happens with the session if count is very large
264
	protected <S extends T > void updateTitleCacheImpl(Class<S> clazz, Integer stepSize, IIdentifiableEntityCacheStrategy<T> cacheStrategy, IProgressMonitor monitor) {
265
		if (stepSize == null){
266
			stepSize = UPDATE_TITLE_CACHE_DEFAULT_STEP_SIZE;
267
		}
268
		if (monitor == null){
269
			monitor = DefaultProgressMonitor.NewInstance();
270
		}
271

    
272
		long count = dao.count(clazz);
273
		long countUpdated = 0;
274
		monitor.beginTask("update titles for " + clazz.getSimpleName(), Long.valueOf(count).intValue());
275
		int worked = 0;
276
		for(int i = 0 ; i < count ; i = i + stepSize){
277
			// not sure if such strict ordering is necessary here, but for safety reasons I do it
278
			ArrayList<OrderHint> orderHints = new ArrayList<>();
279
			orderHints.add( new OrderHint("id", OrderHint.SortOrder.ASCENDING));
280

    
281

    
282
			Map<Class<? extends CdmBase>, AutoPropertyInitializer<CdmBase>> oldAutoInit = switchOfAutoinitializer();
283
			List<S> list = this.list(clazz, stepSize, i, orderHints, null);
284
			switchOnOldAutoInitializer(oldAutoInit);
285

    
286
			List<T> entitiesToUpdate = new ArrayList<>();
287
			for (T entity : list){
288
				entity = HibernateProxyHelper.deproxy(entity);
289
			    if (entity.updateCaches(cacheStrategy)){
290
			        countUpdated++;
291
			    }
292
				worked++;
293
			}
294

    
295
			monitor.worked(list.size());
296
			if (monitor.isCanceled()){
297
				monitor.done();
298
				return;
299
			}
300
		}
301
		monitor.done();
302
	}
303

    
304
	/**
305
	 * Brings back all auto initializers to the bean initializer
306
	 * @see #switchOfAutoinitializer()
307
	 * @param oldAutoInit
308
	 */
309
	protected void switchOnOldAutoInitializer(
310
			Map<Class<? extends CdmBase>, AutoPropertyInitializer<CdmBase>> oldAutoInit) {
311
		HibernateBeanInitializer initializer = (HibernateBeanInitializer)this.appContext.getBean("defaultBeanInitializer");
312
		initializer.setBeanAutoInitializers(oldAutoInit);
313
	}
314

    
315
	/**
316
	 * Removes all auto initializers from the bean initializer
317
	 *
318
	 * @see #switchOnOldAutoInitializer(Map)
319
	 * @return
320
	 */
321
	protected Map<Class<? extends CdmBase>, AutoPropertyInitializer<CdmBase>> switchOfAutoinitializer() {
322
		HibernateBeanInitializer initializer = (HibernateBeanInitializer)this.appContext.getBean("defaultBeanInitializer");
323
		Map<Class<? extends CdmBase>, AutoPropertyInitializer<CdmBase>> oldAutoInitializers = initializer.getBeanAutoInitializers();
324
		Map<Class<? extends CdmBase>, AutoPropertyInitializer<CdmBase>> map = new HashMap<>();
325
		initializer.setBeanAutoInitializers(map);
326
		return oldAutoInitializers;
327
	}
328

    
329
	private class DeduplicateState{
330
		String lastTitleCache;
331
		Integer pageSize = 50;
332
		int nPages = 3;
333
		int startPage = 0;
334
		boolean isCompleted = false;
335
		int result;
336
	}
337

    
338
	@Override
339
	@Transactional(readOnly = false)
340
	public int deduplicate(Class<? extends T> clazz, IMatchStrategyEqual matchStrategy, IMergeStrategy mergeStrategy) {
341
		DeduplicateState dedupState = new DeduplicateState();
342

    
343
		if (clazz == null){
344
			logger.warn("Deduplication clazz must not be null!");
345
			return 0;
346
		}
347
		if (! ( IMatchable.class.isAssignableFrom(clazz) && IMergable.class.isAssignableFrom(clazz) )  ){
348
			logger.warn("Deduplication implemented only for classes implementing IMatchable and IMergeable. No deduplication performed!");
349
			return 0;
350
		}
351
		Class matchableClass = clazz;
352
		if (matchStrategy == null){
353
			matchStrategy = DefaultMatchStrategy.NewInstance(matchableClass);
354
		}
355
		List<T> nextGroup = new ArrayList<>();
356

    
357
		int result = 0;
358
//		double countTotal = count(clazz);
359
//
360
//		Number countPagesN = Math.ceil(countTotal/dedupState.pageSize.doubleValue()) ;
361
//		int countPages = countPagesN.intValue();
362
//
363

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

    
366
		while (! dedupState.isCompleted){
367
			//get x page sizes
368
			List<? extends T> objectList = getPages(clazz, dedupState, orderHints);
369
			//after each page check if any changes took place
370
			int nUnEqualPages = handleAllPages(objectList, dedupState, nextGroup, matchStrategy, mergeStrategy);
371
			nUnEqualPages = nUnEqualPages + dedupState.pageSize * dedupState.startPage;
372
			//refresh start page counter
373
			int finishedPages = nUnEqualPages / dedupState.pageSize;
374
			dedupState.startPage = finishedPages;
375
		}
376

    
377
		result += handleLastGroup(nextGroup, matchStrategy, mergeStrategy);
378
		return result;
379
	}
380

    
381

    
382
	private int handleAllPages(List<? extends T> objectList, DeduplicateState dedupState, List<T> nextGroup, IMatchStrategyEqual matchStrategy, IMergeStrategy mergeStrategy) {
383
		int nUnEqual = 0;
384
		for (T object : objectList){
385
			String currentTitleCache = object.getTitleCache();
386
			if (currentTitleCache != null && currentTitleCache.equals(dedupState.lastTitleCache)){
387
				//=titleCache
388
				nextGroup.add(object);
389
			}else{
390
				//<> titleCache
391
				dedupState.result += handleLastGroup(nextGroup, matchStrategy, mergeStrategy);
392
				nextGroup = new ArrayList<>();
393
				nextGroup.add(object);
394
				nUnEqual++;
395
			}
396
			dedupState.lastTitleCache = currentTitleCache;
397
		}
398
		handleLastGroup(nextGroup, matchStrategy, mergeStrategy);
399
		return nUnEqual;
400
	}
401

    
402
	private <S extends T> List<S> getPages(Class<S> clazz, DeduplicateState dedupState, List<OrderHint> orderHints) {
403
		List<S> result = new ArrayList<>();
404
		for (int pageNo = dedupState.startPage; pageNo < dedupState.startPage + dedupState.nPages; pageNo++){
405
			List<S> objectList = listByTitleWithRestrictions(clazz, null, null, null, dedupState.pageSize, pageNo, orderHints, null);
406
			result.addAll(objectList);
407
		}
408
		if (result.size()< dedupState.nPages * dedupState.pageSize ){
409
			dedupState.isCompleted = true;
410
		}
411
		return result;
412
	}
413

    
414
	private int handleLastGroup(List<T> group, IMatchStrategyEqual matchStrategy, IMergeStrategy mergeStrategy) {
415
		int result = 0;
416
		int size = group.size();
417
		Set<Integer> exclude = new HashSet<>();  //set to collect all objects, that have been merged already
418
		for (int i = 0; i < size - 1; i++){
419
			if (exclude.contains(i)){
420
				continue;
421
			}
422
			for (int j = i + 1; j < size; j++){
423
				if (exclude.contains(j)){
424
					continue;
425
				}
426
				T firstObject = group.get(i);
427
				T secondObject = group.get(j);
428

    
429
				try {
430
					if (matchStrategy.invoke((IMatchable)firstObject, (IMatchable)secondObject).isSuccessful()){
431
						commonService.merge((IMergable)firstObject, (IMergable)secondObject, mergeStrategy);
432
						exclude.add(j);
433
						result++;
434
					}
435
				} catch (MatchException e) {
436
					logger.warn("MatchException when trying to match " + firstObject.getTitleCache());
437
					e.printStackTrace();
438
				} catch (MergeException e) {
439
					logger.warn("MergeException when trying to merge " + firstObject.getTitleCache());
440
					e.printStackTrace();
441
				}
442
			}
443
		}
444
		return result;
445
	}
446

    
447
    @Transactional(readOnly = true)
448
    @Override
449
    public long countByTitle(Class<? extends T> clazz, String queryString,MatchMode matchmode, List<Criterion> criteria){
450
         long numberOfResults = dao.countByTitle(clazz, queryString, matchmode, criteria);
451

    
452
         return numberOfResults;
453
    }
454

    
455
	@Transactional(readOnly = true)
456
	@Override
457
	public long countByTitleWithRestrictions(Class<? extends T> clazz, String queryString, MatchMode matchmode,  List<Restriction<?>> restrictions){
458
		 long numberOfResults = dao.countByTitleWithRestrictions(clazz, queryString, matchmode, restrictions);
459

    
460
		 return numberOfResults;
461
	}
462

    
463
	@Transactional(readOnly = true)
464
	@Override
465
	public long countByTitle(IIdentifiableEntityServiceConfigurator<T> config){
466

    
467
        boolean withRestrictions = config.getRestrictions() != null && !config.getRestrictions().isEmpty();
468
        boolean withCriteria = config.getCriteria() != null && !config.getCriteria().isEmpty();
469

    
470
        if(withCriteria && withRestrictions){
471
            throw new RuntimeException("Restrictions and Criteria can not be used at the same time");
472
        } else if(withRestrictions){
473
            return countByTitleWithRestrictions(config.getClazz(), config.getTitleSearchStringSqlized(), config.getMatchMode(), config.getRestrictions());
474
        } else {
475
            return countByTitle(config.getClazz(), config.getTitleSearchStringSqlized(), config.getMatchMode(), config.getCriteria());
476
        }
477

    
478
	}
479

    
480
	@Override
481
	@Transactional(readOnly = true)
482
	public <S extends T> Pager<IdentifiedEntityDTO<S>> findByIdentifier(
483
			Class<S> clazz, String identifier, DefinedTerm identifierType, MatchMode matchmode,
484
			boolean includeEntity, Integer pageSize,
485
			Integer pageNumber,	List<String> propertyPaths) {
486

    
487
		long numberOfResults = dao.countByIdentifier(clazz, identifier, identifierType, matchmode);
488
        List<Object[]> daoResults = new ArrayList<>();
489
        if(numberOfResults > 0) { // no point checking again
490
        	daoResults = dao.findByIdentifier(clazz, identifier, identifierType,
491
    				matchmode, includeEntity, pageSize, pageNumber, propertyPaths);
492
        }
493

    
494
        List<IdentifiedEntityDTO<S>> result = new ArrayList<>();
495
        for (Object[] daoObj : daoResults){
496
        	if (includeEntity){
497
        		result.add(new IdentifiedEntityDTO<>((DefinedTerm)daoObj[0], (String)daoObj[1], (S)daoObj[2]));
498
        	}else{
499
        		result.add(new IdentifiedEntityDTO<>((DefinedTerm)daoObj[0], (String)daoObj[1], (UUID)daoObj[2], (String)daoObj[3], null));
500
        	}
501
        }
502
		return new DefaultPagerImpl<>(pageNumber, numberOfResults, pageSize, result);
503
	}
504

    
505
	@Override
506
    @Transactional(readOnly = true)
507
    public <S extends T> List<IdentifiedEntityDTO<S>> listByIdentifier(
508
            Class<S> clazz, String identifier, DefinedTerm identifierType, MatchMode matchmode,
509
            boolean includeEntity, List<String> propertyPaths, Integer limit) {
510

    
511
        long numberOfResults = dao.countByIdentifier(clazz, identifier, identifierType, matchmode);
512
        List<Object[]> daoResults = new ArrayList<>();
513
        if(numberOfResults > 0) { // no point checking again
514
            daoResults = dao.findByIdentifier(clazz, identifier, identifierType,
515
                    matchmode, includeEntity, limit, 0, propertyPaths);
516
        }
517

    
518
        List<IdentifiedEntityDTO<S>> result = new ArrayList<>();
519
        for (Object[] daoObj : daoResults){
520
            if (includeEntity){
521
                result.add(new IdentifiedEntityDTO<>((DefinedTerm)daoObj[0], (String)daoObj[1], (S)daoObj[2]));
522
            }else{
523
                result.add(new IdentifiedEntityDTO<>((DefinedTerm)daoObj[0], (String)daoObj[1], (UUID)daoObj[2], (String)daoObj[3], null));
524
            }
525
        }
526
        return result;
527
    }
528

    
529

    
530
    @Override
531
    @Transactional(readOnly = true)
532
    public <S extends T> Pager<MarkedEntityDTO<S>> findByMarker(
533
            Class<S> clazz, MarkerType markerType, Boolean markerValue,
534
            boolean includeEntity, Integer pageSize,
535
            Integer pageNumber, List<String> propertyPaths) {
536

    
537
        Long numberOfResults = dao.countByMarker(clazz, markerType, markerValue);
538
        List<Object[]> daoResults = new ArrayList<>();
539
        if(numberOfResults > 0) { // no point checking again
540
            daoResults = dao.findByMarker(clazz, markerType, markerValue, includeEntity,
541
                    pageSize, pageNumber, propertyPaths);
542
        }
543

    
544
        List<MarkedEntityDTO<S>> result = new ArrayList<>();
545
        for (Object[] daoObj : daoResults){
546
            if (includeEntity){
547
                result.add(new MarkedEntityDTO<>((MarkerType)daoObj[0], (Boolean)daoObj[1], (S)daoObj[2]));
548
            }else{
549
                result.add(new MarkedEntityDTO<>((MarkerType)daoObj[0], (Boolean)daoObj[1], (UUID)daoObj[2], (String)daoObj[3]));
550
            }
551
        }
552
        return new DefaultPagerImpl<>(pageNumber, numberOfResults, pageSize, result);
553
    }
554

    
555
    @Override
556
    @Transactional(readOnly = true)
557
    public List<UuidAndTitleCache<T>> findUuidAndTitleCacheByMarker(Integer limit, String pattern,
558
            MarkerType markerType){
559

    
560
        Long numberOfResults = dao.countByMarker(null, markerType, null);
561
        List<UuidAndTitleCache<T>> daoResults = new ArrayList<>();
562
        if(numberOfResults > 0) { // no point checking again
563
            daoResults = dao.getUuidAndTitleCacheByMarker(limit, pattern, markerType);
564
        }
565

    
566

    
567
        return daoResults;
568
    }
569

    
570

    
571

    
572
}
573

    
(71-71/103)