2 * Copyright (C) 2007 EDIT
3 * European Distributed Institute of Taxonomy
4 * http://www.e-taxonomy.eu
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.
10 package eu
.etaxonomy
.cdm
.api
.service
;
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
;
19 import java
.util
.UUID
;
21 import org
.apache
.log4j
.Logger
;
22 import org
.hibernate
.criterion
.Criterion
;
23 import org
.springframework
.transaction
.annotation
.Transactional
;
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
;
62 public abstract class IdentifiableServiceBase
<T
extends IdentifiableEntity
, DAO
extends IIdentifiableDao
<T
>> extends AnnotatableServiceBase
<T
,DAO
>
63 implements IIdentifiableEntityService
<T
>{
66 protected static final int UPDATE_TITLE_CACHE_DEFAULT_STEP_SIZE
= 1000;
67 protected static final Logger logger
= Logger
.getLogger(IdentifiableServiceBase
.class);
70 @Transactional(readOnly
= true)
71 public Pager
<Rights
> getRights(T t
, Integer pageSize
, Integer pageNumber
, List
<String
> propertyPaths
) {
72 Integer numberOfResults
= dao
.countRights(t
);
74 List
<Rights
> results
= new ArrayList
<Rights
>();
75 if(numberOfResults
> 0) { // no point checking again //TODO use AbstractPagerImpl.hasResultsInRange(numberOfResults, pageNumber, pageSize)
76 results
= dao
.getRights(t
, pageSize
, pageNumber
,propertyPaths
);
79 return new DefaultPagerImpl
<Rights
>(pageNumber
, numberOfResults
, pageSize
, results
);
83 @Transactional(readOnly
= true)
84 public Pager
<IdentifiableSource
> getSources(T t
, Integer pageSize
, Integer pageNumber
, List
<String
> propertyPaths
) {
85 Integer numberOfResults
= dao
.countSources(t
);
87 List
<IdentifiableSource
> results
= new ArrayList
<IdentifiableSource
>();
88 if(numberOfResults
> 0) { // no point checking again //TODO use AbstractPagerImpl.hasResultsInRange(numberOfResults, pageNumber, pageSize)
89 results
= dao
.getSources(t
, pageSize
, pageNumber
,propertyPaths
);
92 return new DefaultPagerImpl
<IdentifiableSource
>(pageNumber
, numberOfResults
, pageSize
, results
);
96 @Transactional(readOnly
= false)
98 public T
replace(T x
, T y
) {
99 return dao
.replace(x
, y
);
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.
110 @Transactional(readOnly
= true)
111 protected List
<T
> findCdmObjectsByTitle(String title
){
112 return ((IIdentifiableDao
)dao
).findByTitle(title
);
115 @Transactional(readOnly
= true)
116 protected List
<T
> findCdmObjectsByTitle(String title
, Class
<T
> clazz
){
117 return ((IIdentifiableDao
)dao
).findByTitleAndClass(title
, clazz
);
119 @Transactional(readOnly
= true)
120 protected List
<T
> findCdmObjectsByTitle(String title
, CdmBase sessionObject
){
121 return ((IIdentifiableDao
)dao
).findByTitle(title
, sessionObject
);
125 * TODO - Migrated from CommonServiceBase
127 * @see eu.etaxonomy.cdm.api.service.ICommonService#getSourcedObjectById(java.lang.String, java.lang.String)
129 @Transactional(readOnly
= true)
131 public ISourceable
getSourcedObjectByIdInSource(Class clazz
, String idInSource
, String idNamespace
) {
132 ISourceable result
= null;
134 List
<T
> list
= dao
.findOriginalSourceByIdInSource(idInSource
, idNamespace
);
135 if (! list
.isEmpty()){
136 result
= list
.get(0);
141 @Transactional(readOnly
= true)
143 public List
<UuidAndTitleCache
<T
>> getUuidAndTitleCache(Integer limit
, String pattern
) {
144 return dao
.getUuidAndTitleCache(limit
, pattern
);
147 @Transactional(readOnly
= true)
149 public <S
extends T
> List
<UuidAndTitleCache
<S
>> getUuidAndTitleCache(Class
<S
> clazz
,Integer limit
, String pattern
) {
150 return dao
.getUuidAndTitleCache(clazz
, limit
, pattern
);
153 @Transactional(readOnly
= true)
155 public String
getTitleCache(UUID uuid
, boolean refresh
){
156 return dao
.getTitleCache(uuid
, refresh
);
159 @Transactional(readOnly
= true)
161 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
) {
162 long numberOfResults
= dao
.countByTitle(clazz
, queryString
, matchmode
, criteria
);
164 List
<T
> results
= new ArrayList
<T
>();
165 if(numberOfResults
> 0) { // no point checking again //TODO use AbstractPagerImpl.hasResultsInRange(numberOfResults, pageNumber, pageSize)
166 results
= dao
.findByTitle(clazz
, queryString
, matchmode
, criteria
, pageSize
, pageNumber
, orderHints
, propertyPaths
);
169 return new DefaultPagerImpl
<T
>(pageNumber
, numberOfResults
, pageSize
, results
);
172 @Transactional(readOnly
= true)
174 public Pager
<T
> findByTitle(IIdentifiableEntityServiceConfigurator
<T
> config
){
175 return findByTitle(config
.getClazz(), config
.getTitleSearchStringSqlized(), config
.getMatchMode(), config
.getCriteria(), config
.getPageSize(), config
.getPageNumber(), config
.getOrderHints(), config
.getPropertyPaths());
178 @Transactional(readOnly
= true)
180 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
) {
181 long numberOfResults
= dao
.countByTitle(clazz
, queryString
, matchmode
, criteria
);
183 List
<T
> results
= new ArrayList
<T
>();
184 if(numberOfResults
> 0) { // no point checking again //TODO use AbstractPagerImpl.hasResultsInRange(numberOfResults, pageNumber, pageSize)
185 results
= dao
.findByTitle(clazz
, queryString
, matchmode
, criteria
, pageSize
, pageNumber
, orderHints
, propertyPaths
);
190 @Transactional(readOnly
= true)
192 public Pager
<T
> findTitleCache(Class
<?
extends T
> clazz
, String queryString
, Integer pageSize
, Integer pageNumber
, List
<OrderHint
> orderHints
, MatchMode matchMode
){
193 long numberOfResults
= dao
.countTitleCache(clazz
, queryString
, matchMode
);
195 List
<T
> results
= new ArrayList
<T
>();
196 if(numberOfResults
> 0) { // no point checking again //TODO use AbstractPagerImpl.hasResultsInRange(numberOfResults, pageNumber, pageSize)
197 results
= dao
.findTitleCache(clazz
, queryString
, pageSize
, pageNumber
, orderHints
, matchMode
);
200 r
+= numberOfResults
;
202 return new DefaultPagerImpl
<T
>(pageNumber
, r
, pageSize
, results
);
205 @Transactional(readOnly
= true)
207 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
) {
208 long numberOfResults
= dao
.countByReferenceTitle(clazz
, queryString
, matchmode
, criteria
);
210 List
<T
> results
= new ArrayList
<T
>();
211 if(numberOfResults
> 0) { // no point checking again //TODO use AbstractPagerImpl.hasResultsInRange(numberOfResults, pageNumber, pageSize)
212 results
= dao
.findByReferenceTitle(clazz
, queryString
, matchmode
, criteria
, pageSize
, pageNumber
, orderHints
, propertyPaths
);
217 @Transactional(readOnly
= true)
219 public T
find(LSID lsid
) {
220 return dao
.find(lsid
);
223 @Transactional(readOnly
= true)
225 public Pager
<T
> search(Class
<?
extends T
> clazz
, String queryString
, Integer pageSize
, Integer pageNumber
, List
<OrderHint
> orderHints
, List
<String
> propertyPaths
) {
226 Integer numberOfResults
= dao
.count(clazz
,queryString
);
228 List
<T
> results
= new ArrayList
<T
>();
229 if(numberOfResults
> 0) { // no point checking again //TODO use AbstractPagerImpl.hasResultsInRange(numberOfResults, pageNumber, pageSize)
230 results
= dao
.search(clazz
,queryString
, pageSize
, pageNumber
, orderHints
, propertyPaths
);
233 return new DefaultPagerImpl
<T
>(pageNumber
, numberOfResults
, pageSize
, results
);
237 @Transactional(readOnly
= false)
238 public void updateTitleCache() {
239 updateTitleCache(null, null, null, null);
242 @Transactional(readOnly
= false) //TODO check transactional behaviour, e.g. what happens with the session if count is very large
243 protected <S
extends T
> void updateTitleCacheImpl(Class
<S
> clazz
, Integer stepSize
, IIdentifiableEntityCacheStrategy
<T
> cacheStrategy
, IProgressMonitor monitor
) {
244 if (stepSize
== null){
245 stepSize
= UPDATE_TITLE_CACHE_DEFAULT_STEP_SIZE
;
247 if (monitor
== null){
248 monitor
= DefaultProgressMonitor
.NewInstance();
251 int count
= dao
.count(clazz
);
252 monitor
.beginTask("update titles", count
);
254 for(int i
= 0 ; i
< count
; i
= i
+ stepSize
){
255 // not sure if such strict ordering is necessary here, but for safety reasons I do it
256 ArrayList
<OrderHint
> orderHints
= new ArrayList
<OrderHint
>();
257 orderHints
.add( new OrderHint("id", OrderHint
.SortOrder
.ASCENDING
));
260 Map
<Class
<?
extends CdmBase
>, AutoPropertyInitializer
<CdmBase
>> oldAutoInit
= switchOfAutoinitializer();
261 List
<S
> list
= this.list(clazz
, stepSize
, i
, orderHints
, null);
262 switchOnOldAutoInitializer(oldAutoInit
);
264 List
<T
> entitiesToUpdate
= new ArrayList
<T
>();
265 for (T entity
: list
){
266 HibernateProxyHelper
.deproxy(entity
, clazz
);
267 if (entity
.isProtectedTitleCache() == false){
268 updateTitleCacheForSingleEntity(cacheStrategy
, entitiesToUpdate
, entity
);
272 for (T entity
: entitiesToUpdate
){
273 if (entity
.getTitleCache() != null){
274 //System.err.println(entity.getTitleCache());
276 //System.err.println("no titleCache" + ((TaxonName)entity).getNameCache());
279 saveOrUpdate(entitiesToUpdate
);
280 monitor
.worked(list
.size());
281 if (monitor
.isCanceled()){
290 * Brings back all auto initializers to the bean initializer
291 * @see #switchOfAutoinitializer()
294 protected void switchOnOldAutoInitializer(
295 Map
<Class
<?
extends CdmBase
>, AutoPropertyInitializer
<CdmBase
>> oldAutoInit
) {
296 HibernateBeanInitializer initializer
= (HibernateBeanInitializer
)this.appContext
.getBean("defaultBeanInitializer");
297 initializer
.setBeanAutoInitializers(oldAutoInit
);
301 * Removes all auto initializers from the bean initializer
303 * @see #switchOnOldAutoInitializer(Map)
306 protected Map
<Class
<?
extends CdmBase
>, AutoPropertyInitializer
<CdmBase
>> switchOfAutoinitializer() {
307 HibernateBeanInitializer initializer
= (HibernateBeanInitializer
)this.appContext
.getBean("defaultBeanInitializer");
308 Map
<Class
<?
extends CdmBase
>, AutoPropertyInitializer
<CdmBase
>> oldAutoInitializers
= initializer
.getBeanAutoInitializers();
309 Map
<Class
<?
extends CdmBase
>, AutoPropertyInitializer
<CdmBase
>> map
= new HashMap
<Class
<?
extends CdmBase
>, AutoPropertyInitializer
<CdmBase
>>();
310 initializer
.setBeanAutoInitializers(map
);
311 return oldAutoInitializers
;
315 * @param cacheStrategy
316 * @param entitiesToUpdate
320 * @param cacheStrategy
321 * @param entitiesToUpdate
324 @SuppressWarnings("unchecked")
325 private void updateTitleCacheForSingleEntity(
326 IIdentifiableEntityCacheStrategy
<T
> cacheStrategy
,
327 List
<T
> entitiesToUpdate
,
330 //assert (entity.isProtectedTitleCache() == false );
331 entity
= CdmBase
.deproxy(entity
);
333 //exclude recursive inreferences
334 if (entity
.isInstanceOf(Reference
.class)){
335 Reference ref
= (Reference
)entity
;
336 if (ref
.getInReference() != null && ref
.getInReference().equals(ref
)){
342 //define the correct cache strategy
343 IIdentifiableEntityCacheStrategy entityCacheStrategy
= cacheStrategy
;
344 if (entityCacheStrategy
== null){
345 entityCacheStrategy
= entity
.getCacheStrategy();
346 //FIXME find out why the wrong cache strategy is loaded here, see #1876
347 if (entity
instanceof Reference
){
348 entityCacheStrategy
= ReferenceFactory
.newReference(((Reference
)entity
).getType()).getCacheStrategy();
354 entity
.setProtectedTitleCache(true);
356 String oldTitleCache
= entity
.getTitleCache();
357 entity
.setTitleCache(oldTitleCache
, false); //before we had entity.setProtectedTitleCache(false) but this deleted the titleCache itself
358 entity
.setCacheStrategy(entityCacheStrategy
);
359 //NonViralNames and Reference have more caches //TODO handle in NameService
360 String oldNameCache
= null;
361 String oldFullTitleCache
= null;
362 String oldAbbrevTitleCache
= null;
363 if (entity
instanceof TaxonName
){
364 if (((TaxonName
) entity
).isNonViral()) {
366 INonViralName nvn
= (INonViralName
) entity
;
367 if (!nvn
.isProtectedNameCache()){
368 nvn
.setProtectedNameCache(true);
369 oldNameCache
= nvn
.getNameCache();
370 nvn
.setProtectedNameCache(false);
372 if (!nvn
.isProtectedFullTitleCache()){
373 nvn
.setProtectedFullTitleCache(true);
374 oldFullTitleCache
= nvn
.getFullTitleCache();
375 nvn
.setProtectedFullTitleCache(false);
377 }catch(ClassCastException e
){
378 System
.out
.println("entity: " + entity
.getTitleCache());
382 }else if (entity
instanceof Reference
){
383 Reference ref
= (Reference
) entity
;
384 if (!ref
.isProtectedAbbrevTitleCache()){
385 ref
.setProtectedAbbrevTitleCache(true);
386 oldAbbrevTitleCache
= ref
.getAbbrevTitleCache();
387 ref
.setProtectedAbbrevTitleCache(false);
390 setOtherCachesNull(entity
);
391 String newTitleCache
= null;
392 INonViralName nvn
= null; //TODO find better solution
394 if (entity
instanceof TaxonName
){
395 nvn
= (ITaxonName
) entity
;
396 newTitleCache
= entityCacheStrategy
.getTitleCache(nvn
);
398 newTitleCache
= entityCacheStrategy
.getTitleCache(entity
);
400 }catch (ClassCastException e
){
401 nvn
= HibernateProxyHelper
.deproxy(entity
, TaxonName
.class);
402 newTitleCache
= entityCacheStrategy
.getTitleCache(nvn
);
405 if ( oldTitleCache
== null || ! oldTitleCache
.equals(newTitleCache
) ){
406 entity
.setTitleCache(null, false);
407 String newCache
= entity
.getTitleCache();
409 if (newCache
== null){
410 logger
.warn("newCache should never be null");
412 if (oldTitleCache
== null){
413 logger
.info("oldTitleCache should never be null");
417 nvn
.getFullTitleCache();
419 if (entity
instanceof Reference
){
420 Reference ref
= (Reference
) entity
;
421 ref
.getAbbrevTitleCache();
423 entitiesToUpdate
.add(entity
);
424 }else if (nvn
!= null){
425 String newNameCache
= nvn
.getNameCache();
426 String newFullTitleCache
= nvn
.getFullTitleCache();
427 if ((oldNameCache
== null && !nvn
.isProtectedNameCache()) || (oldNameCache
!= null && !oldNameCache
.equals(newNameCache
))){
428 entitiesToUpdate
.add(entity
);
429 }else if ((oldFullTitleCache
== null && !nvn
.isProtectedFullTitleCache()) || (oldFullTitleCache
!= null && !oldFullTitleCache
.equals(newFullTitleCache
))){
430 entitiesToUpdate
.add(entity
);
432 }else if (entity
instanceof Reference
){
433 Reference ref
= (Reference
) entity
;
434 String newAbbrevTitleCache
= ref
.getAbbrevTitleCache();
435 if ( (oldAbbrevTitleCache
== null && !ref
.isProtectedAbbrevTitleCache() ) || (oldAbbrevTitleCache
!= null && !oldAbbrevTitleCache
.equals(newAbbrevTitleCache
))){
436 entitiesToUpdate
.add(entity
);
444 * Needs override if not only the title cache should be set to null to
445 * generate the correct new title cache
447 protected void setOtherCachesNull(T entity
) {
451 private class DeduplicateState
{
452 String lastTitleCache
;
453 Integer pageSize
= 50;
456 boolean isCompleted
= false;
461 @Transactional(readOnly
= false)
462 public int deduplicate(Class
<?
extends T
> clazz
, IMatchStrategy matchStrategy
, IMergeStrategy mergeStrategy
) {
463 DeduplicateState dedupState
= new DeduplicateState();
466 logger
.warn("Deduplication clazz must not be null!");
469 if (! ( IMatchable
.class.isAssignableFrom(clazz
) && IMergable
.class.isAssignableFrom(clazz
) ) ){
470 logger
.warn("Deduplication implemented only for classes implementing IMatchable and IMergeable. No deduplication performed!");
473 Class matchableClass
= clazz
;
474 if (matchStrategy
== null){
475 matchStrategy
= DefaultMatchStrategy
.NewInstance(matchableClass
);
477 List
<T
> nextGroup
= new ArrayList
<T
>();
480 // double countTotal = count(clazz);
482 // Number countPagesN = Math.ceil(countTotal/dedupState.pageSize.doubleValue()) ;
483 // int countPages = countPagesN.intValue();
486 List
<OrderHint
> orderHints
= Arrays
.asList(new OrderHint
[]{new OrderHint("titleCache", SortOrder
.ASCENDING
)});
488 while (! dedupState
.isCompleted
){
490 List
<T
> objectList
= getPages(clazz
, dedupState
, orderHints
);
491 //after each page check if any changes took place
492 int nUnEqualPages
= handleAllPages(objectList
, dedupState
, nextGroup
, matchStrategy
, mergeStrategy
);
493 nUnEqualPages
= nUnEqualPages
+ dedupState
.pageSize
* dedupState
.startPage
;
494 //refresh start page counter
495 int finishedPages
= nUnEqualPages
/ dedupState
.pageSize
;
496 dedupState
.startPage
= finishedPages
;
499 result
+= handleLastGroup(nextGroup
, matchStrategy
, mergeStrategy
);
504 private int handleAllPages(List
<T
> objectList
, DeduplicateState dedupState
, List
<T
> nextGroup
, IMatchStrategy matchStrategy
, IMergeStrategy mergeStrategy
) {
506 for (T object
: objectList
){
507 String currentTitleCache
= object
.getTitleCache();
508 if (currentTitleCache
!= null && currentTitleCache
.equals(dedupState
.lastTitleCache
)){
510 nextGroup
.add(object
);
513 dedupState
.result
+= handleLastGroup(nextGroup
, matchStrategy
, mergeStrategy
);
514 nextGroup
= new ArrayList
<T
>();
515 nextGroup
.add(object
);
518 dedupState
.lastTitleCache
= currentTitleCache
;
520 handleLastGroup(nextGroup
, matchStrategy
, mergeStrategy
);
524 private List
<T
> getPages(Class
<?
extends T
> clazz
, DeduplicateState dedupState
, List
<OrderHint
> orderHints
) {
525 List
<T
> result
= new ArrayList
<T
>();
526 for (int pageNo
= dedupState
.startPage
; pageNo
< dedupState
.startPage
+ dedupState
.nPages
; pageNo
++){
527 List
<T
> objectList
= listByTitle(clazz
, null, null, null, dedupState
.pageSize
, pageNo
, orderHints
, null);
528 result
.addAll(objectList
);
530 if (result
.size()< dedupState
.nPages
* dedupState
.pageSize
){
531 dedupState
.isCompleted
= true;
536 private int handleLastGroup(List
<T
> group
, IMatchStrategy matchStrategy
, IMergeStrategy mergeStrategy
) {
538 int size
= group
.size();
539 Set
<Integer
> exclude
= new HashSet
<Integer
>(); //set to collect all objects, that have been merged already
540 for (int i
= 0; i
< size
- 1; i
++){
541 if (exclude
.contains(i
)){
544 for (int j
= i
+ 1; j
< size
; j
++){
545 if (exclude
.contains(j
)){
548 T firstObject
= group
.get(i
);
549 T secondObject
= group
.get(j
);
552 if (matchStrategy
.invoke((IMatchable
)firstObject
, (IMatchable
)secondObject
)){
553 commonService
.merge((IMergable
)firstObject
, (IMergable
)secondObject
, mergeStrategy
);
557 } catch (MatchException e
) {
558 logger
.warn("MatchException when trying to match " + firstObject
.getTitleCache());
560 } catch (MergeException e
) {
561 logger
.warn("MergeException when trying to merge " + firstObject
.getTitleCache());
569 @Transactional(readOnly
= true)
571 public Integer
countByTitle(Class
<?
extends T
> clazz
, String queryString
,MatchMode matchmode
, List
<Criterion
> criteria
){
572 long numberOfResults
= dao
.countByTitle(clazz
, queryString
, matchmode
, criteria
);
574 return ((Number
)numberOfResults
).intValue();
577 @Transactional(readOnly
= true)
579 public Integer
countByTitle(IIdentifiableEntityServiceConfigurator
<T
> config
){
580 return countByTitle(config
.getClazz(), config
.getTitleSearchStringSqlized(),
581 config
.getMatchMode(), config
.getCriteria());
586 @Transactional(readOnly
= true)
587 public <S
extends T
> Pager
<IdentifiedEntityDTO
<S
>> findByIdentifier(
588 Class
<S
> clazz
, String identifier
, DefinedTerm identifierType
, MatchMode matchmode
,
589 boolean includeEntity
, Integer pageSize
,
590 Integer pageNumber
, List
<String
> propertyPaths
) {
592 Integer numberOfResults
= dao
.countByIdentifier(clazz
, identifier
, identifierType
, matchmode
);
593 List
<Object
[]> daoResults
= new ArrayList
<Object
[]>();
594 if(numberOfResults
> 0) { // no point checking again
595 daoResults
= dao
.findByIdentifier(clazz
, identifier
, identifierType
,
596 matchmode
, includeEntity
, pageSize
, pageNumber
, propertyPaths
);
599 List
<IdentifiedEntityDTO
<S
>> result
= new ArrayList
<IdentifiedEntityDTO
<S
>>();
600 for (Object
[] daoObj
: daoResults
){
602 result
.add(new IdentifiedEntityDTO
<S
>((DefinedTerm
)daoObj
[0], (String
)daoObj
[1], (S
)daoObj
[2]));
604 result
.add(new IdentifiedEntityDTO
<S
>((DefinedTerm
)daoObj
[0], (String
)daoObj
[1], (UUID
)daoObj
[2], (String
)daoObj
[3]));
607 return new DefaultPagerImpl
<IdentifiedEntityDTO
<S
>>(pageNumber
, numberOfResults
, pageSize
, result
);
611 @Transactional(readOnly
= true)
612 public <S
extends T
> Pager
<MarkedEntityDTO
<S
>> findByMarker(
613 Class
<S
> clazz
, MarkerType markerType
, Boolean markerValue
,
614 boolean includeEntity
, Integer pageSize
,
615 Integer pageNumber
, List
<String
> propertyPaths
) {
617 Long numberOfResults
= dao
.countByMarker(clazz
, markerType
, markerValue
);
618 List
<Object
[]> daoResults
= new ArrayList
<>();
619 if(numberOfResults
> 0) { // no point checking again
620 daoResults
= dao
.findByMarker(clazz
, markerType
, markerValue
, includeEntity
,
621 pageSize
, pageNumber
, propertyPaths
);
624 List
<MarkedEntityDTO
<S
>> result
= new ArrayList
<>();
625 for (Object
[] daoObj
: daoResults
){
627 result
.add(new MarkedEntityDTO
<S
>((MarkerType
)daoObj
[0], (Boolean
)daoObj
[1], (S
)daoObj
[2]));
629 result
.add(new MarkedEntityDTO
<S
>((MarkerType
)daoObj
[0], (Boolean
)daoObj
[1], (UUID
)daoObj
[2], (String
)daoObj
[3]));
632 return new DefaultPagerImpl
<MarkedEntityDTO
<S
>>(pageNumber
, numberOfResults
, pageSize
, result
);