org.apache.camel.component.file.remote.FtpEndpoint Maven / Gradle / Ivy
/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You 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.apache.camel.component.file.remote;
import java.util.HashMap;
import java.util.Map;
import org.apache.camel.Category;
import org.apache.camel.FailedToCreateConsumerException;
import org.apache.camel.FailedToCreateProducerException;
import org.apache.camel.LoggingLevel;
import org.apache.camel.Processor;
import org.apache.camel.api.management.ManagedAttribute;
import org.apache.camel.api.management.ManagedResource;
import org.apache.camel.component.file.GenericFileConfiguration;
import org.apache.camel.component.file.GenericFileProcessStrategy;
import org.apache.camel.component.file.GenericFileProducer;
import org.apache.camel.component.file.remote.RemoteFileConfiguration.PathSeparator;
import org.apache.camel.component.file.remote.strategy.FtpProcessStrategyFactory;
import org.apache.camel.component.file.strategy.FileMoveExistingStrategy;
import org.apache.camel.spi.ClassResolver;
import org.apache.camel.spi.Metadata;
import org.apache.camel.spi.UriEndpoint;
import org.apache.camel.spi.UriParam;
import org.apache.camel.support.PlatformHelper;
import org.apache.camel.util.ObjectHelper;
import org.apache.commons.net.ftp.FTPClient;
import org.apache.commons.net.ftp.FTPClientConfig;
import org.apache.commons.net.ftp.FTPFile;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
/**
* Upload and download files to/from FTP servers.
*/
@UriEndpoint(firstVersion = "1.1.0", scheme = "ftp", extendsScheme = "file", title = "FTP",
syntax = "ftp:host:port/directoryName", alternativeSyntax = "ftp:username:password@host:port/directoryName",
category = { Category.FILE })
@Metadata(excludeProperties = "appendChars,readLockIdempotentReleaseAsync,readLockIdempotentReleaseAsyncPoolSize,"
+ "readLockIdempotentReleaseDelay,readLockIdempotentReleaseExecutorService,"
+ "directoryMustExist,extendedAttributes,probeContentType,startingDirectoryMustExist,"
+ "startingDirectoryMustHaveAccess,chmodDirectory,forceWrites,copyAndDeleteOnRenameFail,"
+ "renameUsingCopy")
@ManagedResource(description = "Managed FtpEndpoint")
public class FtpEndpoint extends RemoteFileEndpoint {
private static final Logger LOG = LoggerFactory.getLogger(FtpEndpoint.class);
protected int soTimeout;
protected int dataTimeout;
@UriParam
protected FtpConfiguration configuration;
@UriParam(label = "advanced")
protected FTPClientConfig ftpClientConfig;
@UriParam(label = "advanced", prefix = "ftpClientConfig.", multiValue = true)
protected Map ftpClientConfigParameters;
@UriParam(label = "advanced", prefix = "ftpClient.", multiValue = true)
protected Map ftpClientParameters;
@UriParam(label = "advanced")
protected FTPClient ftpClient;
@UriParam(label = "common", defaultValue = "DEBUG")
protected LoggingLevel transferLoggingLevel = LoggingLevel.DEBUG;
@UriParam(label = "common", defaultValue = "5")
protected int transferLoggingIntervalSeconds = 5;
@UriParam(label = "common")
protected boolean transferLoggingVerbose;
@UriParam(label = "consumer")
protected boolean resumeDownload;
public FtpEndpoint() {
}
public FtpEndpoint(String uri, RemoteFileComponent component, FtpConfiguration configuration) {
super(uri, component, configuration);
this.configuration = configuration;
}
@Override
public String getScheme() {
return "ftp";
}
@Override
public RemoteFileConsumer createConsumer(Processor processor) throws Exception {
if (isResumeDownload() && ObjectHelper.isEmpty(getLocalWorkDirectory())) {
throw new IllegalArgumentException("The option localWorkDirectory must be configured when resumeDownload=true");
}
if (isResumeDownload() && !getConfiguration().isBinary()) {
throw new IllegalArgumentException("The option binary must be enabled when resumeDownload=true");
}
return super.createConsumer(processor);
}
@Override
protected RemoteFileConsumer buildConsumer(Processor processor) {
try {
return new FtpConsumer(
this, processor, createRemoteFileOperations(),
processStrategy != null ? processStrategy : createGenericFileStrategy());
} catch (Exception e) {
throw new FailedToCreateConsumerException(this, e);
}
}
@Override
protected GenericFileProducer buildProducer() {
try {
if (this.getMoveExistingFileStrategy() == null) {
this.setMoveExistingFileStrategy(createDefaultFtpMoveExistingFileStrategy());
}
return new RemoteFileProducer<>(this, createRemoteFileOperations());
} catch (Exception e) {
throw new FailedToCreateProducerException(this, e);
}
}
/**
* Default Existing File Move Strategy
*
* @return the default implementation for ftp components
*/
private FileMoveExistingStrategy createDefaultFtpMoveExistingFileStrategy() {
return new FtpDefaultMoveExistingFileStrategy();
}
@Override
protected GenericFileProcessStrategy createGenericFileStrategy() {
return new FtpProcessStrategyFactory().createGenericFileProcessStrategy(getCamelContext(), getParamsAsMap());
}
@Override
public RemoteFileOperations createRemoteFileOperations() throws Exception {
// configure ftp client
FTPClient client = ftpClient;
if (client == null) {
// must use a new client if not explicit configured to use a custom
// client
client = createFtpClient();
}
// use configured buffer size which is larger and therefore faster (as
// the default is no buffer)
if (getBufferSize() > 0) {
client.setBufferSize(getBufferSize());
}
// set any endpoint configured timeouts
if (getConfiguration().getConnectTimeout() > -1) {
client.setConnectTimeout(getConfiguration().getConnectTimeout());
}
if (getConfiguration().getSoTimeout() > -1) {
soTimeout = getConfiguration().getSoTimeout();
}
dataTimeout = getConfiguration().getTimeout();
if (getConfiguration().getActivePortRange() != null) {
// parse it as min-max
String[] parts = getConfiguration().getActivePortRange().split("-");
if (parts.length != 2) {
throw new IllegalArgumentException("The option activePortRange should have syntax: min-max");
}
int min = getCamelContext().getTypeConverter().mandatoryConvertTo(int.class, parts[0]);
int max = getCamelContext().getTypeConverter().mandatoryConvertTo(int.class, parts[1]);
LOG.debug("Using active port range: {}-{}", min, max);
client.setActivePortRange(min, max);
}
// then lookup ftp client parameters and set those
if (ftpClientParameters != null) {
Map localParameters = new HashMap<>(ftpClientParameters);
// setting soTimeout has to be done later on FTPClient (after it has
// connected)
Object timeout = localParameters.remove("soTimeout");
if (timeout != null) {
soTimeout = getCamelContext().getTypeConverter().convertTo(int.class, timeout);
}
// and we want to keep data timeout so we can log it later
timeout = localParameters.remove("dataTimeout");
if (timeout != null) {
dataTimeout = getCamelContext().getTypeConverter().convertTo(int.class, timeout);
}
setProperties(client, localParameters);
}
if (ftpClientConfigParameters != null) {
// client config is optional so create a new one if we have
// parameter for it
if (ftpClientConfig == null) {
ftpClientConfig = new FTPClientConfig();
}
Map localConfigParameters = new HashMap<>(ftpClientConfigParameters);
setProperties(ftpClientConfig, localConfigParameters);
}
if (dataTimeout > 0) {
client.setDataTimeout(dataTimeout);
}
if (LOG.isDebugEnabled()) {
LOG.debug("Created FTPClient[connectTimeout: {}, soTimeout: {}, dataTimeout: {}, bufferSize: {}"
+ ", receiveDataSocketBufferSize: {}, sendDataSocketBufferSize: {}]: {}",
client.getConnectTimeout(), getSoTimeout(), dataTimeout, client.getBufferSize(),
client.getReceiveDataSocketBufferSize(), client.getSendDataSocketBufferSize(), client);
}
FtpOperations operations = new FtpOperations(client, getFtpClientConfig());
operations.setEndpoint(this);
return operations;
}
protected FTPClient createFtpClient() throws Exception {
FTPClient client = new FTPClient();
// If we're in an OSGI environment, set the parser factory to
// OsgiParserFactory, because commons-net uses Class.forName in their
// default ParserFactory
if (isOsgi()) {
ClassResolver cr = getCamelContext().getClassResolver();
OsgiParserFactory opf = new OsgiParserFactory(cr);
client.setParserFactory(opf);
}
return client;
}
private boolean isOsgi() {
return PlatformHelper.isOsgiContext(getCamelContext());
}
@Override
public FtpConfiguration getConfiguration() {
if (configuration == null) {
configuration = new FtpConfiguration();
}
return configuration;
}
@Override
public void setConfiguration(GenericFileConfiguration configuration) {
if (configuration == null) {
throw new IllegalArgumentException("FtpConfiguration expected");
}
// need to set on both
this.configuration = (FtpConfiguration) configuration;
super.setConfiguration(configuration);
}
public FTPClient getFtpClient() {
return ftpClient;
}
/**
* To use a custom instance of FTPClient
*/
public void setFtpClient(FTPClient ftpClient) {
this.ftpClient = ftpClient;
}
public FTPClientConfig getFtpClientConfig() {
return ftpClientConfig;
}
/**
* To use a custom instance of FTPClientConfig to configure the FTP client the endpoint should use.
*/
public void setFtpClientConfig(FTPClientConfig ftpClientConfig) {
this.ftpClientConfig = ftpClientConfig;
}
public Map getFtpClientParameters() {
return ftpClientParameters;
}
/**
* Used by FtpComponent to provide additional parameters for the FTPClient
*/
void setFtpClientParameters(Map ftpClientParameters) {
this.ftpClientParameters = ftpClientParameters;
}
public Map getFtpClientConfigParameters() {
return ftpClientConfigParameters;
}
/**
* Used by FtpComponent to provide additional parameters for the FTPClientConfig
*/
void setFtpClientConfigParameters(Map ftpClientConfigParameters) {
this.ftpClientConfigParameters = new HashMap<>(ftpClientConfigParameters);
}
public int getSoTimeout() {
return soTimeout;
}
/**
* Sets the soTimeout on the FTP client.
*/
public void setSoTimeout(int soTimeout) {
this.soTimeout = soTimeout;
}
public int getDataTimeout() {
return dataTimeout;
}
/**
* Sets the data timeout on the FTP client.
*/
public void setDataTimeout(int dataTimeout) {
this.dataTimeout = dataTimeout;
}
public LoggingLevel getTransferLoggingLevel() {
return transferLoggingLevel;
}
/**
* Configure the logging level to use when logging the progress of upload and download operations.
*/
public void setTransferLoggingLevel(LoggingLevel transferLoggingLevel) {
this.transferLoggingLevel = transferLoggingLevel;
}
@ManagedAttribute(description = "Logging level to use when logging the progress of upload and download operations")
public void setTransferLoggingLevelName(String transferLoggingLevel) {
this.transferLoggingLevel = getCamelContext().getTypeConverter().convertTo(LoggingLevel.class, transferLoggingLevel);
}
@ManagedAttribute
public String getTransferLoggingLevelName() {
return transferLoggingLevel.name();
}
@ManagedAttribute
public int getTransferLoggingIntervalSeconds() {
return transferLoggingIntervalSeconds;
}
/**
* Configures the interval in seconds to use when logging the progress of upload and download operations that are
* in-flight. This is used for logging progress when operations takes longer time.
*/
@ManagedAttribute(description = "Interval in seconds to use when logging the progress of upload and download operations that are in-flight")
public void setTransferLoggingIntervalSeconds(int transferLoggingIntervalSeconds) {
this.transferLoggingIntervalSeconds = transferLoggingIntervalSeconds;
}
@ManagedAttribute
public boolean isTransferLoggingVerbose() {
return transferLoggingVerbose;
}
/**
* Configures whether the perform verbose (fine grained) logging of the progress of upload and download operations.
*/
@ManagedAttribute(description = "Whether the perform verbose (fine grained) logging of the progress of upload and download operations")
public void setTransferLoggingVerbose(boolean transferLoggingVerbose) {
this.transferLoggingVerbose = transferLoggingVerbose;
}
public boolean isResumeDownload() {
return resumeDownload;
}
/**
* Configures whether resume download is enabled. This must be supported by the FTP server (almost all FTP servers
* support it). In addition the options localWorkDirectory must be configured so downloaded files are
* stored in a local directory, and the option binary must be enabled, which is required to support
* resuming of downloads.
*/
public void setResumeDownload(boolean resumeDownload) {
this.resumeDownload = resumeDownload;
}
@Override
public char getFileSeparator() {
// the regular ftp component should use the configured separator
// as FTP servers may require you to use windows or unix style
// and therefore you need to be able to control that
PathSeparator pathSeparator = getConfiguration().getSeparator();
switch (pathSeparator) {
case Windows:
return '\\';
case UNIX:
return '/';
default:
return super.getFileSeparator();
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy