ActivityPostServiceImpl.java

/*
 * Copyright 2024 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 com.querydsl.core.BooleanBuilder;

import org.apache.commons.lang3.StringUtils;
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.RepositoryFile;
import org.genesys.filerepository.model.RepositoryFolder;
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.model.UserRole;
import org.genesys.server.model.impl.ActivityPost;
import org.genesys.server.model.impl.ActivityPostLang;
import org.genesys.server.model.impl.QActivityPost;
import org.genesys.server.persistence.ActivityPostLangRepository;
import org.genesys.server.persistence.ActivityPostRepository;
import org.genesys.server.service.ActivityPostService;
import org.genesys.server.service.ActivityPostTranslationService;
import org.genesys.server.service.HtmlSanitizer;
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.ActivityPostTranslationService.TranslatedActivityPost;
import org.genesys.server.service.filter.ActivityPostFilter;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.cache.annotation.CacheEvict;
import org.springframework.cache.annotation.Cacheable;
import org.springframework.data.domain.Page;
import org.springframework.data.domain.PageImpl;
import org.springframework.data.domain.Pageable;
import org.springframework.data.domain.Sort;
import org.springframework.security.access.prepost.PreAuthorize;
import org.springframework.stereotype.Component;
import org.springframework.stereotype.Service;
import org.springframework.transaction.annotation.Transactional;
import org.springframework.validation.annotation.Validated;
import org.springframework.web.multipart.MultipartFile;

import java.io.IOException;
import java.nio.file.Path;
import java.nio.file.Paths;
import java.time.Instant;
import java.util.Locale;
import java.util.Map;
import java.util.Objects;
import java.util.UUID;

@Service
@Transactional(readOnly = true)
@Validated
public class ActivityPostServiceImpl 
		extends FilteredTranslatedCRUDServiceImpl<ActivityPost, ActivityPostLang, ActivityPostTranslationService.TranslatedActivityPost, ActivityPostFilter, ActivityPostRepository>
		implements ActivityPostService {

	@Autowired
	private SecurityUtils securityUtils;

	@Autowired
	private HtmlSanitizer htmlSanitizer;

	@Autowired
	protected RepositoryService repositoryService;

	@Autowired
	private CustomAclService aclService;

	@Autowired
	protected AsAdminInvoker asAdminInvoker;
	
	@Autowired(required = false)
	private TranslatorService translatorService;

	@Component(value = "ActivityPostTranslationSupport")
	protected static class ActivityPostTranslationSupport
		extends BaseTranslationSupport<ActivityPost, ActivityPostLang, ActivityPostTranslationService.TranslatedActivityPost, ActivityPostFilter, ActivityPostLangRepository>
		implements ActivityPostTranslationService {

		@Override
		@Transactional
		@PreAuthorize("hasRole('ADMINISTRATOR') or hasRole('CONTENTMANAGER')")
		public ActivityPostLang create(ActivityPostLang source) {
			return super.create(source);
		}

		@Override
		@Transactional
		@PreAuthorize("hasRole('ADMINISTRATOR') or hasRole('CONTENTMANAGER')")
		public ActivityPostLang update(ActivityPostLang updated, ActivityPostLang target) {
			return super.update(updated, target);
		}

		@Override
		@Transactional
		@PreAuthorize("hasRole('ADMINISTRATOR') or hasRole('CONTENTMANAGER')")
		public ActivityPostLang remove(ActivityPostLang entity) {
			return super.remove(entity);
		}
	}

	@Override
	@Cacheable(value = "contentcache", key = "'activityPost-' + #id")
	public ActivityPost get(long id) {
		var inst = Instant.now();
		ActivityPost activityPost;
		if (! (securityUtils.hasRole(UserRole.ADMINISTRATOR) || securityUtils.hasRole(UserRole.CONTENTMANAGER))) {
			QActivityPost qActivityPost = QActivityPost.activityPost;
			activityPost = repository.findOne(qActivityPost.publishDate.before(inst)
				.andAnyOf(qActivityPost.expirationDate.isNull(), qActivityPost.expirationDate.after(inst))
				.and(qActivityPost.id.eq(id))).orElse(null);
		} else {
			activityPost = repository.findById(id).orElse(null);
		}
		if (activityPost == null) {
			throw new NotFoundElement("Record not found by id=" + id);
		}
		return lazyLoad(activityPost);
	}

	/**
	 * Create and persist a new {@link ActivityPost}
	 */
	@Transactional
	@PreAuthorize("hasRole('ADMINISTRATOR') or hasRole('CONTENTMANAGER')")
	@CacheEvict(value = "contentcache", allEntries = true)
	@Override
	public ActivityPost create(ActivityPost activityPost) {
		return lazyLoad(createFast(activityPost));
	}

	@Override
	@Transactional
	@PreAuthorize("hasRole('ADMINISTRATOR') or hasRole('CONTENTMANAGER')")
	@CacheEvict(value = "contentcache", allEntries = true)
	public ActivityPost createFast(ActivityPost source) {
		final ActivityPost newPost = new ActivityPost();
		newPost.apply(source);
		return updatePostData(newPost, newPost.getTitle(), newPost.getSummary(), newPost.getBody());
	}

	@Override
	@Transactional(readOnly = false)
	@PreAuthorize("hasRole('ADMINISTRATOR') or hasRole('CONTENTMANAGER')")
	@CacheEvict(value = "contentcache", allEntries = true)
	public ActivityPost updateFast(ActivityPost updated, ActivityPost target) {
		target.apply(updated);
		return updatePostData(target, target.getTitle(), target.getSummary(), target.getBody());
	}

	@Override
	@Transactional(readOnly = false)
	@PreAuthorize("hasRole('ADMINISTRATOR') or hasRole('CONTENTMANAGER')")
	@CacheEvict(value = "contentcache", allEntries = true)
	public ActivityPost update(ActivityPost updated, ActivityPost target) {
		return lazyLoad(updateFast(updated, target));
	}

	@Override
	@Cacheable(value = "contentcache")
	public Page<ActivityPost> list(ActivityPostFilter filter, Pageable pageable) {
		final BooleanBuilder predicate = new BooleanBuilder();
		if (filter != null) {
			predicate.and(filter.buildPredicate());
		}
		if (! (securityUtils.hasRole(UserRole.ADMINISTRATOR) || securityUtils.hasRole(UserRole.CONTENTMANAGER))) {
			QActivityPost qActivityPost = QActivityPost.activityPost;
			predicate.and(qActivityPost.publishDate.before(Instant.now())
				.andAnyOf(qActivityPost.expirationDate.isNull(), qActivityPost.expirationDate.after(Instant.now())));
		}
		Page<ActivityPost> activityPostPage = repository.findAll(predicate, pageable);
		activityPostPage.forEach(this::lazyLoad);
		return activityPostPage;
	}

	private ActivityPost updatePostData(ActivityPost post, String title, String summary, String body) {
		post.setTitle(htmlSanitizer.sanitize(title));
		post.setSummary(summary); // summary is in Markdown
		post.setBody(htmlSanitizer.sanitize(body));

		return lazyLoad(repository.save(post));
	}

	@Override
	@Transactional(readOnly = false)
	@PreAuthorize("hasRole('ADMINISTRATOR') or hasRole('CONTENTMANAGER')")
	@CacheEvict(value = "contentcache", allEntries = true)
	public ActivityPost remove(ActivityPost activityPost) {
		if (activityPost == null || activityPost.getId() == null)
			throw new InvalidApiUsageException("An existing ActivityPost must be provided.");

		// Reload
		activityPost = get(activityPost.getId());

		final Path repositoryPath = getActivityPostFolderPath(activityPost);
		try {
			if (! repositoryService.hasPath(repositoryPath)) {
				// No related repository folder, so just delete activity post
				repository.delete(activityPost);
				return activityPost;
			}

			try {
				// Remove all files related to activity post
				for (RepositoryFile file : repositoryService.getFiles(repositoryPath, Sort.unsorted())) {
					repositoryService.removeFile(file);
				}
			} catch (NoSuchRepositoryFileException | IOException e) {
				LOG.warn("Could not delete file from ActivityPost {}", activityPost.getId());
			}

			// Delete activity post
			repository.delete(activityPost);
			// Delete folder
			repositoryService.deleteFolder(repositoryPath);
		} catch (InvalidRepositoryPathException e) {
			// Must be OK
			LOG.warn("Invalid repository path: {}", e.getMessage());
		} catch (FolderNotEmptyException e) {
			LOG.warn("Could not remove news folder: {}", e.getMessage(), e);
		}
		return activityPost;
	}

	@Override
	@Transactional
	@PreAuthorize("hasRole('ADMINISTRATOR') or hasRole('CONTENTMANAGER') or hasPermission(#activityPost, 'ADMINISTRATION')")
	public RepositoryFile uploadActivityPostFile(ActivityPost activityPost, final MultipartFile file) throws IOException, InvalidRepositoryPathException, InvalidRepositoryFileDataException {
		if (file == null)
			throw new InvalidRepositoryPathException("Nothing to upload. File must be provided.");
		if (activityPost == null || activityPost.getId() == null)
			throw new InvalidApiUsageException("An existing ActivityPost must be provided.");

		// Reload
		activityPost = get(activityPost.getId());

		// Ensure news folder
		var activityPostFolder = ensureNewsFolder(activityPost);

		String originalFilename = UUID.randomUUID().toString().concat("_").concat(file.getOriginalFilename());
		LOG.info("Upload file {} to path {}", originalFilename, activityPostFolder.getFolderPath());
		var savedFile = repositoryService.addFile(activityPostFolder.getFolderPath(), originalFilename, file.getContentType(), file.getInputStream(), null);

		if (savedFile != null) {
			// Relax permissions on descriptor image: allow USERS and ANONYMOUS to read the image
			aclService.makePubliclyReadable(savedFile, true);
		}

		return savedFile;
	}

	private RepositoryFolder ensureNewsFolder(final ActivityPost activityPost) {
		try {
			final Path repositoryPath = getActivityPostFolderPath(activityPost);
			return asAdminInvoker.invoke(() -> {
				// Ensure target folder exists for the ActivityPost
				return repositoryService.ensureFolder(repositoryPath);
			});
		} catch (Exception e) {
			LOG.warn("Could not create folder: {}", e.getMessage());
			throw new InvalidApiUsageException("Could not create folder", e);
		}
	}

	private Path getActivityPostFolderPath(final ActivityPost activityPost) {
		assert (activityPost != null);
		assert (activityPost.getId() != null);

		return Paths.get("/content", "news", activityPost.getId().toString()).toAbsolutePath();
	}

	protected ActivityPost lazyLoad(ActivityPost activityPost) {
		if(activityPost != null) {
			if (activityPost.getCoverImage() != null) {
				activityPost.getCoverImage().getId();
			}
		}
		return activityPost;
	}


	@Override
	public ActivityPostLang machineTranslate(ActivityPost 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 ActivityPostLang();
		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(ActivityPost.class, "title", enTranslation.getTitle(), Locale.ENGLISH))
				.texts(Map.of(
					"title", new FormattedText(TextFormat.markdown, enTranslation.getTitle()),
					"summary", new FormattedText(TextFormat.html, enTranslation.getSummary()),
					"body", new FormattedText(TextFormat.markdown, enTranslation.getBody())
				));

		} else {
			// Translations to English use the original texts

			var originLocale = Locale.forLanguageTag(original.getOriginalLanguageTag());
			builder
				.sourceLang(original.getOriginalLanguageTag())
				.context(buildTranslationContext(ActivityPost.class, "title", original.getTitle(), originLocale))
				.texts(Map.of(
					"title", new FormattedText(TextFormat.html, original.getTitle()),
					"summary", new FormattedText(TextFormat.html, original.getSummary()),
					"body", new FormattedText(TextFormat.html, original.getBody())
				));
		}

		var translations = translatorService.translate(builder.build());

		if (StringUtils.isNotBlank(original.getTitle())) {
			mt.setTitle(translations.getTexts().get("title"));
		}
		if (StringUtils.isNotBlank(original.getSummary())) {
			mt.setSummary(translations.getTexts().get("summary"));
		}
		if (StringUtils.isNotBlank(original.getBody())) {
			mt.setBody(translations.getTexts().get("body"));
		}
		return mt;
	}

	@Override
	public Page<TranslatedActivityPost> allNews(Pageable page) {
		var inst = Instant.now();
		QActivityPost qActivityPost = QActivityPost.activityPost;
		var posts = repository.findAll(qActivityPost.publishDate.before(inst)
				.andAnyOf(qActivityPost.expirationDate.isNull(), (qActivityPost.expirationDate.after(inst))), page);
		return new PageImpl<>(translationSupport.getTranslated(posts.getContent()), page, posts.getTotalElements());
	}
	
}