dashboard/components/form/SerialConsole/index.vue

268 lines
5.6 KiB
Vue

<script>
import { allHash } from '@/utils/promise';
import debounce from 'lodash/debounce';
import Socket, {
EVENT_CONNECTED,
EVENT_CONNECTING,
EVENT_DISCONNECTED,
EVENT_MESSAGE,
EVENT_CONNECT_ERROR,
} from '@/utils/socket';
export default {
props: {
value: {
type: Object,
required: true,
},
},
data() {
return {
socket: null,
terminal: null,
fitAddon: null,
searchAddon: null,
webglAddon: null,
isOpen: false,
isOpening: false,
backlog: [],
firstTime: true,
queue: []
};
},
computed: {
xtermConfig() {
return {
cursorBlink: true,
useStyle: true,
fontSize: 12,
};
},
},
watch: {
queue: {
handler: debounce(async function(neu) {
if (neu.length === 0) {
return;
}
const msg = await Promise.all(neu);
(msg || []).forEach((m) => {
this.terminal.write(m);
});
this.queue = [];
}, 5)
}
},
beforeDestroy() {
this.close();
},
async mounted() {
await this.setupTerminal();
await this.connect();
},
methods: {
async setupTerminal() {
const docStyle = getComputedStyle(document.querySelector('body'));
const xterm = await import(/* webpackChunkName: "xterm" */ 'xterm');
const addons = await allHash({
fit: import(/* webpackChunkName: "xterm" */ 'xterm-addon-fit'),
webgl: import(/* webpackChunkName: "xterm" */ 'xterm-addon-webgl'),
weblinks: import(/* webpackChunkName: "xterm" */ 'xterm-addon-web-links'),
search: import(/* webpackChunkName: "xterm" */ 'xterm-addon-search'),
});
const terminal = new xterm.Terminal({
theme: {
background: docStyle.getPropertyValue('--terminal-bg').trim(),
cursor: docStyle.getPropertyValue('--terminal-cursor').trim(),
foreground: docStyle.getPropertyValue('--terminal-text').trim()
},
...this.xtermConfig,
});
this.fitAddon = new addons.fit.FitAddon();
this.searchAddon = new addons.search.SearchAddon();
try {
this.webglAddon = new addons.webgl.WebGlAddon();
} catch (e) {
// Some browsers (Safari) don't support the webgl renderer, so don't use it.
this.webglAddon = null;
}
terminal.loadAddon(this.fitAddon);
terminal.loadAddon(this.searchAddon);
terminal.loadAddon(new addons.weblinks.WebLinksAddon());
terminal.open(this.$refs.xterm);
if ( this.webglAddon ) {
terminal.loadAddon(this.webglAddon);
}
this.fit();
this.flush();
terminal.onData((input) => {
const msg = this.str2ab(input);
this.write(msg);
});
this.terminal = terminal;
},
str2ab(str) {
const enc = new TextEncoder();
return enc.encode(str);
},
write(msg) {
if ( this.isOpen ) {
this.socket.send(msg);
} else {
this.backlog.push(msg);
}
},
clear() {
this.terminal.clear();
},
getSocketUrl() {
return `${ this.value?.getSerialConsolePath }`;
},
async connect() {
if ( this.socket ) {
await this.socket.disconnect();
this.socket = null;
this.terminal.reset();
}
const url = this.getSocketUrl();
if ( !url ) {
return;
}
this.socket = new Socket(url);
this.socket.addEventListener(EVENT_CONNECTING, (e) => {
this.isOpen = false;
this.isOpening = true;
});
this.socket.addEventListener(EVENT_CONNECT_ERROR, (e) => {
this.isOpen = false;
this.isOpening = false;
console.error('Connect Error', e); // eslint-disable-line no-console
});
this.socket.addEventListener(EVENT_CONNECTED, (e) => {
this.isOpen = true;
this.isOpening = false;
if (this.show) {
this.fit();
this.flush();
}
if (this.firstTime) {
this.socket.send(this.str2ab('\n'));
this.firstTime = false;
}
});
this.socket.addEventListener(EVENT_DISCONNECTED, (e) => {
this.isOpen = false;
this.isOpening = false;
this.$emit('close');
});
this.socket.addEventListener(EVENT_MESSAGE, (e) => {
this.queue.push(e.detail.data.text());
});
this.socket.connect();
this.terminal.focus();
},
flush() {
const backlog = this.backlog.slice();
this.backlog = [];
for ( const data of backlog ) {
this.socket.send(data);
}
},
fit(arg) {
if ( !this.fitAddon ) {
return;
}
this.fitAddon.fit();
const { rows, cols } = this.fitAddon.proposeDimensions();
if ( !this.isOpen ) {
return;
}
const message = JSON.stringify({
Width: cols,
Height: rows
});
this.socket.send(this.str2ab(message));
},
close() {
if ( this.socket ) {
this.socket.disconnect();
}
if ( this.terminal ) {
this.terminal.dispose();
}
},
}
};
</script>
<template>
<div class="harvester-shell-container">
<div ref="xterm" class="shell-body" />
<resize-observer @notify="fit" />
</div>
</template>
<style lang="scss">
@import '@/node_modules/xterm/css/xterm.css';
body, #__nuxt, #__layout, MAIN {
height: 100%;
}
.harvester-shell-container {
height: 100%;
overflow: hidden;
.shell-body, .terminal.xterm {
height: 100%;
}
}
</style>