Java JSch遠程執行Shell命令的方法

背景

項目需求,需要遠程 ssh 登錄到某個節點執行 shell 命令來完成任務。對於這種需求,如果不用 java 程序,直接 linux 的 ssh 命令就可以完成,但是在編碼到程序中時需要相關的程序包來完成,本文主要介紹在 java 中如何使用 JSch 包實現 ssh 遠程連接並執行命令。

JSch 簡介

JSch 是Java Secure Channel的縮寫。JSch是一個SSH2的純Java實現。它允許你連接到一個SSH服務器,並且可以使用端口轉發,X11轉發,文件傳輸等,當然你也可以集成它的功能到你自己的應用程序。框架jsch很老的框架,更新到2016年,現在也不更新瞭。

JSch 使用 shell 執行命令,有兩種方法

  • ChannelExec: 一次執行一條命令,一般我們用這個就夠瞭。
  • ChannelShell: 可執行多條命令,平時開發用的不多,根據需要來吧;
ChannelExec channelExec = (ChannelExec) session.openChannel("exec");//隻能執行一條指令(也可執行符合指令)
ChannelShell channelShell = (ChannelShell) session.openChannel("shell");//可執行多條指令 不過需要輸入輸出流

1. ChannelExec

每個命令之間用 ; 隔開。說明:各命令的執行給果,不會影響其它命令的執行。換句話說,各個命令都會執行,但不保證每個命令都執行成功。
每個命令之間用 && 隔開。說明:若前面的命令執行成功,才會去執行後面的命令。這樣可以保證所有的命令執行完畢後,執行過程都是成功的。
每個命令之間用 || 隔開。說明:|| 是或的意思,隻有前面的命令執行失敗後才去執行下一條命令,直到執行成功一條命令為止。

2. ChannelShell

對於ChannelShell,以輸入流的形式,可執行多條指令,這就像在本地計算機上使用交互式shell(它通常用於:交互式使用)。如要要想停止,有兩種方式:

發送一個exit命令,告訴程序本次交互結束;
使用字節流中的available方法,來獲取數據的總大小,然後循環去讀。

使用示例

1. 引入 pom 依賴

<dependency>
   <groupId>com.jcraft</groupId>
   <artifactId>jsch</artifactId>
   <version>0.1.53</version>
</dependency>

2. jsch 使用示例

在此封裝瞭一個 Shell 工具類,用來執行 shell 命令,具體使用細節在代碼註釋中有說明,可以直接拷貝並使用,代碼如下:

package org.example.shell;
/**
 * Created by qianghaohao on 2021/3/28
 */


import com.jcraft.jsch.ChannelExec;
import com.jcraft.jsch.JSch;
import com.jcraft.jsch.Session;

import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;

/**
 * @description:
 * @author: qianghaohao
 * @time: 2021/3/28
 */
public class Shell {
    private String host;
    private String username;
    private String password;
    private int port = 22;
    private int timeout = 60 * 60 * 1000;

    public Shell(String host, String username, String password, int port, int timeout) {
        this.host = host;
        this.username = username;
        this.password = password;
        this.port = port;
        this.timeout = timeout;
    }

    public Shell(String host, String username, String password) {
        this.host = host;
        this.username = username;
        this.password = password;
    }

    public String execCommand(String cmd) {
        JSch jSch = new JSch();
        Session session = null;
        ChannelExec channelExec = null;
        BufferedReader inputStreamReader = null;
        BufferedReader errInputStreamReader = null;
        StringBuilder runLog = new StringBuilder("");
        StringBuilder errLog = new StringBuilder("");
        try {
            // 1. 獲取 ssh session
            session = jSch.getSession(username, host, port);
            session.setPassword(password);
            session.setTimeout(timeout);
            session.setConfig("StrictHostKeyChecking", "no");
            session.connect();  // 獲取到 ssh session

            // 2. 通過 exec 方式執行 shell 命令
            channelExec = (ChannelExec) session.openChannel("exec");
            channelExec.setCommand(cmd);
            channelExec.connect();  // 執行命令

            // 3. 獲取標準輸入流
            inputStreamReader = new BufferedReader(new InputStreamReader(channelExec.getInputStream()));
            // 4. 獲取標準錯誤輸入流
            errInputStreamReader = new BufferedReader(new InputStreamReader(channelExec.getErrStream()));

            // 5. 記錄命令執行 log
            String line = null;
            while ((line = inputStreamReader.readLine()) != null) {
                runLog.append(line).append("\n");
            }

            // 6. 記錄命令執行錯誤 log
            String errLine = null;
            while ((errLine = errInputStreamReader.readLine()) != null) {
                errLog.append(errLine).append("\n");
            }

            // 7. 輸出 shell 命令執行日志
            System.out.println("exitStatus=" + channelExec.getExitStatus() + ", openChannel.isClosed="
                    + channelExec.isClosed());
            System.out.println("命令執行完成,執行日志如下:");
            System.out.println(runLog.toString());
            System.out.println("命令執行完成,執行錯誤日志如下:");
            System.out.println(errLog.toString());
        } catch (Exception e) {
            e.printStackTrace();
        } finally {
            try {
                if (inputStreamReader != null) {
                    inputStreamReader.close();
                }
                if (errInputStreamReader != null) {
                    errInputStreamReader.close();
                }

                if (channelExec != null) {
                    channelExec.disconnect();
                }
                if (session != null) {
                    session.disconnect();
                }
            } catch (IOException e) {
                e.printStackTrace();
            }
        }

        return runLog.toString();
    }
}

上述工具類使用:

package org.example;

import org.example.shell.Shell;

/**
 * Hello world!
 *
 */
public class App {
    public static void main( String[] args ) {
        String cmd = "ls -1";
        Shell shell = new Shell("192.168.10.10", "ubuntu", "11111");
        String execLog = shell.execCommand(cmd);
        System.out.println(execLog);
    }
}

需要註意的點

如果需要後臺執行某個命令,不能直接 <命令> + & 的方式執行,這樣在 JSch 中不生效,需要寫成這樣的格式:<命令> > /dev/null 2>&1 &。比如要後臺執行 sleep 60,需要寫成 sleep 60 > /dev/null 2>&1

具體 issue 見這裡:https://stackoverflow.com/questions/37833683/running-programs-using-jsch-in-the-background

參考文檔

https://www.cnblogs.com/slankka/p/11988477.html

https://blog.csdn.net/sinat_24928447/article/details/83022818

到此這篇關於Java JSch遠程執行Shell命令的方法的文章就介紹到這瞭,更多相關Java JSch遠程執行Shell命令內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!

推薦閱讀: