我正在使用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的转换中,我可能会失去输入按钮功能!我在等你的建议,解决方案和建议。
答案 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=">" />
</h:form>
</h:body>
</html>
答案 1 :(得分:0)
设置in
变量对频道没有任何影响,调用out.toString()
不会得到任何结果。
您需要将命令(及其参数)写入通道,然后从那里读取输出。另外,不要使用setInputStream
或setOutputStream
方法(如果您有要读取的内容或要编写的内容,则它们很好,但您不会这样做。)
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
原则是否是创建交互式终端的正确设计,终端不需要知道命令何时完成。当命令需要更多输入时会发生什么?