DescriptorServiceImpl.java
/*
* Copyright 2018 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.genesys.server.service.impl;
import static org.genesys.server.model.traits.QDescriptor.descriptor;
import java.io.IOException;
import java.io.OutputStream;
import java.nio.file.Path;
import java.nio.file.Paths;
import java.time.Instant;
import java.time.temporal.ChronoUnit;
import java.util.ArrayList;
import java.util.HashSet;
import java.util.List;
import java.util.Locale;
import java.util.Map;
import java.util.Objects;
import java.util.UUID;
import java.util.Map.Entry;
import java.util.stream.Collectors;
import javax.persistence.EntityManager;
import lombok.extern.slf4j.Slf4j;
import org.apache.commons.collections4.CollectionUtils;
import org.apache.commons.lang3.StringUtils;
import org.genesys.blocks.security.model.AclSid;
import org.genesys.blocks.security.serialization.Permissions;
import org.genesys.blocks.security.service.CustomAclService;
import org.genesys.filerepository.FolderNotEmptyException;
import org.genesys.filerepository.InvalidRepositoryFileDataException;
import org.genesys.filerepository.InvalidRepositoryPathException;
import org.genesys.filerepository.NoSuchRepositoryFileException;
import org.genesys.filerepository.model.RepositoryFolder;
import org.genesys.filerepository.model.RepositoryImage;
import org.genesys.filerepository.service.RepositoryService;
import org.genesys.server.component.security.AsAdminInvoker;
import org.genesys.server.component.security.SecurityUtils;
import org.genesys.server.exception.InvalidApiUsageException;
import org.genesys.server.exception.NotFoundElement;
import org.genesys.server.exception.SearchException;
import org.genesys.server.model.Partner;
import org.genesys.server.model.PublishState;
import org.genesys.server.model.UserRole;
import org.genesys.server.model.dataset.Dataset;
import org.genesys.server.model.filters.DescriptorFilter;
import org.genesys.server.model.filters.FilterHelpers;
import org.genesys.server.model.traits.Descriptor;
import org.genesys.server.model.traits.DescriptorLang;
import org.genesys.server.model.traits.DescriptorList;
import org.genesys.server.model.traits.QDescriptor;
import org.genesys.server.model.vocab.QVocabularyTerm;
import org.genesys.server.model.vocab.VocabularyTerm;
import org.genesys.server.model.vocab.VocabularyTermLang;
import org.genesys.server.persistence.DescriptorLangRepository;
import org.genesys.server.persistence.PartnerRepository;
import org.genesys.server.persistence.traits.DescriptorRepository;
import org.genesys.server.persistence.vocab.VocabularyTermRepository;
import org.genesys.server.service.DescriptorService;
import org.genesys.server.service.DescriptorTranslationService;
import org.genesys.server.service.DescriptorTranslationService.TranslatedDescriptor;
import org.genesys.server.service.DownloadService;
import org.genesys.server.service.ElasticsearchService;
import org.genesys.server.service.TranslatorService;
import org.genesys.server.service.TranslatorService.FormattedText;
import org.genesys.server.service.TranslatorService.TextFormat;
import org.genesys.server.service.TranslatorService.TranslatorException;
import org.genesys.server.service.VersionManager;
import org.genesys.server.service.VocabularyTermTranslationService;
import org.genesys.server.service.VocabularyTermTranslationService.TranslatedVocabularyTerm;
import org.genesys.util.HibernateUtil;
import org.genesys.util.JPAUtils;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.annotation.Lazy;
import org.springframework.dao.ConcurrencyFailureException;
import org.springframework.data.domain.Page;
import org.springframework.data.domain.PageImpl;
import org.springframework.data.domain.Pageable;
import org.springframework.security.access.prepost.PostAuthorize;
import org.springframework.security.access.prepost.PostFilter;
import org.springframework.security.access.prepost.PreAuthorize;
import org.springframework.security.acls.domain.BasePermission;
import org.springframework.stereotype.Component;
import org.springframework.stereotype.Service;
import org.springframework.transaction.annotation.Isolation;
import org.springframework.transaction.annotation.Propagation;
import org.springframework.transaction.annotation.Transactional;
import org.springframework.validation.annotation.Validated;
import org.springframework.web.multipart.MultipartFile;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.google.common.collect.Lists;
import com.querydsl.core.BooleanBuilder;
import com.querydsl.core.types.Predicate;
import com.querydsl.core.types.dsl.BooleanExpression;
/**
* The Class DescriptorServiceImpl.
*/
@Slf4j
@Service("catalogDescriptorService")
@Transactional(readOnly = true)
@Validated
public class DescriptorServiceImpl extends FilteredTranslatedCRUDServiceImpl<
Descriptor, DescriptorLang, DescriptorTranslationService.TranslatedDescriptor, DescriptorFilter, DescriptorRepository>
implements DescriptorService {
private static final Predicate[] EMPTY_PREDICATE_ARRAY = new Predicate[] {};
/** The descriptor repository. */
@Autowired
private DescriptorRepository descriptorRepository;
@Autowired
private VocabularyTermRepository termRepository;
@Autowired
private DownloadService downloadService;
/** The securityUtils. */
@Autowired
private SecurityUtils securityUtils;
@Autowired
private EntityManager entityManager;
@Autowired
private VersionManager versionManager;
@Autowired(required = false)
private ElasticsearchService elasticsearchService;
/** The file repo service. */
@Autowired
private RepositoryService repositoryService;
@Autowired
private CustomAclService aclService;
/** Execute code as admin */
@Autowired
protected AsAdminInvoker asAdminInvoker;
@Autowired
private PartnerRepository partnerRepository;
@Autowired(required = false)
private TranslatorService translatorService;
@Autowired
@Lazy
private VocabularyTermTranslationService vocabularyTermTranslationService;
@Component(value = "DescriptorTranslationSupport")
protected static class DescriptorTranslationSupport
extends BaseTranslationSupport<
Descriptor, DescriptorLang, DescriptorTranslationService.TranslatedDescriptor, DescriptorFilter, DescriptorLangRepository>
implements DescriptorTranslationService {
@Autowired
@Lazy
private VocabularyTermTranslationService vocabularyTermTranslationService;
@Override
@PreAuthorize("hasRole('ADMINISTRATOR') or hasPermission(#source.entity, 'write')")
public DescriptorLang create(DescriptorLang source) {
return super.create(source);
}
@Override
@PreAuthorize("hasRole('ADMINISTRATOR') or hasPermission(#target.entity, 'write')")
public DescriptorLang update(DescriptorLang updated, DescriptorLang target) {
return super.update(updated, target);
}
@Override
@PreAuthorize("hasRole('ADMINISTRATOR') or hasPermission(#source.entity, 'write')")
public DescriptorLang remove(DescriptorLang source) {
return super.remove(source);
}
@Override
public TranslatedDescriptor getTranslated(Descriptor input, Locale locale) {
var td = super.getTranslated(input, locale);
td.setTerms(vocabularyTermTranslationService.getTranslated(td.entity.getTerms(), locale));
return td;
}
@Override
public List<TranslatedDescriptor> getTranslated(List<Descriptor> input, Locale locale) {
var tds = super.getTranslated(input, locale);
tds.forEach(td -> {
td.setTerms(vocabularyTermTranslationService.getTranslated(td.entity.getTerms(), locale));
});
return tds;
}
}
/**
* {@inheritDoc}
*/
@Transactional
@Override
@PreAuthorize("hasRole('ADMINISTRATOR') or hasPermission(#input.owner, 'write')")
public Descriptor create(final Descriptor input) {
return lazyLoad(createFast(input));
}
@Transactional
@Override
@PreAuthorize("hasRole('ADMINISTRATOR') or hasPermission(#input.owner, 'write')")
public Descriptor createFast(Descriptor input) {
log.info("Creating descriptor: {} - {}", input.getTitle(), input.getDataType());
input.setOwner(partnerRepository.findById(input.getOwner().getId()).orElseThrow());
updateTerms(input);
final Descriptor descriptor = new Descriptor();
descriptor.apply(input);
descriptor.setUuid(input.getUuid());
descriptor.setVersionTag(input.getVersionTag());
// can not be published when creating
descriptor.setState(PublishState.DRAFT);
final Descriptor saved = super.createFast(descriptor);
// Grant all permissions to the Partner's SID
final AclSid sid = aclService.ensureAuthoritySid(saved.getOwner().getAuthorityName());
aclService.setPermissions(saved, sid, new Permissions().grantAll());
return saved;
}
private Descriptor getDescriptor(final Descriptor input) {
Descriptor descriptor = null;
if (input.getUuid() != null) {
descriptor = descriptorRepository.findByUuid(input.getUuid());
} else {
descriptor = descriptorRepository.findById(input.getId()).orElse(null);
}
if (descriptor == null) {
throw new NotFoundElement("Record not found by uuid=" + input.getUuid() + " or id=" + input.getId());
}
if (input.getVersion() != null && !descriptor.getVersion().equals(input.getVersion())) {
throw new ConcurrencyFailureException("Object version changed to " + descriptor.getVersion() + ", you provided " + input.getVersion());
}
return descriptor;
}
private Descriptor getUnpublishedDescriptor(Descriptor descriptor) {
descriptor = getDescriptor(descriptor);
if (descriptor.isPublished()) {
throw new InvalidApiUsageException("Cannot modify a published Descriptor.");
}
return descriptor;
}
/**
* {@inheritDoc}
*/
@Override
@PostFilter("hasRole('ADMINISTRATOR') or filterObject.published or hasPermission(filterObject, 'READ')")
public List<Descriptor> searchMatchingDescriptor(final Descriptor input) {
try {
input.trimStringsToNull();
log.debug("searchMatching: {}", new ObjectMapper().writerWithDefaultPrettyPrinter().writeValueAsString(input));
} catch (IOException e) {
}
List<Predicate> searches = Lists.newArrayList();
// version + title
searches.add(descriptor.versionTag.eq(input.getVersionTag()).and(descriptor.title.eq(input.getTitle())));
if (StringUtils.isNotBlank(input.getDescription())) {
// description
searches.add(descriptor.description.eq(input.getDescription()));
}
if (StringUtils.isNotBlank(input.getDescription())) {
// title + dataType
searches.add(descriptor.title.eq(input.getTitle()).and(descriptor.dataType.eq(input.getDataType())));
}
if (input.getCategory() != null) {
// title + category
searches.add(descriptor.title.eq(input.getTitle()).and(descriptor.category.eq(input.getCategory())));
}
if (StringUtils.isNotBlank(input.getUom())) {
// uom
searches.add(descriptor.dataType.eq(input.getDataType()).and(descriptor.uom.eq(input.getUom())));
}
final Predicate predicate = StringUtils.isBlank(input.getCrop()) ? descriptor.crop.isNull().andAnyOf(searches.toArray(EMPTY_PREDICATE_ARRAY)) : descriptor.crop.eq(input.getCrop()).andAnyOf(searches.toArray(EMPTY_PREDICATE_ARRAY));
List<Descriptor> matches = new ArrayList<>();
descriptorRepository.findAll(predicate).forEach(match -> matches.add(lazyLoad(match)));
return matches;
}
@Override
@Transactional
@PreAuthorize("hasRole('ADMINISTRATOR') or hasPermission(#descriptor, 'write')")
public Descriptor updateImage(Descriptor descriptor, final MultipartFile file, final RepositoryImage imageMetadata) throws IOException, InvalidRepositoryPathException,
InvalidRepositoryFileDataException, NoSuchRepositoryFileException {
if (securityUtils.hasRole(UserRole.ADMINISTRATOR)) {
descriptor = getDescriptor(descriptor); // Editorial changes work regardless of state
} else {
descriptor = getUnpublishedDescriptor(descriptor);
}
if (!file.getContentType().startsWith("image/")) {
throw new InvalidApiUsageException("Invalid image type: " + file.getContentType() + " is not permitted.");
}
RepositoryImage repositoryImage = descriptor.getImage();
if (repositoryImage != null) {
descriptor = removeImage(descriptor);
}
String originalFilename = file.getOriginalFilename();
String ext = "";
if (originalFilename != null) {
int dotIndex = originalFilename.lastIndexOf(".");
ext = dotIndex > 0 ? originalFilename.substring(dotIndex) : "";
}
// Make descriptor folder
var descriptorFolder = ensureDescriptorFolder(descriptor);
repositoryImage = repositoryService.addFile(descriptorFolder.getFolderPath(), descriptor.getUuid().toString() + ext, file.getContentType(), file.getInputStream(), imageMetadata);
descriptor.setImage(repositoryImage);
return lazyLoad(descriptorRepository.save(descriptor));
}
private RepositoryFolder ensureDescriptorFolder(final Descriptor descriptor) {
try {
final Path descriptorPath = getRepositoryImageFolder(descriptor);
final Partner partner = descriptor.getOwner();
// Ensure folder ownership
return asAdminInvoker.invoke(() -> {
// Ensure target folder exists for the Descriptor
return repositoryService.ensureFolder(descriptorPath, partner);
});
} catch (Exception e) {
log.warn("Could not create folder properties: {}", e.getMessage());
throw new InvalidApiUsageException("Could not create folder", e);
}
}
@Override
@Transactional(rollbackFor = Throwable.class)
@PreAuthorize("hasRole('ADMINISTRATOR') or hasPermission(#descriptor, 'write')")
public Descriptor removeImage(Descriptor descriptor) throws InvalidRepositoryPathException, IOException {
if (securityUtils.hasRole(UserRole.ADMINISTRATOR)) {
descriptor = getDescriptor(descriptor); // Editorial changes work regardless of state
} else {
descriptor = getUnpublishedDescriptor(descriptor);
}
deleteDescriptorImage(descriptor);
return lazyLoad(descriptorRepository.save(descriptor));
}
private void deleteDescriptorImage(Descriptor descriptor) throws IOException {
RepositoryImage descriptorImage = descriptor.getImage();
descriptor.setImage(null);
if (descriptorImage != null) {
descriptorImage = HibernateUtil.unproxy(descriptorImage);
Path folderPath = descriptorImage.getFolder().getFolderPath();
try {
repositoryService.removeFile(descriptorImage);
} catch (NoSuchRepositoryFileException e) {
// That's fine
log.warn("Could not remove descriptor image: {}", e.getMessage());
}
try {
repositoryService.deleteFolder(folderPath);
} catch (Throwable e) {
// Weird
log.warn("Could not remove descriptor image folder: {}", e.getMessage());
}
}
}
private Path getRepositoryImageFolder(final Descriptor descriptor) {
assert (descriptor != null);
assert (descriptor.getUuid() != null);
String uuid = descriptor.getUuid().toString();
return Paths.get("/descriptor", uuid.substring(0, 3), uuid).toAbsolutePath();
}
/**
* Persist or update terms in descriptor itself. It updates descriptor's own
* terms List
*
* @param descriptor the descriptor
*/
protected void updateTerms(final Descriptor descriptor) {
final List<VocabularyTerm> terms = descriptor.getTerms();
if (terms != null && !terms.isEmpty()) {
terms.stream().collect(Collectors.groupingBy(VocabularyTerm::getCode))
.entrySet().stream().filter(e -> e.getValue().size() > 1).findFirst()
.ifPresent(e -> {
throw new InvalidApiUsageException("Terms with duplicate codes are not allowed.");
}
);
if (descriptor.getId() != null) {
var remainingTermIds = terms.stream().map(VocabularyTerm::getId).filter(id -> id != null).collect(Collectors.toSet());
var existedForRemove = termRepository.findAll(
QVocabularyTerm.vocabularyTerm.descriptor().id.eq(descriptor.getId())
.and(QVocabularyTerm.vocabularyTerm.id.notIn(remainingTermIds))
);
if (existedForRemove.iterator().hasNext()) {
termRepository.deleteAll(existedForRemove);
}
}
terms.forEach(term -> term.setOriginalLanguageTag(descriptor.getOriginalLanguageTag())); // Force set same language
final List<VocabularyTerm> r = termRepository.saveAllAndFlush(terms);
terms.clear();
log.debug("Adding terms {}", r);
terms.addAll(r);
}
}
/**
* {@inheritDoc}
*/
@Transactional
@Override
public Descriptor update(final Descriptor descriptor) {
final Descriptor loaded = getUnpublishedDescriptor(descriptor);
return update(descriptor, loaded);
}
@Override
@Transactional
public Descriptor update(Descriptor updated, Descriptor target) {
return lazyLoad(updateFast(updated, target));
}
@Override
@Transactional
@PreAuthorize("hasRole('ADMINISTRATOR') or hasPermission(#target, 'WRITE')")
public Descriptor updateFast(Descriptor updated, Descriptor target) {
log.info("Updating descriptor uuid={} id={}", updated.getUuid(), updated.getId());
final Partner owner = target.getOwner();
if (updated.getOwner() != null && !target.getOwner().getUuid().equals(updated.getOwner().getUuid())) {
throw new InvalidApiUsageException("Descriptor owner can't be changed");
}
updateTerms(updated);
target.apply(updated);
// Keep owner
target.setOwner(owner);
target.setState(PublishState.DRAFT);
return descriptorRepository.save(target);
}
/**
* {@inheritDoc}
*/
@Transactional
@Override
@PreAuthorize("hasRole('ADMINISTRATOR')")
public Descriptor forceUpdateDescriptor(final Descriptor descriptor) {
final Descriptor loaded = getDescriptor(descriptor); // Editorial changes work regardless of state
log.info("Updating descriptor uuid={} id={}", descriptor.getUuid(), descriptor.getId());
final Partner owner = loaded.getOwner();
if (descriptor.getOwner() != null && !loaded.getOwner().equals(descriptor.getOwner())) {
throw new InvalidApiUsageException("Descriptor owner can't be changed");
}
updateTerms(descriptor);
loaded.apply(descriptor);
// Keep owner
loaded.setOwner(owner);
// loaded.setState(PublishState.DRAFT); // Editoral changes do not modify state
return lazyLoad(descriptorRepository.save(loaded));
}
/**
* {@inheritDoc}
*/
@Override
@PostAuthorize("hasRole('ADMINISTRATOR') || returnObject==null || returnObject.isPublished() || hasPermission(returnObject, 'write')")
public Descriptor getDescriptor(final UUID uuid) {
return descriptorRepository.findByUuid(uuid);
}
/**
* {@inheritDoc}
*/
@Override
@PostAuthorize("hasRole('ADMINISTRATOR') || returnObject==null || returnObject.isPublished() || hasPermission(returnObject, 'write')")
public Descriptor loadDescriptor(final UUID uuid) {
return lazyLoad(descriptorRepository.findByUuid(uuid));
}
@Override
@PostAuthorize("hasRole('ADMINISTRATOR') || returnObject==null || returnObject.entity.isPublished() || hasPermission(returnObject.entity, 'write')")
public DescriptorTranslationService.TranslatedDescriptor loadTranslatedDescriptor(UUID uuid) throws NotFoundElement {
return translate(descriptorRepository.findByUuid(uuid));
}
/**
* {@inheritDoc}
*/
@Override
@PostAuthorize("hasRole('ADMINISTRATOR') || returnObject==null || returnObject.isPublished() || hasPermission(returnObject, 'read')")
public Descriptor loadDescriptor(final UUID uuid, final int version) {
return lazyLoad(descriptorRepository.findByUuidAndVersion(uuid, version));
}
/**
* {@inheritDoc}
*/
@Override
@PreAuthorize("isAuthenticated()")
public Page<Descriptor> listDescriptorsForCurrentUser(final DescriptorFilter filter, final Pageable page) throws IOException, SearchException {
Pageable markdownSortPageRequest = JPAUtils.toMarkdownSort(page, "title");
Page<Descriptor> res;
if (securityUtils.hasRole(UserRole.ADMINISTRATOR)) {
if (filter.isFulltextQuery()) {
res = elasticsearchService.findAll(Descriptor.class, filter, markdownSortPageRequest);
} else {
res = descriptorRepository.findAll(filter.buildPredicate(), markdownSortPageRequest);
}
return new PageImpl<>(res.getContent(), page, res.getTotalElements());
} else {
final HashSet<Long> partners = new HashSet<>(securityUtils.listObjectIdentityIdsForCurrentUser(Partner.class, BasePermission.WRITE));
DescriptorFilter partnerFilter = filter.copy(DescriptorFilter.class);
partnerFilter.owner().id(partners);
if (filter.isFulltextQuery()) {
res = elasticsearchService.findAll(Descriptor.class, partnerFilter, markdownSortPageRequest);
} else {
res = descriptorRepository.findAll(partnerFilter.buildPredicate(), markdownSortPageRequest);
}
}
return new PageImpl<>(res.getContent(), page, res.getTotalElements());
}
@Override
@PreAuthorize("isAuthenticated()")
public Page<Descriptor> listAccessibleDescriptors(DescriptorFilter descriptorFilter, Pageable page) throws IOException, SearchException {
Pageable markdownSortPageRequest = JPAUtils.toMarkdownSort(page, "title");
Page<Descriptor> res;
if (securityUtils.hasRole(UserRole.ADMINISTRATOR)) {
res = listDescriptorsForCurrentUser(descriptorFilter, markdownSortPageRequest);
} else {
final HashSet<Long> partners = new HashSet<>(securityUtils.listObjectIdentityIdsForCurrentUser(Partner.class, BasePermission.WRITE));
BooleanExpression and = QDescriptor.descriptor.state.in(PublishState.PUBLISHED).or(descriptor.owner().id.in(partners)).and(descriptorFilter.buildPredicate());
if (descriptorFilter.isFulltextQuery()) {
and = and.andAnyOf(FilterHelpers.containsAll(descriptorFilter.get_text(), QDescriptor.descriptor.title, QDescriptor.descriptor.columnName, QDescriptor.descriptor.description, QDescriptor.descriptor.crop));
}
res = descriptorRepository.findAll(and, markdownSortPageRequest);
}
return new PageImpl<>(res.getContent(), page, res.getTotalElements());
}
/**
* {@inheritDoc}
*/
@Override
public Page<Descriptor> listDescriptors(final DescriptorFilter descriptorFilter, final Pageable page) throws SearchException {
Pageable markdownSortPageRequest = JPAUtils.toMarkdownSort(page, "title");
Page<Descriptor> res;
if (descriptorFilter.isFulltextQuery()) {
res = elasticsearchService.findAll(Descriptor.class, descriptorFilter, page);
} else {
res = descriptorRepository.findAll(new BooleanBuilder().and(descriptorFilter.buildPredicate()).and(QDescriptor.descriptor.state.in(PublishState.PUBLISHED)),
markdownSortPageRequest);
}
return new PageImpl<>(res.getContent(), page, res.getTotalElements());
}
@Override
public Page<TranslatedDescriptor> listDescriptorsDetails(DescriptorFilter filter, Pageable page) throws SearchException {
var descriptors = listDescriptors(filter, page);
descriptors.getContent().forEach(d -> d.getTerms().size());
return new PageImpl<>(translationSupport.getTranslated(descriptors.getContent()), page, descriptors.getTotalElements());
}
/**
* {@inheritDoc}
*
* @throws IOException
* @throws InvalidRepositoryPathException
* @throws FolderNotEmptyException
* @throws NoSuchRepositoryFileException
*/
@Transactional(rollbackFor = Throwable.class)
@Override
@PreAuthorize("hasRole('ADMINISTRATOR') or hasPermission(#descriptor, 'delete')")
public Descriptor removeDescriptor(final Descriptor descriptor) throws IOException {
final Descriptor loadedDescriptor = getUnpublishedDescriptor(descriptor);
deleteDescriptorImage(loadedDescriptor);
descriptorRepository.delete(loadedDescriptor);
return loadedDescriptor;
}
@Transactional(rollbackFor = Throwable.class)
@Override
@PreAuthorize("hasRole('ADMINISTRATOR') or hasPermission(#entity, 'delete')")
public Descriptor remove(Descriptor entity) {
try {
return removeDescriptor(entity);
} catch (Exception e) {
throw new InvalidApiUsageException(e);
}
}
/**
* Lazy load for objects in descriptor.
*
* @param descriptor descriptor
* @return descriptor with loaded inner objects
*/
private Descriptor lazyLoad(final Descriptor descriptor) {
if (descriptor == null) {
throw new NotFoundElement("No such descriptor");
}
descriptor.lazyLoad();
return descriptor;
}
@Override
@Transactional
@PreAuthorize("hasRole('ADMINISTRATOR')")
public List<Descriptor> upsertDescriptors(final List<Descriptor> sources) {
final List<Descriptor> updates = new ArrayList<>();
for (final Descriptor source : sources) {
updates.add(upsertDescriptor(source));
}
return updates;
}
@Override
@Transactional
@PreAuthorize("hasRole('ADMINISTRATOR')")
public Descriptor upsertDescriptor(final Descriptor source) {
Descriptor target;
if (source.getVersion() != null) {
target = getUnpublishedDescriptor(source);
} else {
target = new Descriptor();
target.setUuid(source.getUuid());
}
updateTerms(source);
target.apply(source);
return descriptorRepository.save(target);
}
@Override
@Transactional
@PreAuthorize("hasRole('ADMINISTRATOR')")
public Descriptor approveDescriptor(final Descriptor descriptor) {
final Descriptor loaded = getUnpublishedDescriptor(descriptor);
if (loaded.getState() == PublishState.DRAFT) {
throw new InvalidApiUsageException("Descriptor should be sent for review before publication");
}
loaded.setState(PublishState.PUBLISHED);
if (loaded.getImage() != null) {
// Relax permissions on descriptor image: allow USERS and ANONYMOUS to read the image
aclService.makePubliclyReadable(HibernateUtil.unproxy(loaded.getImage()), true);
}
return lazyLoad(descriptorRepository.saveAndFlush(loaded));
}
@Override
@Transactional
@PreAuthorize("hasRole('ADMINISTRATOR') or hasPermission(#descriptor, 'administration')")
public Descriptor reviewDescriptor(final Descriptor descriptor) {
final Descriptor loaded = getUnpublishedDescriptor(descriptor);
if (loaded.getState() == PublishState.REVIEWING) {
throw new InvalidApiUsageException("The Descriptor is already under approval");
}
loaded.setState(PublishState.REVIEWING);
return lazyLoad(descriptorRepository.save(loaded));
}
/**
* {@inheritDoc}
*/
@Override
@Transactional(isolation = Isolation.READ_UNCOMMITTED, noRollbackFor = InvalidApiUsageException.class)
@PreAuthorize("hasRole('ADMINISTRATOR') or hasPermission(#descriptor, 'administration')")
public Descriptor rejectDescriptor(final Descriptor descriptor) {
final Descriptor loaded = getDescriptor(descriptor);
if (loaded.isPublished() && !securityUtils.hasRole(UserRole.ADMINISTRATOR)) {
if (loaded.getLastModifiedDate() != null && Instant.now().minus(24, ChronoUnit.HOURS).isAfter(loaded.getLastModifiedDate())) {
throw new InvalidApiUsageException("Cannot be un-published. More than 24 hours have passed since the publication.");
}
}
{
for (final Dataset referencedDataset : loaded.getDatasets()) {
if (referencedDataset.getState() == PublishState.PUBLISHED) {
throw new InvalidApiUsageException("Cannot be un-published. The descriptor is referenced by a published dataset.");
}
}
for (final DescriptorList referencedDescriptorList : loaded.getDescriptorLists()) {
if (referencedDescriptorList.getState() == PublishState.PUBLISHED) {
throw new InvalidApiUsageException("Cannot be un-published. The descriptor is referenced by a published descriptor list.");
}
}
}
loaded.setState(PublishState.DRAFT);
if (loaded.getImage() != null) {
// Tighten permissions on descriptor image
aclService.makePubliclyReadable(HibernateUtil.unproxy(loaded.getImage()), false);
}
return lazyLoad(descriptorRepository.save(loaded));
}
/**
* {@inheritDoc}
*/
@Override
@PreAuthorize("#descriptor.isPublished() || hasRole('ADMINISTRATOR') || hasPermission(#descriptor, 'read')")
@PostFilter("hasRole('ADMINISTRATOR') || filterObject==null || filterObject.isPublished() || hasPermission(filterObject, 'write')")
public List<DescriptorList> getDescriptorLists(final Descriptor descriptor) {
final List<DescriptorList> list = descriptorRepository.listDescriptorLists(descriptor);
list.forEach(d -> entityManager.detach(d));
return list;
}
/**
* {@inheritDoc}
*/
@Override
@PreAuthorize("hasRole('ADMINISTRATOR') || #descriptor.isPublished() || hasPermission(#descriptor, 'read')")
@PostFilter("hasRole('ADMINISTRATOR') || filterObject==null || filterObject.isPublished() || hasPermission(filterObject, 'write')")
public List<Dataset> getDatasets(final Descriptor descriptor) {
final List<Dataset> list = descriptorRepository.listDatasets(descriptor);
list.forEach(d -> entityManager.detach(d));
return list;
}
@Override
@Transactional
@PreAuthorize("#descriptor.isPublished() || hasRole('ADMINISTRATOR') || hasPermission(#descriptor, 'read')")
public Descriptor nextVersion(final Descriptor descriptor, final boolean major) {
final Descriptor source = getDescriptor(descriptor);
log.info("Creating new version for descriptor uuid={} id={}", descriptor.getUuid(), descriptor.getId());
final Descriptor copy = new Descriptor();
copy.apply(source);
copy.setUuid(null);
copy.setState(PublishState.DRAFT);
copy.setVersionTag(versionManager.next(descriptor.getVersionTag(), major));
copy.setOwner(descriptor.getOwner());
return lazyLoad(descriptorRepository.save(copy));
}
@Override
public void exportDescriptors(DescriptorFilter filter, OutputStream outputStream) throws IOException {
List<Descriptor> descriptors = (List<Descriptor>) descriptorRepository.findAll(filter.buildPredicate());
downloadService.writeXlsxDescriptor(descriptors, outputStream);
}
@Override
public long countDescriptors(DescriptorFilter filter) throws SearchException {
if (filter.isFulltextQuery()) {
return elasticsearchService.count(DescriptorList.class, filter);
}
return descriptorRepository.count(filter.buildPredicate());
}
@Override
@PreAuthorize("hasRole('ADMINISTRATOR') || hasPermission(#original, 'write')")
public DescriptorLang machineTranslate(Descriptor original, String targetLanguage) throws TranslatorService.TranslatorException {
if (Objects.equals(original.getOriginalLanguageTag(), targetLanguage)) {
throw new InvalidApiUsageException("Source and target language are the same");
}
var mt = new DescriptorLang();
mt.setMachineTranslated(true);
mt.setLanguageTag(targetLanguage);
mt.setEntity(original);
if (translatorService == null) return mt;
var builder = TranslatorService.TranslationStructuredRequest.builder()
.targetLang(targetLanguage);
// Translations to other languages use the English version (either original or translated)
if (!Objects.equals(Locale.ENGLISH.getLanguage(), targetLanguage) && !Objects.equals(Locale.ENGLISH.getLanguage(), original.getOriginalLanguageTag())) {
var enTranslation = translationSupport.getLang(original, Locale.ENGLISH.getLanguage());
if (enTranslation == null) {
throw new InvalidApiUsageException("English text is not available.");
}
builder
.sourceLang(enTranslation.getLanguageTag())
.context(buildTranslationContext(Descriptor.class, "title", enTranslation.getTitle(), Locale.ENGLISH))
.texts(Map.of(
"title", new FormattedText(TextFormat.markdown, enTranslation.getTitle()),
"description", new FormattedText(TextFormat.markdown, enTranslation.getDescription())
));
} else {
// Translations to English use the original texts
var originLocale = Locale.forLanguageTag(original.getOriginalLanguageTag());
builder
.sourceLang(original.getOriginalLanguageTag())
.context(buildTranslationContext(Descriptor.class, "title", original.getTitle(), originLocale))
.texts(Map.of(
"title", new FormattedText(TextFormat.markdown, original.getTitle()),
"description", new FormattedText(TextFormat.markdown, original.getDescription())
));
}
var translations = translatorService.translate(builder.build());
if (StringUtils.isNotBlank(original.getTitle())) {
mt.setTitle(translations.getTexts().get("title"));
}
if (StringUtils.isNotBlank(original.getDescription())) {
mt.setDescription(translations.getTexts().get("description"));
}
return mt;
}
@Override
@Transactional(propagation = Propagation.MANDATORY, readOnly = true)
@PreAuthorize("hasRole('ADMINISTRATOR') || hasPermission(#descriptor, 'write')")
public List<VocabularyTermLang> machineTranslateTerms(Descriptor descriptor, String targetLanguage) {
var builder = TranslatorService.TranslationStructuredRequest.builder()
.targetLang(targetLanguage);
// Translations to other languages use the English version (either original or translated)
if (!Objects.equals(Locale.ENGLISH.getLanguage(), targetLanguage) && !Objects.equals(Locale.ENGLISH.getLanguage(), descriptor.getOriginalLanguageTag())) {
var enTranslation = translationSupport.getTranslated(descriptor, Locale.ENGLISH);
if (enTranslation == null) {
throw new InvalidApiUsageException("English text is not available.");
}
var context = buildTranslationContext(Descriptor.class, "title", enTranslation.getTranslation().getTitle(), Locale.ENGLISH);
if (Descriptor.DataType.CODED.equals(descriptor.getDataType()) && CollectionUtils.isNotEmpty(descriptor.getTerms())) {
var codes = makeDescriptorCodes(descriptor, Locale.ENGLISH);
if (StringUtils.isNotBlank(codes)) {
var codesContext = buildTranslationContext(Descriptor.class, "codes", codes, Locale.ENGLISH);
if (codesContext != null) {
context = context != null ? context + "\n" + codesContext : codesContext;
}
}
}
builder
.sourceLang(enTranslation.getTranslation().getLanguageTag())
.context(context)
.texts(enTranslation.getTerms().stream().map(term -> {
return List.of(
Map.entry("title" + term.getEntity().getId(), new FormattedText(TextFormat.markdown, term.getTranslation().getTitle())),
Map.entry("description" + term.getEntity().getId(), new FormattedText(TextFormat.markdown, term.getTranslation().getDescription()))
);
}).flatMap(x -> x.stream()).filter(entry -> Objects.nonNull(entry.getValue().getText())).collect(Collectors.toMap(Entry::getKey, Entry::getValue)));
} else {
// Translations to English use the original texts
var originLocale = Locale.forLanguageTag(descriptor.getOriginalLanguageTag());
var context = buildTranslationContext(Descriptor.class, "title", descriptor.getTitle(), originLocale);
if (Descriptor.DataType.CODED.equals(descriptor.getDataType()) && CollectionUtils.isNotEmpty(descriptor.getTerms())) {
var codes = makeDescriptorCodes(descriptor, originLocale);
if (StringUtils.isNotBlank(codes)) {
var codesContext = buildTranslationContext(Descriptor.class, "codes", codes, originLocale);
if (codesContext != null) {
context = context != null ? context + "\n" + codesContext : codesContext;
}
}
}
builder
.sourceLang(descriptor.getOriginalLanguageTag())
.context(context)
.texts(descriptor.getTerms().stream().map(term -> {
return List.of(
Map.entry("title" + term.getId(), new FormattedText(TextFormat.markdown, term.getTitle())),
Map.entry("description" + term.getId(), new FormattedText(TextFormat.markdown, term.getDescription()))
);
}).flatMap(x -> x.stream()).filter(entry -> Objects.nonNull(entry.getValue().getText())).collect(Collectors.toMap(Entry::getKey, Entry::getValue)));
}
try {
var translations = translatorService.translate(builder.build());
return descriptor.getTerms().stream().map(term -> {
var mt = new VocabularyTermLang();
mt.setEntity(term);
mt.setMachineTranslated(true);
mt.setLanguageTag(targetLanguage);
if (StringUtils.isNotBlank(term.getTitle())) {
mt.setTitle(translations.getTexts().get("title" + term.getId()));
}
if (StringUtils.isNotBlank(term.getDescription())) {
mt.setDescription(translations.getTexts().get("description" + term.getId()));
}
return mt;
}).filter(Objects::nonNull).collect(Collectors.toList());
} catch (TranslatorException e) {
return null;
}
}
private String makeDescriptorCodes(Descriptor original, Locale lang) {
if (original.getOriginalLanguageTag().equals(lang.getLanguage())) {
return original.getTerms().stream()
.map(VocabularyTerm::getTitle)
.filter(Objects::nonNull)
.collect(Collectors.joining(", ")).trim();
} else {
return vocabularyTermTranslationService.getTranslated(original.getTerms(), lang).stream()
.map(TranslatedVocabularyTerm::getTranslation)
.filter(Objects::nonNull)
.map(VocabularyTermLang::getTitle)
.filter(Objects::nonNull)
.collect(Collectors.joining(", ")).trim();
}
}
}