连接到远程SSH服务器(通过Node.js / html5控制台)

时间:2016-08-01 00:59:46

标签: javascript node.js windows html5 ssh

我一直在网上搜索,想要回答一个我认为很简单的问题。我的目标很直接。我想使用Node.js模块构建一个简单的基于Web的SSH客户端。如果我想连接到节点服务器本身,我找到了几个选项,但似乎找不到连接到REMOTE服务器的任何示例。

基本上我正在寻找的结果是这样的工作流程:连接到网络服务器 - >单击服务器列表中的服务器名称 - >输入我点击

的服务器的SSH会话

我发现的唯一远远接近我所寻找的是guacamole。但是,我不想使用鳄梨酱,因为我希望这个应用程序独立于操作系统。目前我正在Windows 10平台上构建它,并在完成后将其移植到fedora。

我找到了creating an SSH terminal的这个教程。但是,所有这一切都是创建(或尝试创建)与本地系统的SSH连接。

另一个看起来非常棒的选项是tty.js。唉,底线与上面的教程相同。该模块仅允许您连接到node.js服务器,而不是远程服务器。

任何人都有关于实现这一目标的可能途径的信息吗?

4 个答案:

答案 0 :(得分:30)

使用Outlook Mail REST APIssh2xterm等模块很容易做到这一点。

以下是一个例子:

  1. npm install ssh2 xterm socket.io
  2. 创建index.html
  3. <html>
      <head>
        <title>SSH Terminal</title>
        <link rel="stylesheet" href="/src/xterm.css" />
        <script src="/src/xterm.js"></script>
        <script src="/addons/fit/fit.js"></script>
        <script src="/socket.io/socket.io.js"></script>
        <script>
          window.addEventListener('load', function() {
            var terminalContainer = document.getElementById('terminal-container');
            var term = new Terminal({ cursorBlink: true });
            term.open(terminalContainer);
            term.fit();
    
            var socket = io.connect();
            socket.on('connect', function() {
              term.write('\r\n*** Connected to backend***\r\n');
    
              // Browser -> Backend
              term.on('data', function(data) {
                socket.emit('data', data);
              });
    
              // Backend -> Browser
              socket.on('data', function(data) {
                term.write(data);
              });
    
              socket.on('disconnect', function() {
                term.write('\r\n*** Disconnected from backend***\r\n');
              });
            });
          }, false);
        </script>
        <style>
          body {
            font-family: helvetica, sans-serif, arial;
            font-size: 1em;
            color: #111;
          }
          h1 {
            text-align: center;
          }
          #terminal-container {
            width: 960px;
            height: 600px;
            margin: 0 auto;
            padding: 2px;
          }
          #terminal-container .terminal {
            background-color: #111;
            color: #fafafa;
            padding: 2px;
          }
          #terminal-container .terminal:focus .terminal-cursor {
            background-color: #fafafa;
          }
        </style>
      </head>
      <body>
        <div id="terminal-container"></div>
      </body>
    </html>
    
    1. 创建server.js
    2. var fs = require('fs');
      var path = require('path');
      var server = require('http').createServer(onRequest);
      
      var io = require('socket.io')(server);
      var SSHClient = require('ssh2').Client;
      
      // Load static files into memory
      var staticFiles = {};
      var basePath = path.join(require.resolve('xterm'), '..');
      [ 'addons/fit/fit.js',
        'src/xterm.css',
        'src/xterm.js'
      ].forEach(function(f) {
        staticFiles['/' + f] = fs.readFileSync(path.join(basePath, f));
      });
      staticFiles['/'] = fs.readFileSync('index.html');
      
      // Handle static file serving
      function onRequest(req, res) {
        var file;
        if (req.method === 'GET' && (file = staticFiles[req.url])) {
          res.writeHead(200, {
            'Content-Type': 'text/'
                            + (/css$/.test(req.url)
                               ? 'css'
                               : (/js$/.test(req.url) ? 'javascript' : 'html'))
          });
          return res.end(file);
        }
        res.writeHead(404);
        res.end();
      }
      
      io.on('connection', function(socket) {
        var conn = new SSHClient();
        conn.on('ready', function() {
          socket.emit('data', '\r\n*** SSH CONNECTION ESTABLISHED ***\r\n');
          conn.shell(function(err, stream) {
            if (err)
              return socket.emit('data', '\r\n*** SSH SHELL ERROR: ' + err.message + ' ***\r\n');
            socket.on('data', function(data) {
              stream.write(data);
            });
            stream.on('data', function(d) {
              socket.emit('data', d.toString('binary'));
            }).on('close', function() {
              conn.end();
            });
          });
        }).on('close', function() {
          socket.emit('data', '\r\n*** SSH CONNECTION CLOSED ***\r\n');
        }).on('error', function(err) {
          socket.emit('data', '\r\n*** SSH CONNECTION ERROR: ' + err.message + ' ***\r\n');
        }).connect({
          host: '192.168.100.105',
          username: 'foo',
          password: 'barbaz'
        });
      });
      
      server.listen(8000);
      
      1. .connect()
      2. 中编辑传递给server.js的SSH服务器配置
      3. node server.js
      4. 访问浏览器中的socket.io

