/** * MailArchiver is an application that provides services for storing and managing e-mail messages through a Web Services SOAP interface. * Copyright (C) 2012 Marcio Andre Scholl Levien and Fernando Alberto Reuter Wendt and Jose Ronaldo Nogueira Fonseca Junior * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU Affero General Public License as * published by the Free Software Foundation, either version 3 of the * License, or (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU Affero General Public License for more details. * * You should have received a copy of the GNU Affero General Public License * along with this program. If not, see . */ /******************************************************************************\ * * This product was developed by * * SERVIÇO FEDERAL DE PROCESSAMENTO DE DADOS (SERPRO), * * a government company established under Brazilian law (5.615/70), * at Department of Development of Porto Alegre. * \******************************************************************************/ package serpro.mailarchiver.service.web; import java.io.IOException; import java.nio.file.Files; import java.nio.file.Path; import java.nio.file.StandardCopyOption; import javax.jdo.annotations.PersistenceAware; import org.springframework.beans.factory.annotation.Autowired; import serpro.mailarchiver.domain.metaarchive.Folder; import serpro.mailarchiver.service.BaseService; import serpro.mailarchiver.service.dto.TFolder; import serpro.mailarchiver.service.find.FFolder; import serpro.mailarchiver.util.Logger; import serpro.mailarchiver.util.transaction.WithReadWriteTx; @PersistenceAware public class DefaultMoveFolderOperation extends BaseService implements MoveFolderOperation { private static final Logger log = Logger.getLocalLogger(); @Autowired private FFolder findFolder; @WithReadWriteTx @Override public TFolder apply(String newParentFolderId, String folderId) throws ServiceFault { Folder folder = findFolder.byId(folderId); if(folder == null) { ServiceFault.folderNotFound() .setActor("moveFolder") .setMessage("Folder not found.") .addValue("folderId", folderId) .raise(); } if(folder.isUserHomeFolder() || folder.isSpecialFolder()) { ServiceFault.invalidFolderId() .setActor("moveFolder") .setMessage("Invalid move of home folder or special folder.") .addValue("folderId", folderId) .raise(); } Folder newParentFolder = findFolder.byId(newParentFolderId); if(newParentFolder == null) { ServiceFault.folderNotFound() .setActor("moveFolder") .setMessage("New parent folder not found.") .addValue("newParentFolderId", newParentFolderId) .raise(); } Folder currentParentFolder = folder.getParent(); if(newParentFolder != currentParentFolder) { for(Folder child : newParentFolder.getChildren()) { if(child.getName().equalsIgnoreCase(folder.getName())) { ServiceFault.folderNameAlreadyExists() .setActor("moveFolder") .setMessage("An folder with the same name already exist inside the new parent folder.") .addValue("folderId", folderId) .addValue("folderName", folder.getName()) .addValue("newParentFolderId", newParentFolderId) .addValue("newParentFolderName", newParentFolder.getName()) .raise(); } } try { Path source = folder.getAbsolutePath(); folder.setParent(newParentFolder); Path target = folder.getAbsolutePath(); Files.move(source, target, StandardCopyOption.ATOMIC_MOVE); } catch(IOException e) { folder.setParent(currentParentFolder); ServiceFault.fileSystemFailure() .setActor("moveFolder") .setMessage("Filesystem move folder failure.") .addValue("folderId", folderId) .addValue("currentFolderPath", folder.getRelativePath()) .addValue("newParentFolderId", newParentFolderId) .addValue("newParentFolderPath", newParentFolder.getRelativePath()) .setCause(e) .raise(); } } return new TFolder(folder); } }