org.apache.sshd.client.subsystem.sftp.SftpClient 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.sshd.client.subsystem.sftp;
import java.io.Closeable;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.nio.channels.Channel;
import java.nio.channels.FileChannel;
import java.nio.charset.Charset;
import java.nio.charset.StandardCharsets;
import java.nio.file.OpenOption;
import java.nio.file.StandardOpenOption;
import java.nio.file.attribute.AclEntry;
import java.nio.file.attribute.FileTime;
import java.util.Arrays;
import java.util.Collection;
import java.util.Collections;
import java.util.Comparator;
import java.util.EnumSet;
import java.util.List;
import java.util.Map;
import java.util.NavigableMap;
import java.util.Objects;
import java.util.Set;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicReference;
import org.apache.sshd.client.subsystem.SubsystemClient;
import org.apache.sshd.client.subsystem.sftp.extensions.SftpClientExtension;
import org.apache.sshd.common.subsystem.sftp.SftpConstants;
import org.apache.sshd.common.subsystem.sftp.SftpHelper;
import org.apache.sshd.common.util.GenericUtils;
import org.apache.sshd.common.util.ValidateUtils;
import org.apache.sshd.common.util.buffer.BufferUtils;
/**
* @author Apache MINA Project
*/
public interface SftpClient extends SubsystemClient {
/**
* Used to indicate the {@link Charset} (or its name) for decoding referenced files/folders names - extracted from
* the client session when 1st initialized.
*
* @see #DEFAULT_NAME_DECODING_CHARSET
* @see #getNameDecodingCharset()
* @see #setNameDecodingCharset(Charset)
*/
String NAME_DECODING_CHARSET = "sftp-name-decoding-charset";
/**
* Default value of {@value #NAME_DECODING_CHARSET}
*/
Charset DEFAULT_NAME_DECODING_CHARSET = StandardCharsets.UTF_8;
enum OpenMode {
Read,
Write,
Append,
Create,
Truncate,
Exclusive;
/**
* The {@link Set} of {@link OpenOption}-s supported by {@link #fromOpenOptions(Collection)}
*/
public static final Set SUPPORTED_OPTIONS = Collections.unmodifiableSet(EnumSet.of(
StandardOpenOption.READ,
StandardOpenOption.APPEND,
StandardOpenOption.CREATE,
StandardOpenOption.TRUNCATE_EXISTING,
StandardOpenOption.WRITE,
StandardOpenOption.CREATE_NEW,
StandardOpenOption.SPARSE));
/**
* Converts {@link StandardOpenOption}-s into {@link OpenMode}-s
*
* @param options The original options - ignored if {@code null}/empty
* @return A {@link Set} of the equivalent modes
* @throws IllegalArgumentException If an unsupported option is requested
* @see #SUPPORTED_OPTIONS
*/
public static Set fromOpenOptions(Collection extends OpenOption> options) {
if (GenericUtils.isEmpty(options)) {
return Collections.emptySet();
}
Set modes = EnumSet.noneOf(OpenMode.class);
for (OpenOption option : options) {
if (option == StandardOpenOption.READ) {
modes.add(Read);
} else if (option == StandardOpenOption.APPEND) {
modes.add(Append);
} else if (option == StandardOpenOption.CREATE) {
modes.add(Create);
} else if (option == StandardOpenOption.TRUNCATE_EXISTING) {
modes.add(Truncate);
} else if (option == StandardOpenOption.WRITE) {
modes.add(Write);
} else if (option == StandardOpenOption.CREATE_NEW) {
modes.add(Create);
modes.add(Exclusive);
} else if (option == StandardOpenOption.SPARSE) {
/*
* As per the Javadoc:
*
* The option is ignored when the file system does not support the creation of sparse files
*/
continue;
} else {
throw new IllegalArgumentException("Unsupported open option: " + option);
}
}
return modes;
}
}
enum CopyMode {
Atomic,
Overwrite
}
enum Attribute {
Size,
UidGid,
Perms,
OwnerGroup,
AccessTime,
ModifyTime,
CreateTime,
Acl,
Extensions
}
class Handle {
private final String path;
private final byte[] id;
Handle(String path, byte[] id) {
// clone the original so the handle is immutable
this.path = ValidateUtils.checkNotNullAndNotEmpty(path, "No remote path");
this.id = ValidateUtils.checkNotNullAndNotEmpty(id, "No handle ID").clone();
}
/**
* @return The remote path represented by this handle
*/
public String getPath() {
return path;
}
public int length() {
return id.length;
}
/**
* @return A cloned instance of the identifier in order to avoid inadvertent modifications to the handle
* contents
*/
public byte[] getIdentifier() {
return id.clone();
}
@Override
public int hashCode() {
return Arrays.hashCode(id);
}
@Override
public boolean equals(Object obj) {
if (obj == null) {
return false;
}
if (obj == this) {
return true;
}
// we do not ask getClass() == obj.getClass() in order to allow for derived classes equality
if (!(obj instanceof Handle)) {
return false;
}
return Arrays.equals(id, ((Handle) obj).id);
}
@Override
public String toString() {
return getPath() + ": " + BufferUtils.toHex(BufferUtils.EMPTY_HEX_SEPARATOR, id);
}
}
// CHECKSTYLE:OFF
abstract class CloseableHandle extends Handle implements Channel, Closeable {
protected CloseableHandle(String path, byte[] id) {
super(path, id);
}
}
// CHECKSTYLE:ON
class Attributes {
private Set flags = EnumSet.noneOf(Attribute.class);
private int type = SftpConstants.SSH_FILEXFER_TYPE_UNKNOWN;
private int perms;
private int uid;
private int gid;
private String owner;
private String group;
private long size;
private FileTime accessTime;
private FileTime createTime;
private FileTime modifyTime;
private List acl;
private Map extensions = Collections.emptyMap();
public Attributes() {
super();
}
public Set getFlags() {
return flags;
}
public Attributes addFlag(Attribute flag) {
flags.add(flag);
return this;
}
public Attributes removeFlag(Attribute flag) {
flags.remove(flag);
return this;
}
public int getType() {
return type;
}
public void setType(int type) {
this.type = type;
}
public long getSize() {
return size;
}
public Attributes size(long size) {
setSize(size);
return this;
}
public void setSize(long size) {
this.size = size;
addFlag(Attribute.Size);
}
public String getOwner() {
return owner;
}
public Attributes owner(String owner) {
setOwner(owner);
return this;
}
public void setOwner(String owner) {
this.owner = owner;
/*
* According to https://tools.ietf.org/wg/secsh/draft-ietf-secsh-filexfer/draft-ietf-secsh-filexfer-13.txt
* section 7.5
*
* If either the owner or group field is zero length, the field should be considered absent, and no change
* should be made to that specific field during a modification operation.
*/
if (GenericUtils.isEmpty(owner)) {
removeFlag(Attribute.OwnerGroup);
} else {
addFlag(Attribute.OwnerGroup);
}
}
public String getGroup() {
return group;
}
public Attributes group(String group) {
setGroup(group);
return this;
}
public void setGroup(String group) {
this.group = group;
/*
* According to https://tools.ietf.org/wg/secsh/draft-ietf-secsh-filexfer/draft-ietf-secsh-filexfer-13.txt
* section 7.5
*
* If either the owner or group field is zero length, the field should be considered absent, and no change
* should be made to that specific field during a modification operation.
*/
if (GenericUtils.isEmpty(group)) {
removeFlag(Attribute.OwnerGroup);
} else {
addFlag(Attribute.OwnerGroup);
}
}
public int getUserId() {
return uid;
}
public int getGroupId() {
return gid;
}
public Attributes owner(int uid, int gid) {
this.uid = uid;
this.gid = gid;
addFlag(Attribute.UidGid);
return this;
}
public int getPermissions() {
return perms;
}
public Attributes perms(int perms) {
setPermissions(perms);
return this;
}
public void setPermissions(int perms) {
this.perms = perms;
addFlag(Attribute.Perms);
}
public FileTime getAccessTime() {
return accessTime;
}
public Attributes accessTime(long atime) {
return accessTime(atime, TimeUnit.SECONDS);
}
public Attributes accessTime(long atime, TimeUnit unit) {
return accessTime(FileTime.from(atime, unit));
}
public Attributes accessTime(FileTime atime) {
setAccessTime(atime);
return this;
}
public void setAccessTime(FileTime atime) {
accessTime = Objects.requireNonNull(atime, "No access time");
addFlag(Attribute.AccessTime);
}
public FileTime getCreateTime() {
return createTime;
}
public Attributes createTime(long ctime) {
return createTime(ctime, TimeUnit.SECONDS);
}
public Attributes createTime(long ctime, TimeUnit unit) {
return createTime(FileTime.from(ctime, unit));
}
public Attributes createTime(FileTime ctime) {
setCreateTime(ctime);
return this;
}
public void setCreateTime(FileTime ctime) {
createTime = Objects.requireNonNull(ctime, "No create time");
addFlag(Attribute.CreateTime);
}
public FileTime getModifyTime() {
return modifyTime;
}
public Attributes modifyTime(long mtime) {
return modifyTime(mtime, TimeUnit.SECONDS);
}
public Attributes modifyTime(long mtime, TimeUnit unit) {
return modifyTime(FileTime.from(mtime, unit));
}
public Attributes modifyTime(FileTime mtime) {
setModifyTime(mtime);
return this;
}
public void setModifyTime(FileTime mtime) {
modifyTime = Objects.requireNonNull(mtime, "No modify time");
addFlag(Attribute.ModifyTime);
}
public List getAcl() {
return acl;
}
public Attributes acl(List acl) {
setAcl(acl);
return this;
}
public void setAcl(List acl) {
this.acl = Objects.requireNonNull(acl, "No ACLs");
addFlag(Attribute.Acl);
}
public Map getExtensions() {
return extensions;
}
public Attributes extensions(Map extensions) {
setExtensions(extensions);
return this;
}
public void setStringExtensions(Map extensions) {
setExtensions(SftpHelper.toBinaryExtensions(extensions));
}
public void setExtensions(Map extensions) {
this.extensions = Objects.requireNonNull(extensions, "No extensions");
addFlag(Attribute.Extensions);
}
public boolean isRegularFile() {
return (getPermissions() & SftpConstants.S_IFMT) == SftpConstants.S_IFREG;
}
public boolean isDirectory() {
return (getPermissions() & SftpConstants.S_IFMT) == SftpConstants.S_IFDIR;
}
public boolean isSymbolicLink() {
return (getPermissions() & SftpConstants.S_IFMT) == SftpConstants.S_IFLNK;
}
public boolean isOther() {
return !isRegularFile() && !isDirectory() && !isSymbolicLink();
}
@Override
public String toString() {
return "type=" + getType() + ";size=" + getSize() + ";uid=" + getUserId() + ";gid=" + getGroupId() + ";perms=0x"
+ Integer.toHexString(getPermissions()) + ";flags=" + getFlags() + ";owner=" + getOwner() + ";group="
+ getGroup() + ";aTime=" + getAccessTime() + ";cTime=" + getCreateTime() + ";mTime=" + getModifyTime()
+ ";extensions=" + getExtensions().keySet();
}
}
class DirEntry {
public static final Comparator BY_CASE_SENSITIVE_FILENAME = new Comparator() {
@Override
public int compare(DirEntry o1, DirEntry o2) {
if (o1 == o2) {
return 0;
} else if (o1 == null) {
return 1;
} else if (o2 == null) {
return -1;
} else {
return GenericUtils.safeCompare(o1.getFilename(), o2.getFilename(), true);
}
}
};
public static final Comparator BY_CASE_INSENSITIVE_FILENAME = new Comparator() {
@Override
public int compare(DirEntry o1, DirEntry o2) {
if (o1 == o2) {
return 0;
} else if (o1 == null) {
return 1;
} else if (o2 == null) {
return -1;
} else {
return GenericUtils.safeCompare(o1.getFilename(), o2.getFilename(), false);
}
}
};
private final String filename;
private final String longFilename;
private final Attributes attributes;
public DirEntry(String filename, String longFilename, Attributes attributes) {
this.filename = filename;
this.longFilename = longFilename;
this.attributes = attributes;
}
public String getFilename() {
return filename;
}
public String getLongFilename() {
return longFilename;
}
public Attributes getAttributes() {
return attributes;
}
@Override
public String toString() {
return getFilename() + "[" + getLongFilename() + "]: " + getAttributes();
}
}
DirEntry[] EMPTY_DIR_ENTRIES = new DirEntry[0];
// default values used if none specified
int MIN_BUFFER_SIZE = 256;
int MIN_READ_BUFFER_SIZE = MIN_BUFFER_SIZE;
int MIN_WRITE_BUFFER_SIZE = MIN_BUFFER_SIZE;
int IO_BUFFER_SIZE = 32 * 1024;
long DEFAULT_WAIT_TIMEOUT = TimeUnit.SECONDS.toMillis(15L);
/**
* Property that can be used on the {@link org.apache.sshd.common.FactoryManager} to control the internal timeout
* used by the client to open a channel. If not specified then {@link #DEFAULT_CHANNEL_OPEN_TIMEOUT} value is used
*/
String SFTP_CHANNEL_OPEN_TIMEOUT = "sftp-channel-open-timeout";
long DEFAULT_CHANNEL_OPEN_TIMEOUT = DEFAULT_WAIT_TIMEOUT;
/**
* Default modes for opening a channel if no specific modes specified
*/
Set DEFAULT_CHANNEL_MODES = Collections.unmodifiableSet(EnumSet.of(OpenMode.Read, OpenMode.Write));
/**
* @return The negotiated SFTP protocol version
*/
int getVersion();
@Override
default String getName() {
return SftpConstants.SFTP_SUBSYSTEM_NAME;
}
/**
* @return The (never {@code null}) {@link Charset} used to decode referenced files/folders names
* @see #NAME_DECODING_CHARSET
*/
Charset getNameDecodingCharset();
void setNameDecodingCharset(Charset cs);
/**
* @return An (unmodifiable) {@link NavigableMap} of the reported server extensions. where key=extension name (case
* insensitive)
*/
NavigableMap getServerExtensions();
boolean isClosing();
//
// Low level API
//
/**
* Opens a remote file for read
*
* @param path The remote path
* @return The file's {@link CloseableHandle}
* @throws IOException If failed to open the remote file
* @see #open(String, Collection)
*/
default CloseableHandle open(String path) throws IOException {
return open(path, Collections.emptySet());
}
/**
* Opens a remote file with the specified mode(s)
*
* @param path The remote path
* @param options The desired mode - if none specified then {@link OpenMode#Read} is assumed
* @return The file's {@link CloseableHandle}
* @throws IOException If failed to open the remote file
* @see #open(String, Collection)
*/
default CloseableHandle open(String path, OpenMode... options) throws IOException {
return open(path, GenericUtils.of(options));
}
/**
* Opens a remote file with the specified mode(s)
*
* @param path The remote path
* @param options The desired mode - if none specified then {@link OpenMode#Read} is assumed
* @return The file's {@link CloseableHandle}
* @throws IOException If failed to open the remote file
*/
CloseableHandle open(String path, Collection options) throws IOException;
/**
* Close the handle obtained from one of the {@code open} methods
*
* @param handle The {@code Handle} to close
* @throws IOException If failed to execute
*/
void close(Handle handle) throws IOException;
/**
* @param path The remote path to remove
* @throws IOException If failed to execute
*/
void remove(String path) throws IOException;
default void rename(String oldPath, String newPath) throws IOException {
rename(oldPath, newPath, Collections.emptySet());
}
default void rename(String oldPath, String newPath, CopyMode... options) throws IOException {
rename(oldPath, newPath, GenericUtils.of(options));
}
void rename(String oldPath, String newPath, Collection options) throws IOException;
/**
* Reads data from the open (file) handle
*
* @param handle The file {@link Handle} to read from
* @param fileOffset The file offset to read from
* @param dst The destination buffer
* @return Number of read bytes - {@code -1} if EOF reached
* @throws IOException If failed to read the data
* @see #read(Handle, long, byte[], int, int)
*/
default int read(Handle handle, long fileOffset, byte[] dst) throws IOException {
return read(handle, fileOffset, dst, null);
}
/**
* Reads data from the open (file) handle
*
* @param handle The file {@link Handle} to read from
* @param fileOffset The file offset to read from
* @param dst The destination buffer
* @param eofSignalled If not {@code null} then upon return holds a value indicating whether EOF was reached due to
* the read. If {@code null} indicator value then this indication is not available
* @return Number of read bytes - {@code -1} if EOF reached
* @throws IOException If failed to read the data
* @see #read(Handle, long, byte[], int, int, AtomicReference)
* @see SFTP v6 -
* section 9.3
*/
default int read(Handle handle, long fileOffset, byte[] dst, AtomicReference eofSignalled) throws IOException {
return read(handle, fileOffset, dst, 0, dst.length, eofSignalled);
}
default int read(Handle handle, long fileOffset, byte[] dst, int dstOffset, int len) throws IOException {
return read(handle, fileOffset, dst, dstOffset, len, null);
}
/**
* Reads data from the open (file) handle
*
* @param handle The file {@link Handle} to read from
* @param fileOffset The file offset to read from
* @param dst The destination buffer
* @param dstOffset Offset in destination buffer to place the read data
* @param len Available destination buffer size to read
* @param eofSignalled If not {@code null} then upon return holds a value indicating whether EOF was reached due to
* the read. If {@code null} indicator value then this indication is not available
* @return Number of read bytes - {@code -1} if EOF reached
* @throws IOException If failed to read the data
* @see SFTP v6 -
* section 9.3
*/
int read(Handle handle, long fileOffset, byte[] dst, int dstOffset, int len, AtomicReference eofSignalled)
throws IOException;
default void write(Handle handle, long fileOffset, byte[] src) throws IOException {
write(handle, fileOffset, src, 0, src.length);
}
/**
* Write data to (open) file handle
*
* @param handle The file {@link Handle}
* @param fileOffset Zero-based offset to write in file
* @param src Data buffer
* @param srcOffset Offset of valid data in buffer
* @param len Number of bytes to write
* @throws IOException If failed to write the data
*/
void write(Handle handle, long fileOffset, byte[] src, int srcOffset, int len) throws IOException;
/**
* Create remote directory
*
* @param path Remote directory path
* @throws IOException If failed to execute
*/
void mkdir(String path) throws IOException;
/**
* Remove remote directory
*
* @param path Remote directory path
* @throws IOException If failed to execute
*/
void rmdir(String path) throws IOException;
/**
* Obtain a handle for a directory
*
* @param path Remote directory path
* @return The associated directory {@link Handle}
* @throws IOException If failed to execute
*/
CloseableHandle openDir(String path) throws IOException;
/**
* @param handle Directory {@link Handle} to read from
* @return A {@link List} of entries - {@code null} to indicate no more entries Note: the list
* may be incomplete since the client and server have some internal imposed limit on the
* number of entries they can process. Therefore several calls to this method may be required
* (until {@code null}). In order to iterate over all the entries use {@link #readDir(String)}
* @throws IOException If failed to access the remote site
*/
default List readDir(Handle handle) throws IOException {
return readDir(handle, null);
}
/**
* @param handle Directory {@link Handle} to read from
* @return A {@link List} of entries - {@code null} to indicate no more entries
* @param eolIndicator An indicator that can be used to get information whether end of list has been reached -
* ignored if {@code null}. Upon return, set value indicates whether all entries have been
* exhausted - a {@code null} value means that this information cannot be provided and another
* call to {@code readDir} is necessary in order to verify that no more entries are pending
* @throws IOException If failed to access the remote site
* @see SFTP v6 -
* section 9.4
*/
List readDir(Handle handle, AtomicReference eolIndicator) throws IOException;
/**
* @param handle A directory {@link Handle}
* @return An {@link Iterable} that can be used to iterate over all the directory entries (like
* {@link #readDir(String)}). Note: the iterable instance is not re-usable - i.e., files
* can be iterated only once
* @throws IOException If failed to access the directory
*/
Iterable listDir(Handle handle) throws IOException;
/**
* The effective "normalized" remote path
*
* @param path The requested path - may be relative, and/or contain dots - e.g., ".",
* "..", "./foo", "../bar"
*
* @return The effective "normalized" remote path
* @throws IOException If failed to execute
*/
String canonicalPath(String path) throws IOException;
/**
* Retrieve remote path meta-data - follow symbolic links if encountered
*
* @param path The remote path
* @return The associated {@link Attributes}
* @throws IOException If failed to execute
*/
Attributes stat(String path) throws IOException;
/**
* Retrieve remote path meta-data - do not follow symbolic links
*
* @param path The remote path
* @return The associated {@link Attributes}
* @throws IOException If failed to execute
*/
Attributes lstat(String path) throws IOException;
/**
* Retrieve file/directory handle meta-data
*
* @param handle The {@link Handle} obtained via one of the {@code open} calls
* @return The associated {@link Attributes}
* @throws IOException If failed to execute
*/
Attributes stat(Handle handle) throws IOException;
/**
* Update remote node meta-data
*
* @param path The remote path
* @param attributes The {@link Attributes} to update
* @throws IOException If failed to execute
*/
void setStat(String path, Attributes attributes) throws IOException;
/**
* Update remote node meta-data
*
* @param handle The {@link Handle} obtained via one of the {@code open} calls
* @param attributes The {@link Attributes} to update
* @throws IOException If failed to execute
*/
void setStat(Handle handle, Attributes attributes) throws IOException;
/**
* Retrieve target of a link
*
* @param path Remote path that represents a link
* @return The link target
* @throws IOException If failed to execute
*/
String readLink(String path) throws IOException;
/**
* Create symbolic link
*
* @param linkPath The link location
* @param targetPath The referenced target by the link
* @throws IOException If failed to execute
* @see #link(String, String, boolean)
*/
default void symLink(String linkPath, String targetPath) throws IOException {
link(linkPath, targetPath, true);
}
/**
* Create a link
*
* @param linkPath The link location
* @param targetPath The referenced target by the link
* @param symbolic If {@code true} then make this a symbolic link, otherwise a hard one
* @throws IOException If failed to execute
*/
void link(String linkPath, String targetPath, boolean symbolic) throws IOException;
// see SSH_FXP_BLOCK / SSH_FXP_UNBLOCK for byte range locks
void lock(Handle handle, long offset, long length, int mask) throws IOException;
void unlock(Handle handle, long offset, long length) throws IOException;
//
// High level API
//
default FileChannel openRemotePathChannel(String path, OpenOption... options) throws IOException {
return openRemotePathChannel(path, GenericUtils.isEmpty(options) ? Collections.emptyList() : Arrays.asList(options));
}
default FileChannel openRemotePathChannel(String path, Collection extends OpenOption> options) throws IOException {
return openRemoteFileChannel(path, OpenMode.fromOpenOptions(options));
}
default FileChannel openRemoteFileChannel(String path, OpenMode... modes) throws IOException {
return openRemoteFileChannel(path, GenericUtils.isEmpty(modes) ? Collections.emptyList() : Arrays.asList(modes));
}
/**
* Opens an {@link FileChannel} on the specified remote path
*
* @param path The remote path
* @param modes The access mode(s) - if {@code null}/empty then the {@link #DEFAULT_CHANNEL_MODES} are used
* @return The open {@link FileChannel} - Note: do not close this owner client instance until the
* channel is no longer needed since it uses the client for providing the channel's
* functionality.
* @throws IOException If failed to open the channel
* @see java.nio.channels.Channels#newInputStream(java.nio.channels.ReadableByteChannel)
* @see java.nio.channels.Channels#newOutputStream(java.nio.channels.WritableByteChannel)
*/
FileChannel openRemoteFileChannel(String path, Collection modes) throws IOException;
/**
* @param path The remote directory path
* @return An {@link Iterable} that can be used to iterate over all the directory entries (unlike
* {@link #readDir(Handle)})
* @throws IOException If failed to access the remote site
* @see #readDir(Handle)
*/
Iterable readDir(String path) throws IOException;
default InputStream read(String path) throws IOException {
return read(path, 0);
}
default InputStream read(String path, int bufferSize) throws IOException {
return read(path, bufferSize, EnumSet.of(OpenMode.Read));
}
default InputStream read(String path, OpenMode... mode) throws IOException {
return read(path, 0, mode);
}
default InputStream read(String path, int bufferSize, OpenMode... mode) throws IOException {
return read(path, bufferSize, GenericUtils.of(mode));
}
default InputStream read(String path, Collection mode) throws IOException {
return read(path, 0, mode);
}
/**
* Read a remote file's data via an input stream
*
* @param path The remote file path
* @param bufferSize The internal read buffer size
* @param mode The remote file {@link OpenMode}s
* @return An {@link InputStream} for reading the remote file data
* @throws IOException If failed to execute
*/
InputStream read(String path, int bufferSize, Collection mode) throws IOException;
default OutputStream write(String path) throws IOException {
return write(path, 0);
}
default OutputStream write(String path, int bufferSize) throws IOException {
return write(path, bufferSize, EnumSet.of(OpenMode.Write, OpenMode.Create, OpenMode.Truncate));
}
default OutputStream write(String path, OpenMode... mode) throws IOException {
return write(path, 0, mode);
}
default OutputStream write(String path, int bufferSize, OpenMode... mode) throws IOException {
return write(path, bufferSize, GenericUtils.of(mode));
}
default OutputStream write(String path, Collection mode) throws IOException {
return write(path, 0, mode);
}
/**
* Write to a remote file via an output stream
*
* @param path The remote file path
* @param bufferSize The internal write buffer size
* @param mode The remote file {@link OpenMode}s
* @return An {@link OutputStream} for writing the data
* @throws IOException If failed to execute
*/
OutputStream write(String path, int bufferSize, Collection mode) throws IOException;
/**
* @param The generic extension type
* @param extensionType The extension type
* @return The extension instance - Note: it is up to the caller to invoke
* {@link SftpClientExtension#isSupported()} - {@code null} if this extension type is not
* implemented by the client
* @see #getServerExtensions()
*/
E getExtension(Class extends E> extensionType);
/**
* @param extensionName The extension name
* @return The extension instance - Note: it is up to the caller to invoke
* {@link SftpClientExtension#isSupported()} - {@code null} if this extension type is not
* implemented by the client
* @see #getServerExtensions()
*/
SftpClientExtension getExtension(String extensionName);
}
© 2015 - 2024 Weber Informatics LLC | Privacy Policy