答案 1 :(得分:3)

只是将更新的代码添加到 @mscdex 很好的答案,因为这些年来这些库已经发生了变化。

图书馆:

npm install express socket.io ssh2 xterm xterm-addon-fit

index.html:

<html>
  <head>
    <title>SSH Terminal</title>
    <link rel="stylesheet" href="/xterm.css" />
    <script src="/xterm.js"></script>
    <script src="/xterm-addon-fit.js"></script>
    <script src="/socket.io/socket.io.js"></script>
    <script>
      window.addEventListener('load', function() {
        var terminalContainer = document.getElementById('terminal-container');
        const term = new Terminal({ cursorBlink: true });        
        const fitAddon = new FitAddon.FitAddon();
        term.loadAddon(fitAddon);
        term.open(terminalContainer);
        fitAddon.fit();

        var socket = io() //.connect();
        socket.on('connect', function() {
          term.write('\r\n*** Connected to backend ***\r\n');
        });

        // Browser -> Backend
        term.onKey(function (ev) {
          socket.emit('data', ev.key);
        });

        // Backend -> Browser
        socket.on('data', function(data) {
          term.write(data);
        });

        socket.on('disconnect', function() {
          term.write('\r\n*** Disconnected from backend ***\r\n');
        });
      }, false);
    </script>
    <style>
      body {
        font-family: helvetica, sans-serif, arial;
        font-size: 1em;
        color: #111;
      }
      h1 {
        text-align: center;
      }
      #terminal-container {
        width: 960px;
        height: 600px;
        margin: 0 auto;
        padding: 2px;
      }
      #terminal-container .terminal {
        background-color: #111;
        color: #fafafa;
        padding: 2px;
      }
      #terminal-container .terminal:focus .terminal-cursor {
        background-color: #fafafa;
      }
    </style>
  </head>
  <body>
    <h3>WebSSH</h3>
    <div id="terminal-container"></div>
  </body>
</html>

server.js:

var fs = require('fs');
var path = require('path');
var server = require('http').createServer(onRequest);

var io = require('socket.io')(server);
var SSHClient = require('ssh2').Client;

// Load static files into memory
var staticFiles = {};
var basePath = path.join(require.resolve('xterm'), '..');
staticFiles['/xterm.css'] = fs.readFileSync(path.join(basePath, '../css/xterm.css'));
staticFiles['/xterm.js'] = fs.readFileSync(path.join(basePath, 'xterm.js'));
basePath = path.join(require.resolve('xterm-addon-fit'), '..');
staticFiles['/xterm-addon-fit.js'] = fs.readFileSync(path.join(basePath, 'xterm-addon-fit.js'));
staticFiles['/'] = fs.readFileSync('index.html');

// Handle static file serving
function onRequest(req, res) {
  var file;
  if (req.method === 'GET' && (file = staticFiles[req.url])) {
    res.writeHead(200, {
      'Content-Type': 'text/'
        + (/css$/.test(req.url)
        ? 'css'
        : (/js$/.test(req.url) ? 'javascript' : 'html'))
    });
    return res.end(file);
  }
  res.writeHead(404);
  res.end();
}

io.on('connection', function(socket) {
  var conn = new SSHClient();
  conn.on('ready', function() {
    socket.emit('data', '\r\n*** SSH CONNECTION ESTABLISHED ***\r\n');
    conn.shell(function(err, stream) {
      if (err)
        return socket.emit('data', '\r\n*** SSH SHELL ERROR: ' + err.message + ' ***\r\n');
      socket.on('data', function(data) {
        stream.write(data);
      });
      stream.on('data', function(d) {
        socket.emit('data', d.toString('binary'));
      }).on('close', function() {
        conn.end();
      });
    });
  }).on('close', function() {
    socket.emit('data', '\r\n*** SSH CONNECTION CLOSED ***\r\n');
  }).on('error', function(err) {
    socket.emit('data', '\r\n*** SSH CONNECTION ERROR: ' + err.message + ' ***\r\n');
  }).connect({
    host: 'domain.tld',
    port: 22,
    username: 'root',
    privateKey: require('fs').readFileSync('path/to/keyfile')
  });
});

