在浏览器中模拟 Linux 终端



我读过Fabrice Bellard在浏览器中的linux模拟。

Fabrice Bellard 的 Javascript 中的 Linux 模拟器如何工作?

今天我偶然发现了这个网站,他们正在浏览器中模拟完整的 linux 终端,我能够运行 python、perl 等。我知道他们在node.js上运行他们的站点,但我无法弄清楚他们如何准确模拟终端。

http://runnable.com/UWRl3KlLuONCAACG/read-files-from-filesystem-in-python

完整的 linux 是 http://docker.io 的,其余的都是 https://github.com/Runnable/dockworker

我们不是在模拟终端,而是正如 Kyle 所说,通过 websocket 复制终端(带有 ajax 回退)。

在浏览器中,我们使用 https://github.com/chjj/term.js 它来自Fabrice Bellard的模拟器。 它处理输出以及击键捕获。

让我在前面说这样做不是一个好主意。

但是,您可以生成一个shell并使用web-sockets或XMLHttpRequests将按键推送到生成的服务器进程。下面是一个在窗口上运行的工作示例。不幸的是,我没有开始连接/弄清楚 Ctrl+c。但是,您应该了解它的要点。

  require("underscore");
  var Server = {},
      express = require("express"),
      path = require("path"),
      sys = require("sys"),
      application_root = __dirname;
  global.Server = Server;
  Server.root = application_root;
  global.app = express();
  Server.setup = require("./lib/setup.js").setup({
    //redis: require("./lib/redis-client").createClient(),
    app: app, 
    //mongoose : require("mongoose"),
    io : require("socket.io"),
    express : express,
    port: 1773,
    paths : {
      views :  path.join(application_root,"app","views"),
      root : path.join(application_root,"public"),
      controllers : path.join(application_root,"app","controllers"),
      models : path.join(application_root,"app","models")
    }
  });
  var proc = require('child_process'),
      cmd;
  app.socket.on('connection', function(socket) {
    if (!cmd) {
      //console.log('spawning cmd');
      cmd = proc.spawn('cmd');
      //console.log(cmd?'CMD started':'CMD not started');
      if (cmd.stdout) {
        //console.log('stdout present');
        cmd.stdout.on('data',function(data) {
          if (data) {
            //console.log("data: "+data);
            socket.emit('cmd', ""+data);
          }
        });
      }
      if (cmd.stderr) {
        cmd.stderr.on('data', function(data) {
          //console.log('stderr present');
          if (data) {
            socket.emit('cmd', ""+data);
          }
        });
      }
      cmd.on('exit', function() {
        //console.log('cmd exited');
        socket.emit('cmd', '[CMD Shutdown]');
        if (cmd) {
          cmd.kill();
          cmd = null;
        }
      });
    }
    socket.on('sendCmd', function(data) {
      if (data && data.buffer) {
        var kB = data.buffer.replace("r","n");
        if (cmd && cmd.stdin) {
          cmd.stdin.write(kB);
        }
      }
    });
    socket.on('disconnect', function() {
      console.log('connection closed');
      if (cmd) {
        cmd.stdin.end(); //.kill();
        if (cmd) {
          cmd.kill();
          cmd = null;
        }
      }
    });
  });

编辑:实际上,这是一个工作示例的一部分。它缺少捕获击键并将其发送到服务器的客户端。但是,它应该给你一个大致的想法。

最新更新