pom
<dependency>
<groupId>com.jcraft</groupId>
<artifactId>jsch</artifactId>
<version>0.1.54</version>
</dependency>
配置
####################sftp配置#########################
sftp.username=xxx
sftp.passward=xx.xx
sftp.port=22
sftp.host=xx.xx.xx
sftp.proxy.ip=xx
sftp.proxy.port=xx
sftp.timeout=30000
代码
监控类
package xxxxx
import com.jcraft.jsch.SftpProgressMonitor;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
public class FileProgressMonitor implements SftpProgressMonitor {
Logger logger= LoggerFactory.getLogger(getClass());
private long transfered;
@Override
public boolean count(long count) {
transfered = transfered + count;
if (transfered < 1024) {
logger.info("Currently transferred total size:{} bytes",transfered);
}
if ((transfered > 1024) && (transfered < 1048576)) {
logger.info("Currently transferred total size: {}KB",transfered/1024);
} else {
String size = String.format("%.2f", transfered / 1024.0 / 1024.0);
logger.info("Currently transferred total size: " + size + "MB");
}
return true;
}
@Override
public void end() {
logger.info("文件下载完成");
}
@Override
public void init(int op, String src, String dest, long max) {
logger.info("文件开始下载->{}",src);
}
}
下载类
package xxxxx;
import com.cntaiping.tpi.dmp.config.FileProgressMonitor;
import com.jcraft.jsch.*;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.context.annotation.Configuration;
import org.springframework.util.StringUtils;
import java.io.*;
import java.time.LocalDate;
import java.time.format.DateTimeFormatter;
import java.util.ArrayList;
import java.util.List;
import java.util.Properties;
import java.util.Vector;
@Configuration
public class SftpUtil {
Logger logger = LoggerFactory.getLogger(getClass());
@Value("${sftp.username}")
private String username;
@Value("${sftp.passward}")
private String passward;
@Value("${sftp.port}")
private int port;
@Value("${sftp.host}")
private String host;
@Value("${sftp.proxy.ip}")
private String proxyIp;
@Value("${sftp.proxy.port}")
private int proxyPort;
@Value("${sftp.timeout}")
private int timeOut;
ChannelSftp sftp = null;
/**
* 通过sftp上传文件
*
* @param timeOut
* @throws JSchException
*/
public void sendFile(String serverpath, String localpath, int timeOut) throws JSchException {
if (serverpath == null || localpath == null) {
return;
}
if (sftp == null || sftp.isClosed()) {
sftp = getChannel();
}
try {
try {
sftp.cd(serverpath);
} catch (SftpException sException) {
if (ChannelSftp.SSH_FX_NO_SUCH_FILE == sException.id) {
sftp.mkdir(serverpath);
sftp.cd(serverpath);
}
}
sftp.put(localpath, serverpath);
} catch (SftpException e) {
logger.error("上传文件失败", e);
throw new RuntimeException("平台文件夹创建失败,请检查权限:" + e);
} finally {
disconnect(sftp);
}
}
/**
* 获取平台文件列表
*
* @param srcPath
* @return
* @throws JSchException
* @throws IOException
*/
public List getAllFileNames(String srcPath) throws JSchException, IOException {
srcPath = srcPath.replace("\\", "/"); //目标文件路径
ChannelSftp sftp = getChannel();
logger.info("开始获取文件文件列表。。。");
List temp = new ArrayList();
try {
Vector vector = sftp.ls(srcPath);
for (Object obj : vector) {
if ((obj instanceof ChannelSftp.LsEntry) && ((ChannelSftp.LsEntry) obj).getFilename().matches("^[A-Za-z0-9_]+.[A-Za-z0-9_]+$")) {
String fileName = ((ChannelSftp.LsEntry) obj).getFilename();
//sftp.get(srcPath+"/"+fileName, dstPath);
temp.add(srcPath + fileName);
}
}
} catch (SftpException e) {
logger.error("获取文件列表失败", e);
} finally {
disconnect(sftp);
}
return temp;
}
/**
* 获取平台的文件
*
* @param srcPath 平台的文件路径
* @param destFilePathFolder 下载文件保存到本地的路径
* @param dealDate yyyy-mm-dd
* @throws JSchException
* @throws IOException
*/
public boolean downloadFile(String srcPath, String destFilePathFolder, String dealDate) throws JSchException, IOException, SftpException {
//默认当前日期
if (StringUtils.isEmpty(dealDate)) {
dealDate = LocalDate.now().format(DateTimeFormatter.ISO_LOCAL_DATE);
}
srcPath = srcPath.replace("\\", "/");
destFilePathFolder = destFilePathFolder.replace("\\", "/");
String destFilePath ="";
String destFileSize="";
try {
//本地路径如果不存在,则创建
File destFile = new File(destFilePathFolder, dealDate);
if (!destFile.exists()) {
logger.info("创建文件夹");
destFile.mkdirs();
}
File srcFile = new File(srcPath);
logger.info("开始获取文件大小");
if (sftp == null || sftp.isClosed()) {
sftp = getChannel();
}
long start = System.currentTimeMillis();
SftpATTRS attr = sftp.stat(srcPath);
//2位小数
destFileSize = String.format("%.2f", attr.getSize() / 1024.0 / 1024.0);
destFilePath = destFile.getAbsolutePath() + File.separator + srcFile.getName();
logger.info("开始下载文件->{}大小->{}MB,本地存放地址->{}", srcPath, destFileSize, destFilePath);
sftp.get(srcPath, destFile.getAbsolutePath());
long end = System.currentTimeMillis();
logger.info("文件->{}下载耗时{}分钟", srcPath, (end - start) / 1000.0 / 60.0);
return true;
} catch (SftpException e) {
//获取已经下载文件的大小
File file = new File(destFilePath);
if (file.exists()) {
String fileSize = String.format("%.2f", file.length() / 1024.0 / 1024.0);
logger.error("当前下载文件->{}已经下载{}MB,总大小{}MB", srcPath, fileSize, destFileSize);
}
logger.error("下载异常", e);
e.printStackTrace();
} finally {
disconnect(sftp);
}
return false;
}
/**
* 获取平台的文件
*
* @param srcPath 平台的文件路径
* @param dstPathFolder 文件夹路径
* @throws JSchException
* @throws IOException
*/
public boolean downloadFileMonitor(String srcPath, String dstPathFolder) throws JSchException, IOException, SftpException {
srcPath = srcPath.replace("\\", "/");
dstPathFolder = dstPathFolder.replace("\\", "/");
try {
//本地路径如果不存在,则创建
File destfile = new File(dstPathFolder);
File srcfile = new File(srcPath);
if (!destfile.exists()) {
destfile.mkdirs();
}
long start = System.currentTimeMillis();
if (sftp == null || sftp.isClosed()) {
sftp = getChannel();
}
SftpATTRS attr = sftp.stat(srcPath);
//2位小数
String fileSize = String.format("%.2f", attr.getSize() / 1024.0 / 1024.0);
logger.info("开始下载文件->{}大小->{}MB,本地存放地址->{}", srcPath, fileSize, dstPathFolder);
//System.out.println(dstPathFolder + File.separator + srcfile.getName());
OutputStream out = new FileOutputStream(dstPathFolder + File.separator + srcfile.getName());
//添加回调函数监控进度
InputStream is = sftp.get(srcPath, new FileProgressMonitor());
byte[] buff = new byte[1024 * 5];
int read;
if (is != null) {
logger.info("Start to read input stream");
do {
read = is.read(buff, 0, buff.length);
if (read > 0) {
out.write(buff, 0, read);
}
out.flush();
} while (read >= 0);
System.out.println("input stream read done.");
}
long end = System.currentTimeMillis();
logger.info("文件->{}下载耗时{}分钟", srcPath, (end - start) / 1000.0 / 60.0);
return true;
} catch (SftpException e) {
logger.error("下载异常", e);
} finally {
disconnect(sftp);
}
return false;
}
/**
* 获取SFTP通道
*
* @return
* @throws JSchException
*/
public ChannelSftp getChannel() throws JSchException {
logger.info("开始获取ChannelSftp----------");
Session session = null;
Channel channel = null;
JSch jsch = new JSch(); // 创建JSch对象
session = jsch.getSession(username, host, port); // 按照用户名,主机ip,端口获取一个Session对象
session.setPassword(passward); // 设置暗码
Properties config = new Properties();
config.put("StrictHostKeyChecking", "no");
session.setConfig(config);
//session.setConfig("userauth.gssapi-with-mic", "no");
session.setTimeout(timeOut); //
Proxy proxy = new ProxyHTTP(proxyIp, proxyPort);
session.setProxy(proxy);
// ProxySOCKS5 proxySOCKS5= new ProxySOCKS5("xxx",Integer.valueOf(proxyPortStr));
session.connect(); // 经由过程Session建树链接
channel = session.openChannel("sftp"); // 打开SFTP通道
channel.connect(); // 建树SFTP通道的连接
logger.info("已经到获取ChannelSftp----------");
return (ChannelSftp) channel;
}
/**
* 关闭连接
*
* @param sftp
*/
private void disconnect(ChannelSftp sftp) {
try {
if (sftp != null) {
sftp.quit();
if (sftp.isConnected()) {
sftp.disconnect();
} else if (sftp.isClosed()) {
logger.info("sftp is closed already");
}
if (null != sftp.getSession()) {
sftp.getSession().disconnect();
}
}
} catch (JSchException e) {
e.printStackTrace();
}
}
}
优化版:上面的sftp只能配置一个,具体局限性,而且有些代码不够精简
package com.xxxxx.tpi.dmp.util;
import com.cntaiping.tpi.dmp.config.FileProgressMonitor;
import com.jcraft.jsch.*;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import java.io.*;
import java.net.InetAddress;
import java.util.ArrayList;
import java.util.List;
import java.util.Properties;
import java.util.Vector;
public class SftpUtil {
Logger logger = LoggerFactory.getLogger(getClass());
private String host;
private int port;
private String username;
private String password;
private String proxyIp;
private int proxyPort;
private int timeout;
public SftpUtil(String host,int port,String username,String password,int timeout){
this.host = host;
this.port = port;
this.username = username;
this.password = password;
this.timeout = timeout;
}
/**
* 有代理
* @param host
* @param port
* @param username
* @param password
* @param proxyIp
* @param proxyPort
* @param timeout
*/
public SftpUtil(String host,int port,String username,String password,String proxyIp,int proxyPort,int timeout){
this.host = host;
this.port = port;
this.username = username;
this.password = password;
this.timeout = timeout;
this.proxyIp = proxyIp;
this.proxyPort = proxyPort;
}
ChannelSftp sftp = null;
/**
* 通过sftp上传文件
* @throws JSchException
*/
public void sendFile(String serverpath, String localpath) throws JSchException {
logger.info("开始上传文件");
if (serverpath == null || localpath == null) {
return;
}
if (sftp == null || sftp.isClosed()) {
sftp = getChannel();
}
try {
try {
sftp.cd(serverpath);
} catch (SftpException sException) {
if (ChannelSftp.SSH_FX_NO_SUCH_FILE == sException.id) {
logger.info("在服务器创建文件夹");
sftp.mkdir(serverpath);
sftp.cd(serverpath);
}
}
sftp.put(localpath, serverpath);
} catch (SftpException e) {
logger.error("上传文件失败", e);
throw new RuntimeException("平台文件夹创建失败,请检查权限:" + e);
} finally {
disconnect(sftp);
}
}
/**
* 获取平台文件列表
*
* @param srcPath
* @return
* @throws JSchException
* @throws IOException
*/
public List<String> getAllFileNames(String srcPath) throws JSchException {
srcPath = srcPath.replace("\\", "/"); //目标文件路径
sftp = getChannel();
logger.info("开始获取文件文件列表->{}",srcPath);
List<String> temp = new ArrayList();
try {
Vector<ChannelSftp.LsEntry> vector = sftp.ls(srcPath);
for (ChannelSftp.LsEntry entry : vector) {
String fileName = entry.getFilename();
//除掉无用的..
if (".".equals(fileName) || "..".equals(fileName)) {
continue;
}
if(srcPath.endsWith("/")){
temp.add(srcPath+fileName);
}else{
temp.add(srcPath +File.separator+ fileName);
}
}
return temp;
} catch (SftpException e) {
logger.error("获取文件列表失败", e);
disconnect(sftp);
} finally {
//disconnect(sftp);
}
return null;
}
/**
* 获取平台的文件
*
* @param srcPath 平台的文件路径
* @param destFilePathFolder 下载文件保存到本地的路径
* @throws JSchException
* @throws IOException
*/
public String downloadFile(String srcPath, String destFilePathFolder) throws JSchException {
srcPath = srcPath.replace("\\", "/");
destFilePathFolder = destFilePathFolder.replace("\\", "/");
String destFilePath ="";
String destFileSize="";
try {
//本地路径如果不存在,则创建
File destFile = new File(destFilePathFolder);
if (!destFile.exists()) {
logger.info("创建本地文件夹");
destFile.mkdirs();
}
File srcFile = new File(srcPath);
logger.info("开始获取文件大小");
if (sftp == null || sftp.isClosed()) {
sftp = getChannel();
}
long start = System.currentTimeMillis();
SftpATTRS attr = sftp.stat(srcPath);
//2为小数
destFileSize = String.format("%.2f", attr.getSize() / 1024.0 / 1024.0);
destFilePath = destFile.getAbsolutePath() + File.separator + srcFile.getName();
logger.info("开始下载文件->{}大小->{}MB,本地存放地址->{}", srcPath, destFileSize, destFilePath);
sftp.get(srcPath, destFile.getAbsolutePath());
long end = System.currentTimeMillis();
logger.info("文件->{}下载耗时{}分钟", srcPath, (end - start) / 1000.0 / 60.0);
return destFilePath;
} catch (SftpException e) {
//获取已经下载文件的大小
File file = new File(destFilePath);
if (file.exists()) {
String fileSize = String.format("%.2f", file.length() / 1024.0 / 1024.0);
logger.error("当前下载文件异常->{}已经下载{}MB,总大小{}MB", srcPath, fileSize, destFileSize);
}
logger.error("下载异常", e);
e.printStackTrace();
} finally {
disconnect(sftp);
}
return null;
}
/**
* 获取平台的文件
*
* @param srcPath 平台的文件路径
* @param dstPathFolder 文件夹路径
* @throws JSchException
* @throws IOException
*/
public boolean downloadFileMonitor(String srcPath, String dstPathFolder) throws JSchException, IOException, SftpException {
srcPath = srcPath.replace("\\", "/");
dstPathFolder = dstPathFolder.replace("\\", "/");
try {
//本地路径如果不存在,则创建
File destfile = new File(dstPathFolder);
File srcfile = new File(srcPath);
if (!destfile.exists()) {
destfile.mkdirs();
}
long start = System.currentTimeMillis();
if (sftp == null || sftp.isClosed()) {
sftp = getChannel();
}
SftpATTRS attr = sftp.stat(srcPath);
//2为小数
String fileSize = String.format("%.2f", attr.getSize() / 1024.0 / 1024.0);
logger.info("开始下载文件->{}大小->{}MB,本地存放地址->{}", srcPath, fileSize, dstPathFolder);
System.out.println(dstPathFolder + File.separator + srcfile.getName());
OutputStream out = new FileOutputStream(dstPathFolder + File.separator + srcfile.getName());
//添加回调函数监控进度
InputStream is = sftp.get(srcPath, new FileProgressMonitor());
byte[] buff = new byte[1024 * 5];
int read;
if (is != null) {
logger.info("Start to read input stream");
do {
read = is.read(buff, 0, buff.length);
if (read > 0) {
out.write(buff, 0, read);
}
out.flush();
} while (read >= 0);
System.out.println("input stream read done.");
}
long end = System.currentTimeMillis();
logger.info("文件->{}下载耗时{}分钟", srcPath, (end - start) / 1000.0 / 60.0);
return true;
} catch (SftpException e) {
logger.error("下载异常", e);
} finally {
disconnect(sftp);
}
return false;
}
/**
* 获取SFTP通道
*
* @return
* @throws JSchException
*/
public ChannelSftp getChannel() throws JSchException {
logger.info("开始获取ChannelSftp----------");
Session session = null;
Channel channel = null;
JSch jsch = new JSch(); // 创建JSch对象
session = jsch.getSession(username, host, port); // 按照用户名,主机ip,端口获取一个Session对象
session.setPassword(password); // 设置暗码
Properties config = new Properties();
config.put("StrictHostKeyChecking", "no");
session.setConfig(config);
//session.setConfig("userauth.gssapi-with-mic", "no");
session.setTimeout(timeout); //
Proxy proxy = new ProxyHTTP(proxyIp, proxyPort);
session.setProxy(proxy);
// ProxySOCKS5 proxySOCKS5= new ProxySOCKS5("172.30.30.12",Integer.valueOf(proxyPortStr));
session.connect(); // 经由过程Session建树链接
channel = session.openChannel("sftp"); // 打开SFTP通道
channel.connect(); // 建树SFTP通道的连接
logger.info("已经到获取ChannelSftp----------");
return (ChannelSftp) channel;
}
/**
* 关闭连接
*
* @param sftp
*/
private void disconnect(ChannelSftp sftp) {
try {
if (sftp != null) {
sftp.quit();
if (sftp.isConnected()) {
sftp.disconnect();
} else if (sftp.isClosed()) {
logger.info("sftp is closed already");
}
if (null != sftp.getSession()) {
sftp.getSession().disconnect();
}
}
} catch (JSchException e) {
e.printStackTrace();
}
}
}