All Downloads are FREE. Search and download functionalities are using the official Maven repository.

net.yadaframework.components.YadaFileManager Maven / Gradle / Ivy

There is a newer version: 0.7.7.R4
Show newest version
package net.yadaframework.components;

import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.nio.file.Files;
import java.nio.file.StandardCopyOption;
import java.util.Date;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

import org.apache.tomcat.util.http.fileupload.IOUtils;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.annotation.Lazy;
import org.springframework.stereotype.Service;
import org.springframework.web.multipart.MultipartFile;

import net.yadaframework.core.YadaConfiguration;
import net.yadaframework.exceptions.YadaInvalidUsageException;
import net.yadaframework.persistence.entity.YadaAttachedFile;
import net.yadaframework.persistence.entity.YadaManagedFile;
import net.yadaframework.persistence.repository.YadaAttachedFileDao;
import net.yadaframework.persistence.repository.YadaFileManagerDao;
import net.yadaframework.raw.YadaIntDimension;

/**
 * The File Manager handles uploaded files. They are kept in a specific folder where they can be
 * chosen to be attached to entities.
 *
 */
// Not in YadaWebCMS because used by YadaSession and YadaUtil
@Service
public class YadaFileManager {
	private final transient Logger log = LoggerFactory.getLogger(this.getClass());

    // Use @Lazy to resolve circular reference error
	@Autowired @Lazy private YadaAttachedFileDao yadaAttachedFileDao;
	@Autowired private YadaConfiguration config;
	@Autowired private YadaUtil yadaUtil;
	// Use @Lazy to resolve circular reference error
	@Autowired @Lazy private YadaWebUtil yadaWebUtil;
	@Autowired private YadaFileManagerDao yadaFileManagerDao;
	
	protected String COUNTER_SEPARATOR="_";

	// Image to return when no image is available
	public final static String NOIMAGE_DATA="";

	// TODO distinguere tra mobile portrait e mobile landscape
	// TODO le dimensioni mobile/desktop devono essere configurabili
	// TODO mantenere l'immagine caricata nella versione originale

	/**
	 * Move the file to the public temp folder for later processing.
	 * @param yadaManagedFile
	 * @return
	 * @throws IOException
	 */
	public YadaManagedFile moveToTemp(YadaManagedFile yadaManagedFile) throws IOException {
		File destinationFile = new File(config.getTempImageDir(), yadaManagedFile.getFilename());
		destinationFile = YadaUtil.findAvailableName(destinationFile, null);
		return yadaManagedFile.move(destinationFile);
	}

	/**
	 * Remove a managed file from disk and database
	 * @param managedFile
	 * @return true when deleted from disk, false when not deleted from disk (could have been deleted from db though)
	 */
	public boolean delete(YadaManagedFile managedFile) {
		return yadaFileManagerDao.delete(managedFile);
	}

	/**
	 * Returns the absolute path of a managed file
	 * @param yadaAttachedFile the attachment
	 * @param filename the relative file name, can be yadaAttachedFile.getFilename(), yadaAttachedFile.getFilenameDesktop(), yadaAttachedFile.getFilenameMobile()
	 * @return the File or null
	 */
	public File getAbsoluteFile(YadaManagedFile managedFile) {
		if (managedFile==null) {
			return null;
		}
		return managedFile.getAbsoluteFile();
	}

//	/**
//	 * Deletes a file from disk and database
//	 * @param managedFile the file to delete
//	 */
//	public void delete(YadaManagedFile managedFile) {
//		yadaFileManagerDao.delete(managedFile);
//	}
	
//	private YadaAttachedFile moveFolderByPattern(YadaAttachedFile toMove, List fileRenames) {
//		if (fileRenames!=null) {
//			String relativeFolderPath = toMove.getRelativeFolderPath();
//			for (YadaPatternAndReplace yadaPatternAndReplace : fileRenames) {
//				Matcher matcher = yadaPatternAndReplace.getMatcher(relativeFolderPath);
//				if (matcher.find()) {
//					String newRelativeFolderPath = matcher.replaceAll(yadaPatternAndReplace.getReplace());
//					toMove.setRelativeFolderPath(newRelativeFolderPath);
//					break; // Stop at the first replacement
//				}
//			}
//		}
//		return toMove;
//	}

