DescriptorListServiceImpl.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.QDescriptorList.descriptorList;
import java.io.IOException;
import java.io.OutputStream;
import java.util.Arrays;
import java.util.HashMap;
import java.util.HashSet;
import java.util.List;
import java.util.Locale;
import java.util.Map;
import java.util.Objects;
import java.util.Set;
import java.util.UUID;
import java.util.stream.Collectors;
import lombok.extern.slf4j.Slf4j;
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.server.component.aspect.NotifyForReview;
import org.genesys.server.component.aspect.NotifyOnPublished;
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.filters.DescriptorListFilter;
import org.genesys.server.model.traits.Descriptor;
import org.genesys.server.model.traits.DescriptorList;
import org.genesys.server.model.traits.DescriptorListLang;
import org.genesys.server.persistence.DescriptorListLangRepository;
import org.genesys.server.persistence.traits.DescriptorListRepository;
import org.genesys.server.service.DescriptorListService;
import org.genesys.server.service.DescriptorListTranslationService;
import org.genesys.server.service.DescriptorService;
import org.genesys.server.service.DescriptorTranslationService;
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.util.JPAUtils;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.dao.ConcurrencyFailureException;
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.data.domain.Sort;
import org.springframework.security.access.prepost.PostAuthorize;
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.Transactional;
import org.springframework.validation.annotation.Validated;
import com.google.common.collect.Sets;
import com.querydsl.core.BooleanBuilder;
import com.querydsl.core.types.Predicate;
/**
* Service for working with {@link DescriptorList}.
*
* @author Andrey Lugovskoy
*/
@Slf4j
@Service
@Transactional(readOnly = true)
@Validated
public class DescriptorListServiceImpl extends FilteredTranslatedCRUDServiceImpl<
DescriptorList, DescriptorListLang, DescriptorListTranslationService.TranslatedDescriptorList, DescriptorListFilter, DescriptorListRepository>
implements DescriptorListService {
@Autowired
private DescriptorListRepository descriptorListRepository;
/** The securityUtils. */
@Autowired
private SecurityUtils securityUtils;
@Autowired
private DescriptorService descriptorService;
@Autowired
private DownloadService downloadService;
@Autowired(required = false)
private ElasticsearchService elasticsearchService;
@Autowired
private CustomAclService aclService;
@Autowired(required = false)
private TranslatorService translatorService;
@Autowired
DescriptorTranslationService descriptorTranslationService;
@Component(value = "DescriptorListTranslationSupport")
protected static class DescriptorListTranslationSupport
extends BaseTranslationSupport<
DescriptorList, DescriptorListLang, DescriptorListTranslationService.TranslatedDescriptorList, DescriptorListFilter, DescriptorListLangRepository>
implements DescriptorListTranslationService {
@Override
@PreAuthorize("hasRole('ADMINISTRATOR') or hasPermission(#source.entity, 'ADMINISTRATION')")
public DescriptorListLang create(DescriptorListLang source) {
return super.create(source);
}
@Override
@PreAuthorize("hasRole('ADMINISTRATOR') or hasPermission(#target.entity, 'ADMINISTRATION')")
public DescriptorListLang update(DescriptorListLang updated, DescriptorListLang target) {
return super.update(updated, target);
}
@Override
@PreAuthorize("hasRole('ADMINISTRATOR') or hasPermission(#source.entity, 'ADMINISTRATION')")
public DescriptorListLang remove(DescriptorListLang source) {
return super.remove(source);
}
}
@Override
@PostAuthorize("returnObject == null || returnObject.isPublished() || hasRole('ADMINISTRATOR') || hasPermission(returnObject, 'read')")
public DescriptorList get(UUID uuid) {
return repository.findByUuid(uuid);
}
@Override
public long countDescriptorLists(DescriptorListFilter filter) throws SearchException {
if (filter.isFulltextQuery()) {
return elasticsearchService.count(DescriptorList.class, filter);
}
return descriptorListRepository.count(filter.buildPredicate());
}
private DescriptorList getDescriptorList(final DescriptorList input) {
final DescriptorList descriptorList = descriptorListRepository.findById(input.getId()).orElse(null);
if (descriptorList == null) {
throw new NotFoundElement("Record not found by id=" + input.getId());
}
if (!descriptorList.getVersion().equals(input.getVersion())) {
log.warn("Subset versions don't match anymore");
throw new ConcurrencyFailureException("Object version changed to " + descriptorList.getVersion() + ", you provided " + input.getVersion());
}
return descriptorList;
}
private DescriptorList getUnpublishedDescriptorList(final DescriptorList input) {
DescriptorList loadedDescriptorList = getDescriptorList(input);
if (loadedDescriptorList.isPublished()) {
throw new InvalidApiUsageException("Cannot modify a published Descriptor List.");
}
return loadedDescriptorList;
}
/**
* {@inheritDoc}
*/
@Transactional
@Override
@PreAuthorize("hasRole('ADMINISTRATOR') or hasPermission(#input.owner, 'write')")
public DescriptorList create(final DescriptorList input) {
return createFast(input);
}
@Override
@Transactional
@PreAuthorize("hasRole('ADMINISTRATOR') or hasPermission(#input.owner, 'write')")
public DescriptorList createFast(DescriptorList input) {
log.info("Create descriptor list {}", input);
final DescriptorList descriptorList = new DescriptorList();
copyValues(descriptorList, input);
descriptorList.setOwner(input.getOwner());
// can not be published when creating
descriptorList.setState(PublishState.DRAFT);
final DescriptorList saved = super.createFast(descriptorList);
// 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;
}
/**
* {@inheritDoc}
*/
@Override
@Transactional
@PreAuthorize("hasRole('ADMINISTRATOR') or (! #input.isPublished() and hasPermission(#input, 'write'))")
public DescriptorList update(final DescriptorList input) {
final DescriptorList descriptorList = getUnpublishedDescriptorList(input);
return update(input, descriptorList);
}
@Override
@Transactional
@PreAuthorize("hasRole('ADMINISTRATOR') or (! #target.isPublished() and hasPermission(#target, 'write'))")
public DescriptorList update(DescriptorList updated, DescriptorList target) {
return lazyLoad(updateFast(updated, target));
}
@Override
@Transactional
@PreAuthorize("hasRole('ADMINISTRATOR') or (! #target.isPublished() and hasPermission(#target, 'write'))")
public DescriptorList updateFast(DescriptorList updated, DescriptorList target) {
log.info("Updating descriptor list {}", updated);
if (updated.getOwner() != null && !target.getOwner().equals(updated.getOwner())) {
throw new InvalidApiUsageException("Descriptor list owner can't be changed");
}
copyValues(target, updated);
return descriptorListRepository.save(target);
}
@Override
@Transactional
@PreAuthorize("hasRole('ADMINISTRATOR') or (! #descriptorList.isPublished() and hasPermission(#descriptorList, 'DELETE'))")
public DescriptorList remove(DescriptorList descriptorList) {
descriptorList = getUnpublishedDescriptorList(descriptorList);
descriptorListRepository.delete(descriptorList);
return descriptorList;
}
/**
* {@inheritDoc}
*/
@Transactional
@Override
@PreAuthorize("hasRole('ADMINISTRATOR') or (! #descriptorList.isPublished() and hasPermission(#descriptorList, 'WRITE'))")
public DescriptorList removeDescriptors(final DescriptorList descriptorList, final Descriptor... descriptors) {
final DescriptorList loaded = getUnpublishedDescriptorList(descriptorList);
log.info("Remove descriptors {} of descriptor list{}.", descriptors, descriptorList);
if (descriptors == null || descriptors.length == 0) {
// Noop
return descriptorList;
}
// Which UUIDs to remove?
final Set<UUID> descriptorUuids = Arrays.stream(descriptors).map(descriptor -> descriptor.getUuid()).collect(Collectors.toSet());
// Keep descriptors that are not in the list
loaded.setDescriptors(loaded.getDescriptors().stream().filter(descriptor -> !descriptorUuids.contains(descriptor.getUuid())).collect(Collectors.toList()));
return lazyLoad(descriptorListRepository.save(loaded));
}
/**
* {@inheritDoc}
*/
@Transactional
@Override
@PreAuthorize("hasRole('ADMINISTRATOR') or (! #descriptorList.isPublished() and hasPermission(#descriptorList, 'WRITE'))")
public DescriptorList addDescriptors(final DescriptorList descriptorList, final Descriptor... descriptors) {
final DescriptorList loaded = getUnpublishedDescriptorList(descriptorList);
if (descriptors == null || descriptors.length == 0) {
// Noop
return descriptorList;
}
// append to end
for (final Descriptor descriptor : descriptors) {
if (loaded.getDescriptors().contains(descriptor)) {
log.info("Not adding existing descriptor uuid={} title={} to descriptionList uuid={} title={}", descriptor.getUuid(), descriptor.getTitle(), descriptorList.getUuid(), descriptorList
.getTitle());
} else {
log.info("Add descriptor uuid={} title={} to descriptionList uuid={} title={}", descriptor.getUuid(), descriptor.getTitle(), descriptorList.getUuid(), descriptorList
.getTitle());
loaded.getDescriptors().add(descriptor);
}
}
return lazyLoad(descriptorListRepository.save(loaded));
}
/**
* {@inheritDoc}
*/
@Transactional
@Override
@PreAuthorize("hasRole('ADMINISTRATOR') or (! #descriptorList.isPublished() and hasPermission(#descriptorList, 'WRITE'))")
public DescriptorList setDescriptors(final DescriptorList descriptorList, final Descriptor[] descriptors) {
final DescriptorList loaded = getUnpublishedDescriptorList(descriptorList);
if (descriptors == null || descriptors.length == 0) {
// Noop
return descriptorList;
}
// Re-add them in order
loaded.getDescriptors().clear();
for (final Descriptor descriptor : descriptors) {
log.info("Add descriptor uuid={} title={} to descriptionList uuid={} title={}", descriptor.getUuid(), descriptor.getTitle(), descriptorList.getUuid(), descriptorList
.getTitle());
loaded.getDescriptors().add(descriptor);
}
return lazyLoad(descriptorListRepository.save(loaded));
}
private DescriptorList lazyLoad(final DescriptorList descriptorList) {
if (descriptorList == null) {
throw new NotFoundElement("No such descriptor list");
}
if (descriptorList.getOwner() != null) {
descriptorList.getOwner().getId();
}
return descriptorList;
}
private List<Descriptor> lazyLoadDescriptors(List<Descriptor> descriptors) {
if (descriptors != null) {
descriptors.size();
descriptors.forEach(descriptor -> {
descriptor.getTerms().size();
if (descriptor.getImage() != null) {
descriptor.getImage().getId();
}
});
}
return descriptors;
}
@Override
@PostAuthorize("hasRole('ADMINISTRATOR') || returnObject==null || returnObject.isPublished() || hasPermission(returnObject, 'read')")
public DescriptorList load(final long id) {
return lazyLoad(descriptorListRepository.findById(id).orElse(null));
}
@Override
@PostAuthorize("hasRole('ADMINISTRATOR') || returnObject==null || returnObject.isPublished() || hasPermission(returnObject, 'read')")
public DescriptorList loadDescriptorList(final UUID uuid) {
return lazyLoad(descriptorListRepository.findByUuid(uuid));
}
@Override
@PostAuthorize("hasRole('ADMINISTRATOR') || returnObject==null || returnObject.entity.isPublished() || hasPermission(returnObject.entity, 'read')")
public DescriptorListTranslationService.TranslatedDescriptorList loadTranslatedDescriptorList(final UUID uuid) {
return translate(lazyLoad(descriptorListRepository.findByUuid(uuid)));
}
@Override
@PreAuthorize("hasRole('ADMINISTRATOR') || #descriptorList.isPublished() || hasPermission(#descriptorList, 'read')")
public List<Descriptor> loadDescriptors(DescriptorList descriptorList) {
descriptorList = descriptorListRepository.findByUuid(descriptorList.getUuid());
return lazyLoadDescriptors(descriptorList.getDescriptors());
}
@Override
@PostAuthorize("hasRole('ADMINISTRATOR') || returnObject==null || returnObject.isPublished() || hasPermission(returnObject, 'read')")
public DescriptorList loadDescriptorList(final UUID uuid, final int version) {
return lazyLoad(descriptorListRepository.findByUuidAndVersion(uuid, version));
}
/**
* {@inheritDoc}
*/
private void copyValues(final DescriptorList target, final DescriptorList source) {
// do not copy source#owner
target.setCrop(source.getCrop());
target.setDescription(source.getDescription());
target.setState(source.getState());
target.setTitle(source.getTitle());
target.setUrl(source.getUrl());
target.setVersionTag(source.getVersionTag());
target.setUuid(source.getUuid());
target.setBibliographicCitation(source.getBibliographicCitation());
target.setPublisher(source.getPublisher());
target.setOriginalLanguageTag(source.getOriginalLanguageTag());
}
@Override
@PostAuthorize("hasRole('ADMINISTRATOR') || returnObject==null || returnObject.isPublished() || hasPermission(returnObject, 'read')")
public DescriptorList loadDescriptorList(final DescriptorList input) {
return loadDescriptorList(input.getUuid(), input.getVersion());
}
private DescriptorList loadDescriptorList(final UUID uuid, final Integer version) {
if (uuid == null) {
throw new InvalidApiUsageException("Required parameter uuid is missing");
}
final DescriptorList descriptorList = descriptorListRepository.findByUuid(uuid);
if (descriptorList == null) {
throw new NotFoundElement("Record not found by uuid=" + uuid);
}
if (version != null && !descriptorList.getVersion().equals(version)) {
throw new ConcurrencyFailureException("Object version changed to " + descriptorList.getVersion() + ", you provided " + version);
}
return descriptorList;
}
@Override
public Page<DescriptorList> list(final DescriptorListFilter filters, final Pageable page) throws SearchException {
final BooleanBuilder published = new BooleanBuilder();
published.and(descriptorList.state.in(PublishState.PUBLISHED));
if (filters.isFulltextQuery()) {
return elasticsearchService.findAll(DescriptorList.class, filters, published, page);
}
Pageable markdownSortPageRequest = JPAUtils.toMarkdownSort(page, "title");
Page<DescriptorList> res = descriptorListRepository.findAll(published.and(filters.buildPredicate()), markdownSortPageRequest);
return new PageImpl<DescriptorList>(res.getContent(), page, res.getTotalElements());
}
@Override
public Page<DescriptorListTranslationService.TranslatedDescriptorList> listFiltered(final DescriptorListFilter filters, final Pageable page) throws SearchException {
final BooleanBuilder published = new BooleanBuilder();
published.and(descriptorList.state.in(PublishState.PUBLISHED));
Page<DescriptorList> res;
if (filters.isFulltextQuery()) {
res = elasticsearchService.findAll(DescriptorList.class, filters, published, page);
} else {
Pageable markdownSortPageRequest = JPAUtils.toMarkdownSort(page, "title");
res = descriptorListRepository.findAll(published.and(filters.buildPredicate()), markdownSortPageRequest);
}
return new PageImpl<>(translationSupport.getTranslated(res.getContent()), page, res.getTotalElements());
}
/**
* {@inheritDoc}
*/
@Override
public Map<String, ElasticsearchService.TermResult> getSuggestions(DescriptorListFilter filter) throws SearchException, IOException {
assert (filter != null);
Set<String> suggestions = Sets.newHashSet("crop");
Map<String, ElasticsearchService.TermResult> suggestionRes = new HashMap<>(suggestions.size());
for (String suggestionKey : suggestions) {
DescriptorListFilter suggestionFilter = filter.copy(DescriptorListFilter.class);
suggestionFilter.state(PublishState.PUBLISHED);
try {
suggestionFilter.clearFilter(suggestionKey);
} catch (NoSuchFieldException | IllegalAccessException e) {
log.error("Error while clearing filter: ", e.getMessage());
}
ElasticsearchService.TermResult suggestion = elasticsearchService.termStatisticsAuto(DescriptorList.class, suggestionFilter, 100, suggestionKey);
suggestionRes.put(suggestionKey, suggestion);
}
return suggestionRes;
}
@Override
public Page<DescriptorList> listDescriptorListsForCurrentUser(final DescriptorListFilter filter, final Pageable page) throws SearchException {
final Pageable markdownSortPageRequest = JPAUtils.toMarkdownSort(page, "title");
Page<DescriptorList> res;
if (securityUtils.hasRole(UserRole.ADMINISTRATOR)) {
if (filter.isFulltextQuery()) {
res = elasticsearchService.findAll(DescriptorList.class, filter, markdownSortPageRequest);
} else {
res = descriptorListRepository.findAll(filter.buildPredicate(), markdownSortPageRequest);
}
} else {
final HashSet<Long> partners = new HashSet<>(securityUtils.listObjectIdentityIdsForCurrentUser(Partner.class, BasePermission.WRITE));
if (filter.isFulltextQuery()) {
res = elasticsearchService.findAll(DescriptorList.class, filter, descriptorList.owner().id.in(partners), markdownSortPageRequest);
} else {
res = descriptorListRepository.findAll(descriptorList.owner().id.in(partners).and(filter.buildPredicate()), markdownSortPageRequest);
}
}
return new PageImpl<DescriptorList>(res.getContent(), page, res.getTotalElements());
}
/**
* {@inheritDoc}
*/
@Override
@Transactional
@PreAuthorize("hasRole('ADMINISTRATOR')")
@NotifyOnPublished
public DescriptorList approveDescriptorList(final DescriptorList descriptorList) {
final DescriptorList loaded = getUnpublishedDescriptorList(descriptorList);
if (loaded.getState() == PublishState.DRAFT) {
throw new InvalidApiUsageException("Descriptor List should be sent for review before publication");
}
for (final Descriptor descriptor : loaded.getDescriptors()) {
if (descriptor.getState() != PublishState.PUBLISHED) {
log.info("Publishing descriptor {}", descriptor);
try {
descriptorService.approveDescriptor(descriptor);
} catch (InvalidApiUsageException e) {
throw e;
}
}
}
loaded.setState(PublishState.PUBLISHED);
return lazyLoad(descriptorListRepository.saveAndFlush(loaded));
}
/**
* {@inheritDoc}
*/
@Override
@Transactional
@PreAuthorize("hasRole('ADMINISTRATOR') or hasPermission(#descriptorList, 'WRITE')")
@NotifyForReview
public DescriptorList reviewDescriptorList(final DescriptorList descriptorList) {
final DescriptorList loaded = getUnpublishedDescriptorList(descriptorList);
if (loaded.getState() == PublishState.REVIEWING) {
throw new InvalidApiUsageException("The DescriptorList is already under approval");
}
for (final Descriptor descriptor : loaded.getDescriptors()) {
if (descriptor.getState() == PublishState.DRAFT) {
log.info("Send to review descriptor {}", descriptor);
try {
descriptorService.reviewDescriptor(descriptor);
} catch (InvalidApiUsageException e) {
throw e;
}
}
}
loaded.setState(PublishState.REVIEWING);
return lazyLoad(descriptorListRepository.save(loaded));
}
/**
* {@inheritDoc}
*/
@Override
@Transactional
@PreAuthorize("hasRole('ADMINISTRATOR') or hasPermission(#descriptorList, 'administration')")
public DescriptorList rejectDescriptorList(final DescriptorList descriptorList) {
final DescriptorList loaded = descriptorListRepository.findByUuidAndVersion(descriptorList.getUuid(), descriptorList.getVersion());
if (loaded == null) {
throw new NotFoundElement("No DescriptorList with specified uuid and version");
}
if (loaded.isPublished() && !securityUtils.hasRole(UserRole.ADMINISTRATOR)) {
long oneDay = 24 * 60 * 60 * 1000;
if (loaded.getLastModifiedDate() != null && loaded.getLastModifiedDate().toEpochMilli() <= (System.currentTimeMillis() - oneDay)) {
throw new InvalidApiUsageException("Cannot be un-published. More than 24 hours have passed since the publication.");
}
}
loaded.setState(PublishState.DRAFT);
descriptorListRepository.save(loaded);
for (final Descriptor referencedDescriptor : loaded.getDescriptors()) {
log.info("Rejecting descriptor {}", referencedDescriptor);
try {
descriptorService.rejectDescriptor(referencedDescriptor);
} catch (final InvalidApiUsageException e) {
log.info("Not unpublishing a descriptor referenced in a published dataset or descriptor list: {}", e.getMessage());
}
}
return lazyLoad(descriptorListRepository.save(loaded));
}
/**
* {@inheritDoc}
*/
@Override
public List<DescriptorList> autocompleteDescriptorLists(final String text) {
Pageable markdownSortPageRequest = JPAUtils.toMarkdownSort(PageRequest.of(0, 15, Sort.by("title")), "title");
if (StringUtils.isBlank(text)) {
final Predicate predicate = descriptorList.state.in(PublishState.PUBLISHED);
return descriptorListRepository.findAll(predicate, markdownSortPageRequest).getContent();
} else {
final Predicate predicate = descriptorList.state.in(PublishState.PUBLISHED).and(descriptorList.title.containsIgnoreCase(text));
return descriptorListRepository.findAll(predicate, markdownSortPageRequest).getContent();
}
}
/**
* {@inheritDoc}
*/
@Override
@PreAuthorize("#descriptorList.isPublished() || hasRole('ADMINISTRATOR') || hasPermission(#descriptorList, 'read')")
public void exportDescriptorList(final DescriptorList descriptorList, final OutputStream outputStream) throws IOException {
downloadService.writeXlsxDescriptorList(descriptorListRepository.getReferenceById(descriptorList.getId()), outputStream);
}
@Override
@PreAuthorize("hasRole('ADMINISTRATOR') || hasPermission(#original, 'write')")
public DescriptorListLang machineTranslate(DescriptorList 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 DescriptorListLang();
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(DescriptorList.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(DescriptorList.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
@PreAuthorize("hasRole('ADMINISTRATOR') || #descriptorList.isPublished() || hasPermission(#descriptorList, 'read')")
public List<DescriptorTranslationService.TranslatedDescriptor> loadTranslatedDescriptors(DescriptorList descriptorList) {
var reloaded = descriptorListRepository.findById(descriptorList.getId()).orElseThrow();
return descriptorTranslationService.getTranslated(reloaded.getDescriptors());
}
}