RepositoryFileSystemFactory.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.filerepository.service.ftp;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.nio.file.Path;
import java.nio.file.Paths;
import java.util.ArrayList;
import java.util.Comparator;
import java.util.List;
import java.util.concurrent.TimeUnit;
import org.apache.commons.lang3.time.StopWatch;
import org.apache.ftpserver.ftplet.AuthenticationFailedException;
import org.apache.ftpserver.ftplet.FileSystemFactory;
import org.apache.ftpserver.ftplet.FileSystemView;
import org.apache.ftpserver.ftplet.FtpException;
import org.apache.ftpserver.ftplet.FtpFile;
import org.apache.ftpserver.ftplet.User;
import org.genesys.filerepository.FileRepositoryException;
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.springframework.beans.factory.InitializingBean;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.data.domain.Sort;
import org.springframework.security.core.AuthenticationException;
import org.springframework.stereotype.Component;
import lombok.extern.slf4j.Slf4j;
/**
* A factory for creating RepositoryFileSystem objects.
*/
@Component
@Slf4j
public class RepositoryFileSystemFactory implements FileSystemFactory, InitializingBean {
/** The repository service. */
@Autowired(required = true)
private RepositoryService repositoryService;
/** The bytes manager. */
@Autowired(required = true)
private TemporaryBytesManager bytesManager;
/**
* File.
*
* @param repositoryFile the repository file
* @return the repository ftp file
*/
private RepositoryFtpFile file(final RepositoryFile repositoryFile, final RepositoryFileSystemView session) {
log.trace("Making RepositoryFtpFile repositoryFile={}", repositoryFile);
return new RepositoryFtpFile(repositoryFile) {
@Override
public String getOwnerName() {
// TODO Auto-generated method stub
return "root";
}
@Override
public String getGroupName() {
// TODO Auto-generated method stub
return "wheel";
}
@Override
public boolean mkdir() {
log.debug("MKDIR on file not possible");
return false;
}
@Override
public boolean delete() {
return FtpRunAs.asFtpUser(session.user, () -> {
log.info("Delete file={}", this.getAbsolutePath());
try {
repositoryService.removeFile(repositoryFile);
return true;
} catch (NoSuchRepositoryFileException | IOException e) {
log.warn(e.getMessage());
return false;
}
});
}
@Override
public boolean move(final FtpFile destination) {
return FtpRunAs.asFtpUser(session.user, () -> {
log.info("Move file={} to dest={}", this.getAbsolutePath(), Paths.get(destination.getAbsolutePath()));
try {
repositoryService.moveAndRenameFile(repositoryFile, Paths.get(destination.getAbsolutePath()));
return true;
} catch (InvalidRepositoryPathException | InvalidRepositoryFileDataException e) {
log.warn("Error moving file: {}", e.getMessage());
return false;
}
});
}
@Override
public OutputStream createOutputStream(final long offset) throws IOException {
return FtpRunAs.asFtpUser(session.user, () -> {
log.info("Creating output stream for file={} at offset={}", getAbsolutePath(), offset);
return bytesManager.createOutputStream(session.user, repositoryFile, offset);
});
}
@Override
public InputStream createInputStream(final long offset) throws IOException {
return FtpRunAs.asFtpUser(session.user, () -> {
log.info("Creating input stream for file={} at offset={}", getAbsolutePath(), offset);
return bytesManager.createInputStream(repositoryFile, offset);
});
}
};
}
/**
* Directory.
*
* @param path the path
* @param session the session
* @return the repository ftp directory
*/
private RepositoryFtpDirectory directory(final Path path, final RepositoryFileSystemView session) {
log.trace("Viewing RepositoryFtpDirectory path={}", path);
return new RepositoryFtpDirectory(path) {
@Override
public boolean move(final FtpFile destination) {
return FtpRunAs.asFtpUser(session.user, () -> {
log.info("Move directory={} to dest={}", this.getAbsolutePath(), destination.getAbsolutePath());
try {
repositoryService.renamePath(Paths.get(this.getAbsolutePath()), Paths.get(destination.getAbsolutePath()));
return true;
} catch (final InvalidRepositoryPathException e) {
log.error("Failed to rename directory", e);
return false;
}
});
}
@Override
public boolean mkdir() {
log.info("Mkdir directory={}", this.getAbsolutePath());
try {
return FtpRunAs.asFtpUser(session.user, () -> {
repositoryService.ensureFolder(path);
return true;
});
} catch (InvalidRepositoryPathException e) {
log.error("{}", e.getMessage(), e);
return false;
}
}
@Override
public List<? extends FtpFile> listFiles() {
try {
return FtpRunAs.asFtpUser(session.user, this::_listFiles);
} catch (FileRepositoryException e) {
throw new RuntimeException(e.getMessage(), e);
}
}
private List<? extends FtpFile> _listFiles() throws InvalidRepositoryPathException {
final Path root = path.normalize().toAbsolutePath();
log.debug("Listing files in path={}", root);
final ArrayList<FtpFile> all = new ArrayList<>();
var stopWatch = StopWatch.createStarted();
var folders = repositoryService.getFolders(root, Sort.unsorted());
log.trace("Got {} folders in {} in {}ms", folders.size(), root, stopWatch.getTime(TimeUnit.MILLISECONDS));
folders.stream().peek(rf -> {
// System.err.println("repoFolder " + rf.getPath());
}).map(rf -> directory(rf.getFolderPath(), session)).forEach(all::add);
log.trace("Got FTP directories at {}ms", stopWatch.getTime(TimeUnit.MILLISECONDS));
var files = repositoryService.getFiles(root, Sort.unsorted());
log.trace("Got {} files in {} at {}ms", files.size(), root, stopWatch.getTime(TimeUnit.MILLISECONDS));
files.stream().peek(rf -> {
// System.err.println("repoFile " + rf.getStorageFullPath());
}).map(rf -> file(rf, session)).forEach(all::add);
log.trace("Got FTP files at {}ms", stopWatch.getTime(TimeUnit.MILLISECONDS));
// Sorted list of everything
all.sort(Comparator.comparing(FtpFile::getName));
log.debug("Done sorting {} FTP directories and files at {}ms", all.size(), stopWatch.getTime(TimeUnit.MILLISECONDS));
return all;
}
@Override
public String getOwnerName() {
// TODO Auto-generated method stub
return "root";
}
@Override
public String getGroupName() {
// TODO Auto-generated method stub
return "wheel";
}
@Override
public boolean delete() {
return FtpRunAs.asFtpUser(session.user, () -> {
log.info("Delete this={}", path);
try {
repositoryService.deleteFolder(path);
return true;
} catch (FileRepositoryException e) {
return false;
}
});
}
@Override
public boolean changeWorkingDirectory(final String dir) {
final Path normalized = Paths.get(getAbsolutePath()).resolve(dir).normalize().toAbsolutePath();
log.info("CWD this={} dir={} normalized={}", getAbsolutePath(), dir, normalized);
if ("/".equals(normalized.toString())) {
this.cwd(normalized);
return true;
}
// Check if such path exists?
try {
RepositoryFolder folder = repositoryService.getFolder(normalized);
if (folder != null) {
this.cwd(normalized);
return true;
} else {
log.warn("CWD to non-existent folder {} normalized={}", dir, normalized);
return false;
}
} catch (InvalidRepositoryPathException e) {
log.warn("CWD to non-existent folder {} normalized={}", dir, normalized, e);
return false;
}
}
};
}
/*
* (non-Javadoc)
* @see
* org.apache.ftpserver.ftplet.FileSystemFactory#createFileSystemView(org.apache
* .ftpserver.ftplet.User)
*/
@Override
public FileSystemView createFileSystemView(final User user) {
log.info("Creating new repository view for {}", user.getName());
return new RepositoryFileSystemView((FtpUser) user) {
@Override
public FtpFile getFile(final String file) throws FtpException {
log.debug("getFile file={} for user={}", file, username);
final Path path = file.startsWith("/") ? Paths.get(file).normalize() : Paths.get(cwd.getAbsolutePath(), file).normalize();
log.trace("Resolved normalized={}", path);
try {
return isDirectory(path) ?
// directory
directory("/".equals(path.toString()) ? path : FtpRunAs.asFtpUser(user, () -> repositoryService.getFolder(path).getFolderPath()), this)
// or file
: file(FtpRunAs.asFtpUser(user, () -> repositoryService.getFile(path.getParent(), path.getFileName().toString())), this);
} catch (final AuthenticationException e) {
log.warn("Authentication problem {}", e.getMessage(), e);
throw new AuthenticationFailedException(e.getMessage(), e);
} catch (NoSuchRepositoryFileException e) {
log.debug("No such file {}: {}", file, e.getMessage());
assert path.getFileName() != null;
return new CanBeAnythingFile(path.getParent(), path.getFileName().toString()) {
@Override
public boolean mkdir() {
log.debug("MKDIR {}", path);
try {
RepositoryFolder repositoryFolder = FtpRunAs.asFtpUser(user, () -> repositoryService.ensureFolder(path));
return repositoryFolder != null;
} catch (InvalidRepositoryPathException e) {
log.warn("{}", e.getMessage(), e);
return false;
}
}
@Override
public boolean delete() {
return true;
}
@Override
public OutputStream createOutputStream(long offset) throws IOException {
log.debug("STOR {}", path);
if (path.getParent() != null) {
log.debug("MKDIR {}", path.getParent());
try {
RepositoryFolder repositoryFolder = FtpRunAs.asFtpUser(user, () -> repositoryService.ensureFolder(path.getParent()));
if (repositoryFolder == null) {
throw new InvalidRepositoryPathException("Folder not created " + path.getParent());
}
return bytesManager.newFile(user, path);
} catch (InvalidRepositoryPathException e) {
log.warn("{}", e.getMessage(), e);
throw new IOException(e.getMessage(), e);
}
} else {
throw new IOException("Cannot store files to /");
}
}
};
} catch (FileRepositoryException e) {
throw new FtpException(e.getMessage(), e);
}
}
private boolean isDirectory(final Path path) throws FtpException {
if ("/".equals(path.toString())) {
return true;
}
try {
log.trace("isDirectory {}", path);
return repositoryService.hasPath(path);
} catch (final InvalidRepositoryPathException e) {
log.debug("Invalid repository path {}: {}", path, e.getMessage());
throw new FtpException(e.getMessage(), e);
}
}
};
// AspectJProxyFactory factory = new AspectJProxyFactory(userView);
// factory.addAspect(new FtpSpringSecurityAspect((FtpUser) user));
// return factory.getProxy();
}
/*
* (non-Javadoc)
* @see org.springframework.beans.factory.InitializingBean#afterPropertiesSet()
*/
@Override
public void afterPropertiesSet() throws Exception {
assert (this.repositoryService != null);
log.warn("Initialized RFSF with service={}", this.repositoryService);
}
/**
* The Class RepositoryFileSystemView.
*/
private abstract class RepositoryFileSystemView implements FileSystemView {
/** The user. */
protected FtpUser user;
/** The username. */
protected String username;
/** The cwd. */
protected RepositoryFtpDirectory cwd = directory(Paths.get("/"), this);
/** The home dir. */
protected RepositoryFtpDirectory homeDir = directory(Paths.get("/"), this);
/**
* Instantiates a new repository file system view.
*
* @param user the user
*/
public RepositoryFileSystemView(final FtpUser user) {
this.user = user;
username = user.getName();
}
/*
* (non-Javadoc)
* @see org.apache.ftpserver.ftplet.FileSystemView#isRandomAccessible()
*/
@Override
public boolean isRandomAccessible() throws FtpException {
// TODO Auto-generated method stub
return false;
}
/*
* (non-Javadoc)
* @see org.apache.ftpserver.ftplet.FileSystemView#getWorkingDirectory()
*/
@Override
public FtpFile getWorkingDirectory() throws FtpException {
log.debug("getWorkingDirectory for user={}", username);
return this.cwd;
}
/*
* (non-Javadoc)
* @see org.apache.ftpserver.ftplet.FileSystemView#getHomeDirectory()
*/
@Override
public FtpFile getHomeDirectory() throws FtpException {
log.debug("getHomeDirectory for user={}", username);
return this.homeDir;
}
/*
* (non-Javadoc)
* @see org.apache.ftpserver.ftplet.FileSystemView#dispose()
*/
@Override
public void dispose() {
log.info("Disposing repository view for user={}", username);
}
/*
* (non-Javadoc)
* @see
* org.apache.ftpserver.ftplet.FileSystemView#changeWorkingDirectory(java.lang.
* String)
*/
@Override
public boolean changeWorkingDirectory(final String dir) throws FtpException {
log.debug("CWD dir={} for user={}", dir, username);
return this.cwd.changeWorkingDirectory(dir);
}
}
}