🔨 执行日志.

This commit is contained in:
lijiahang
2024-03-20 15:28:20 +08:00
parent 3d853eb7d4
commit 8a7976a4dd
9 changed files with 392 additions and 75 deletions

View File

@@ -93,5 +93,5 @@ export function getExecLogTailToken(request: ExecTailRequest) {
* 下载执行日志文件 * 下载执行日志文件
*/ */
export function downloadExecLogFile(id: number) { export function downloadExecLogFile(id: number) {
return axios.get<Blob>('/asset/exec/download-log', { unwrap: true, params: { id } }); return axios.get('/asset/exec/download-log', { unwrap: true, params: { id } });
} }

View File

@@ -332,6 +332,7 @@
display: flex; display: flex;
align-items: center; align-items: center;
padding-left: 20px; padding-left: 20px;
color: var(--color-text-1);
&-logo { &-logo {
width: 32px; width: 32px;

View File

@@ -33,6 +33,7 @@ export interface LogDomRef {
export interface LogAppenderConf { export interface LogAppenderConf {
id: number; id: number;
el: HTMLElement; el: HTMLElement;
fixed: boolean;
terminal: Terminal; terminal: Terminal;
addons: LogAddons; addons: LogAddons;
} }
@@ -49,8 +50,38 @@ export interface ILogAppender {
// 初始化 // 初始化
init(refs: Array<LogDomRef>): Promise<void>; init(refs: Array<LogDomRef>): Promise<void>;
// 自适应 // 设置当前元素
fit(): void; setCurrent(id: number): void;
// 查找关键字
find(word: string, next: boolean, options: any): void;
// 聚焦
focus(): void;
// 设置固定
setFixed(fixed: boolean): void;
// 去顶部
toTop(): void;
// 去底部
toBottom(): void;
// 添加字体大小
addFontSize(addSize: number): void;
// 复制
copy(): void;
// 复制全部
copyAll(): void;
// 选中全部
selectAll(): void;
// 清空
clear(): void;
// 关闭 client // 关闭 client
closeClient(): void; closeClient(): void;

View File

@@ -1,38 +1,38 @@
import type { ExecTailRequest } from '@/api/exec/exec';
import { getExecLogTailToken } from '@/api/exec/exec';
import type { ILogAppender, LogAddons, LogAppenderConf, LogDomRef } from './appender.const'; import type { ILogAppender, LogAddons, LogAppenderConf, LogDomRef } from './appender.const';
import type { ExecTailRequest } from '@/api/exec/exec';
import { AppenderOptions } from './appender.const'; import { AppenderOptions } from './appender.const';
import { getExecLogTailToken } from '@/api/exec/exec';
import { webSocketBaseUrl } from '@/utils/env'; import { webSocketBaseUrl } from '@/utils/env';
import { Message } from '@arco-design/web-vue'; import { Message } from '@arco-design/web-vue';
import { createWebSocket } from '@/utils'; import { createWebSocket } from '@/utils';
import { useDebounceFn } from '@vueuse/core';
import { addEventListen, removeEventListen } from '@/utils/event';
import { copy as copyText } from '@/hooks/copy';
import { Terminal } from 'xterm'; import { Terminal } from 'xterm';
import { FitAddon } from 'xterm-addon-fit'; import { FitAddon } from 'xterm-addon-fit';
import { SearchAddon } from 'xterm-addon-search'; import { SearchAddon } from 'xterm-addon-search';
import { CanvasAddon } from 'xterm-addon-canvas'; import { CanvasAddon } from 'xterm-addon-canvas';
import { useDebounceFn } from '@vueuse/core';
import { addEventListen, removeEventListen } from '@/utils/event';
// todo SEARCH addon setfixed
// todo font-size totop copy tobottom selectall clear
// todo 批量执行的 warn
// 执行日志 appender 实现 // 执行日志 appender 实现
export default class LogAppender implements ILogAppender { export default class LogAppender implements ILogAppender {
private config: ExecTailRequest; private current: LogAppenderConf;
private client?: WebSocket; private client?: WebSocket;
private readonly config: ExecTailRequest;
private readonly appenderRel: Record<string, LogAppenderConf>; private readonly appenderRel: Record<string, LogAppenderConf>;
private keepAliveTask?: number; private keepAliveTask?: number;
private readonly fitFn: () => {}; private readonly fitAllFn: () => {};
constructor(config: ExecTailRequest) { constructor(config: ExecTailRequest) {
this.current = undefined as unknown as LogAppenderConf;
this.config = config; this.config = config;
this.appenderRel = {}; this.appenderRel = {};
this.fitFn = useDebounceFn(this.fit).bind(this); this.fitAllFn = useDebounceFn(this.fitAll).bind(this);
} }
// 初始化 // 初始化
@@ -49,6 +49,8 @@ export default class LogAppender implements ILogAppender {
for (let logDomRef of logDomRefs) { for (let logDomRef of logDomRefs) {
// 初始化 terminal // 初始化 terminal
const terminal = new Terminal(AppenderOptions); const terminal = new Terminal(AppenderOptions);
// 初始化快捷键
this.initCustomKey(terminal);
// 初始化插件 // 初始化插件
const addons = this.initAddons(terminal); const addons = this.initAddons(terminal);
// 打开终端 // 打开终端
@@ -57,12 +59,46 @@ export default class LogAppender implements ILogAppender {
addons.fit.fit(); addons.fit.fit();
this.appenderRel[logDomRef.id] = { this.appenderRel[logDomRef.id] = {
...logDomRef, ...logDomRef,
fixed: false,
terminal, terminal,
addons addons
}; };
} }
// 设置当前对象
this.current = this.appenderRel[logDomRefs[0].id];
// 注册自适应事件 // 注册自适应事件
addEventListen(window, 'resize', this.fitFn); addEventListen(window, 'resize', this.fitAllFn);
}
// 初始化快捷键操作
initCustomKey(terminal: Terminal) {
terminal.attachCustomKeyEventHandler((e: KeyboardEvent) => {
if (e.type !== 'keydown') {
return true;
}
if (e.ctrlKey && e.code === 'KeyC') {
// 复制
e.preventDefault();
this.copy();
return false;
} else if (e.ctrlKey && e.code === 'KeyL') {
// 清空
e.preventDefault();
this.clear();
return false;
} else if (e.ctrlKey && e.code === 'KeyA') {
// 全选
e.preventDefault();
this.selectAll();
return false;
} else if (e.ctrlKey && e.code === 'KeyF') {
// 搜索
e.preventDefault();
// TODO open search
return false;
}
return true;
});
} }
// 初始化插件 // 初始化插件
@@ -102,10 +138,92 @@ export default class LogAppender implements ILogAppender {
}, 15000); }, 15000);
} }
// 自适应 // 设置当前元素
fit(): void { setCurrent(id: number): void {
const rel = this.appenderRel[id];
if (!rel) {
return;
}
this.current = rel;
// 自适应
rel.addons.fit.fit();
// 非固定跳转到最底部
if (!rel.fixed) {
rel.terminal.scrollToBottom();
}
this.focus();
}
// 查找关键字
find(word: string, next: boolean, options: any) {
if (next) {
this.current.addons.search.findNext(word, options);
} else {
this.current.addons.search.findPrevious(word, options);
}
}
// 设置固定
setFixed(fixed: boolean): void {
this.current.fixed = fixed;
this.focus();
}
// 去顶部
toTop(): void {
this.current.terminal.scrollToTop();
this.focus();
}
// 去底部
toBottom(): void {
this.current.terminal.scrollToBottom();
this.focus();
}
// 添加字体大小
addFontSize(addSize: number): void {
this.current.terminal.options['fontSize'] = this.current.terminal.options['fontSize'] as number + addSize;
this.current.addons.fit.fit();
this.focus();
}
// 复制
copy(): void {
copyText(this.current.terminal.getSelection(), '已复制');
this.focus();
}
// 复制全部
copyAll(): void {
this.selectAll();
this.copy();
this.current.terminal.clearSelection();
this.focus();
}
// 选中全部
selectAll(): void {
this.current.terminal.selectAll();
this.focus();
}
// 清空
clear(): void {
this.current.terminal.clear();
this.current.terminal.clearSelection();
this.focus();
}
// 聚焦
focus(): void {
this.current.terminal.focus();
}
// 自适应全部
fitAll(): void {
Object.values(this.appenderRel).forEach(s => { Object.values(this.appenderRel).forEach(s => {
s.addons?.fit?.fit(); s.addons.fit.fit();
}); });
} }
@@ -129,7 +247,7 @@ export default class LogAppender implements ILogAppender {
} }
}); });
// 移除自适应事件 // 移除自适应事件
removeEventListen(window, 'resize', this.fitFn); removeEventListen(window, 'resize', this.fitAllFn);
} }
// 关闭 // 关闭

