通过 Java 编程语言执行 scp 传输的最佳方法是什么?看来我可以通过 JSSE、JSch 或 bouncy castle java 库来执行此操作。这些解决方案似乎都没有一个简单的答案。
15 回答
我最终使用了Jsch——它非常简单,而且似乎可以很好地扩展(我每隔几分钟就抓取几千个文件)。
看看这里
那是 Ants 的 SCP 任务的源代码。“执行”方法中的代码是它的具体细节。这应该让您对所需的内容有一个公平的了解。我相信它使用JSch。
或者,您也可以直接从您的 java 代码中执行此 Ant 任务。
我用一些实用方法包装了 Jsch 以使其更友好并调用它
jscp
可在此处获得:https ://github.com/willwarren/jscp
SCP 实用程序 tar 一个文件夹,将其压缩,并将其 scp 到某个位置,然后解压缩。
用法:
// create secure context
SecureContext context = new SecureContext("userName", "localhost");
// set optional security configurations.
context.setTrustAllHosts(true);
context.setPrivateKeyFile(new File("private/key"));
// Console requires JDK 1.7
// System.out.println("enter password:");
// context.setPassword(System.console().readPassword());
Jscp.exec(context,
"src/dir",
"destination/path",
// regex ignore list
Arrays.asList("logs/log[0-9]*.txt",
"backups")
);
还包括有用的类 - Scp 和 Exec,以及 TarAndGzip,它们的工作方式几乎相同。
这是高级解决方案,无需重新发明。又快又脏!
1)首先,访问http://ant.apache.org/bindownload.cgi并下载最新的 Apache Ant 二进制文件。(现在,apache-ant-1.9.4-bin.zip)。
2)解压下载的文件,找到JAR ant-jsch.jar(“apache-ant-1.9.4/lib/ant-jsch.jar”)。在您的项目中添加这个 JAR。还有 ant-launcher.jar 和 ant.jar。
3)转到Jcraft jsch SouceForge 项目并下载 jar。如今,jsch-0.1.52.jar。还要将此 JAR 添加到您的项目中。
现在,您可以轻松地将 Ant Classes Scp用于通过网络复制文件或SSHExec用于 SSH 服务器中的命令的 java 代码。
4)代码示例 Scp:
// This make scp copy of
// one local file to remote dir
org.apache.tools.ant.taskdefs.optional.ssh.Scp scp = new Scp();
int portSSH = 22;
String srvrSSH = "ssh.your.domain";
String userSSH = "anyuser";
String pswdSSH = new String ( jPasswordField1.getPassword() );
String localFile = "C:\\localfile.txt";
String remoteDir = "/uploads/";
scp.setPort( portSSH );
scp.setLocalFile( localFile );
scp.setTodir( userSSH + ":" + pswdSSH + "@" + srvrSSH + ":" + remoteDir );
scp.setProject( new Project() );
scp.setTrust( true );
scp.execute();
openssh 项目列出了几个 Java 替代方案,Trilead SSH for Java似乎符合您的要求。
我使用这个 SFTP API,它有一个叫做 Zehon 的 SCP,它很棒,很容易使用,有很多示例代码。这是网站http://www.zehon.com
我查看了很多这样的解决方案,但其中很多都不喜欢。主要是因为必须识别已知主机的烦人步骤。相对于 scp 命令,这和 JSCH 处于可笑的低水平。
我找到了一个不需要这个但它被捆绑并用作命令行工具的库。 https://code.google.com/p/scp-java-client/
我查看了源代码,发现了如何在没有命令行的情况下使用它。下面是一个上传示例:
uk.co.marcoratto.scp.SCP scp = new uk.co.marcoratto.scp.SCP(new uk.co.marcoratto.scp.listeners.SCPListenerPrintStream());
scp.setUsername("root");
scp.setPassword("blah");
scp.setTrust(true);
scp.setFromUri(file.getAbsolutePath());
scp.setToUri("root@host:/path/on/remote");
scp.execute();
最大的缺点是它不在 maven repo 中(我可以找到)。但是,易用性对我来说是值得的。
这是使用JSch上传文件的示例:
ScpUploader.java
:
import com.jcraft.jsch.ChannelSftp;
import com.jcraft.jsch.JSch;
import com.jcraft.jsch.JSchException;
import com.jcraft.jsch.Session;
import com.jcraft.jsch.SftpException;
import java.io.ByteArrayInputStream;
import java.util.Properties;
public final class ScpUploader
{
public static ScpUploader newInstance()
{
return new ScpUploader();
}
private volatile Session session;
private volatile ChannelSftp channel;
private ScpUploader(){}
public synchronized void connect(String host, int port, String username, String password) throws JSchException
{
JSch jsch = new JSch();
Properties config = new Properties();
config.put("StrictHostKeyChecking", "no");
session = jsch.getSession(username, host, port);
session.setPassword(password);
session.setConfig(config);
session.setInputStream(System.in);
session.connect();
channel = (ChannelSftp) session.openChannel("sftp");
channel.connect();
}
public synchronized void uploadFile(String directoryPath, String fileName, byte[] fileBytes, boolean overwrite) throws SftpException
{
if(session == null || channel == null)
{
System.err.println("No open session!");
return;
}
// a workaround to check if the directory exists. Otherwise, create it
channel.cd("/");
String[] directories = directoryPath.split("/");
for(String directory : directories)
{
if(directory.length() > 0)
{
try
{
channel.cd(directory);
}
catch(SftpException e)
{
// swallowed exception
System.out.println("The directory (" + directory + ") seems to be not exist. We will try to create it.");
try
{
channel.mkdir(directory);
channel.cd(directory);
System.out.println("The directory (" + directory + ") is created successfully!");
}
catch(SftpException e1)
{
System.err.println("The directory (" + directory + ") is failed to be created!");
e1.printStackTrace();
return;
}
}
}
}
channel.put(new ByteArrayInputStream(fileBytes), directoryPath + "/" + fileName, overwrite ? ChannelSftp.OVERWRITE : ChannelSftp.RESUME);
}
public synchronized void disconnect()
{
if(session == null || channel == null)
{
System.err.println("No open session!");
return;
}
channel.exit();
channel.disconnect();
session.disconnect();
channel = null;
session = null;
}
}
AppEntryPoint.java
:
import com.jcraft.jsch.JSchException;
import com.jcraft.jsch.SftpException;
import java.io.IOException;
import java.nio.file.Files;
import java.nio.file.Paths;
public final class AppEntryPoint
{
private static final String HOST = "192.168.1.1";
private static final int PORT = 22;
private static final String USERNAME = "root";
private static final String PASSWORD = "root";
public static void main(String[] args) throws IOException
{
ScpUploader scpUploader = ScpUploader.newInstance();
try
{
scpUploader.connect(HOST, PORT, USERNAME, PASSWORD);
}
catch(JSchException e)
{
System.err.println("Failed to connect the server!");
e.printStackTrace();
return;
}
System.out.println("Successfully connected to the server!");
byte[] fileBytes = Files.readAllBytes(Paths.get("C:/file.zip"));
try
{
scpUploader.uploadFile("/test/files", "file.zip", fileBytes, true); // if overwrite == false, it won't throw exception if the file exists
System.out.println("Successfully uploaded the file!");
}
catch(SftpException e)
{
System.err.println("Failed to upload the file!");
e.printStackTrace();
}
scpUploader.disconnect();
}
}
JSch 是一个很好的库。对于您的问题,它有一个非常简单的答案。
JSch jsch=new JSch();
Session session=jsch.getSession(user, host, 22);
session.setPassword("password");
Properties config = new Properties();
config.put("StrictHostKeyChecking","no");
session.setConfig(config);
session.connect();
boolean ptimestamp = true;
// exec 'scp -t rfile' remotely
String command="scp " + (ptimestamp ? "-p" :"") +" -t "+rfile;
Channel channel=session.openChannel("exec");
((ChannelExec)channel).setCommand(command);
// get I/O streams for remote scp
OutputStream out=channel.getOutputStream();
InputStream in=channel.getInputStream();
channel.connect();
if(checkAck(in)!=0){
System.exit(0);
}
File _lfile = new File(lfile);
if(ptimestamp){
command="T "+(_lfile.lastModified()/1000)+" 0";
// The access time should be sent here,
// but it is not accessible with JavaAPI ;-<
command+=(" "+(_lfile.lastModified()/1000)+" 0\n");
out.write(command.getBytes()); out.flush();
if(checkAck(in)!=0){
System.exit(0);
}
}
您可以在以下位置找到完整代码
http://faisalbhagat.blogspot.com/2013/09/java-uploading-file-remotely-via-scp.html
jsCH 对我来说非常有用。下面是连接到 sftp 服务器并将文件下载到指定目录的方法示例。建议不要禁用 StrictHostKeyChecking。虽然设置起来有点困难,但出于安全原因,指定已知主机应该是规范。
jsch.setKnownHosts("C:\Users\test\known_hosts"); 受到推崇的
JSch.setConfig("StrictHostKeyChecking", "no"); - 不建议
import com.jcraft.jsch.*;
public void downloadFtp(String userName, String password, String host, int port, String path) {
Session session = null;
Channel channel = null;
try {
JSch ssh = new JSch();
JSch.setConfig("StrictHostKeyChecking", "no");
session = ssh.getSession(userName, host, port);
session.setPassword(password);
session.connect();
channel = session.openChannel("sftp");
channel.connect();
ChannelSftp sftp = (ChannelSftp) channel;
sftp.get(path, "specify path to where you want the files to be output");
} catch (JSchException e) {
System.out.println(userName);
e.printStackTrace();
} catch (SftpException e) {
System.out.println(userName);
e.printStackTrace();
} finally {
if (channel != null) {
channel.disconnect();
}
if (session != null) {
session.disconnect();
}
}
}
像这里的一些人一样,我最终为 JSch 库编写了一个包装器。
它被称为 way-secshell,它托管在 GitHub 上:
https://github.com/objectos/way-secshell
// scp myfile.txt localhost:/tmp
File file = new File("myfile.txt");
Scp res = WaySSH.scp()
.file(file)
.toHost("localhost")
.at("/tmp")
.send();
- :如果您使用 Maven 进行依赖管理,请稍微完善费尔南多的答案:-
pom.xml:
<dependency>
<groupId>org.apache.ant</groupId>
<artifactId>ant-jsch</artifactId>
<version>${ant-jsch.version}</version>
</dependency>
在您的项目中添加此依赖项。最新版本可以在这里找到。
Java代码:
public void scpUpload(String source, String destination) {
Scp scp = new Scp();
scp.setPort(port);
scp.setLocalFile(source);
scp.setTodir(username + ":" + password + "@" + host + ":" + destination);
scp.setProject(new Project());
scp.setTrust(true);
scp.execute();
}
我写了一个 scp 服务器,它比其他的要容易得多。我使用 Apache MINA 项目(Apache SSHD)来开发它。您可以在这里查看:https ://github.com/boomz/JSCP
您也可以从/jar
目录下载 jar 文件。如何使用?看看:https ://github.com/boomz/JSCP/blob/master/src/Main.java
我需要递归复制文件夹,尝试不同的解决方案后,最终由 ProcessBuilder + expect/spawn
scpFile("192.168.1.1", "root","password","/tmp/1","/tmp");
public void scpFile(String host, String username, String password, String src, String dest) throws Exception {
String[] scpCmd = new String[]{"expect", "-c", String.format("spawn scp -r %s %s@%s:%s\n", src, username, host, dest) +
"expect \"?assword:\"\n" +
String.format("send \"%s\\r\"\n", password) +
"expect eof"};
ProcessBuilder pb = new ProcessBuilder(scpCmd);
System.out.println("Run shell command: " + Arrays.toString(scpCmd));
Process process = pb.start();
int errCode = process.waitFor();
System.out.println("Echo command executed, any errors? " + (errCode == 0 ? "No" : "Yes"));
System.out.println("Echo Output:\n" + output(process.getInputStream()));
if(errCode != 0) throw new Exception();
}