AccessionServiceImpl.java
/*
* Copyright 2020 Global Crop Diversity Trust
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.gringlobal.service.impl;
import java.io.IOException;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.concurrent.TimeUnit;
import java.util.stream.Collectors;
import lombok.extern.slf4j.Slf4j;
import org.apache.commons.collections4.CollectionUtils;
import org.apache.commons.lang3.StringUtils;
import org.gringlobal.api.exception.InvalidApiUsageException;
import org.gringlobal.api.exception.NotFoundElement;
import org.gringlobal.api.v1.MultiOp;
import org.gringlobal.application.config.GGCESecurityConfig;
import org.gringlobal.component.GGCE;
import org.gringlobal.custom.elasticsearch.SearchException;
import org.gringlobal.model.Accession;
import org.gringlobal.model.AccessionAction;
import org.gringlobal.model.AccessionInvGroup;
import org.gringlobal.model.AccessionInvName;
import org.gringlobal.model.AccessionSource;
import org.gringlobal.model.AccessionSourceMap;
import org.gringlobal.model.Inventory;
import org.gringlobal.model.InventoryMaintenancePolicy;
import org.gringlobal.model.Method;
import org.gringlobal.model.QAccession;
import org.gringlobal.model.QAccessionAction;
import org.gringlobal.model.QAccessionInvGroup;
import org.gringlobal.model.QAccessionInvName;
import org.gringlobal.model.QAccessionPedigree;
import org.gringlobal.model.QAccessionSource;
import org.gringlobal.model.QInventory;
import org.gringlobal.model.Site;
import org.gringlobal.model.community.AccessionMCPD;
import org.gringlobal.model.community.CommunityCodeValues;
import org.gringlobal.model.workflow.WorkflowActionStep;
import org.gringlobal.persistence.AccessionActionRepository;
import org.gringlobal.persistence.AccessionInvAnnotationRepository;
import org.gringlobal.persistence.AccessionInvAttachRepository;
import org.gringlobal.persistence.AccessionInvGroupRepository;
import org.gringlobal.persistence.AccessionInvNameRepository;
import org.gringlobal.persistence.AccessionRepository;
import org.gringlobal.persistence.AccessionSourceRepository;
import org.gringlobal.persistence.InventoryMaintenancePolicyRepository;
import org.gringlobal.persistence.InventoryRepository;
import org.gringlobal.persistence.MethodRepository;
import org.gringlobal.persistence.SiteRepository;
import org.gringlobal.service.AccessionActionService;
import org.gringlobal.service.AccessionActionService.AccessionActionRequest;
import org.gringlobal.service.AccessionActionService.AccessionActionScheduleFilter;
import org.gringlobal.service.ShortFilterService.FilterInfo;
import org.gringlobal.service.AccessionInvGroupService;
import org.gringlobal.service.AccessionService;
import org.gringlobal.service.AccessionSourceMapService;
import org.gringlobal.service.InventoryAttachmentService;
import org.gringlobal.service.InventoryService;
import org.gringlobal.service.MethodService;
import org.gringlobal.service.filter.AccessionActionFilter;
import org.gringlobal.service.filter.AccessionFilter;
import org.gringlobal.spring.TransactionHelper;
import org.gringlobal.util.CoordUtil;
import org.gringlobal.worker.AccessionMCPDConverter;
import org.hibernate.Hibernate;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.annotation.Lazy;
import org.springframework.data.domain.Page;
import org.springframework.data.domain.PageImpl;
import org.springframework.data.domain.PageRequest;
import org.springframework.data.domain.Pageable;
import org.springframework.security.access.AccessDeniedException;
import org.springframework.security.access.prepost.PostAuthorize;
import org.springframework.security.access.prepost.PreAuthorize;
import org.springframework.stereotype.Component;
import org.springframework.stereotype.Service;
import org.springframework.transaction.annotation.Transactional;
import com.google.common.base.Stopwatch;
import com.google.common.collect.Lists;
import com.querydsl.core.types.EntityPath;
import com.querydsl.core.types.Predicate;
import com.querydsl.core.types.dsl.NumberPath;
import com.querydsl.jpa.impl.JPAQuery;
import com.querydsl.jpa.impl.JPAQueryFactory;
import com.querydsl.core.types.ExpressionUtils;
@Service
@Transactional(readOnly = true)
@Slf4j
public class AccessionServiceImpl extends FilteredCRUDService2Impl<Accession, AccessionFilter, AccessionRepository> implements AccessionService {
private static final String[] BOOST_FIELDS = { "accessionNumber", "names.plantName", "taxonomySpecies.name" };
@Autowired
protected JPAQueryFactory jpaQueryFactory;
@Autowired
private InventoryService inventoryService;
@Autowired
private AccessionSourceRepository accessionSourceRepository;
@Autowired
private AccessionInvNameRepository accessionInvNameRepository;
@Autowired
private AccessionInvAttachRepository accessionInvAttachRepository;
@Autowired
private AccessionInvAnnotationRepository annotationRepository;
@Autowired
private AccessionInvGroupService accessionInvGroupService;
@Autowired
private AccessionInvGroupRepository accessionInvGroupRepository;
@Autowired
private AccessionSourceMapService accessionSourceMapService;
@Autowired
private SiteRepository siteRepository;
@Autowired
private InventoryMaintenancePolicyRepository inventoryMaintenancePolicyRepository;
@Autowired
private InventoryRepository inventoryRepository;
@Autowired
private MethodRepository methodRepository;
@Autowired
private OverviewHelper overviewHelper;
@Autowired
private AccessionMCPDConverter accessionMCPDConverter;
@Autowired
private GGCESecurityConfig.GgceSec ggceSec;
@Autowired
private InventoryAttachmentService attachmentService;
@Component
protected static class ActionSupport extends BaseActionSupport<Accession, AccessionAction, AccessionActionFilter, AccessionActionRepository, AccessionActionRequest, AccessionActionScheduleFilter>
implements AccessionActionService {
@Autowired
private AccessionRepository accessionRepository;
@Autowired
private MethodService methodService;
@Override
protected EntityPath<Accession> getOwningEntityPath() {
return QAccessionAction.accessionAction.accession();
}
@Override
protected void initializeActionDetails(List<AccessionAction> actions) {
actions.forEach(action -> Hibernate.initialize(action.getAccession()));
}
@Override
protected void applyOwningEntityFilter(AccessionActionScheduleFilter filter, String owningEntityAlias, List<Predicate> predicates) {
QAccession qAccession = new QAccession(owningEntityAlias);
if (predicates != null && filter.accession != null) {
predicates.addAll(filter.accession.collectPredicates(qAccession));
}
}
@Override
@PostAuthorize("@ggceSec.actionAllowed('PassportData', 'ADMINISTRATION', returnObject.accession.site)")
protected AccessionAction createAction(Accession owningEntity) {
AccessionAction action = new AccessionAction();
action.setAccession(owningEntity);
action.setIsWebVisible("N");
return action;
}
@Override
protected void updateAction(AccessionAction action, AccessionActionRequest request) {
action.setIsWebVisible(request.webVisible);
if (request.method != null && !request.method.isNew()) {
action.setMethod(methodService.get(request.method.getId()));
}
}
@Override
@Transactional
@PostAuthorize("@ggceSec.actionAllowed('PassportData', 'ADMINISTRATION', returnObject.accession.site)")
public AccessionAction create(AccessionAction source) {
assert (source.getId() == null);
AccessionAction saved = repository.save(source);
return _lazyLoad(saved);
}
@Override
@Transactional
@PostAuthorize("@ggceSec.actionAllowed('PassportData', 'ADMINISTRATION', returnObject.accession.site)")
public AccessionAction remove(AccessionAction entity) {
return super.remove(entity);
}
@Override
@Transactional
@PostAuthorize("@ggceSec.actionAllowed('PassportData', 'ADMINISTRATION', returnObject.accession.site)")
public AccessionAction update(AccessionAction updated) {
return super.update(updated);
}
@Override
protected Iterable<Accession> findOwningEntities(Set<Long> id) {
return accessionRepository.findAll(QAccession.accession.id.in(id));
}
@Override
public AccessionAction prepareNextWorkflowStepAction(WorkflowActionStep nextStep, AccessionAction completedAction) {
AccessionAction nextAction = new AccessionAction();
nextAction.setAccession(new Accession(completedAction.getAccession().getId()));
nextAction.setIsWebVisible("N");
return nextAction;
}
}
@Override
protected NumberPath<Long> entityIdPredicate() {
return QAccession.accession.id;
}
@Override
protected JPAQuery<Accession> entityListQuery() {
return jpaQueryFactory.selectFrom(QAccession.accession)
// site (@OneToOne)
.join(QAccession.accession.site()).fetchJoin()
// species (@OneToOne)
.join(QAccession.accession.taxonomySpecies()).fetchJoin()
// pedigree (@OneToOne optional)
.leftJoin(QAccession.accession.accessionPedigree()).fetchJoin();
}
@Override
@Transactional
@PostAuthorize("@ggceSec.actionAllowed('PassportData', 'CREATE', returnObject.site)")
public Accession create(Accession source) {
assert (source.getId() == null);
Accession accession = repository.save(source);
if (CollectionUtils.isNotEmpty(source.getAccessionSources())) {
// Clone and fix reference
List<AccessionSource> sources = source.getAccessionSources().stream().map((accessionSource) -> {
AccessionSource copy = new AccessionSource();
copy.apply(accessionSource);
copy.setAccession(accession);
return copy;
}).collect(Collectors.toList());
accession.setAccessionSources(accessionSourceRepository.saveAll(sources));
}
// we assure system inventory inside the aspect, so mustn't be null
Inventory systemInventory = inventoryRepository.getSystemInventory(accession);
if (CollectionUtils.isNotEmpty(source.getNames())) {
// Clone and fix reference
List<AccessionInvName> names = source.getNames().stream().map((accessionInvName) -> {
AccessionInvName copy = new AccessionInvName();
copy.apply(accessionInvName);
copy.setInventory(systemInventory);
return copy;
}).collect(Collectors.toList());
accessionInvNameRepository.saveAll(names);
}
return accession;
}
@Override
@Transactional
@PreAuthorize("@ggceSec.actionAllowed('PassportData', 'WRITE', #updated.site)")
@PostAuthorize("@ggceSec.actionAllowed('PassportData', 'WRITE', returnObject.site)")
public Accession update(Accession updated) {
return super.update(updated);
}
@Override
@Transactional
@PreAuthorize("@ggceSec.actionAllowed('PassportData', 'WRITE', #target.site)")
public Accession update(Accession updated, Accession target) {
assert (target.getId() != null);
assert (target.getId().equals(updated.getId()));
log.debug("Update Accession. Input data {}", updated);
target.apply(updated);
final Accession saved = repository.save(target);
// FIXME This must be revised
if (Hibernate.isInitialized(updated.getAccessionSources()) && CollectionUtils.isNotEmpty(updated.getAccessionSources())) {
// Update references
updated.getAccessionSources().forEach((accessionSource) -> {
accessionSource.setAccession(saved);
});
accessionSourceRepository.saveAll(updated.getAccessionSources());
}
return saved;
}
@Override
@Transactional
@PreAuthorize("@ggceSec.actionAllowed('PassportData', 'WRITE', #target.site)")
public Accession updateFast(Accession updated, Accession target) {
target.apply(updated);
return repository.save(target);
}
@Override
@Transactional
@PreAuthorize("@ggceSec.actionAllowed('PassportData', 'DELETE', #entity.site)")
public Accession remove(Accession entity) {
return super.remove(entity);
}
@Override
@Transactional
@PreAuthorize("@ggceSec.actionAllowed('PassportData', 'DELETE', #accession.site)")
public void deleteDefaultInventory(Accession accession) {
final Inventory systemInventory = inventoryRepository.getSystemInventory(accession);
if (systemInventory != null) {
log.info("Removing the SYSTEM inventory for removed accession {}", accession.getId());
inventoryRepository.delete(systemInventory);
}
}
@Override
public Page<Accession> list(AccessionFilter filter, Pageable page) throws SearchException {
return super.list(Accession.class, filter, page, BOOST_FIELDS);
}
@Override
public Page<AccessionMCPD> listMCPD(AccessionFilter filter, Pageable page) throws SearchException {
var stopWatch = Stopwatch.createStarted();
var r = super.list(Accession.class, filter, page, BOOST_FIELDS);
log.warn("Loaded page {} in {}ms", page.getPageNumber(), stopWatch.elapsed(TimeUnit.MILLISECONDS));
// Populate JPA caches
var accessionIds = new ArrayList<Long>(); // r.getContent().stream().map(Accession::getId).collect(Collectors.toList())
Map<Long, Accession> accessionsById = new HashMap<>();
r.getContent().forEach(a -> {
accessionIds.add(a.getId());
a.setInventories(new ArrayList<>());
a.setAccessionSources(new ArrayList<>());
a.setNames(new ArrayList<>());
accessionsById.put(a.getId(), a);
});
var maxIdBatchSize = 2000; // MSSQL supports 2100 parameters
var totalAccessionIds = accessionIds.size();
for (var startIndex = 0; startIndex < totalAccessionIds; startIndex += maxIdBatchSize) {
var sublist = accessionIds.subList(startIndex, Math.min(accessionIds.size(), startIndex + maxIdBatchSize));
log.warn("Made sublist of {} accession IDs for page {} at {} ms", sublist.size(), page.getPageNumber(), stopWatch.elapsed(TimeUnit.MILLISECONDS));
var accessionInventories = jpaQueryFactory.selectFrom(QInventory.inventory)
.leftJoin(QInventory.inventory.extra()).fetchJoin() // Avoid N+1
.where(QInventory.inventory.accession().id.in(sublist)).fetch();
log.warn("Loaded {} inventories for page {} at {} ms", accessionInventories.size(), page.getPageNumber(), stopWatch.elapsed(TimeUnit.MILLISECONDS));
accessionInventories.forEach(inv -> {
var a = accessionsById.get(inv.getAccession().getId());
if (a != null) {
a.getInventories().add(inv);
}
});
log.warn("Generated inventories of accessions in page {} at {} ms", page.getPageNumber(), stopWatch.elapsed(TimeUnit.MILLISECONDS));
var accessionSources = jpaQueryFactory.selectFrom(QAccessionSource.accessionSource)
.leftJoin(QAccessionSource.accessionSource.cooperators).fetchJoin()
.where(QAccessionSource.accessionSource.accession().id.in(sublist)).fetch();
log.warn("Loaded {} accessionSources for page {} at {} ms", accessionSources.size(), page.getPageNumber(), stopWatch.elapsed(TimeUnit.MILLISECONDS));
accessionSources.forEach(source -> {
var a = accessionsById.get(source.getAccession().getId());
if (a != null) {
a.getAccessionSources().add(source);
}
});
log.warn("Generated accessionSources of accessions in page {} at {} ms", page.getPageNumber(), stopWatch.elapsed(TimeUnit.MILLISECONDS));
var accessionInvNames = jpaQueryFactory.selectFrom(QAccessionInvName.accessionInvName)
.join(QAccessionInvName.accessionInvName.inventory()).fetchJoin()
.where(QAccessionInvName.accessionInvName.inventory().accession().id.in(sublist).and(QAccessionInvName.accessionInvName.inventory().formTypeCode.eq(Inventory.SYSTEM_INVENTORY_FTC)))
.orderBy(QAccessionInvName.accessionInvName.plantNameRank.asc())
.fetch();
log.warn("Loaded {} accessionInvName for page {} at {} ms", accessionInvNames.size(), page.getPageNumber(), stopWatch.elapsed(TimeUnit.MILLISECONDS));
accessionInvNames.forEach(accessionInvName -> {
var a = accessionsById.get(accessionInvName.getInventory().getAccession().getId());
if (a != null) {
a.getNames().add(accessionInvName);
}
});
log.warn("Generated accessionInvName of accessions in page {} at {} ms", page.getPageNumber(), stopWatch.elapsed(TimeUnit.MILLISECONDS));
var accessionPedigrees = jpaQueryFactory.selectFrom(QAccessionPedigree.accessionPedigree)
.where(QAccessionPedigree.accessionPedigree.accession().id.in(sublist)).fetch();
log.warn("Loaded {} accessionPedigree for page {} at {} ms", accessionPedigrees.size(), page.getPageNumber(), stopWatch.elapsed(TimeUnit.MILLISECONDS));
accessionPedigrees.forEach(accessionPedigree -> {
var a = accessionsById.get(accessionPedigree.getAccession().getId());
if (a != null) {
accessionPedigree.lazyLoad();
a.setAccessionPedigree(accessionPedigree);
}
});
log.warn("Generated accessionPedigree of accessions in page {} at {} ms", page.getPageNumber(), stopWatch.elapsed(TimeUnit.MILLISECONDS));
}
var result = new PageImpl<>(r.getContent().stream().map(accessionMCPDConverter::convert).collect(Collectors.toList()), page, r.getTotalElements());
log.warn("Converted page {} to MCPD in {}ms", page.getPageNumber(), stopWatch.elapsed(TimeUnit.MILLISECONDS));
return result;
}
@Override
@Transactional
@PreAuthorize("@ggceSec.actionAllowed('PassportData', 'ADMINISTRATION')")
public MultiOp<Accession> update(List<Accession> accessions) {
List<AccessionSource> accessionSources = accessions.stream().map(Accession::getAccessionSources)
// filter
.filter((s) -> s != null && s.size() > 0)
// merge to single list
.reduce(new ArrayList<>(), (aggregated, sources) -> {
aggregated.addAll(sources);
return aggregated;
});
accessions = accessions.stream().map(a -> repository.save(a)).collect(Collectors.toList());
accessionSourceRepository.saveAll(accessionSources);
return new MultiOp<>(accessions);
}
@Override
public AccessionDetails getAccessionDetails(Accession accession) {
if (accession == null) {
throw new NotFoundElement();
}
accession = this.reload(accession);
// initialize lazy data
Hibernate.initialize(accession.getAccessionSources());
Hibernate.initialize(accession.getAccessionActions());
Hibernate.initialize(accession.getAccessionIprs());
Hibernate.initialize(accession.getAccessionPedigree());
Hibernate.initialize(accession.getAccessionQuarantines());
Hibernate.initialize(accession.getCitations());
Hibernate.initialize(accession.getBackupLocation1Site());
Hibernate.initialize(accession.getBackupLocation2Site());
Hibernate.initialize(accession.getExploration());
AccessionDetails accessionDetails = new AccessionDetails();
accessionDetails.accession = accession;
accessionDetails.sources = accession.getAccessionSources();
if (accessionDetails.sources != null) {
accessionDetails.sources.forEach((source) -> source.lazyLoad());
}
accessionDetails.actions = accession.getAccessionActions();
accessionDetails.ipr = accession.getAccessionIprs();
accessionDetails.pedigree = accession.getAccessionPedigree();
if (accessionDetails.pedigree != null) {
accessionDetails.pedigree.lazyLoad();
}
accessionDetails.quarantine = accession.getAccessionQuarantines();
if (accessionDetails.quarantine != null) {
accessionDetails.quarantine.forEach((quarantine) -> quarantine.lazyLoad());
}
accessionDetails.citations = accession.getCitations();
accessionDetails.names = accessionInvNameRepository.findAccessionNames(accession, Inventory.SYSTEM_INVENTORY_FTC);
accessionDetails.groups = accessionInvGroupService.listAccessionGroups(accession);
accessionDetails.attachments = accessionInvAttachRepository.findAccessionAttachments(accession, Inventory.SYSTEM_INVENTORY_FTC);
accessionDetails.attachments.forEach(accessionInvAttach -> {
if (accessionInvAttach.getAttachCooperator() != null) {
accessionInvAttach.getAttachCooperator().getId();
}
});
accessionDetails.annotations = annotationRepository.findAccessionAnnotations(accession, Inventory.SYSTEM_INVENTORY_FTC);
accessionDetails.sourceCooperators = accessionSourceMapService.listAccessionSourceMaps(accession);
return accessionDetails;
}
@Override
public AccessionMCPD getMCPD(Long id) {
return accessionMCPDConverter.convert(get(id));
}
@Override
@Transactional
@PreAuthorize("@ggceSec.actionAllowed('Acquisition', 'CREATE')")
public AccessionInvGroup acquire(AcquisitionData acquisitionBatch) {
if (acquisitionBatch.accessions == null || acquisitionBatch.accessions.size() == 0) {
throw new InvalidApiUsageException("No accessions in acquisition batch");
}
Method method = acquisitionBatch.methodId == null ? null : methodRepository.getReferenceById(acquisitionBatch.methodId);
if (method != null) {
log.info("Registering new material with method {}", method.getName());
}
Site site = siteRepository.getReferenceById(acquisitionBatch.siteId);
log.info("Registering new material with site {}", site.getSiteShortName());
InventoryMaintenancePolicy inventoryMaintPolicy = inventoryMaintenancePolicyRepository.getReferenceById(acquisitionBatch.inventoryMaintenancePolicyId);
log.info("Using policy {}", inventoryMaintPolicy.getMaintenanceName());
// Make accessions
List<Accession> accessions = acquisitionBatch.accessions.stream().map((newAcce) -> {
newAcce.setSite(site);
var acce = create(newAcce);
acce.setInitialReceivedDate(acquisitionBatch.sourceDate);
if (acce.getInitialReceivedDate() != null) {
acce.setInitialReceivedDateCode(StringUtils.defaultIfBlank(acquisitionBatch.sourceDateCode, CommunityCodeValues.DATE_FORMAT_DATE.value));
}
if (acquisitionBatch.sourceTypeCode != null) {
var accessionSource = new AccessionSource();
accessionSource.setAccession(acce);
accessionSource.setSourceTypeCode(acquisitionBatch.sourceTypeCode);
accessionSource.setSourceDate(acquisitionBatch.sourceDate);
accessionSource.setSourceDateCode(StringUtils.defaultIfBlank(acquisitionBatch.sourceDateCode, CommunityCodeValues.DATE_FORMAT_DATE.value));
accessionSource.setNote(acquisitionBatch.sourceNote);
accessionSource = accessionSourceRepository.save(accessionSource);
if (acquisitionBatch.sourceCooperator != null) {
accessionSourceMapService.create(new AccessionSourceMap(accessionSource, acquisitionBatch.sourceCooperator));
}
}
return acce;
}).collect(Collectors.toList());
log.warn("Created {} accessions", accessions.size());
// Make inventories
List<Inventory> inventories = accessions.stream().map((accession) -> {
Inventory inventory = new Inventory();
inventory.setAccession(accession);
inventory.setSite(site);
inventory.setInventoryNumberPart1(acquisitionBatch.inventoryNumberPart1);
inventory.setInventoryNumberPart2(AccessionService.AUTO_GENERATE_VALUE);
inventory.setInventoryMaintenancePolicy(inventoryMaintPolicy);
inventory.setFormTypeCode(acquisitionBatch.formTypeCode);
inventory.setAvailabilityStatusCode(acquisitionBatch.availabilityStatusCode);
return inventoryService.create(inventory);
}).collect(Collectors.toList());
log.warn("Created {} inventories", inventories.size());
// Register group
var groupName = acquisitionBatch.groupName;
if (StringUtils.isBlank(groupName)) {
throw new InvalidApiUsageException("Empty group name in acquisition batch");
}
AccessionInvGroup group;
group = accessionInvGroupRepository.findOne(QAccessionInvGroup.accessionInvGroup.groupName.eq(groupName)).orElse(null);
if (group == null) {
group = new AccessionInvGroup();
group.setGroupName(groupName);
group.setNote(acquisitionBatch.note);
group.setMethod(method);
group = accessionInvGroupService.create(group);
log.info("Registered group id={} {}", group.getId(), group.getGroupName());
}
group = accessionInvGroupService.addMembers(group, inventories, null);
return accessionInvGroupService.load(group.getId());
}
@Override
public Map<Object, Number> accessionOverview(String groupBy, AccessionFilter filter) {
return overviewHelper.getOverview(Accession.class, QAccession.accession, QAccession.accession.id.countDistinct(), groupBy, filter);
}
@Override
@PreAuthorize("hasRole('ADMINISTRATOR')")
public void recalculateAllAccessionNumbers() {
boolean lastPage = false;
int pageNumber = 0;
do {
Page<Accession> page = repository.findAll(PageRequest.of(pageNumber, 1000));
lastPage = page.isLast();
pageNumber += 1;
log.warn("Updating accessionNumber for {} records, page {}", page.getNumberOfElements(), pageNumber);
Lists.partition(page.getContent(), 100).forEach(batch -> TransactionHelper.executeInTransaction(false, () -> {
batch.forEach(a -> repository.setAccessionNumber(a.getId(), GGCE.accessionNumber(a)));
return true;
}));
} while (!lastPage);
log.info("Done.");
}
@Override
@PreAuthorize("hasRole('ADMINISTRATOR')")
public void assignMissingAccessionNumbers() {
boolean lastPage = false;
int pageNumber = 0;
do {
Page<Accession> page = repository.findAll(QAccession.accession.accessionNumber.isNull(), PageRequest.of(0, 1000));
lastPage = page.isLast();
pageNumber++;
log.warn("Assigning accessionNumber for {} records, page {}", page.getNumberOfElements(), pageNumber);
Lists.partition(page.getContent(), 100).forEach(batch -> TransactionHelper.executeInTransaction(false, () -> {
batch.forEach(a -> repository.setAccessionNumber(a.getId(), GGCE.accessionNumber(a)));
return true;
}));
} while (!lastPage && pageNumber < 1000); // at most 1000 loops
log.info("Done.");
}
@Override
@Transactional
public void shareAttachment(Long attachId, List<Long> accessionIds) {
var accessions = repository.findAll(QAccession.accession.id.in(accessionIds), Pageable.unpaged()).getContent();
var hasUnavailableSite = accessions.stream()
.map(Accession::getSite)
.distinct()
.anyMatch(site -> !ggceSec.actionAllowed("PassportData", "ADMINISTRATION", site));
if (hasUnavailableSite) {
throw new AccessDeniedException("Don't have permission to create attachments for the selected accessions");
}
var attachment = attachmentService.get(attachId);
var accessionInventories = accessions.stream()
.map(a -> a.getInventories().stream().filter(inv -> Inventory.SYSTEM_INVENTORY_FTC.equals(inv.getFormTypeCode())).findFirst().orElse(null))
.collect(Collectors.toList());
attachmentService.shareAttachment(attachment, accessionInventories);
}
@Autowired
@Lazy
private TileMaker tileMaker;
@Override
@Transactional(readOnly = true)
public byte[] getTile(AccessionFilter filter, int zoom, int xtile, int ytile) throws IOException {
final double latN = CoordUtil.tileToLat(zoom, ytile);
final double latS = CoordUtil.tileToLat(zoom, ytile + 1);
final double diffLat = latN - latS;
final double lonW = CoordUtil.tileToLon(zoom, xtile);
final double lonE = CoordUtil.tileToLon(zoom, xtile + 1);
final double diffLon = lonE - lonW;
if (log.isDebugEnabled()) {
log.debug("{} <= lat <= {} corr={}", latS, latN, diffLat * .1);
log.debug("{} <= lon <= {} corr={}", lonW, lonE, diffLon * .1);
}
return tileMaker.makeTile(zoom, xtile, ytile, () -> {
var qAccSrc = QAccessionSource.accessionSource;
return jpaQueryFactory.select(qAccSrc.latitude, qAccSrc.longitude).distinct().from(qAccSrc)
.where(qAccSrc.isOrigin.eq("Y").and(qAccSrc.latitude.between(latS - diffLat * .1, latN + diffLat * .1)).and(qAccSrc.longitude.between(lonW - diffLon * .1, lonE + diffLon * .1)).and(ExpressionUtils.allOf(filter.collectPredicates(qAccSrc.accession()))))
.stream().map(tuple -> new Double[] { tuple.get(qAccSrc.latitude), tuple.get(qAccSrc.longitude) });
});
}
@Override
@Transactional(readOnly = true)
public MapInfo<AccessionFilter> mapInfo(FilterInfo<AccessionFilter> filterInfo) {
var mapInfo = new MapInfo<AccessionFilter>();
mapInfo.filterCode = filterInfo.filterCode;
mapInfo.filter = filterInfo.filter;
var qAccSrc = QAccessionSource.accessionSource;
mapInfo.count = jpaQueryFactory.select(qAccSrc.accession().countDistinct()).from(qAccSrc).where(qAccSrc.isOrigin.eq("Y").and(ExpressionUtils.allOf(filterInfo.filter.collectPredicates(qAccSrc.accession())))).fetchOne();
var bounds = jpaQueryFactory.select(qAccSrc.latitude.min(), qAccSrc.longitude.min(), qAccSrc.latitude.max(), qAccSrc.longitude.max()).from(qAccSrc).where(qAccSrc.isOrigin.eq("Y").and(ExpressionUtils.allOf(filterInfo.filter.collectPredicates(qAccSrc.accession())))).fetchOne();
mapInfo.bounds = new Double[][] {
{ bounds.get(0, Double.class), bounds.get(1, Double.class) },
{ bounds.get(2, Double.class), bounds.get(3, Double.class) }
};
return mapInfo;
}
}