View File

@@ -53,7 +53,7 @@
<script lang="ts"> <script lang="ts">
export default { export default {
name: 'sshSearchModal' name: 'xtermSearchModal'
}; };
</script> </script>
@@ -130,26 +130,26 @@
display: flex; display: flex;
align-items: center; align-items: center;
justify-content: space-between; justify-content: space-between;
background: var(--search-bg); background: var(--bg);
transition: background-color .2s; transition: background-color .2s;
&:focus-within, &:hover { &:focus-within, &:hover {
background: var(--search-bg-focus); background: var(--bg-focus);
.search-input { .search-input {
color: var(--search-color-text-focus); color: var(--color-text-focus);
} }
.icon-wrapper { .icon-wrapper {
color: var(--search-color-text-focus); color: var(--color-text-focus);
transition: background-color .2s; transition: background-color .2s;
&:hover { &:hover {
background: var(--search-bg-icon-hover-focus); background: var(--bg-icon-hover-focus);
} }
&.selected { &.selected {
background: var(--search-bg-icon-selected-focus); background: var(--bg-icon-selected-focus);
} }
} }
} }
@@ -163,7 +163,7 @@
outline: none; outline: none;
height: 18px; height: 18px;
font-size: 12px; font-size: 12px;
color: var(--search-color-text); color: var(--color-text);
} }
.word-option { .word-option {
@@ -183,18 +183,18 @@
display: flex; display: flex;
align-items: center; align-items: center;
justify-content: center; justify-content: center;
color: var(--search-color-text); color: var(--color-text);
&:not(:first-child) { &:not(:first-child) {
margin-left: 2px; margin-left: 2px;
} }
&:hover { &:hover {
background: var(--search-bg-icon-hover); background: var(--bg-icon-hover);
} }
&.selected { &.selected {
background: var(--search-bg-icon-selected); background: var(--bg-icon-selected);
} }
} }
} }

