refactor term/ws into separate modules

This commit is contained in:
sawka 2022-06-16 00:31:54 -07:00
parent 76ac2c4ff9
commit 45cd98ef00
4 changed files with 254 additions and 181 deletions

View File

@ -1,12 +1,13 @@
import * as React from "react"; import * as React from "react";
import * as mobxReact from "mobx-react"; import * as mobxReact from "mobx-react";
import * as mobx from "mobx"; import * as mobx from "mobx";
import {Terminal} from 'xterm';
import {sprintf} from "sprintf-js"; import {sprintf} from "sprintf-js";
import {boundMethod} from "autobind-decorator"; import {boundMethod} from "autobind-decorator";
import * as dayjs from 'dayjs' import * as dayjs from 'dayjs'
import {If, For, When, Otherwise, Choose} from "tsx-control-statements/components"; import {If, For, When, Otherwise, Choose} from "tsx-control-statements/components";
import cn from "classnames" import cn from "classnames"
import {GlobalWS} from "./ws";
import {TermWrap} from "./term";
type LineType = { type LineType = {
lineid : number, lineid : number,
@ -24,8 +25,6 @@ var GlobalLines = mobx.observable.box([
{lineid: 2, userid: "sawka", ts: 1654631125000, linetype: "text", text: "again"}, {lineid: 2, userid: "sawka", ts: 1654631125000, linetype: "text", text: "again"},
]); ]);
var GlobalWS : any = null;
var TermMap = {}; var TermMap = {};
window.TermMap = TermMap; window.TermMap = TermMap;
@ -99,43 +98,21 @@ class LineText extends React.Component<{line : LineType}, {}> {
} }
} }
function loadPtyOut(term : Terminal, sessionId : string, cmdId : string, callback?: () => void) {
term.clear()
let url = sprintf("http://localhost:8080/api/ptyout?sessionid=%s&cmdid=%s", sessionId, cmdId);
fetch(url).then((resp) => {
if (!resp.ok) {
throw new Error(sprintf("Bad fetch response for /api/ptyout: %d %s", resp.status, resp.statusText));
}
return resp.text()
}).then((resptext) => {
setTimeout(() => term.write(resptext, callback), 0);
});
}
@mobxReact.observer @mobxReact.observer
class LineCmd extends React.Component<{line : LineType}, {}> { class LineCmd extends React.Component<{line : LineType}, {}> {
terminal : mobx.IObservableValue<Terminal> = mobx.observable.box(null, {name: "terminal"}); termWrap : TermWrap;
focus : mobx.IObservableValue<boolean> = mobx.observable.box(false, {name: "focus"});
version : mobx.IObservableValue<int> = mobx.observable.box(0, {name: "lineversion"}); constructor(props) {
super(props);
let {line, sessionid} = this.props;
this.termWrap = new TermWrap(sessionid, line.cmdid);
}
componentDidMount() { componentDidMount() {
let {line, sessionid} = this.props; let {line, sessionid} = this.props;
let terminal = new Terminal({rows: 2, cols: 80});
TermMap[line.cmdid] = terminal;
let termElem = document.getElementById(this.getId()); let termElem = document.getElementById(this.getId());
terminal.open(termElem); this.termWrap.connectToElem(termElem);
mobx.action(() => this.terminal.set(terminal))(); this.termWrap.reloadTerminal();
this.reloadTerminal();
terminal.textarea.addEventListener("focus", () => {
mobx.action(() => {
this.focus.set(true);
})();
});
terminal.textarea.addEventListener("blur", () => {
mobx.action(() => {
this.focus.set(false);
})();
});
if (line.isnew) { if (line.isnew) {
setTimeout(() => { setTimeout(() => {
let lineElem = document.getElementById("line-" + this.getId()); let lineElem = document.getElementById("line-" + this.getId());
@ -145,22 +122,11 @@ class LineCmd extends React.Component<{line : LineType}, {}> {
})(); })();
}, 100); }, 100);
setTimeout(() => { setTimeout(() => {
this.reloadTerminal(); this.termWrap.reloadTerminal();
}, 1000); }, 1000);
} }
} }
reloadTerminal() {
let {line, sessionid} = this.props;
let terminal = this.terminal.get();
loadPtyOut(terminal, sessionid, line.cmdid, this.incVersion);
}
@boundMethod
incVersion() : void {
mobx.action(() => this.version.set(this.version.get() + 1))();
}
getId() : string { getId() : string {
let {line} = this.props; let {line} = this.props;
return "cmd-" + line.lineid + "-" + line.cmdid; return "cmd-" + line.lineid + "-" + line.cmdid;
@ -168,7 +134,7 @@ class LineCmd extends React.Component<{line : LineType}, {}> {
@boundMethod @boundMethod
doRefresh() { doRefresh() {
this.reloadTerminal(); this.termWRap.reloadTerminal();
} }
@boundMethod @boundMethod
@ -191,21 +157,11 @@ class LineCmd extends React.Component<{line : LineType}, {}> {
let {line} = this.props; let {line} = this.props;
let lineid = line.lineid.toString(); let lineid = line.lineid.toString();
let running = false; let running = false;
let term = this.terminal.get();
let version = this.version.get();
let rows = 0; let rows = 0;
let cols = 0; let cols = 0;
if (term != null) { let renderVersion = this.termWrap.getRenderVersion();
let termNumLines = term._core.buffer.lines.length; this.termWrap.resizeToContent();
let termYPos = term._core.buffer.y; let termSize = this.termWrap.getSize();
if (term.rows < 25 && termNumLines > term.rows) {
term.resize(80, Math.min(25, termNumLines));
} else if (term.rows < 25 && termYPos >= term.rows) {
term.resize(80, Math.min(25, termYPos+1));
}
rows = term.rows;
cols = term.cols;
}
return ( return (
<div className="line line-cmd" id={"line-" + this.getId()}> <div className="line line-cmd" id={"line-" + this.getId()}>
<div className={cn("avatar",{"num4": lineid.length == 4}, {"num5": lineid.length >= 5}, {"running": running})}> <div className={cn("avatar",{"num4": lineid.length == 4}, {"num5": lineid.length >= 5}, {"running": running})}>
@ -215,10 +171,10 @@ class LineCmd extends React.Component<{line : LineType}, {}> {
<div className="meta"> <div className="meta">
<div className="user">{line.userid}</div> <div className="user">{line.userid}</div>
<div className="ts">{dayjs(line.ts).format("hh:mm:ss a")}</div> <div className="ts">{dayjs(line.ts).format("hh:mm:ss a")}</div>
<div className="cmdid">{line.cmdid} <If condition={rows > 0}>({rows}x{cols})</If> v{version}</div> <div className="cmdid">{line.cmdid} <If condition={termSize.rows > 0}>({termSize.rows}x{termSize.cols})</If> v{renderVersion}</div>
<div className="cmdtext">&gt; {this.singleLineCmdText(line.cmdtext)}</div> <div className="cmdtext">&gt; {this.singleLineCmdText(line.cmdtext)}</div>
</div> </div>
<div className={cn("terminal-wrapper", {"focus": this.focus.get()})}> <div className={cn("terminal-wrapper", {"focus": this.termWrap.isFocused.get()})}>
<div className="terminal" id={this.getId()}></div> <div className="terminal" id={this.getId()}></div>
</div> </div>
</div> </div>
@ -277,7 +233,6 @@ class CmdInput extends React.Component<{line : LineType, sessionid : string}, {}
let url = sprintf("http://localhost:8080/api/run-command"); let url = sprintf("http://localhost:8080/api/run-command");
let data = {sessionid: this.props.sessionid, command: commandStr}; let data = {sessionid: this.props.sessionid, command: commandStr};
fetch(url, {method: "post", body: JSON.stringify(data)}).then((resp) => handleJsonFetchResponse(url, resp)).then((data) => { fetch(url, {method: "post", body: JSON.stringify(data)}).then((resp) => handleJsonFetchResponse(url, resp)).then((data) => {
console.log("got success data", data);
mobx.action(() => { mobx.action(() => {
let lines = GlobalLines.get(); let lines = GlobalLines.get();
data.data.line.isnew = true; data.data.line.isnew = true;
@ -333,131 +288,13 @@ class SessionView extends React.Component<{sessionid : string}, {}> {
} }
} }
class WSControl {
wsConn : any;
openCallback : any;
open : boolean;
opening : boolean;
reconnectTimes : int;
constructor(openCallback : any) {
this.reconnectTimes = 0;
this.open = false;
this.opening = false;
this.openCallback = openCallback;
setInterval(this.sendPing, 5000);
this.reconnect();
}
reconnect() {
if (this.open) {
this.wsConn.close();
return;
}
this.reconnectTimes++;
let timeoutArr = [0, 0, 2, 5, 10, 10, 30, 60];
let timeout = 60;
if (this.reconnectTimes < timeoutArr.length) {
timeout = timeoutArr[this.reconnectTimes];
}
if (timeout > 0 || true) {
console.log(sprintf("websocket reconnect(%d), sleep %ds", this.reconnectTimes, timeout));
}
setTimeout(() => {
console.log(sprintf("websocket reconnect(%d)", this.reconnectTimes));
this.opening = true;
this.wsConn = new WebSocket("ws://localhost:8081/ws");
this.wsConn.onopen = this.onopen;
this.wsConn.onmessage = this.onmessage;
this.wsConn.onerror = this.onerror;
this.wsConn.onclose = this.onclose;
}, timeout*1000);
}
@boundMethod
onerror(event : any) {
console.log("websocket error", event);
if (this.open || this.opening) {
this.open = false;
this.opening = false;
this.reconnect();
}
}
@boundMethod
onclose(event : any) {
console.log("websocket closed", event);
if (this.open || this.opening) {
this.open = false;
this.opening = false;
this.reconnect();
}
}
@boundMethod
onopen() {
console.log("websocket open");
this.open = true;
this.opening = false;
this.reconnectTimes = 0;
if (this.openCallback != null) {
this.openCallback();
}
}
@boundMethod
onmessage(event : any) {
let eventData = null;
if (event.data != null) {
eventData = JSON.parse(event.data);
}
if (eventData == null) {
return;
}
if (eventData.type == "ping") {
this.wsConn.send(JSON.stringify({type: "pong", stime: parseInt(Date.now()/1000)}));
return;
}
if (eventData.type == "pong") {
// nothing
return;
}
console.log("websocket message", event);
}
@boundMethod
sendPing() {
if (!this.open) {
return;
}
this.wsConn.send(JSON.stringify({type: "ping", stime: Date.now()}));
}
sendMessage(data : any){
if (!this.open) {
return;
}
this.wsConn.send(JSON.stringify(data));
}
}
@mobxReact.observer @mobxReact.observer
class Main extends React.Component<{sessionid : string}, {}> { class Main extends React.Component<{sessionid : string}, {}> {
version : mobx.IObservableValue<int> = mobx.observable.box(false);
constructor(props : any) { constructor(props : any) {
super(props); super(props);
GlobalWS = new WSControl(this.updateVersion);
window.GlobalWS = GlobalWS;
}
@boundMethod
updateVersion() {
mobx.action(() => this.version.set(this.version.get()+1))();
} }
render() { render() {
let version = this.version.get();
return ( return (
<div className="main"> <div className="main">
<h1 className="title scripthaus-logo-small"> <h1 className="title scripthaus-logo-small">

View File

@ -3,12 +3,14 @@ import {createRoot} from 'react-dom/client';
import {sprintf} from "sprintf-js"; import {sprintf} from "sprintf-js";
import {Terminal} from 'xterm'; import {Terminal} from 'xterm';
import {Main} from "./main"; import {Main} from "./main";
import {GlobalWS} from "./ws";
let VERSION = __SHVERSION__; let VERSION = __SHVERSION__;
let terminal = null; let terminal = null;
let sessionId = "47445c53-cfcf-4943-8339-2c04447f20a1"; let sessionId = "47445c53-cfcf-4943-8339-2c04447f20a1";
document.addEventListener("DOMContentLoaded", () => { document.addEventListener("DOMContentLoaded", () => {
GlobalWS.reconnect();
let reactElem = React.createElement(Main, {sessionid: sessionId}, null); let reactElem = React.createElement(Main, {sessionid: sessionId}, null);
let elem = document.getElementById("main"); let elem = document.getElementById("main");
let root = createRoot(elem); let root = createRoot(elem);

91
src/term.ts Normal file
View File

@ -0,0 +1,91 @@
import * as mobx from "mobx";
import {Terminal} from 'xterm';
import {sprintf} from "sprintf-js";
import {boundMethod} from "autobind-decorator";
import {GlobalWS} from "./ws";
var TermMap : Record<string, TermWrap>;
function loadPtyOut(term : Terminal, sessionId : string, cmdId : string, callback?: () => void) {
term.clear()
let url = sprintf("http://localhost:8080/api/ptyout?sessionid=%s&cmdid=%s", sessionId, cmdId);
fetch(url).then((resp) => {
if (!resp.ok) {
throw new Error(sprintf("Bad fetch response for /api/ptyout: %d %s", resp.status, resp.statusText));
}
return resp.text()
}).then((resptext) => {
setTimeout(() => term.write(resptext, callback), 0);
});
}
class TermWrap {
terminal : Terminal;
sessionId : string;
cmdId : string;
ptyPos : number;
runPos : number;
runData : string;
renderVersion : mobx.IObservableValue<number> = mobx.observable.box(1, {name: "renderVersion"});
isFocused : mobx.IObservableValue<boolean> = mobx.observable.box(false, {name: "focus"});
constructor(sessionId : string, cmdId : string) {
this.terminal = new Terminal({rows: 2, cols: 80});
this.sessionId = sessionId;
this.cmdId = cmdId;
this.ptyPos = 0;
this.runPos = 0;
this.runData = "";
TermMap[cmdId] = this;
}
resizeToContent() {
let term = this.terminal;
let termNumLines = term._core.buffer.lines.length;
let termYPos = term._core.buffer.y;
if (term.rows < 25 && termNumLines > term.rows) {
term.resize(80, Math.min(25, termNumLines));
} else if (term.rows < 25 && termYPos >= term.rows) {
term.resize(80, Math.min(25, termYPos+1));
}
}
getSize() : {rows : number, cols : number} {
return {rows: this.terminal.rows, cols: this.terminal.cols};
}
@boundMethod
setFocus(val : boolean) {
mobx.action(() => this.isFocused.set(val))();
}
getRenderVersion() : number {
return this.renderVersion.get();
}
@boundMethod
incRenderVersion() {
mobx.action(() => this.renderVersion.set(this.renderVersion.get() + 1))();
}
reloadTerminal() {
loadPtyOut(this.terminal, this.sessionId, this.cmdId, this.incRenderVersion);
}
connectToElem(elem : Element) {
this.terminal.open(elem);
this.terminal.textarea.addEventListener("focus", () => {
this.setFocus(true);
});
this.terminal.textarea.addEventListener("blur", () => {
this.setFocus(false);
});
}
}
if (window.TermMap == null) {
TermMap = {};
window.TermMap = TermMap;
}
export {TermWrap, TermMap};

143
src/ws.ts Normal file
View File

@ -0,0 +1,143 @@
import * as mobx from "mobx";
import {sprintf} from "sprintf-js";
import {boundMethod} from "autobind-decorator";
class WSControl {
wsConn : any;
open : mobx.IObservableValue<boolean>;
opening : boolean;
reconnectTimes : int;
msgQueue : any[];
constructor() {
this.reconnectTimes = 0;
this.open = mobx.observable.box(false, {name: "WSOpen"});
this.opening = false;
this.msgQueue = [];
setInterval(this.sendPing, 5000);
}
@mobx.action
setOpen(val : boolean) {
this.open.set(val);
}
reconnect() {
if (this.open.get()) {
this.wsConn.close();
return;
}
this.reconnectTimes++;
let timeoutArr = [0, 0, 2, 5, 10, 10, 30, 60];
let timeout = 60;
if (this.reconnectTimes < timeoutArr.length) {
timeout = timeoutArr[this.reconnectTimes];
}
if (timeout > 0) {
console.log(sprintf("websocket reconnect(%d), sleep %ds", this.reconnectTimes, timeout));
}
setTimeout(() => {
console.log(sprintf("websocket reconnect(%d)", this.reconnectTimes));
this.opening = true;
this.wsConn = new WebSocket("ws://localhost:8081/ws");
this.wsConn.onopen = this.onopen;
this.wsConn.onmessage = this.onmessage;
this.wsConn.onerror = this.onerror;
this.wsConn.onclose = this.onclose;
}, timeout*1000);
}
@boundMethod
onerror(event : any) {
console.log("websocket error", event);
if (this.open.get() || this.opening) {
this.setOpen(false);
this.opening = false;
this.reconnect();
}
}
@boundMethod
onclose(event : any) {
console.log("websocket closed", event);
if (this.open.get() || this.opening) {
this.setOpen(false);
this.opening = false;
this.reconnect();
}
}
@boundMethod
onopen() {
console.log("websocket open");
this.setOpen(true);
this.opening = false;
this.reconnectTimes = 0;
this.runMsgQueue();
}
runMsgQueue() {
if (!this.open.get()) {
return;
}
if (this.msgQueue.length == 0) {
return;
}
let msg = this.msgQueue.shift();
this.sendMessage(msg);
setTimeout(() => {
this.runMsgQueue();
}, 100);
}
@boundMethod
onmessage(event : any) {
let eventData = null;
if (event.data != null) {
eventData = JSON.parse(event.data);
}
if (eventData == null) {
return;
}
if (eventData.type == "ping") {
this.wsConn.send(JSON.stringify({type: "pong", stime: parseInt(Date.now()/1000)}));
return;
}
if (eventData.type == "pong") {
// nothing
return;
}
console.log("websocket message", event);
}
@boundMethod
sendPing() {
if (!this.open.get()) {
return;
}
this.wsConn.send(JSON.stringify({type: "ping", stime: Date.now()}));
}
sendMessage(data : any) {
if (!this.open.get()) {
return;
}
this.wsConn.send(JSON.stringify(data));
}
pushMessage(data : any) {
if (!this.open.get()) {
this.msgQueue.push(data);
return;
}
this.sendMessage(data);
}
}
var GlobalWS : WSControl;
if (window.GlobalWS == null) {
GlobalWS = new WSControl();
window.GlobalWS = GlobalWS;
}
export {GlobalWS};