题记:
最近弄一个视频有关的项目,有很多的视频存储,所以存储这块就有必要进行一些监控,及内容淘汰等策略。存储设备都是分布式部署的,我们不希望在 server上安装特别的插件,如何获取远程磁盘信息,成了我最大的障碍。目前采用的方式是java建立telnet连接,发送操作系统命令,解析返回信 息,在此整理下。
摘要:
这是我第一次接触到java获取远程操作系统信息的工作,刚拿到需求有点两眼一抹黑,百度、Google的不少资料。总结起来大概三种类型:
1:Apache的commons.io中有一个工具类方法:FileSystemUtils.freeSpaceKb(path),可以通过共享目录方式获取磁盘剩余信息。
2:JConfig拓展工具,调用系统底层命令,获取磁盘信息。参考资料:http://huangyunzeng.iteye.com/blog/1142873
3:使用jni技术,这个是解决所有和os相关的操作的万能利器了。这个没深入研究过,不会,在此就不多提了。
在看到Apache的FileSystemUtils的时候,在本机测试了下,有点激动了。不过很快发现,这只能获取本地磁盘信息,要获取远程磁盘信息只 能建立共享目录。这显然不能满足我们的要求,不过正是FileSystemUtils工具包的实现原理给了我启发:调用操作系统命令,然后解析,获取信息 是一种不错的方式。
- Runtime.getRuntime().exec(String command) //启动进程执行操作系统命令
现在的问题就变成:建立远程连接,调用操作系统命令,然后解析返回值。我只要找到一种建立连接的方式就能完成任务了。顺着这个思路,后来选择了Telnet,理由如下:
a:Apache中收录了Telnet. org.apache.commons.net.telnet.TelnetClient,可以直接使用Client,不用再考虑socket Server端的实现等。
b:系统中一般都已自带了telnet,只需启动telnet服务就好,不用安装其他的插件。
c:为不用更安全的SSH协议建立连接呢?因为我程序中需要识别操作系统类型。telnet在建立连接后,有一个头信息,根据这个头信息能基本判断是 Window,还是Linux。而换成SSH后,建立连接就不能这么方便的获取操作系统类型了。在此,我就暂时恶心一下,用Telnet吧。
正文:
一、telnet暨磁盘监控原理简介
二、分享部分实现代码
一、telnet暨磁盘监控原理简介
如果你仔细阅读了摘要的内容,相信已经明了本文telnet远程监控磁盘信息的大概原理了。在此再提一遍,是为了补充一些更细节的内容。
二:部分实现代码:
1:成员变量及构造方法
- private TelnetClient telnet = new TelnetClient( "VT220" ); //"VT220" 为处理windows下的乱码问题
- private InputStream in;
- private PrintStream out;
- private char prompt = "$" //普通用户结束
- private int os = - 1 ; //操作系统类型
- public TelnetShell(String ip, int port, String user, String password)
- throws SocketException, IOException {
- telnet.connect(ip, port);
- in = telnet.getInputStream();
- out = new PrintStream(telnet.getOutputStream());
- // 根据root用户设置结束符
- this .prompt = user.equals( "root" ) ? '#' : '>' ;
- login(user, password);
- }
建立连接:
- /**
- * 建立telnet连接
- *
- * <p>
- * 根据telnet 头信息识别操作系统类型,并处理连接异常
- *
- * @param user
- * @param password
- */
- public void login(String user, String password) throws SocketException {
- String thead = readUntil("login:" );
- setOsType(thead); // 根据telnet 头信息获取操作系统类别
- write(user);
- readUntil("password:" );
- write(password);
- // 处理连接异常
- boolean loginSuccess = judgeLogin();
- System.out.println("loginSuccess or not ? " + loginSuccess);
- if (loginSuccess) {
- readUntil(prompt + "" );
- } else {
- throw new SocketException( "Login Failed..." );
- }
- }
读/写操作
- /**
- * 读取分析结果
- *
- * <p>
- * “pattern” 指明IO读取的结束位置
- * <p>
- * 为避免Windows返回中“<dir>”的干扰
- *
- * @param pattern
- * @return
- */
- public String readUntil(String pattern) {
- try {
- char lastChar = pattern.charAt(pattern.length() - 1 );
- StringBuffer sb = new StringBuffer();
- char ch = ( char ) in.read();
- System.out.print(ch);
- while (ch != - 1 ) {
- sb.append(ch);
- if (ch == lastChar) {
- if (sb.toString().toLowerCase().endsWith(pattern)) {
- // 避免windows下因捕获‘>’中断
- if (!sb.toString().toLowerCase().endsWith( "<dir>" )) {
- System.out.println("提取完毕" );
- return sb.toString();
- }
- }
- }
- ch = (char ) in.read();
- System.out.print(ch);
- }
- } catch (Exception e) {
- e.printStackTrace();
- }
- return null ;
- }
- /**
- * 写操作
- *
- * @param value
- */
- public void write(String value) {
- try {
- out.println(value);
- out.flush();
- } catch (Exception e) {
- e.printStackTrace();
- }
- }
识别系统类型,执行命令
- /**
- * 设置操作系统类型
- *
- * @param telnetHeadInfo
- */
- public void setOsType(String telnetHeadInfo) {
- if ( null == telnetHeadInfo || "" .equals(telnetHeadInfo))
- return ;
- if (telnetHeadInfo.indexOf( "Microsoft" ) != - 1 ) {
- os = 1 ;
- } else if (telnetHeadInfo.indexOf( "Kernel" ) != - 1 ) {
- os = 2 ; // x-nix
- } else
- os = 3 ;
- }
- /**
- * 向目标发送命令字符串
- *
- * @param command
- * @return
- */
- public String sendCommand(String command) {
- try {
- write(command);
- return readUntil(prompt + "" );
- } catch (Exception e) {
- e.printStackTrace();
- }
- return null ;
- }
关闭连接
- /**
- * 关闭连接
- */
- public void disconnect() {
- try {
- telnet.disconnect();
- } catch (Exception e) {
- e.printStackTrace();
- }
- }
这样,建立连接、识别操作系统,执行命令,读取返回等都基本完成了。可以说Telnet远程获取操作系统信息的大部分代码已经完成。当然,在提供外层封装上,你也可以仿照Apache的FileSystemUtils写一个工具类,采用一些工厂类的模式方便调用,拓展。
最后,可能您已经发现了,本文的标题是要监控“磁盘信息”,但我完全可以获取跟多信息。有了这个telnetShell.java后,我完全可以发送其他系统命令,获取更多信息。这就看你怎么发挥了,呵呵。