	/**
	 * You don't usually need to call this method but {@link YadaUtil#copyEntity(net.yadaframework.core.CloneableFiltered)} instead.
* Makes a copy of just the filesystem files. New names are generated from the old ones by appending an incremental number. * The input YadaAttachedFile is updated with the new names. The old files are not deleted. * This method is used by YadaUtil.copyEntity() when a field is a YadaAttachedFile so that files are copied too. * @param yadaAttachedFileCopy a copy of some other YadaAttachedFile that will be left unchanged * @param yadaAttachedFileCloneSet when not null, all files are copied to a temp folder. * This is useful when the final path depends on the id * of a cloned object so it can't be determined during cloning. * The method yadaAttachedFileCloneSet.moveAll() will have to be called after the clone has been persisted. * @return the saved YadaAttachedFile with new files on disk, eventually in a different folder * @throws IOException * @see {@link YadaUtil#copyEntity(net.yadaframework.core.CloneableFiltered)} */ public YadaAttachedFile duplicateFiles(YadaAttachedFile yadaAttachedFileCopy, YadaAttachedFileCloneSet yadaAttachedFileCloneSet) throws IOException { if (yadaAttachedFileCopy==null) { return null; } File sourceFileMobile = getAbsoluteMobileFile(yadaAttachedFileCopy); File sourceFileDesktop = getAbsoluteDesktopFile(yadaAttachedFileCopy); File sourceFilePdf = getAbsolutePdfFile(yadaAttachedFileCopy); File sourceFile = getAbsoluteFile(yadaAttachedFileCopy); // Sometimes sourceFileDesktop and sourceFile have the same value: don't copy the file twice! boolean deskSameAsDefault = sourceFileDesktop!=null && sourceFile!=null && sourceFileDesktop.getAbsolutePath().equals(sourceFile.getAbsolutePath()); if (yadaAttachedFileCloneSet!=null) { yadaAttachedFileCloneSet.handle(yadaAttachedFileCopy); // Moved to temp folder } if (sourceFileMobile!=null) { File targetFileMobile = getAbsoluteMobileFile(yadaAttachedFileCopy); targetFileMobile = YadaUtil.findAvailableName(targetFileMobile, null); try (InputStream inputStream = new FileInputStream(sourceFileMobile); OutputStream outputStream = new FileOutputStream(targetFileMobile)) { IOUtils.copy(inputStream, outputStream); } yadaAttachedFileCopy.setFilenameMobile(targetFileMobile.getName()); } File targetFileDesktop = null; if (sourceFileDesktop!=null) { targetFileDesktop = getAbsoluteDesktopFile(yadaAttachedFileCopy); targetFileDesktop = YadaUtil.findAvailableName(targetFileDesktop, null); try (InputStream inputStream = new FileInputStream(sourceFileDesktop); OutputStream outputStream = new FileOutputStream(targetFileDesktop)) { IOUtils.copy(inputStream, outputStream); } yadaAttachedFileCopy.setFilenameDesktop(targetFileDesktop.getName()); } if (sourceFilePdf!=null) { File targetFilePdf = getAbsolutePdfFile(yadaAttachedFileCopy); targetFilePdf = YadaUtil.findAvailableName(targetFilePdf, null); try (InputStream inputStream = new FileInputStream(sourceFilePdf); OutputStream outputStream = new FileOutputStream(targetFilePdf)) { IOUtils.copy(inputStream, outputStream); } yadaAttachedFileCopy.setFilenamePdf(targetFilePdf.getName()); } if (sourceFile!=null && !deskSameAsDefault) { File targetFile = getAbsoluteFile(yadaAttachedFileCopy); targetFile = YadaUtil.findAvailableName(targetFile, null); try (InputStream inputStream = new FileInputStream(sourceFile); OutputStream outputStream = new FileOutputStream(targetFile)) { IOUtils.copy(inputStream, outputStream); } yadaAttachedFileCopy.setFilename(targetFile.getName()); } if (deskSameAsDefault && targetFileDesktop!=null) { // Default file is the same as desktop file yadaAttachedFileCopy.setFilename(targetFileDesktop.getName()); } return yadaAttachedFileDao.save(yadaAttachedFileCopy); } /** * Returns the absolute path of the mobile file * @param yadaAttachedFile the attachment * @return the File or null */ public File getAbsoluteMobileFile(YadaAttachedFile yadaAttachedFile) { if (yadaAttachedFile!=null) { return getAbsoluteFile(yadaAttachedFile, yadaAttachedFile.getFilenameMobile()); } return null; } /** * Returns the absolute path of the desktop file * @param yadaAttachedFile the attachment * @return the File or null */ public File getAbsoluteDesktopFile(YadaAttachedFile yadaAttachedFile) { if (yadaAttachedFile!=null) { return getAbsoluteFile(yadaAttachedFile, yadaAttachedFile.getFilenameDesktop()); } return null; } /** * Returns the absolute path of the pdf file * @param yadaAttachedFile the attachment * @return the File or null */ public File getAbsolutePdfFile(YadaAttachedFile yadaAttachedFile) { if (yadaAttachedFile!=null) { return getAbsoluteFile(yadaAttachedFile, yadaAttachedFile.getFilenamePdf()); } return null; } /** * Returns the absolute path of the default file (no mobile/desktop variant) * @param yadaAttachedFile the attachment * @return the File or null */ public File getAbsoluteFile(YadaAttachedFile yadaAttachedFile) { if (yadaAttachedFile!=null) { return getAbsoluteFile(yadaAttachedFile, yadaAttachedFile.getFilename()); } return null; } /** * Returns the absolute path of a file * @param yadaAttachedFile the attachment * @param filename the relative file name, can be yadaAttachedFile.getFilename(), yadaAttachedFile.getFilenameDesktop(), yadaAttachedFile.getFilenameMobile() * @return the File or null */ public File getAbsoluteFile(YadaAttachedFile yadaAttachedFile, String filename) { if (filename==null || yadaAttachedFile==null) { return null; } File targetFolder = new File(config.getContentPath(), yadaAttachedFile.getRelativeFolderPath()); return new File(targetFolder, filename); } // /** // * Find one from repository // * @param yadaAttachedFileId the attachment id // */ // public YadaAttachedFile findOne(Long yadaAttachedFileId) { // return yadaAttachedFileRepository.findOne(yadaAttachedFileId); // } /** * Deletes from the filesystem all files related to the attachment * @param yadaAttachedFileId the attachment id * @see #deleteFileAttachment(YadaAttachedFile) */ @Deprecated // Removed in yada 0.7.0 because it uses a Spring Data api (?) public void deleteFileAttachment(Long yadaAttachedFileId) { deleteFileAttachment(yadaAttachedFileDao.findById(yadaAttachedFileId).orElse(null)); } /** * Deletes from the filesystem all files related to the attachment * @param yadaAttachedFile the attachment * @see #deleteFileAttachment(Long) */ public void deleteFileAttachment(YadaAttachedFile yadaAttachedFile) { if (yadaAttachedFile==null) { return; } if (yadaAttachedFile.getFilename() != null) { getAbsoluteFile(yadaAttachedFile, yadaAttachedFile.getFilename()).delete(); } if (yadaAttachedFile.getFilenamePdf() != null) { getAbsoluteFile(yadaAttachedFile, yadaAttachedFile.getFilenamePdf()).delete(); } if (yadaAttachedFile.getFilenameDesktop() != null) { getAbsoluteFile(yadaAttachedFile, yadaAttachedFile.getFilenameDesktop()).delete(); } if (yadaAttachedFile.getFilenameMobile() != null) { getAbsoluteFile(yadaAttachedFile, yadaAttachedFile.getFilenameMobile()).delete(); } } /** * Deletes from the filesystem all files related to the attachments * @param yadaAttachedFiles the attachments */ public void deleteFileAttachment(List yadaAttachedFiles) { for (YadaAttachedFile yadaAttachedFile : yadaAttachedFiles) { deleteFileAttachment(yadaAttachedFile); } } /** * Returns the (relative) url of the mobile image if any, or null * @param yadaAttachedFile * @return */ public String getMobileImageUrl(YadaAttachedFile yadaAttachedFile) { if (yadaAttachedFile==null) { return NOIMAGE_DATA; } String imageName = yadaAttachedFile.getFilenameMobile(); if (imageName==null) { return NOIMAGE_DATA; } return computeUrl(yadaAttachedFile, imageName); } /** * Returns the (relative) url of the desktop image. If not defined, falls back to the plain file. * @param yadaAttachedFile * @return */ public String getDesktopImageUrl(YadaAttachedFile yadaAttachedFile) { if (yadaAttachedFile==null) { return NOIMAGE_DATA; } String imageName = yadaAttachedFile.getFilenameDesktop(); if (imageName==null) { return getFileUrl(yadaAttachedFile); } return computeUrl(yadaAttachedFile, imageName); } /** * Returns the (relative) url of the pdf image if any, or null. * @param yadaAttachedFile * @return */ public String getPdfImageUrl(YadaAttachedFile yadaAttachedFile) { if (yadaAttachedFile==null) { return NOIMAGE_DATA; } String imageName = yadaAttachedFile.getFilenamePdf(); if (imageName==null) { return NOIMAGE_DATA; } return computeUrl(yadaAttachedFile, imageName); } /** * Returns the (relative) url of the file, or null. * @param yadaAttachedFile * @return */ public String getFileUrl(YadaAttachedFile yadaAttachedFile) { if (yadaAttachedFile==null) { return null; } String fileName = yadaAttachedFile.getFilename(); if (fileName==null) { return null; } return computeUrl(yadaAttachedFile, fileName); } private String computeUrl(YadaAttachedFile yadaAttachedFile, String filename) { return getFileUrl(yadaAttachedFile.getRelativeFolderPath(), filename); } /** * Returns the url of a YadaAttachedFile that has the given attributes * @param relativeFolderPath * @param filename * @return The URL as a String */ public String getFileUrl(String relativeFolderPath, String filename) { StringBuilder result = new StringBuilder(config.getContentUrl()); result.append(relativeFolderPath) .append("/") .append(filename); return result.toString(); } /** * Uploads a file into the uploads folder. * @param multipartFile * @return * @throws IOException */ private File uploadFileInternal(MultipartFile multipartFile) throws IOException { String originalFilename = multipartFile.getOriginalFilename(); String targetName = yadaUtil.ensureSafeFilename(originalFilename); String[] filenameParts = YadaUtil.splitFileNameAndExtension(targetName); File targetFolder = config.getUploadsFolder(); // Useless: doesn't throw an exception when it fails: targetFolder.mkdirs(); File targetFile = YadaUtil.findAvailableName(new File(targetFolder.getAbsolutePath()), filenameParts[0], filenameParts[1], COUNTER_SEPARATOR); multipartFile.transferTo(targetFile); // try (InputStream inputStream = multipartFile.getInputStream(); OutputStream outputStream = new FileOutputStream(targetFile)) { // IOUtils.copy(inputStream, outputStream); // } catch (IOException e) { // throw e; // } log.debug("File {} uploaded", targetFile.getAbsolutePath()); return targetFile; } /** * Copies a received file to the upload folder. The returned File is the only pointer to the uploaded file. * @param multipartFile file coming from the http request * @return the uploaded file with a unique name, or null if the user did not send any file * @throws IOException */ public File uploadFile(MultipartFile multipartFile) throws IOException { if (yadaWebUtil.isMultipartMissing(multipartFile)) { log.debug("No file sent for upload"); return null; } if (log.isDebugEnabled()) { if (multipartFile.getSize()==0) { log.debug("Empty file sent for upload: {}", multipartFile.getName()); // We still upload it } } File targetFile = uploadFileInternal(multipartFile); return targetFile; } /** * Copies a received file to the upload folder. A pointer to the file is stored in the database as YadaManagedFile * @param multipartFile file coming from the http request * @return the uploaded file with a unique name, or null if the user did not send any file * @throws IOException */ public YadaManagedFile manageFile(MultipartFile multipartFile) throws IOException { return manageFile(multipartFile, null); } /** * Copies a received file to the upload folder. A pointer to the file is stored in the database as YadaManagedFile * @param multipartFile file coming from the http request * @param description a user description for the file * @return the uploaded file with a unique name, or null if the user did not send any file * @throws IOException */ public YadaManagedFile manageFile(MultipartFile multipartFile, String description) throws IOException { if (multipartFile==null || multipartFile.getSize()==0) { log.debug("No file sent for upload"); return null; } File targetFile = uploadFileInternal(multipartFile); YadaManagedFile yadaManagedFile = yadaFileManagerDao.createManagedFile(multipartFile, targetFile, description); return yadaManagedFile; } /** * Replace the file associated with the current attachment * The multipartFile is moved to the destination when config.isFileManagerDeletingUploads() is true, otherwise the original is copied * and left unchanged. * @param currentAttachedFile an existing attachment, never null * @param multipartFile the original uploaded file, to get the client filename. If null, the client filename is not changed. * @return YadaAttachedFile if the file is uploaded, null if no file was sent by the user * @throws IOException */ public YadaAttachedFile attachReplace(YadaAttachedFile currentAttachedFile, MultipartFile multipartFile, String namePrefix) throws IOException { File managedFile = uploadFile(multipartFile); return attachReplace(currentAttachedFile, managedFile, multipartFile, namePrefix); } /** * Replace the file associated with the current attachment * The managedFile is moved to the destination when config.isFileManagerDeletingUploads() is true, otherwise the original is copied * and left unchanged. * @param currentAttachedFile an existing attachment, never null * @param managedFile the new file to set * @param multipartFile the original uploaded file, to get the client filename. If null, the client filename is not changed. * @return YadaAttachedFile if the file has been replaced, null if managedFile is null * @throws IOException */ public YadaAttachedFile attachReplace(YadaAttachedFile currentAttachedFile, File managedFile, MultipartFile multipartFile, String namePrefix) throws IOException { return attachReplace(currentAttachedFile, managedFile, multipartFile, namePrefix, null, null, null); } /** * Replace the file associated with the current attachment, only if a file was actually attached * The managedFile is moved to the destination when config.isFileManagerDeletingUploads() is true, otherwise the original is copied * and left unchanged. * @param currentAttachedFile an existing attachment, never null * @param managedFile the new file to set * @param multipartFile the original uploaded file, to get the client filename. If null, the client filename is not changed. * @param targetExtension optional, to convert image file formats * @param desktopWidth optional width for desktop images - when null, the image is not resized * @param mobileWidth optional width for mobile images - when null, the mobile file is the same as the desktop * @return YadaAttachedFile if the file has been replaced, null if managedFile is null * @throws IOException */ public YadaAttachedFile attachReplace(YadaAttachedFile currentAttachedFile, File managedFile, MultipartFile multipartFile, String namePrefix, String targetExtension, Integer desktopWidth, Integer mobileWidth) throws IOException { boolean needToDeleteOriginal = config.isFileManagerDeletingUploads(); return attachReplace(needToDeleteOriginal, currentAttachedFile, managedFile, multipartFile, namePrefix, targetExtension, desktopWidth, mobileWidth); } /** * Replace the file associated with the current attachment, only if a file was actually attached * @param move the managedFile is moved to the destination when true, otherwise the original is copied * and left unchanged. * @param currentAttachedFile an existing attachment, never null * @param managedFile the new file to set * @param multipartFile the original uploaded file, to get the client filename. If null, the client filename is not changed. * @param namePrefix optional prefix to set before the original file name. Add a separator if you need one. * @param targetExtension optional, to convert image file formats * @param desktopWidth optional width for desktop images - when null, the image is not resized * @param mobileWidth optional width for mobile images - when null, the mobile file is the same as the desktop * @return YadaAttachedFile if the file has been replaced, null if managedFile is null * @throws IOException */ public YadaAttachedFile attachReplace(boolean move, YadaAttachedFile currentAttachedFile, File managedFile, MultipartFile multipartFile, String namePrefix, String targetExtension, Integer desktopWidth, Integer mobileWidth) throws IOException { if (managedFile==null) { return null; } if (currentAttachedFile==null) { // We can't call attachNew() instead, because we don't have relativeFolderPath here throw new YadaInvalidUsageException("currentAttachedFile is missing"); } deleteFileAttachment(currentAttachedFile); // Delete any previous attached files String clientFilename = null; if (multipartFile!=null) { clientFilename = multipartFile.getOriginalFilename(); } return attach(move, currentAttachedFile, managedFile, clientFilename, namePrefix, targetExtension, desktopWidth, mobileWidth); } /** * Copies an uploaded file to the destination folder, creating a database association to assign to an Entity. * The name of the file is in the format [basename]managedFileName_id.ext. * Images are not resized. * @param multipartFile the original uploaded file * @param relativeFolderPath path of the target folder relative to the contents folder, starting with a slash / * @param namePrefix prefix to attach before the original file name. Add a separator if you need one. Can be null. * @return YadaAttachedFile if the file is uploaded, null if no file was sent by the user * @throws IOException * @see {@link #attach(File, String, String, String, Integer, Integer)} */ public YadaAttachedFile attachNew(MultipartFile multipartFile, String relativeFolderPath, String namePrefix) throws IOException { File managedFile = uploadFile(multipartFile); return attachNew(managedFile, multipartFile, relativeFolderPath, namePrefix, null, null, null); } /** * Copies a managed file to the destination folder, creating a database association to assign to an Entity. * The name of the file is in the format [basename]managedFileName_id.ext. * Images are not resized. * @param managedFile an uploaded file, can be an image or not * @param multipartFile the original uploaded file, to get the client filename. If null, the client filename is not set. * @param relativeFolderPath path of the target folder relative to the contents folder * @param namePrefix optional prefix to set before the original file name. Add a separator if you need one. * @return YadaAttachedFile if the file is uploaded, null if no file was sent by the user * @throws IOException * @see {@link #attach(File, String, String, String, Integer, Integer)} */ public YadaAttachedFile attachNew(File managedFile, MultipartFile multipartFile, String relativeFolderPath, String namePrefix) throws IOException { return attachNew(managedFile, multipartFile, relativeFolderPath, namePrefix, null, null, null); } /** * Copies (and resizes) a managed file to the destination folder, creating a database association to assign to an Entity. * The name of the file is in the format [basename]managedFileName_id.ext * @param managedFile an uploaded file, can be an image or not. When null, nothing is done. * @param multipartFile the original uploaded file, to get the client filename. If null, the client filename is not changed. * @param relativeFolderPath path of the target folder relative to the contents folder, starting with a slash / * @param namePrefix prefix to attach before the original file name. Add a separator if you need one. Can be null. * @param targetExtension optional, to convert image file formats * @param desktopWidth optional width for desktop images - when null, the image is not resized * @param mobileWidth optional width for mobile images - when null, the mobile file is the same as the desktop * @return YadaAttachedFile if the file is uploaded, null if no file was sent by the user * @throws IOException * @see {@link #attach(File, String, String, String)} */ public YadaAttachedFile attachNew(File managedFile, MultipartFile multipartFile, String relativeFolderPath, String namePrefix, String targetExtension, Integer desktopWidth, Integer mobileWidth) throws IOException { String clientFilename = null; if (multipartFile!=null) { clientFilename = multipartFile.getOriginalFilename(); } return attachNew(managedFile, clientFilename, relativeFolderPath, namePrefix, targetExtension, desktopWidth, mobileWidth); } /** * Copies (and resizes) a managed file to the destination folder, creating a database association to assign to an Entity. * The name of the file is in the format [basename]managedFileName_id.ext * @param managedFile an uploaded file, can be an image or not. When null, nothing is done. * @param clientFilename the original client filename. If null, the client filename is not changed. * @param relativeFolderPath path of the target folder relative to the contents folder, starting with a slash / * @param namePrefix prefix to attach before the original file name. Add a separator if you need one. Can be null. * @param targetExtension optional, to convert image file formats * @param desktopWidth optional width for desktop images - when null, the image is not resized * @param mobileWidth optional width for mobile images - when null, the mobile file is the same as the desktop * @return YadaAttachedFile if the file is uploaded, null if no file was sent by the user * @throws IOException * @see {@link #attach(File, String, String, String)} */ public YadaAttachedFile attachNew(File managedFile, String clientFilename, String relativeFolderPath, String namePrefix, String targetExtension, Integer desktopWidth, Integer mobileWidth) throws IOException { boolean needToDeleteOriginal = config.isFileManagerDeletingUploads(); return attachNew(needToDeleteOriginal, managedFile, clientFilename, relativeFolderPath, namePrefix, targetExtension, desktopWidth, mobileWidth); } /** * Copies (and resizes) a managed file to the destination folder, creating a database association to assign to an Entity. * The name of the file is in the format [basename]managedFileName_id.ext * @param move true if the original file has to be deleted (moved when not transformed), false to keep it there * @param managedFile an uploaded file, can be an image or not. When null, nothing is done. * @param clientFilename the original client filename. If null, the client filename is not changed. * @param relativeFolderPath path of the target folder relative to the contents folder, starting with a slash / * @param namePrefix prefix to attach before the original file name. Add a separator if you need one. Can be null. * @param targetExtension optional, to convert image file formats * @param desktopWidth optional width for desktop images - when null, the image is not resized * @param mobileWidth optional width for mobile images - when null, the mobile file is the same as the desktop * @return YadaAttachedFile if the file is uploaded, null if no file was sent by the user * @throws IOException * @see {@link #attach(File, String, String, String)} */ public YadaAttachedFile attachNew(boolean move, File managedFile, String clientFilename, String relativeFolderPath, String namePrefix, String targetExtension, Integer desktopWidth, Integer mobileWidth) throws IOException { if (managedFile==null) { return null; } if (!relativeFolderPath.startsWith("/") && !relativeFolderPath.startsWith("\\")) { relativeFolderPath = "/" + relativeFolderPath; log.warn("The relativeFolderPath '{}' should have a leading slash (fixed)", relativeFolderPath); } YadaAttachedFile yadaAttachedFile = new YadaAttachedFile(); // yadaAttachedFile.setAttachedToId(attachToId); yadaAttachedFile.setRelativeFolderPath(relativeFolderPath); // This save should not bee needed anymore because of @PostPersist in YadaAttachedFile yadaAttachedFile = yadaAttachedFileDao.save(yadaAttachedFile); // Get the id File targetFolder = new File(config.getContentPath(), relativeFolderPath); targetFolder.mkdirs(); return attach(move, yadaAttachedFile, managedFile, clientFilename, namePrefix, targetExtension, desktopWidth, mobileWidth); } /** * Performs file copy and (for images) resize to different versions. * The managedFile is moved to the destination when config.isFileManagerDeletingUploads() is true, otherwise the original is copied * and left unchanged. * @param yadaAttachedFile object to fill with values * @param managedFile some file to attach or replace, can be an image or not. When null, nothing is done. * @param clientFilename the client filename. If null, the client filename is not changed. * @param namePrefix prefix to attach before the original file name to make the target name. Add a separator (like a dash) if you need one. Can be null. * @param targetExtension optional, to convert image file formats * @param desktopWidth optional width for desktop images - when null, the image is not resized * @param mobileWidth optional width for mobile images - when null, the mobile file is the same as the desktop * @return * @throws IOException */ private YadaAttachedFile attach(YadaAttachedFile yadaAttachedFile, File managedFile, String clientFilename, String namePrefix, String targetExtension, Integer desktopWidth, Integer mobileWidth) throws IOException { boolean needToDeleteOriginal = config.isFileManagerDeletingUploads(); return attach(needToDeleteOriginal, yadaAttachedFile, managedFile, clientFilename, namePrefix, targetExtension, desktopWidth, mobileWidth); } /** * Performs file copy and (for images) resize to different versions. * The managedFile is moved to the destination when config.isFileManagerDeletingUploads() is true, otherwise the original is copied * and left unchanged. * @param move true if the original file has to be deleted (moved when not transformed), false to keep it there * @param yadaAttachedFile object to fill with values * @param managedFile some file to attach or replace, can be an image or not. When null, nothing is done. * @param clientFilename the client filename. If null, the client filename is not changed. * @param namePrefix prefix to attach before the original file name to make the target name. Add a separator (like a dash) if you need one. Can be null. * @param targetExtension optional, to convert image file formats * @param desktopWidth optional width for desktop images - when null, the image is not resized * @param mobileWidth optional width for mobile images - when null, the mobile file is the same as the desktop * @return * @throws IOException */ public YadaAttachedFile attach(boolean move, YadaAttachedFile yadaAttachedFile, File managedFile, String clientFilename, String namePrefix, String targetExtension, Integer desktopWidth, Integer mobileWidth) throws IOException { // yadaAttachedFile.setUploadTimestamp(new Date()); if (clientFilename!=null) { yadaAttachedFile.setClientFilename(clientFilename); } String origExtension = yadaUtil.getFileExtension(yadaAttachedFile.getClientFilename()); if (targetExtension==null) { targetExtension = origExtension; } YadaIntDimension dimension = yadaUtil.getImageDimension(managedFile); yadaAttachedFile.setImageDimension(dimension); boolean imageExtensionChanged = (origExtension==null && targetExtension!=null) || (targetExtension!=null && targetExtension.compareToIgnoreCase(origExtension)!=0); boolean requiresTransofmation = imageExtensionChanged || desktopWidth!=null || mobileWidth!=null; // // If the file does not need resizing, there is just one default filename like "product-mydoc_2631.pdf" if (!requiresTransofmation) { File targetFile = yadaAttachedFile.calcAndSetTargetFile(namePrefix, targetExtension, null, YadaAttachedFile.YadaAttachedFileType.DEFAULT); // File targetFile = new File(targetFolder, targetFilenamePrefix + "." + targetExtension); if (move) { // Just move the old file to the new destination Files.move(managedFile.toPath(), targetFile.toPath(), StandardCopyOption.REPLACE_EXISTING); } else { // Copy bytes try (InputStream inputStream = new FileInputStream(managedFile); OutputStream outputStream = new FileOutputStream(targetFile)) { IOUtils.copy(inputStream, outputStream); } catch (IOException e) { throw e; } } } else { // Transformation: copy with imagemagick // If desktopWidth is null, the image original size does not change. // The file name is like "product-mydoc_2631_640.jpg" File targetFile = yadaAttachedFile.calcAndSetTargetFile(namePrefix, targetExtension, desktopWidth, YadaAttachedFile.YadaAttachedFileType.DESKTOP); resizeAndConvertImageAsNeeded(managedFile, targetFile, desktopWidth); yadaAttachedFile.setFilename(targetFile.getName()); if (mobileWidth==null) { yadaAttachedFile.setFilenameMobile(null); // No mobile image } else { targetFile = yadaAttachedFile.calcAndSetTargetFile(namePrefix, targetExtension, mobileWidth, YadaAttachedFile.YadaAttachedFileType.MOBILE); resizeAndConvertImageAsNeeded(managedFile, targetFile, mobileWidth); } if (move) { log.debug("Deleting original file {}", managedFile.getAbsolutePath()); managedFile.delete(); } } return yadaAttachedFileDao.save(yadaAttachedFile); } /** * Perform image format conversion and/or resize, when needed * @param sourceFile * @param targetFile * @param targetWidth resize width, can be null for no resize */ private void resizeAndConvertImageAsNeeded(File sourceFile, File targetFile, Integer targetWidth) { if (targetWidth==null) { // Convert only Map params = new HashMap<>(); params.put("FILENAMEIN", sourceFile.getAbsolutePath()); params.put("FILENAMEOUT", targetFile.getAbsolutePath()); boolean convert = yadaUtil.exec("config/shell/convert", params); if (!convert) { log.error("Image not copied when making attachment: {}", targetFile); } } else { // Resize Map params = new HashMap<>(); params.put("FILENAMEIN", sourceFile.getAbsolutePath()); params.put("FILENAMEOUT", targetFile.getAbsolutePath()); params.put("W", Integer.toString(targetWidth)); params.put("H", ""); // the height must be empty to keep the original proportions and resize based on width boolean resized = yadaUtil.exec("config/shell/resize", params); if (!resized) { log.error("Image not resized when making attachment: {}", targetFile); } } } }




© 2015 - 2024 Weber Informatics LLC | Privacy Policy