将JSch的标准iostream重定向到primefaces的终端

时间:2012-09-03 11:59:35

标签: java primefaces inputstream outputstream jsch

我正在使用Primefaces的终端组件和JSch来ssh到远程桌面。使用exec通道执行会花费太多时间,因为会话和通道在每个命令都关闭,我没有设法说明。所以我将频道改为shell,现在我正试图“重定向”标准输入/输出。这是我的代码的样子:

import org.apache.commons.io.IOUtils;
import org.apache.commons.lang3.StringUtils;

@ManagedBean
@SessionScoped
public class TerminalController implements Serializable{  

     public TerminalController(){
     jsch=new JSch();
     InputStream in=null;
     PrintStream out=System.out;
try{
                      session=jsch.getSession(user, ip, port);
                      session.setConfig("StrictHostKeyChecking", "no");
                        session.setPassword(passwd);
                        session.connect();
                       channel=session.openChannel("shell");    
                       channel.setInputStream(in);
                       channel.setOutputStream(out);
                       channel.connect();
}catch(Exception ee){
                      System.out.println(ee);
} }
    public String handleCommand(String command, String[] params) {  
     command=command+StringUtils.join(params," ");

                   in=IOUtils.toInputStream(command);
                  String result=out.toString();
                   out.flush();
                    return result;}

我知道这很乱,我还是java的初学者。 我想到的另一个问题是,在从iostream到string的转换中,我可能会失去输入按钮功能!我在等你的建议,解决方案和建议。

2 个答案:

答案 0 :(得分:1)

import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.io.OutputStream;
import java.io.OutputStreamWriter;
import java.io.PrintWriter;

import javax.faces.bean.ManagedBean;
import javax.faces.bean.SessionScoped;

import org.apache.commons.lang.StringUtils;
import org.fusesource.jansi.AnsiConsole;
import org.fusesource.jansi.AnsiString;

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

@ManagedBean
@SessionScoped
public class TerminalController {
    BufferedReader fromChannel;
    PrintWriter toChannel;
    Channel channel;
    Session session;
    JSch jsch;

    public TerminalController() {
        jsch = new JSch();
        try {
            session = jsch.getSession("leoks", "localhost", 22);
            session.setConfig("StrictHostKeyChecking", "no");
            session.setPassword("xxx");
            session.connect(Integer.MAX_VALUE); // 3 secs timeout
            channel = session.openChannel("shell");
            InputStream inStream = channel.getInputStream();
            fromChannel = new BufferedReader(new InputStreamReader(inStream,
                    "UTF-8"));
            OutputStream outStream = channel.getOutputStream();
            toChannel = new PrintWriter(new OutputStreamWriter(outStream,
                    "UTF-8"));
            channel.connect();

            StringBuilder result = new StringBuilder();
            boolean stop = false;
            while (!stop) {
                // prompt does not end with newline...
                char c = (char) fromChannel.read();
                result.append(c);
                System.out.print(c);
                if (result.toString().endsWith("$")) {
                    System.out.print("<<<");
                    stop = true;
                }
            }

        } catch (Exception ee) {
            ee.printStackTrace();
        }
    }

    public String handleCommand(String command, String[] params)
            throws IOException {
        command = command + " " + StringUtils.join(params, " ");
        toChannel.println(command);
        toChannel.flush();
        StringBuilder result = new StringBuilder();
        AnsiConsole.systemInstall();
        while (true) {
            char c = (char) fromChannel.read();
            result.append(c);
            if (c == '$') {
                AnsiString as = new AnsiString(result.toString());
                String s = as.getPlain().toString();
                s = s.replaceAll("\n", "<br>");
                AnsiConsole.systemUninstall();
                return s;
            }
        }
    }
}

Jansi还有一个HTML渲染类,需要一些工作来正确地用BR替换\ n,因为终端组件呈现HTML并默默地忽略任何格式错误的字符串,其中包含完整的控制代码。您可能还希望使用固定宽度字体替换终端CSS定义,例如,使用CSS,例如

.ui-widget
{
    font-family: "Courier New", Courier, monospace;
    font-size: 0.8em;
}

<html xmlns="http://www.w3.org/1999/xhtml"
    xmlns:h="http://java.sun.com/jsf/html"
    xmlns:p="http://primefaces.org/ui">
<h:head>
    <title>Terminal POC</title>
</h:head>
<h:body>
    <h:outputStylesheet library="css" name="styles.css" />
    <h:form>
        <p:terminal 
            commandHandler="#{terminalController.handleCommand}"
            height="300px" 
            welcomeMessage="Welcome..." 
            prompt="&gt;" />
    </h:form>
</h:body>
</html>

答案 1 :(得分:0)

设置in变量对频道没有任何影响,调用out.toString()不会得到任何结果。

您需要命令(及其参数)写入通道,然后从那里读取输出。另外,不要使用setInputStreamsetOutputStream方法(如果您有要读取的内容或要编写的内容,则它们很好,但您不会这样做。)

public class TerminalController implements Serializable{  

    BufferedReader fromChannel;
    PrintWriter toChannel;
    Channel channel;
    Session session;

     public TerminalController(){
        jsch=new JSch();
        try {
            session=jsch.getSession(user, ip, port);
            session.setConfig("StrictHostKeyChecking", "no");
            session.setPassword(passwd);
            session.connect();
            channel = session.openChannel("shell");    
            InputStream inStream = channel.getInputStream();
            fromChannel = new BufferedReader(new InputStreamReader(inStream, "UTF-8"));
            OutputStream outStream = channel.getOutputStream();
            toChannel = new PrintWriter(new OutputStreamWriter(outStream, "UTF-8"));
            channel.connect();
        } catch(Exception ee) {
            ee.printStackTrace();
        }
    }

    public String handleCommand(String command, String[] params) {  
         command = command + StringUtils.join(params," ");
         toChannel.println(command);
         StringBuilder result = new StringBuilder();
         while(true) {
            String line = fromChannel.readLine();
            result.append(line);
            if(looksLikePrompt(line))
               return result.toString();
            result.append("\n");
         }
    }
}

困难的部分实际上是知道远程命令的输出何时完成,你需要返回(这是looksLikePrompt方法,我不知道怎么写)。

我不确定Primeface终端组件的CommandHandler原则是否是创建交互式终端的正确设计,终端不需要知道命令何时完成。当命令需要更多输入时会发生什么?