; | |
/* | |
* Copyright 2020 Google Inc. All rights reserved. | |
* | |
* Licensed under the Apache License, Version 2.0 (the "License"); you may not | |
* use this file except in compliance with the License. You may obtain a copy of | |
* the License at | |
* | |
* http://www.apache.org/licenses/LICENSE-2.0 | |
* | |
* Unless required by applicable law or agreed to in writing, software | |
* distributed under the License is distributed on an "AS IS" BASIS, WITHOUT | |
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the | |
* License for the specific language governing permissions and limitations under | |
* the License. | |
*/ | |
Object.defineProperty(exports, "__esModule", { value: true }); | |
exports.WebSocketToPty = exports.SocketIoToPty = void 0; | |
var nodePty = require("node-pty"); | |
var socketio = require("socket.io"); | |
var ws_1 = require("ws"); | |
var logging = require("./logging"); | |
var sockets_1 = require("./sockets"); | |
var sessionCounter = 0; | |
// Inspired by | |
// https://xtermjs.org/docs/guides/flowcontrol/#ideas-for-a-better-mechanism. | |
var ACK_CALLBACK_EVERY_BYTES = 100000; | |
var UNACKED_HIGH_WATERMARK = 5; | |
var UNACKED_LOW_WATERMARK = 2; | |
/** Socket<->terminal adapter. */ | |
var Session = /** @class */ (function () { | |
function Session(socket) { | |
var _this = this; | |
this.socket = socket; | |
this.pendingAckCallbacks = 0; | |
this.writtenBytes = 0; | |
this.id = sessionCounter++; | |
this.socket.onClose(function (reason) { | |
logging.getLogger().debug('PTY socket disconnected for session %d reason: %s', _this.id, reason); | |
// Handle client disconnects to close sockets, so as to free up resources. | |
_this.close(); | |
}); | |
this.socket.onStringMessage(function (data) { | |
// Propagate the message over to the pty. | |
logging.getLogger().debug('Send data in session %d\n%s', _this.id, data); | |
var message = JSON.parse(data); | |
if (message.data) { | |
_this.pty.write(message.data); | |
} | |
if (message.cols && message.rows) { | |
_this.pty.resize(message.cols, message.rows); | |
} | |
if (message.ack) { | |
_this.pendingAckCallbacks--; | |
if (_this.pendingAckCallbacks < UNACKED_LOW_WATERMARK) { | |
_this.pty.resume(); | |
} | |
} | |
}); | |
this.pty = nodePty.spawn('tmux', ['new-session', '-A', '-D', '-s', '0'], { | |
name: 'xterm-color', | |
cwd: './content', // Which path should terminal start | |
// Pass environment variables | |
env: process.env, | |
}); | |
this.pty.onData(function (data) { | |
_this.writtenBytes += data.length; | |
if (_this.writtenBytes < ACK_CALLBACK_EVERY_BYTES) { | |
var message = { data: data }; | |
_this.socket.sendString(JSON.stringify(message)); | |
} | |
else { | |
var message = { data: data, ack: true }; | |
_this.socket.sendString(JSON.stringify(message)); | |
_this.pendingAckCallbacks++; | |
_this.writtenBytes = 0; | |
if (_this.pendingAckCallbacks > UNACKED_HIGH_WATERMARK) { | |
_this.pty.pause(); | |
} | |
} | |
}); | |
this.pty.onExit(function (_a) { | |
var exitCode = _a.exitCode, signal = _a.signal; | |
_this.socket.close(false); | |
}); | |
} | |
Session.prototype.close = function () { | |
this.socket.close(false); | |
this.pty.kill(); | |
}; | |
return Session; | |
}()); | |
/** SocketIO to node-pty adapter. */ | |
var SocketIoToPty = /** @class */ (function () { | |
function SocketIoToPty(path, server) { | |
this.path = path; | |
var io = socketio(server, { | |
path: path, | |
transports: ['polling'], | |
allowUpgrades: false, | |
// v2.10 changed default from 60s to 5s, prefer the longer timeout to | |
// avoid errant disconnects. | |
pingTimeout: 60000, | |
}); | |
io.of('/').on('connection', function (socket) { | |
// Session manages its own lifetime. | |
// tslint:disable-next-line:no-unused-expression | |
new Session(new sockets_1.SocketIOAdapter(socket)); | |
}); | |
} | |
/** Return true iff path is handled by socket.io. */ | |
SocketIoToPty.prototype.isPathProxied = function (path) { | |
return path.indexOf(this.path + '/') === 0; | |
}; | |
return SocketIoToPty; | |
}()); | |
exports.SocketIoToPty = SocketIoToPty; | |
/** WebSocket to pty adapter. */ | |
function WebSocketToPty(request, sock, head) { | |
new ws_1.Server({ noServer: true }).handleUpgrade(request, sock, head, function (ws) { | |
// Session manages its own lifetime. | |
// tslint:disable-next-line:no-unused-expression | |
new Session(new sockets_1.WebSocketAdapter(ws)); | |
}); | |
} | |
exports.WebSocketToPty = WebSocketToPty; | |
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"socketio_to_pty.js","sourceRoot":"","sources":["../../../../../../third_party/colab/sources/socketio_to_pty.ts"],"names":[],"mappings":"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","sourcesContent":["/*\n * Copyright 2020 Google Inc. All rights reserved.\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\"); you may not\n * use this file except in compliance with the License. You may obtain a copy of\n * the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS, WITHOUT\n * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the\n * License for the specific language governing permissions and limitations under\n * the License.\n */\n\nimport * as http from 'http';\nimport * as net from 'net';\nimport * as nodePty from 'node-pty';\nimport * as socketio from 'socket.io';\nimport {Server} from 'ws';\n\nimport * as logging from './logging';\nimport {Socket, SocketIOAdapter, WebSocketAdapter} from './sockets';\n\n\n// Pause and resume are missing from the typings.\ninterface Pty {\n  pause(): void;\n  resume(): void;\n}\n\nlet sessionCounter = 0;\n\n// Inspired by\n// https://xtermjs.org/docs/guides/flowcontrol/#ideas-for-a-better-mechanism.\nconst ACK_CALLBACK_EVERY_BYTES = 100000;\nconst UNACKED_HIGH_WATERMARK = 5;\nconst UNACKED_LOW_WATERMARK = 2;\n\n/** Socket<->terminal adapter. */\nclass Session {\n  private readonly id: number;\n  private readonly pty: nodePty.IPty;\n  private pendingAckCallbacks = 0;\n  private writtenBytes = 0;\n\n  constructor(private readonly socket: Socket) {\n    this.id = sessionCounter++;\n\n    this.socket.onClose((reason) => {\n      logging.getLogger().debug(\n          'PTY socket disconnected for session %d reason: %s', this.id, reason);\n\n      // Handle client disconnects to close sockets, so as to free up resources.\n      this.close();\n    });\n\n    this.socket.onStringMessage((data: string) => {\n      // Propagate the message over to the pty.\n      logging.getLogger().debug('Send data in session %d\\n%s', this.id, data);\n      const message = JSON.parse(data) as IncomingMessage;\n      if (message.data) {\n        this.pty.write(message.data);\n      }\n      if (message.cols && message.rows) {\n        this.pty.resize(message.cols, message.rows);\n      }\n      if (message.ack) {\n        this.pendingAckCallbacks--;\n        if (this.pendingAckCallbacks < UNACKED_LOW_WATERMARK) {\n          (this.pty as unknown as Pty).resume();\n        }\n      }\n    });\n\n    this.pty = nodePty.spawn('tmux', ['new-session', '-A', '-D', '-s', '0'], {\n      name: 'xterm-color',\n      cwd: './content',  // Which path should terminal start\n      // Pass environment variables\n      env: process.env as {\n        [key: string]: string;\n      },\n    });\n\n    this.pty.onData((data: string) => {\n      this.writtenBytes += data.length;\n      if (this.writtenBytes < ACK_CALLBACK_EVERY_BYTES) {\n        const message: OutgoingMessage = {data};\n        this.socket.sendString(JSON.stringify(message));\n      } else {\n        const message: OutgoingMessage = {data, ack: true};\n        this.socket.sendString(JSON.stringify(message));\n        this.pendingAckCallbacks++;\n        this.writtenBytes = 0;\n        if (this.pendingAckCallbacks > UNACKED_HIGH_WATERMARK) {\n          (this.pty as unknown as Pty).pause();\n        }\n      }\n    });\n\n    this.pty.onExit(\n        ({exitCode, signal}: {exitCode: number, signal?: number}) => {\n          this.socket.close(false);\n        });\n  }\n\n  private close() {\n    this.socket.close(false);\n    this.pty.kill();\n  }\n}\n\n/** SocketIO to node-pty adapter. */\nexport class SocketIoToPty {\n  constructor(private readonly path: string, server: http.Server) {\n    const io = socketio(server, {\n      path,\n      transports: ['polling'],\n      allowUpgrades: false,\n      // v2.10 changed default from 60s to 5s, prefer the longer timeout to\n      // avoid errant disconnects.\n      pingTimeout: 60000,\n    });\n\n    io.of('/').on('connection', (socket: SocketIO.Socket) => {\n      // Session manages its own lifetime.\n      // tslint:disable-next-line:no-unused-expression\n      new Session(new SocketIOAdapter(socket));\n    });\n  }\n\n  /** Return true iff path is handled by socket.io. */\n  isPathProxied(path: string): boolean {\n    return path.indexOf(this.path + '/') === 0;\n  }\n}\n\n\n/** WebSocket to pty adapter. */\nexport function WebSocketToPty(\n    request: http.IncomingMessage, sock: net.Socket, head: Buffer) {\n  new Server({noServer: true}).handleUpgrade(request, sock, head, (ws) => {\n    // Session manages its own lifetime.\n    // tslint:disable-next-line:no-unused-expression\n    new Session(new WebSocketAdapter(ws));\n  });\n}\n\ndeclare interface IncomingMessage {\n  readonly data?: string;\n  readonly cols?: number;\n  readonly rows?: number;\n  readonly ack?: boolean;\n}\n\ndeclare interface OutgoingMessage {\n  readonly data?: string;\n  readonly ack?: boolean;\n}\n"]} |