背景
工作中需要管理多套环境, 有时需要同时登陆多个节点, 且每个环境用户名密码都一样, 因此需要一个方案来解决动态的批量登录问题.
XShell
Xshell
有session
管理功能:
-
提供了包括记住登录主机、用户名、密码及登录时执行命令或脚本(
js
,py
,vbs
)的功能 -
session
被存储在xsh
文件中, 默认的存储在%USERPROFILE%\Documents\NetSarang Computer\7\Xshell\Sessions
文件夹下 -
使用
xshell
可以直接打开存储在xsh
文件中的用户登录信息, 比如:/d/Program_Files/Xshell/Xshell 192.168.31.6.xsh
-
xsh
文件使用UTF-16LE
编码 -
xsh
采用与ini
相同的格式进行配置 -
xsh
有许多配置项, 这里列举比较重要的:-
[CONNECTION].Host
: 登录用户名 -
[CONNECTION:AUTHENTICATION].UserName
: 登录用户名 -
[CONNECTION:AUTHENTICATION].Password
: 登录密码, 使用XShell
自有加解密算法, 因此在生成时需要先根据加解密算法生成加密后的密码, 参考how-does-Xmanager-encrypt-password1, 我通过pyinstaller -F XShellCryptoHelper.py
将其打包为exe
供java
使用 -
[CONNECTION:AUTHENTICATION].UseInitScript
: 是否使用登录脚本, 1表示开启, 0表示不使用XShell
同时提供了与expect
一样的交互功能, 可以和脚本共同使用, 但由于脚本本身具备这种功能, 并且移植性好, 所以本文不考虑expect
-
[CONNECTION:AUTHENTICATION].ScriptPath
: 登录脚本存储位置
-
需求
通过java
生成一个/d/test.xsh
文件(能生成一个就能生成N
个), 并且在登录的同时执行一个python
脚本, 效果如下:
所需信息:
-
用户名
root
-
密码
test@2023
-
登录主机
192.168.31.6
-
执行脚本
D:\init.py
:def Main(): # 等待root用户登录成功 xsh.Screen.WaitForString('#') xsh.Screen.Send("echo hello word\r")
思路
XShell
提供了一个默认的session
配置文件: %USERPROFILE%\Documents\NetSarang Computer\7\Xshell\Sessions\default
:
- 读取它, 并且根据关键字一一替换:
Host=
->Host=192.168.31.6
UserName=
->UserName=root
Password=
->Password=xxx
, 这里根据自己生成的密码密文进行替换UseInitScript=0
->UseInitScript=1
ScriptPath=
->ScriptPath=D:\init.py
- 使用
UTF-16LE
进行编码并保存在/d/test.xsh
中 - 使用
XShell /d/test.xsh
进行测试, 成功登录并且打印出hello world
即可
实现
使用java17
进行编码
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.nio.charset.StandardCharsets;
import java.nio.file.Files;
import java.nio.file.Path;
import java.nio.file.Paths;
import java.util.concurrent.CompletableFuture;
import java.util.concurrent.TimeUnit;
import java.util.stream.Collectors;
public class Xsh {
private static final String XSHELL_CRYPTO_HELPER_LOCATION = "D:/XShellCryptoHelper.exe";
private static final String[] ENCRYPT_CMD = new String[]{"cmd", "/c", XSHELL_CRYPTO_HELPER_LOCATION, "-e", "test@2023"};
private static final String DEFAULT_SESSION_LOCATION = System.getenv("USERPROFILE") + "\\Documents\\NetSarang Computer\\7\\Xshell\\Sessions\\default";
private static final String TEST_XSH_LOCATION = "D:\\test.xsh";
private static final String[] RUN_XSHELL_CMD = new String[]{"cmd", "/k", "start D:\\Program_Files\\Xshell\\XShell.exe " + TEST_XSH_LOCATION};
public static void main(String[] args) throws IOException, InterruptedException {
String xshContent = Files.readAllLines(Paths.get(DEFAULT_SESSION_LOCATION), StandardCharsets.UTF_16LE).stream().map(line -> {
if (line == null || line.isBlank()) {
return line;
}
return switch (line.trim()) {
case "Host=" -> "Host=192.168.31.6";
case "UserName=" -> "UserName=root";
case "Password=" -> "Password=" + encrypt();
case "UseInitScript=0" -> "UseInitScript=1";
case "ScriptPath=" -> "ScriptPath=D:\\init.py";
default -> line;
};
}).collect(Collectors.joining(System.lineSeparator()));
Path path = Paths.get(TEST_XSH_LOCATION);
Files.deleteIfExists(path);
Files.writeString(path, xshContent, StandardCharsets.UTF_16LE);
CompletableFuture.runAsync(() -> {
try {
Runtime.getRuntime().exec(RUN_XSHELL_CMD);
} catch (IOException e) {
throw new RuntimeException(e);
}
});
TimeUnit.SECONDS.sleep(3);
}
private static String encrypt() {
InputStream is = null;
InputStreamReader isr = null;
BufferedReader br = null;
try {
Process process = Runtime.getRuntime().exec(ENCRYPT_CMD);
process.waitFor();
is = process.getInputStream();
isr = new InputStreamReader(is);
br = new BufferedReader(isr);
// 只有一行输出
return br.readLine();
} catch (Exception e) {
throw new RuntimeException(e);
} finally {
try {
br.close();
isr.close();
is.close();
} catch (IOException e) {
throw new RuntimeException(e);
}
}
}
}
参考
- Using Script
本项目原作者很长时间没有更新, 本来不支持
7.*
版本的加密, 我参考XDecrypt项目对其进行了补充, 当前已支持XShell
全系列加解密!! ↩︎