let port = 8000;
console.log('Listening on port', port)
server.listen(port);

答案 2 :(得分:3)

与上面的答案相同,但实际上使用了快速和现代的语法和库

const express = require('express');
const app = express();
const http = require('http').Server(app);
const io = require('socket.io')(http, {
  cors: {
    origin: "*"
  }
});
app.set('view engine', 'ejs');
app.use(express.urlencoded({
  extended: false,
  limit: '150mb'
}));
app.use(express.static(__dirname + '/public'));
app.use('/xterm.css', express.static(require.resolve('xterm/css/xterm.css')));
app.use('/xterm.js', express.static(require.resolve('xterm')));
app.use('/xterm-addon-fit.js', express.static(require.resolve('xterm-addon-fit')));

const SSHClient = require('ssh2').Client;

app.get('/', (req, res) => {
  // res.sendFile(__dirname + '/index.html');
  res.render('index');
  // I am using ejs as my templating engine but HTML file work just fine.
});

io.on('connection', function(socket) {
  var conn = new SSHClient();
  conn.on('ready', function() {
    socket.emit('data', '\r\n*** SSH CONNECTION ESTABLISHED ***\r\n');
    conn.shell(function(err, stream) {
      if (err)
        return socket.emit('data', '\r\n*** SSH SHELL ERROR: ' + err.message + ' ***\r\n');
      socket.on('data', function(data) {
        stream.write(data);
      });
      stream.on('data', function(d) {
        socket.emit('data', d.toString('binary'));
      }).on('close', function() {
        conn.end();
      });
    });
  }).on('close', function() {
    socket.emit('data', '\r\n*** SSH CONNECTION CLOSED ***\r\n');
  }).on('error', function(err) {
    socket.emit('data', '\r\n*** SSH CONNECTION ERROR: ' + err.message + ' ***\r\n');
  }).connect({
    host: '192.168.0.103',
    port: 22,
    username: 'kali',
    password: 'kali'
  });
});

http.listen(3000, () => {
  console.log('Listening on http://localhost:3000');
});
* {
  padding: 0%;
  margin: 0%;
  box-sizing: border-box;
}

body {
  font-family: Helvetica, sans-serif, arial;
  font-size: 1em;
  color: #111;
}

h1 {
  text-align: center;
}

#terminal-container {
  width: 960px;
  height: 600px;
  margin: 0 auto;
  padding: 2px;
}

#terminal-container .terminal {
  background-color: #111;
  color: #fafafa;
  padding: 2px;
}

#terminal-container .terminal:focus .terminal-cursor {
  background-color: #fafafa;
}
<!DOCTYPE html>
<html lang="en">

<head>
  <meta charset="UTF-8">
  <meta http-equiv="X-UA-Compatible" content="IE=edge">
  <meta name="viewport" content="width=device-width, initial-scale=1.0">
  <title>SSH SERVER</title>
  <link rel="stylesheet" href="/xterm.css" />
  <script defer src="/xterm.js"></script>
  <script defer src="/xterm-addon-fit.js"></script>
  <script defer src="/socket.io/socket.io.js"></script>
  <script defer src='/js/app.js'></script>
  <link rel='stylesheet' href='/css/main.css'>
</head>

<body>
  <h3>WebSSH</h3>
  <div id="terminal-container"></div>
  <script>
  // PLEASE USE A SEPERATE FILE FOR THE JS and defer it
  // like the above app.js file 
    window.addEventListener('load', function() {
      const terminalContainer = document.getElementById('terminal-container');
      const term = new Terminal({
        cursorBlink: true
      });
      const fitAddon = new FitAddon.FitAddon();
      term.loadAddon(fitAddon);
      term.open(terminalContainer);
      fitAddon.fit();

      const socket = io() //.connect();
      socket.on('connect', function() {
        term.write('\r\n*** Connected to backend ***\r\n');
      });

      // Browser -> Backend
      term.onKey(function(ev) {
        socket.emit('data', ev.key);
      });

      // Backend -> Browser
      socket.on('data', function(data) {
        term.write(data);
      });

      socket.on('disconnect', function() {
        term.write('\r\n*** Disconnected from backend ***\r\n');
      });
    }, false);
  </script>
</body>

</html>

答案 3 :(得分:1)

也尝试noVnc。 但是,在xterm.js的页面中进行一些挖掘可以发现其他解决方案,例如

WebSSH2