View File

@@ -87,7 +87,7 @@ export function getParentPath(path: string) {
/** /**
* 下载文件 * 下载文件
*/ */
export function downloadFile(res: any, fileName: string) { export function downloadFile(res: any, fileName: string = '') {
const blob = new Blob([res.data]); const blob = new Blob([res.data]);
const tempLink = document.createElement('a'); const tempLink = document.createElement('a');
const blobURL = window.URL.createObjectURL(blob); const blobURL = window.URL.createObjectURL(blob);

View File

@@ -7,40 +7,159 @@
<!-- 面板头部 --> <!-- 面板头部 -->
<div class="log-header"> <div class="log-header">
<!-- 左侧信息 --> <!-- 左侧信息 -->
<div class="log-header-left"> <a-space class="log-header-left" :size="12">
<a-space :size="12"> <!-- 状态 -->
<!-- 状态 --> <a-tag :color="getDictValue(execHostStatusKey, host.status, 'color')">
<a-tag :color="getDictValue(execHostStatusKey, host.status, 'color')"> {{ getDictValue(execHostStatusKey, host.status) }}
{{ getDictValue(execHostStatusKey, host.status) }} </a-tag>
</a-tag> <!-- exitStatus -->
<!-- exitStatus --> <a-tag v-if="host.exitStatus || host.exitStatus === 0"
<a-tag v-if="host.exitStatus || host.exitStatus === 0" :color="host.exitStatus === 0 ? 'arcoblue' : 'orangered'"
:color="host.exitStatus === 0 ? 'arcoblue' : 'orangered'" title="exit status">
title="exit status"> <template #icon>
<template #icon> <icon-check v-if="host.exitStatus === 0" />
<icon-check v-if="host.exitStatus === 0" /> <icon-exclamation v-else />
<icon-exclamation v-else /> </template>
</template> <span class="tag-value">{{ host.exitStatus }}</span>
<span class="tag-value">{{ host.exitStatus }}</span> </a-tag>
</a-tag> <!-- 持续时间 -->
<!-- 持续时间 --> <a-tag color="arcoblue" title="持续时间">
<a-tag color="arcoblue" title="持续时间"> <template #icon>
<template #icon> <icon-loading v-if="host.status === execHostStatus.WAITING || host.status === execHostStatus.RUNNING" />
<icon-loading v-if="host.status === execHostStatus.WAITING || host.status === execHostStatus.RUNNING" /> <icon-clock-circle v-else />
<icon-clock-circle v-else /> </template>
</template> <span class="tag-value">{{ formatDuration(host.startTime, host.finishTime) || '0s' }}</span>
<span class="tag-value">{{ formatDuration(host.startTime, host.finishTime) || '0s' }}</span> </a-tag>
</a-tag> </a-space>
</a-space>
</div>
<!-- 右侧操作 --> <!-- 右侧操作 -->
<div class="log-header-right">TODO</div> <a-space class="log-header-right" :size="12">
</div> <!-- 搜索 -->
<!-- 日志面板 --> <span class="log-action click-icon-wrapper"
<div class="log-wrapper"> title="搜索"
<div class="log-appender" @click="() => appender?.addFontSize(1)">
:ref="e => addRef(host.id, e) as unknown as VNodeRef" /> <icon-find-replace />
</span>
<!-- 增大字号 -->
<span class="log-action click-icon-wrapper"
title="增大字号"
@click="() => appender?.addFontSize(1)">
<icon-zoom-in />
</span>
<!-- 减小字号 -->
<span class="log-action click-icon-wrapper"
title="减小字号"
@click="() => appender?.addFontSize(-1)">
<icon-zoom-out />
</span>
<!-- 去顶部 -->
<span class="log-action click-icon-wrapper"
title="去顶部"
@click="() => appender?.toTop()">
<icon-up />
</span>
<!-- 去底部 -->
<span class="log-action click-icon-wrapper"
title="去底部"
@click="() => appender?.toBottom()">
<icon-down />
</span>
<!-- 全选 -->
<span class="log-action click-icon-wrapper"
title="全选"
@click="() => appender?.selectAll()">
<icon-expand />
</span>
<!-- 复制 -->
<span class="log-action click-icon-wrapper"
title="复制"
@click="() => appender?.copy()">
<icon-copy />
</span>
<!-- 复制全部 -->
<span class="log-action click-icon-wrapper"
title="复制全部"
@click="() => appender?.copyAll()">
<icon-brush />
</span>
<!-- 清空 -->
<span class="log-action click-icon-wrapper"
title="清空"
@click="() => appender?.clear()">
<icon-delete />
</span>
<!-- 下载 -->
<span class="log-action click-icon-wrapper"
title="下载"
@click="downloadLogFile(host.id)">
<icon-download />
</span>
<!-- 设置固定 -->
<a-switch type="round"
checked-text="固定"
unchecked-text="跟随"
@change="(e: any) => appender?.setFixed(e as boolean)" />
</a-space>
</div> </div>
<!-- 右键菜单 -->
<a-dropdown :popup-max-height="false"
trigger="contextMenu"
position="bl"
alignPoint>
<!-- 日志面板 -->
<div class="log-wrapper">
<!-- terminal -->
<div class="log-appender"
:ref="e => addRef(host.id, e as HTMLElement) as unknown as VNodeRef" />
<!-- 搜索框 -->
<xterm-search-modal ref="searchModal"
class="search-modal"
@find="searchWords"
@close="searchClose" />
</div>
<!-- 右键菜单 -->
<template #content>
<!-- 去顶部 -->
<a-doption style="line-height: 30px; padding: 0 8px;"
@click="() => appender?.toTop()">
<template #icon>
<icon-up />
</template>
<span>去顶部</span>
</a-doption>
<!-- 去底部 -->
<a-doption style="line-height: 30px; padding: 0 8px;"
@click="() => appender?.toBottom()">
<template #icon>
<icon-down />
</template>
<span>去底部</span>
</a-doption>
<!-- 全选 -->
<a-doption style="line-height: 30px; padding: 0 8px;"
@click="() => appender?.selectAll()">
<template #icon>
<icon-expand />
</template>
<span>全选</span>
</a-doption>
<!-- 复制 -->
<a-doption style="line-height: 30px; padding: 0 8px;"
@click="() => appender?.copy()">
<template #icon>
<icon-copy />
</template>
<span>复制</span>
</a-doption>
<!-- 清空 -->
<a-doption style="line-height: 30px; padding: 0 8px;"
@click="() => appender?.clear()">
<template #icon>
<icon-delete />
</template>
<span>清空</span>
</a-doption>
</template>
</a-dropdown>
</div> </div>
</div> </div>
</template> </template>
@@ -56,10 +175,13 @@
import type { ExecCommandResponse } from '@/api/exec/exec'; import type { ExecCommandResponse } from '@/api/exec/exec';
import type { LogDomRef, ILogAppender } from '@/components/xtrem/log-appender/appender.const'; import type { LogDomRef, ILogAppender } from '@/components/xtrem/log-appender/appender.const';
import { nextTick, ref, watch } from 'vue'; import { nextTick, ref, watch } from 'vue';
import { downloadExecLogFile } from '@/api/exec/exec';
import { downloadFile } from '@/utils/file';
import { formatDuration } from '@/utils'; import { formatDuration } from '@/utils';
import { execHostStatus, execHostStatusKey } from '@/views/exec/exec-log/types/const'; import { execHostStatus, execHostStatusKey } from '@/views/exec/exec-log/types/const';
import { useDictStore } from '@/store'; import { useDictStore } from '@/store';
import LogAppender from '@/components/xtrem/log-appender/log-appender'; import LogAppender from '@/components/xtrem/log-appender/log-appender';
import XtermSearchModal from '@/components/xtrem/search-modal/index.vue';
import 'xterm/css/xterm.css'; import 'xterm/css/xterm.css';
const props = defineProps<{ const props = defineProps<{
@@ -71,11 +193,12 @@
const logRefs = ref<Array<LogDomRef>>([]); const logRefs = ref<Array<LogDomRef>>([]);
const appender = ref<ILogAppender>(); const appender = ref<ILogAppender>();
const searchModal = ref();
// 切换标签自适应 // 切换标签
watch(() => props.current, () => { watch(() => props.current, (val) => {
nextTick(() => { nextTick(() => {
appender.value?.fit(); appender.value?.setCurrent(val);
}); });
}); });
@@ -107,10 +230,27 @@
}); });
}; };
// 搜索关键字
const searchWords = (word: string, next: boolean, options: any) => {
appender.value?.find(word, next, options);
};
// 关闭搜索框
const searchClose = () => {
appender.value?.focus();
};
// 下载文件
const downloadLogFile = async (id: number) => {
const data = await downloadExecLogFile(id);
downloadFile(data);
};
</script> </script>
<style lang="less" scoped> <style lang="less" scoped>
@header-height: 38px; @header-height: 40px;
.log-view { .log-view {
width: 100%; width: 100%;
height: 100%; height: 100%;
@@ -136,8 +276,10 @@
font-weight: 600; font-weight: 600;
} }
&-right { .log-action {
width: 24px;
height: 24px;
font-size: 16px;
} }
} }
@@ -157,4 +299,16 @@
} }
} }
} }
.search-modal {
--bg-focus: rgba(255, 255, 255, .85);
--bg: rgba(255, 255, 255, .95);
--color-text: #0E0E0E;
--color-text-focus: #0F0F0F;
--bg-icon-hover: rgba(12, 12, 12, .04);
--bg-icon-hover-focus: rgba(12, 12, 12, .08);
--bg-icon-selected: rgba(12, 12, 12, .06);
--bg-icon-selected-focus: rgba(12, 12, 12, .10);
}
</style> </style>

View File

@@ -138,9 +138,10 @@
@host-real-width: @host-width + 16px; @host-real-width: @host-width + 16px;
.log-panel-container { .log-panel-container {
width: 100%; width: calc(100% - 32px);
height: 100%; height: calc(100% - 32px);
display: flex; display: flex;
position: absolute;
} }
.host-container, .log-container { .host-container, .log-container {

View File

@@ -44,9 +44,10 @@
<!-- 终端实例 --> <!-- 终端实例 -->
<div class="ssh-inst" ref="terminalRef" /> <div class="ssh-inst" ref="terminalRef" />
<!-- 搜索模态框 --> <!-- 搜索模态框 -->
<ssh-search-modal ref="searchModal" <xterm-search-modal ref="searchModal"
@find="findWords" class="search-modal"
@close="focus" /> @find="findWords"
@close="focus" />
</div> </div>
</ssh-context-menu> </ssh-context-menu>
<!-- 命令编辑器 --> <!-- 命令编辑器 -->
@@ -74,8 +75,8 @@
import { ActionBarItems, connectStatusKey } from '../../types/terminal.const'; import { ActionBarItems, connectStatusKey } from '../../types/terminal.const';
import ShellEditorModal from '@/components/view/shell-editor/modal/index.vue'; import ShellEditorModal from '@/components/view/shell-editor/modal/index.vue';
import IconActions from '../layout/icon-actions.vue'; import IconActions from '../layout/icon-actions.vue';
import SshSearchModal from './ssh-search-modal.vue';
import SshContextMenu from './ssh-context-menu.vue'; import SshContextMenu from './ssh-context-menu.vue';
import XtermSearchModal from '@/components/xtrem/search-modal/index.vue';
const props = defineProps<{ const props = defineProps<{
tab: TerminalTabItem tab: TerminalTabItem
@@ -251,4 +252,15 @@
} }
} }
.search-modal {
--bg-focus: var(--search-bg-focus);
--bg: var(--search-bg);
--color-text: var(--search-color-text);
--color-text-focus: var(--search-color-text-focus);
--bg-icon-hover: var(--search-bg-icon-hover);
--bg-icon-hover-focus: var(--search-bg-icon-hover-focus);
--bg-icon-selected: var(--search-bg-icon-selected);
--bg-icon-selected-focus: var(--search-bg-icon-selected-focus);
}
</style> </style>