NEZ-894 feat: endpoint-logTab
This commit is contained in:
@@ -194,8 +194,7 @@
|
|||||||
height: calc(100% - 58px);
|
height: calc(100% - 58px);
|
||||||
|
|
||||||
.el-table:not(.chart-table) {
|
.el-table:not(.chart-table) {
|
||||||
position: absolute;
|
width: 100%;
|
||||||
width: calc(100% - 40px);
|
|
||||||
border: 1px solid $--right-box-border-color;
|
border: 1px solid $--right-box-border-color;
|
||||||
border-bottom: none;
|
border-bottom: none;
|
||||||
|
|
||||||
|
|||||||
@@ -47,7 +47,8 @@
|
|||||||
<panel-tab-new @getTableData="getTableData" :paramsType="'module'" v-if="from === fromRoute.module && targetTab === 'panel'" v-show="subResizeShow" :from="from" :obj="obj" :tabs="tabs.module.moduleTabTitle" :targetTab.sync="targetTab" @changeTab="changeTab"></panel-tab-new>
|
<panel-tab-new @getTableData="getTableData" :paramsType="'module'" v-if="from === fromRoute.module && targetTab === 'panel'" v-show="subResizeShow" :from="from" :obj="obj" :tabs="tabs.module.moduleTabTitle" :targetTab.sync="targetTab" @changeTab="changeTab"></panel-tab-new>
|
||||||
<!--endpoint列表的tab-->
|
<!--endpoint列表的tab-->
|
||||||
<panel-tab-new @getTableData="getTableData" :paramsType="'endpoint'" v-if="from === fromRoute.endpoint && targetTab === 'panelTab'" v-show="subResizeShow" :from="from" :obj="obj" :tabs="tabs.endpoint.endpointTabTitle" :targetTab.sync="targetTab" @changeTab="changeTab"></panel-tab-new>
|
<panel-tab-new @getTableData="getTableData" :paramsType="'endpoint'" v-if="from === fromRoute.endpoint && targetTab === 'panelTab'" v-show="subResizeShow" :from="from" :obj="obj" :tabs="tabs.endpoint.endpointTabTitle" :targetTab.sync="targetTab" @changeTab="changeTab"></panel-tab-new>
|
||||||
<endpointQuery v-if="from === fromRoute.endpoint && targetTab === 'endpointQuery'" v-show="subResizeShow" :from="from" :obj="obj" :tabs="tabs.endpoint.endpointTabTitle" @changeTab="changeTab" :targetTab="targetTab"></endpointQuery>
|
<endpointQuery v-if="from === fromRoute.endpoint && targetTab === 'endpointQuery'" v-show="subResizeShow" :from="from" :obj="obj" :tabs="tabs.endpoint.endpointTabTitle" :targetTab.sync="targetTab" @changeTab="changeTab"></endpointQuery>
|
||||||
|
<log-bottom-tab v-if="from === fromRoute.endpoint && targetTab === 'log'" v-show="subResizeShow" :from="from" :obj="obj" :tabs="tabs.endpoint.endpointTabTitle" :targetTab.sync="targetTab" @changeTab="changeTab"></log-bottom-tab>
|
||||||
<alertMessageTabNew v-if="from === fromRoute.endpoint && targetTab === 'endpointAlertMessage'" v-show="subResizeShow" :from="from" :obj="obj" :tabs="tabs.endpoint.endpointTabTitle" @changeTab="changeTab" :targetTab="targetTab"></alertMessageTabNew>
|
<alertMessageTabNew v-if="from === fromRoute.endpoint && targetTab === 'endpointAlertMessage'" v-show="subResizeShow" :from="from" :obj="obj" :tabs="tabs.endpoint.endpointTabTitle" @changeTab="changeTab" :targetTab="targetTab"></alertMessageTabNew>
|
||||||
<!--chartTemp的Tab-->
|
<!--chartTemp的Tab-->
|
||||||
<panel-tab-new @getTableData="getTableData" :paramsType="'template'" v-if="from === fromRoute.chartTemp && targetTab === 'panel'" v-show="subResizeShow" :from="from" :obj="obj" :tabs="tabs.chartTemp.chartTempTabTitle" :targetTab.sync="targetTab" @changeTab="changeTab"></panel-tab-new>
|
<panel-tab-new @getTableData="getTableData" :paramsType="'template'" v-if="from === fromRoute.chartTemp && targetTab === 'panel'" v-show="subResizeShow" :from="from" :obj="obj" :tabs="tabs.chartTemp.chartTempTabTitle" :targetTab.sync="targetTab" @changeTab="changeTab"></panel-tab-new>
|
||||||
@@ -76,10 +77,12 @@ import operationLogTab from './tabs/operationLogTab'
|
|||||||
import terminalLogTab from './tabs/terminalLogTab'
|
import terminalLogTab from './tabs/terminalLogTab'
|
||||||
import assetTab from '@/components/common/bottomBox/tabs/assetTab'
|
import assetTab from '@/components/common/bottomBox/tabs/assetTab'
|
||||||
import { fromRoute } from '@/components/common/js/constants'
|
import { fromRoute } from '@/components/common/js/constants'
|
||||||
|
import LogBottomTab from '@/components/common/bottomBox/tabs/logBottomTab'
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
name: 'bottomBox',
|
name: 'bottomBox',
|
||||||
components: {
|
components: {
|
||||||
|
LogBottomTab,
|
||||||
cabinetTab,
|
cabinetTab,
|
||||||
alertMessageTab,
|
alertMessageTab,
|
||||||
endpointTab,
|
endpointTab,
|
||||||
@@ -186,6 +189,7 @@ export default {
|
|||||||
endpointTabTitle: [
|
endpointTabTitle: [
|
||||||
{ prop: 'panelTab', name: this.$t('overall.detail') },
|
{ prop: 'panelTab', name: this.$t('overall.detail') },
|
||||||
{ prop: 'endpointQuery', name: 'Query' },
|
{ prop: 'endpointQuery', name: 'Query' },
|
||||||
|
{ prop: 'log', name: 'Log' },
|
||||||
{ prop: 'endpointAlertMessage', name: this.$t('overall.alert') }
|
{ prop: 'endpointAlertMessage', name: this.$t('overall.alert') }
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
@@ -283,12 +287,15 @@ export default {
|
|||||||
}
|
}
|
||||||
.sub-container {
|
.sub-container {
|
||||||
padding: 10px;
|
padding: 10px;
|
||||||
height: 100%;
|
height: calc(100% - 64px);
|
||||||
background-color: #f6f6f6;
|
background-color: #f6f6f6;
|
||||||
overflow-y: auto;
|
overflow-y: auto;
|
||||||
&>div {
|
&>div {
|
||||||
background-color: white;
|
background-color: white;
|
||||||
}
|
}
|
||||||
|
&>.nz-table2 {
|
||||||
|
padding-top: 20px !important;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
.sub-top-tools .top-tool-btn-txt .nz-icon{
|
.sub-top-tools .top-tool-btn-txt .nz-icon{
|
||||||
display: inline-block;
|
display: inline-block;
|
||||||
@@ -321,7 +328,7 @@ export default {
|
|||||||
height: 100%;
|
height: 100%;
|
||||||
background-color: #f6f6f6;
|
background-color: #f6f6f6;
|
||||||
|
|
||||||
.nz-table2 {
|
&>.nz-table2 {
|
||||||
height: calc(100% - 92px);
|
height: calc(100% - 92px);
|
||||||
padding: 20px 20px 0;
|
padding: 20px 20px 0;
|
||||||
}
|
}
|
||||||
@@ -346,6 +353,14 @@ export default {
|
|||||||
padding-bottom: 10px;
|
padding-bottom: 10px;
|
||||||
height: calc(100% - 30px);
|
height: calc(100% - 30px);
|
||||||
}
|
}
|
||||||
|
.bottom-log {
|
||||||
|
padding: 15px;
|
||||||
|
}
|
||||||
|
.bottom-common {
|
||||||
|
padding: 20px;
|
||||||
|
height: 100%;
|
||||||
|
box-sizing: border-box;
|
||||||
|
}
|
||||||
.chart-temp{
|
.chart-temp{
|
||||||
height: calc(100% - 20px);
|
height: calc(100% - 20px);
|
||||||
overflow-y: auto;
|
overflow-y: auto;
|
||||||
|
|||||||
@@ -34,7 +34,7 @@
|
|||||||
></element-set>
|
></element-set>
|
||||||
</transition>
|
</transition>
|
||||||
<div class="sub-container">
|
<div class="sub-container">
|
||||||
<div :class="[targetTab === 'panel' ? 'bottom-panel' : 'nz-table2',from === fromRoute.chartTemp ? 'chart-temp': '']">
|
<div :class="subContentClass">
|
||||||
<slot></slot>
|
<slot></slot>
|
||||||
</div>
|
</div>
|
||||||
<div class="pagination-bottom" v-if="showPagination">
|
<div class="pagination-bottom" v-if="showPagination">
|
||||||
@@ -89,6 +89,24 @@ export default {
|
|||||||
computed: {
|
computed: {
|
||||||
bottomHeaderTitle () {
|
bottomHeaderTitle () {
|
||||||
return this.title || this.$t('overall.name')
|
return this.title || this.$t('overall.name')
|
||||||
|
},
|
||||||
|
subContentClass () {
|
||||||
|
const className = []
|
||||||
|
switch (this.targetTab) {
|
||||||
|
case 'panel':
|
||||||
|
className.push('bottom-panel')
|
||||||
|
break
|
||||||
|
case 'log': {
|
||||||
|
className.push('bottom-log')
|
||||||
|
break
|
||||||
|
}
|
||||||
|
default: {
|
||||||
|
className.push('nz-table2')
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
this.from === fromRoute.chartTemp && className.push('chart-temp')
|
||||||
|
return className.join(' ')
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
data () {
|
data () {
|
||||||
|
|||||||
@@ -222,9 +222,6 @@ export default {
|
|||||||
width: 100%;
|
width: 100%;
|
||||||
height: 100%;
|
height: 100%;
|
||||||
}
|
}
|
||||||
/deep/ .nz-table2{
|
|
||||||
padding: 0 !important;
|
|
||||||
}
|
|
||||||
/deep/ .sub-container {
|
/deep/ .sub-container {
|
||||||
box-sizing: border-box;
|
box-sizing: border-box;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,223 @@
|
|||||||
|
<template>
|
||||||
|
<nz-bottom-data-list
|
||||||
|
:custom-tool="true"
|
||||||
|
:layout="[]"
|
||||||
|
:show-pagination="false"
|
||||||
|
:tabs="tabs"
|
||||||
|
:targetTab="targetTab"
|
||||||
|
@changeTab="changeTab"
|
||||||
|
>
|
||||||
|
<template v-slot:title><span :title="obj.name">{{obj.name}}</span></template>
|
||||||
|
<template v-slot:top-tool-right>
|
||||||
|
<el-input v-model="matchContent" class="margin-r-10" placeholder="" size="small" suffix-icon="el-icon-search" @keyup.enter.native="queryLogData()">
|
||||||
|
<el-select slot="prepend" v-model="matchSymbol" class="symbol-select" size="small" style="width: 60px;">
|
||||||
|
<el-option value="|="></el-option>
|
||||||
|
<el-option value="!="></el-option>
|
||||||
|
<el-option value="|~"></el-option>
|
||||||
|
<el-option value="!~"></el-option>
|
||||||
|
</el-select>
|
||||||
|
</el-input>
|
||||||
|
<pick-time id="explore" ref="pickTime" v-model="filterTime" :refresh-data-func="queryLogData" :use-chart-unit="false" :use-refresh="false">
|
||||||
|
<template slot="added-text">{{$t('overall.query')}}</template>
|
||||||
|
</pick-time>
|
||||||
|
</template>
|
||||||
|
<template v-slot>
|
||||||
|
<log-tab ref="logDetail" :log-data="logData" @exportLog="exportLog" @limitChange="queryLogData"></log-tab>
|
||||||
|
</template>
|
||||||
|
</nz-bottom-data-list>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
import nzBottomDataList from '@/components/common/bottomBox/nzBottomDataList'
|
||||||
|
import logTab from '@/components/page/dashboard/explore/logTab'
|
||||||
|
import subDataListMixin from '@/components/common/mixin/subDataList'
|
||||||
|
import axios from 'axios'
|
||||||
|
import bus from '@/libs/bus'
|
||||||
|
|
||||||
|
export default {
|
||||||
|
name: 'logBottomTab',
|
||||||
|
mixins: [subDataListMixin],
|
||||||
|
components: {
|
||||||
|
nzBottomDataList,
|
||||||
|
logTab
|
||||||
|
},
|
||||||
|
data () {
|
||||||
|
return {
|
||||||
|
logData: [],
|
||||||
|
filterTime: [
|
||||||
|
bus.timeFormate(bus.getOffsetTimezoneData(-1), 'yyyy-MM-dd hh:mm:ss'),
|
||||||
|
bus.timeFormate(bus.getOffsetTimezoneData(), 'yyyy-MM-dd hh:mm:ss')
|
||||||
|
],
|
||||||
|
expressions: [''],
|
||||||
|
matchSymbol: '|=',
|
||||||
|
matchContent: ''
|
||||||
|
}
|
||||||
|
},
|
||||||
|
methods: {
|
||||||
|
exportLog ({ limit, descending }) {
|
||||||
|
const params = {
|
||||||
|
logql: this.expressions,
|
||||||
|
start: this.$stringTimeParseToUnix(this.filterTime[0]),
|
||||||
|
end: this.$stringTimeParseToUnix(this.filterTime[1]),
|
||||||
|
direction: descending ? 'backward' : 'forward',
|
||||||
|
limit
|
||||||
|
}
|
||||||
|
axios.get('/logs/loki/export', { responseType: 'blob', params: params }).then(res => {
|
||||||
|
if (window.navigator.msSaveOrOpenBlob) {
|
||||||
|
// 兼容ie11
|
||||||
|
const blobObject = new Blob([res.data])
|
||||||
|
window.navigator.msSaveOrOpenBlob(blobObject, 'log')
|
||||||
|
} else {
|
||||||
|
const url = URL.createObjectURL(new Blob([res.data]))
|
||||||
|
const a = document.createElement('a')
|
||||||
|
document.body.appendChild(a) // 此处增加了将创建的添加到body当中
|
||||||
|
a.href = url
|
||||||
|
a.download = 'log'
|
||||||
|
a.target = '_blank'
|
||||||
|
a.click()
|
||||||
|
a.remove() // 将a标签移除
|
||||||
|
}
|
||||||
|
}, error => {
|
||||||
|
const $self = this
|
||||||
|
const reader = new FileReader()
|
||||||
|
reader.onload = function (event) {
|
||||||
|
const responseText = reader.result
|
||||||
|
const exception = JSON.parse(responseText)
|
||||||
|
if (exception.message) {
|
||||||
|
$self.$message.error(exception.message)
|
||||||
|
} else {
|
||||||
|
console.error(error)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
reader.readAsText(error.response.data)
|
||||||
|
})
|
||||||
|
},
|
||||||
|
queryLogData (limit) { // log的chart和table是一个请求
|
||||||
|
if (!limit) {
|
||||||
|
limit = this.$refs.logDetail.getLimit()
|
||||||
|
}
|
||||||
|
if (this.expressions.length > 0) {
|
||||||
|
const requestArr = []
|
||||||
|
this.expressions.forEach((item, index) => {
|
||||||
|
if (item != '') {
|
||||||
|
let expr = item
|
||||||
|
this.matchContent && (expr = `${item} ${this.matchSymbol} ${this.matchContent}`)
|
||||||
|
requestArr.push(this.$get('/logs/loki/api/v1/query_range?format=1&query=' + expr + '&start=' + this.$stringTimeParseToUnix(this.filterTime[0]) + '&end=' + this.$stringTimeParseToUnix(this.filterTime[1]) + '&limit=' + limit))
|
||||||
|
}
|
||||||
|
})
|
||||||
|
if (requestArr.length > 0) {
|
||||||
|
this.showIntroduce = false
|
||||||
|
this.saveDisabled = false
|
||||||
|
}
|
||||||
|
axios.all(requestArr).then(res => {
|
||||||
|
this.logData = res.map(r => r.data)
|
||||||
|
const hasGraph = this.logData.some(d => d.resultType === 'matrix')
|
||||||
|
const hasLog = this.logData.some(d => d.resultType === 'streamsFormat')
|
||||||
|
const graphTabIndex = this.showTab.indexOf('1')
|
||||||
|
if (hasGraph) {
|
||||||
|
if (graphTabIndex === -1) {
|
||||||
|
this.showTab.push('1')
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (graphTabIndex > -1) {
|
||||||
|
this.showTab.splice(graphTabIndex, 1)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
const logTabIndex = this.showTab.indexOf('2')
|
||||||
|
if (hasLog) {
|
||||||
|
if (logTabIndex === -1) {
|
||||||
|
this.showTab.push('1')
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (logTabIndex > -1) {
|
||||||
|
this.showTab.splice(logTabIndex, 1)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
this.$nextTick(() => {
|
||||||
|
hasGraph && this.loadLogGraph()
|
||||||
|
})
|
||||||
|
})
|
||||||
|
}
|
||||||
|
},
|
||||||
|
loadLogGraph () {
|
||||||
|
const graphData = this.logData.filter(l => l.resultType === 'matrix')
|
||||||
|
if (graphData && graphData.length > 0) {
|
||||||
|
this.$refs.logChart.startLoading()
|
||||||
|
const promqlInputIndexs = []
|
||||||
|
const queryExpression = []
|
||||||
|
const series = []
|
||||||
|
const legend = []
|
||||||
|
this.expressions.forEach((item, index) => {
|
||||||
|
if (item !== '') {
|
||||||
|
promqlInputIndexs.push(index)
|
||||||
|
queryExpression.push(item)
|
||||||
|
}
|
||||||
|
this.logData.forEach((response, index) => {
|
||||||
|
if (response.resultType === 'matrix') {
|
||||||
|
const promqlIndex = promqlInputIndexs[index]
|
||||||
|
const data = response.result
|
||||||
|
if (!data || data.length < 1) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
data.forEach((result, i) => {
|
||||||
|
const seriesItem = {
|
||||||
|
name: '',
|
||||||
|
symbol: 'emptyCircle', // 去掉点
|
||||||
|
symbolSize: [2, 2],
|
||||||
|
showSymbol: false,
|
||||||
|
smooth: 0.2, // 曲线变平滑
|
||||||
|
data: [],
|
||||||
|
lineStyle: {
|
||||||
|
width: 1,
|
||||||
|
opacity: 0.9
|
||||||
|
},
|
||||||
|
type: 'line'
|
||||||
|
}
|
||||||
|
let legendName = ''
|
||||||
|
seriesItem.data = result.values.map((item) => {
|
||||||
|
return [item[0] * 1000, item[1]]
|
||||||
|
})
|
||||||
|
if (result.metric && Object.keys(result.metric).length > 0) {
|
||||||
|
const metric = Object.assign({}, result.metric)
|
||||||
|
seriesItem.name += metric.__name__ ? metric.__name__ : ''
|
||||||
|
seriesItem.name += '{'
|
||||||
|
delete metric.__name__
|
||||||
|
for (const key in metric) {
|
||||||
|
seriesItem.name += key + '=' + '"' + metric[key] + '",'
|
||||||
|
}
|
||||||
|
legendName = seriesItem.name.substr(0, seriesItem.name.length - 1)
|
||||||
|
legendName += '}'
|
||||||
|
} else {
|
||||||
|
legendName = queryExpression[index]
|
||||||
|
}
|
||||||
|
seriesItem.name = legendName + '-' + index
|
||||||
|
series.push(seriesItem)
|
||||||
|
legend.push({ name: seriesItem.name, alias: legendName, isGray: false })
|
||||||
|
})
|
||||||
|
|
||||||
|
this.$refs['promql-' + promqlIndex][0].setError('')
|
||||||
|
}
|
||||||
|
})
|
||||||
|
})
|
||||||
|
this.$refs.logChart.setLegend(legend)
|
||||||
|
this.$refs.logChart.setRandomColors(series.length)
|
||||||
|
this.$refs.logChart.setSeries(series)
|
||||||
|
this.defaultChartVisible = true
|
||||||
|
this.$nextTick(() => {
|
||||||
|
this.$refs.logChart.endLoading()
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
mounted () {
|
||||||
|
this.expressions = [`{project="${this.obj.project.name}",module="${this.obj.module.name}",endpoint="${this.obj.name}"}`]
|
||||||
|
this.queryLogData()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style>
|
||||||
|
.symbol-select .el-input__inner {
|
||||||
|
padding-left: 8px;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
@@ -101,94 +101,3 @@ export default {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style lang="scss">
|
|
||||||
/* begin--二级顶部工具栏*/
|
|
||||||
.sub-top-tools {
|
|
||||||
display: flex;
|
|
||||||
height: 32px;
|
|
||||||
align-items: center;
|
|
||||||
justify-content: space-between;
|
|
||||||
border-top: 1px solid #DCDFE6;
|
|
||||||
border-bottom: 1px solid #E4E7ED;
|
|
||||||
margin: 0 -6px;
|
|
||||||
padding-right: 80px;
|
|
||||||
background-color: $content-right-background-color;
|
|
||||||
margin-bottom: 10px;
|
|
||||||
}
|
|
||||||
.sub-top-tools>div {
|
|
||||||
margin-top: 2px;
|
|
||||||
}
|
|
||||||
.sub-top-tools .top-tool-search {
|
|
||||||
width: 260px;
|
|
||||||
margin: -1px 0 0 0;
|
|
||||||
.select_input input {
|
|
||||||
background-color: white;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
.sub-container {
|
|
||||||
padding: 10px;
|
|
||||||
height: 100%;
|
|
||||||
background-color: #f6f6f6;
|
|
||||||
&>div {
|
|
||||||
background-color: white;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
.sub-top-tools .top-tool-btn-txt .nz-icon{
|
|
||||||
display: inline-block;
|
|
||||||
font-size: 12px;
|
|
||||||
margin-right: 6px;
|
|
||||||
}
|
|
||||||
.sub-top-tool-right {
|
|
||||||
display: flex;
|
|
||||||
align-content: center;
|
|
||||||
}
|
|
||||||
.has-sub-popper {
|
|
||||||
color: $danger-color;
|
|
||||||
}
|
|
||||||
.sub-box {
|
|
||||||
height: 50%;
|
|
||||||
position: relative;
|
|
||||||
}
|
|
||||||
.sub-list {
|
|
||||||
height: calc(100% - 9px);
|
|
||||||
position: absolute;
|
|
||||||
width: 100%;
|
|
||||||
z-index: 1;
|
|
||||||
top: 9px;
|
|
||||||
|
|
||||||
.sub-list__tabs {
|
|
||||||
height: 100%;
|
|
||||||
background-color: white;
|
|
||||||
|
|
||||||
&>div {
|
|
||||||
height: 100%;
|
|
||||||
background-color: #f6f6f6;
|
|
||||||
|
|
||||||
.nz-table2 {
|
|
||||||
height: calc(100% - 92px);
|
|
||||||
padding: 20px 20px 0;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
.main-and-sub-transition {
|
|
||||||
transition: .4s height;
|
|
||||||
}
|
|
||||||
|
|
||||||
.resize-modal {
|
|
||||||
width: calc(100% - 240px);
|
|
||||||
opacity: 0.6;
|
|
||||||
background-color: #f5f9ff;
|
|
||||||
border: 1px solid #a7d0f7;
|
|
||||||
box-sizing: border-box;
|
|
||||||
position: fixed;
|
|
||||||
cursor: ns-resize;
|
|
||||||
display: none;
|
|
||||||
z-index: 20;
|
|
||||||
vertical-align: bottom;
|
|
||||||
bottom: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* end--二级顶部工具栏*/
|
|
||||||
</style>
|
|
||||||
|
|||||||
@@ -584,11 +584,9 @@ export default {
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
beforeDestroy () {
|
beforeDestroy () {
|
||||||
document.querySelector('#tableList').removeEventListener('mouseenter', this.tableListEnter)
|
document.querySelector('#tableList') && document.querySelector('#tableList').removeEventListener('mouseenter', this.tableListEnter)
|
||||||
document.querySelector('#tableList').removeEventListener('mouseleave', this.tableListLeave)
|
document.querySelector('#tableList') && document.querySelector('#tableList').removeEventListener('mouseleave', this.tableListLeave)
|
||||||
if (this.scrollbarWrap) {
|
this.scrollbarWrap && this.scrollbarWrap.removeEventListener('scroll', bus.debounce)
|
||||||
this.scrollbarWrap.removeEventListener('scroll', bus.debounce)
|
|
||||||
};
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
@@ -659,7 +657,9 @@ export default {
|
|||||||
/* begin-chart list*/
|
/* begin-chart list*/
|
||||||
.table-list {
|
.table-list {
|
||||||
overflow-y: auto;
|
overflow-y: auto;
|
||||||
height: calc(100% - 56px);
|
height: 100%;
|
||||||
|
padding: 20px 0;
|
||||||
|
box-sizing: border-box;
|
||||||
}
|
}
|
||||||
|
|
||||||
.box-content {
|
.box-content {
|
||||||
|
|||||||
@@ -169,7 +169,11 @@ export default {
|
|||||||
.interval-refresh {
|
.interval-refresh {
|
||||||
display: flex;
|
display: flex;
|
||||||
}
|
}
|
||||||
.select-refresh-time-label{
|
.select-refresh-time-label{
|
||||||
margin-left: 5px;
|
margin-left: 5px;
|
||||||
}
|
}
|
||||||
|
.time-picker {
|
||||||
|
padding-left: 8px;
|
||||||
|
display: flex;
|
||||||
|
}
|
||||||
</style>
|
</style>
|
||||||
|
|||||||
@@ -2,127 +2,125 @@
|
|||||||
@import '../../../charts/chart';
|
@import '../../../charts/chart';
|
||||||
</style>
|
</style>
|
||||||
<template>
|
<template>
|
||||||
<div>
|
<!--表格-->
|
||||||
<!--表格-->
|
<el-table
|
||||||
<el-table
|
id="alertMessageTable"
|
||||||
id="alertMessageTable"
|
ref="dataTable"
|
||||||
ref="dataTable"
|
:cell-class-name="labelsClassName"
|
||||||
:cell-class-name="labelsClassName"
|
:data="tableData"
|
||||||
:data="tableData"
|
:height="height"
|
||||||
:height="height"
|
border
|
||||||
border
|
@header-dragend="dragend"
|
||||||
@header-dragend="dragend"
|
@sort-change="tableDataSort"
|
||||||
@sort-change="tableDataSort"
|
@selection-change="selectionChange"
|
||||||
@selection-change="selectionChange"
|
@row-dblclick="(row)=>{$emit('messageDetail', row)}"
|
||||||
@row-dblclick="(row)=>{$emit('messageDetail', row)}"
|
>
|
||||||
|
<el-table-column
|
||||||
|
:resizable="false"
|
||||||
|
align="center"
|
||||||
|
type="selection"
|
||||||
|
width="55">
|
||||||
|
</el-table-column>
|
||||||
|
<el-table-column
|
||||||
|
v-for="(item, index) in customTableTitle"
|
||||||
|
v-if="item.show"
|
||||||
|
:key="`col-${index}`"
|
||||||
|
:fixed="item.fixed"
|
||||||
|
:label="item.label"
|
||||||
|
:min-width="`${item.minWidth}`"
|
||||||
|
:prop="item.prop"
|
||||||
|
:resizable="true"
|
||||||
|
:sort-orders="['ascending', 'descending']"
|
||||||
|
:sortable="item.sortable"
|
||||||
|
:width="`${item.width}`"
|
||||||
>
|
>
|
||||||
<el-table-column
|
<template slot-scope="scope" :column="item">
|
||||||
:resizable="false"
|
<template v-if="item.prop === 'alertRule'">
|
||||||
align="center"
|
<div v-if="scope.row.alertRule.name" >
|
||||||
type="selection"
|
<span
|
||||||
width="55">
|
class="data-column__span"
|
||||||
</el-table-column>
|
@mouseenter="alertMessageHover(scope.row.alertRule, true, $event)"
|
||||||
<el-table-column
|
@mouseleave="alertMessageHover(scope.row.alertRule, false)"
|
||||||
v-for="(item, index) in customTableTitle"
|
>{{scope.row.alertRule.name}}</span>
|
||||||
v-if="item.show"
|
<alertRuleInfo v-if="scope.row.alertRule.loading" :id="scope.row.alertRule.id" :severity-data="severityData" :that="scope.row.alertRule"></alertRuleInfo>
|
||||||
:key="`col-${index}`"
|
</div>
|
||||||
:fixed="item.fixed"
|
|
||||||
:label="item.label"
|
|
||||||
:min-width="`${item.minWidth}`"
|
|
||||||
:prop="item.prop"
|
|
||||||
:resizable="true"
|
|
||||||
:sort-orders="['ascending', 'descending']"
|
|
||||||
:sortable="item.sortable"
|
|
||||||
:width="`${item.width}`"
|
|
||||||
>
|
|
||||||
<template slot-scope="scope" :column="item">
|
|
||||||
<template v-if="item.prop === 'alertRule'">
|
|
||||||
<div v-if="scope.row.alertRule.name" >
|
|
||||||
<span
|
|
||||||
class="data-column__span"
|
|
||||||
@mouseenter="alertMessageHover(scope.row.alertRule, true, $event)"
|
|
||||||
@mouseleave="alertMessageHover(scope.row.alertRule, false)"
|
|
||||||
>{{scope.row.alertRule.name}}</span>
|
|
||||||
<alertRuleInfo v-if="scope.row.alertRule.loading" :id="scope.row.alertRule.id" :severity-data="severityData" :that="scope.row.alertRule"></alertRuleInfo>
|
|
||||||
</div>
|
|
||||||
<template v-else>-</template>
|
|
||||||
</template>
|
|
||||||
<template v-else-if="item.prop === 'summary'">
|
|
||||||
<template v-if="scope.row[item.prop]">{{scope.row[item.prop]}}</template>
|
|
||||||
<template v-else>-</template>
|
|
||||||
</template>
|
|
||||||
<template v-else-if="item.prop === 'description'">
|
|
||||||
<template v-if="scope.row[item.prop]">{{scope.row[item.prop]}}</template>
|
|
||||||
<span v-else>-</span>
|
|
||||||
</template>
|
|
||||||
<template v-else-if="item.prop === 'remark'">
|
|
||||||
<template v-if="scope.row[item.prop]">{{scope.row[item.prop]}}</template>
|
|
||||||
<span v-else>-</span>
|
|
||||||
</template>
|
|
||||||
<span v-else-if="item.prop === 'severityId'&&scope.row['severity']" class="severity">
|
|
||||||
<i class="nz-icon nz-icon-circle" :style="{color:scope.row['severity'].color,'font-size':'12px','margin-right':'5px'}"></i> {{scope.row['severity'].name}}
|
|
||||||
</span>
|
|
||||||
<span v-else-if="item.prop === 'startAt'">{{utcTimeToTimezoneStr(scope.row.startAt)}}</span>
|
|
||||||
<template v-else-if="item.prop === 'duration'">
|
|
||||||
<el-tooltip :disabled="!scope.row.endAt" effect="light" placement="right">
|
|
||||||
<div slot="content">
|
|
||||||
{{$t('config.terminallog.endTime')}}<br/>
|
|
||||||
{{utcTimeToTimezoneStr(scope.row.endAt)}}
|
|
||||||
</div>
|
|
||||||
<span>{{getDuration(scope.row)}}</span>
|
|
||||||
</el-tooltip>
|
|
||||||
</template>
|
|
||||||
<template v-else-if="item.prop === 'labels'" class="labels">
|
|
||||||
<span v-for="(item, i) in labelsSort(scope.row.labels)" :key="i">
|
|
||||||
<span
|
|
||||||
@mouseenter="labelHover(scope.row, item.label, true, $event)"
|
|
||||||
@mouseleave="labelHover(scope.row, item.label, false)">
|
|
||||||
<nz-alert-tag
|
|
||||||
v-if="item.label !== 'alertname' && item.label !== 'severity'" :key="item.label" :cursor-point="tagType(item.label) !== 'info'"
|
|
||||||
:label="item.label"
|
|
||||||
:type="tagType(item.label)"
|
|
||||||
style="margin: 5px 0 5px 5px;"
|
|
||||||
>
|
|
||||||
{{item.value}}
|
|
||||||
</nz-alert-tag>
|
|
||||||
</span>
|
|
||||||
<alertLabel
|
|
||||||
v-if="(item.label === 'asset' ||item.label === 'module' || item.label === 'project'||item.label === 'endpoint' ||item.label === 'cpu' ||item.label === 'user' ||item.label === 'parent_asset') && scope.row[item.label] && scope.row[item.label].loading"
|
|
||||||
:id="scope.row[item.label].id"
|
|
||||||
:that="scope.row[item.label]"
|
|
||||||
:type="item.label"
|
|
||||||
></alertLabel>
|
|
||||||
</span>
|
|
||||||
</template>
|
|
||||||
<span v-else-if="item.prop === 'state'" :class="{'green': scope.row['state'] == 2, 'red': scope.row['state'] == 1}">
|
|
||||||
{{scope.row['state'] == 1 ? "Pending" : ""}}
|
|
||||||
{{scope.row['state'] == 2 ? "Expired" : ""}}
|
|
||||||
</span>
|
|
||||||
<span v-else-if="scope.row[item.prop]">{{scope.row[item.prop]}}</span>
|
|
||||||
<template v-else>-</template>
|
<template v-else>-</template>
|
||||||
</template>
|
</template>
|
||||||
</el-table-column>
|
<template v-else-if="item.prop === 'summary'">
|
||||||
<el-table-column
|
<template v-if="scope.row[item.prop]">{{scope.row[item.prop]}}</template>
|
||||||
v-if="showOption"
|
<template v-else>-</template>
|
||||||
:resizable="false"
|
</template>
|
||||||
:width="operationWidth"
|
<template v-else-if="item.prop === 'description'">
|
||||||
fixed="right">
|
<template v-if="scope.row[item.prop]">{{scope.row[item.prop]}}</template>
|
||||||
<div slot="header" class="table-operation-title">{{$t('overall.option')}}</div>
|
<span v-else>-</span>
|
||||||
<div slot-scope="scope" class="table-operation-items">
|
</template>
|
||||||
<button class="table-operation-item" @click="$emit('messageDetail', scope.row)"><i class="nz-icon nz-icon-view1"></i></button>
|
<template v-else-if="item.prop === 'remark'">
|
||||||
<el-dropdown v-has="['alertMessage_expired']" size="medium" trigger="hover" @command="tableOperation">
|
<template v-if="scope.row[item.prop]">{{scope.row[item.prop]}}</template>
|
||||||
<div class="table-operation-item table-operation-item--more">
|
<span v-else>-</span>
|
||||||
<i class="nz-icon nz-icon-more3"></i>
|
</template>
|
||||||
|
<span v-else-if="item.prop === 'severityId'&&scope.row['severity']" class="severity">
|
||||||
|
<i :style="{color:scope.row['severity'].color,'font-size':'12px','margin-right':'5px'}" class="nz-icon nz-icon-circle"></i> {{scope.row['severity'].name}}
|
||||||
|
</span>
|
||||||
|
<span v-else-if="item.prop === 'startAt'">{{utcTimeToTimezoneStr(scope.row.startAt)}}</span>
|
||||||
|
<template v-else-if="item.prop === 'duration'">
|
||||||
|
<el-tooltip :disabled="!scope.row.endAt" effect="light" placement="right">
|
||||||
|
<div slot="content">
|
||||||
|
{{$t('config.terminallog.endTime')}}<br/>
|
||||||
|
{{utcTimeToTimezoneStr(scope.row.endAt)}}
|
||||||
</div>
|
</div>
|
||||||
<el-dropdown-menu slot="dropdown">
|
<span>{{getDuration(scope.row)}}</span>
|
||||||
<el-dropdown-item v-has="'alertMessage_expired'" :command="['delete', scope.row]"><i class="nz-icon nz-icon-delete"></i><span class="operation-dropdown-text">{{$t('overall.delete')}}</span></el-dropdown-item>
|
</el-tooltip>
|
||||||
<el-dropdown-item v-has="'alertSilence_add'" :command="['fastSilence', scope.row, 'alertMessage']"><i class="nz-icon nz-icon-fast-silence"></i><span class="operation-dropdown-text">{{$t('overall.silenceAlert')}}</span></el-dropdown-item>
|
</template>
|
||||||
</el-dropdown-menu>
|
<template v-else-if="item.prop === 'labels'" class="labels">
|
||||||
</el-dropdown>
|
<span v-for="(item, i) in labelsSort(scope.row.labels)" :key="i">
|
||||||
</div>
|
<span
|
||||||
</el-table-column>
|
@mouseenter="labelHover(scope.row, item.label, true, $event)"
|
||||||
</el-table>
|
@mouseleave="labelHover(scope.row, item.label, false)">
|
||||||
</div>
|
<nz-alert-tag
|
||||||
|
v-if="item.label !== 'alertname' && item.label !== 'severity'" :key="item.label" :cursor-point="tagType(item.label) !== 'info'"
|
||||||
|
:label="item.label"
|
||||||
|
:type="tagType(item.label)"
|
||||||
|
style="margin: 5px 0 5px 5px;"
|
||||||
|
>
|
||||||
|
{{item.value}}
|
||||||
|
</nz-alert-tag>
|
||||||
|
</span>
|
||||||
|
<alertLabel
|
||||||
|
v-if="(item.label === 'asset' ||item.label === 'module' || item.label === 'project'||item.label === 'endpoint' ||item.label === 'cpu' ||item.label === 'user' ||item.label === 'parent_asset') && scope.row[item.label] && scope.row[item.label].loading"
|
||||||
|
:id="scope.row[item.label].id"
|
||||||
|
:that="scope.row[item.label]"
|
||||||
|
:type="item.label"
|
||||||
|
></alertLabel>
|
||||||
|
</span>
|
||||||
|
</template>
|
||||||
|
<span v-else-if="item.prop === 'state'" :class="{'green': scope.row['state'] == 2, 'red': scope.row['state'] == 1}">
|
||||||
|
{{scope.row['state'] == 1 ? "Pending" : ""}}
|
||||||
|
{{scope.row['state'] == 2 ? "Expired" : ""}}
|
||||||
|
</span>
|
||||||
|
<span v-else-if="scope.row[item.prop]">{{scope.row[item.prop]}}</span>
|
||||||
|
<template v-else>-</template>
|
||||||
|
</template>
|
||||||
|
</el-table-column>
|
||||||
|
<el-table-column
|
||||||
|
v-if="showOption"
|
||||||
|
:resizable="false"
|
||||||
|
:width="operationWidth"
|
||||||
|
fixed="right">
|
||||||
|
<div slot="header" class="table-operation-title">{{$t('overall.option')}}</div>
|
||||||
|
<div slot-scope="scope" class="table-operation-items">
|
||||||
|
<button class="table-operation-item" @click="$emit('messageDetail', scope.row)"><i class="nz-icon nz-icon-view1"></i></button>
|
||||||
|
<el-dropdown v-has="['alertMessage_expired']" size="medium" trigger="hover" @command="tableOperation">
|
||||||
|
<div class="table-operation-item table-operation-item--more">
|
||||||
|
<i class="nz-icon nz-icon-more3"></i>
|
||||||
|
</div>
|
||||||
|
<el-dropdown-menu slot="dropdown">
|
||||||
|
<el-dropdown-item v-has="'alertMessage_expired'" :command="['delete', scope.row]"><i class="nz-icon nz-icon-delete"></i><span class="operation-dropdown-text">{{$t('overall.delete')}}</span></el-dropdown-item>
|
||||||
|
<el-dropdown-item v-has="'alertSilence_add'" :command="['fastSilence', scope.row, 'alertMessage']"><i class="nz-icon nz-icon-fast-silence"></i><span class="operation-dropdown-text">{{$t('overall.silenceAlert')}}</span></el-dropdown-item>
|
||||||
|
</el-dropdown-menu>
|
||||||
|
</el-dropdown>
|
||||||
|
</div>
|
||||||
|
</el-table-column>
|
||||||
|
</el-table>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script>
|
<script>
|
||||||
|
|||||||
@@ -557,9 +557,7 @@ export default {
|
|||||||
}
|
}
|
||||||
.query-page-option{
|
.query-page-option{
|
||||||
width: 100%;
|
width: 100%;
|
||||||
position: absolute;
|
|
||||||
background: #fff;
|
background: #fff;
|
||||||
bottom: -34px;
|
|
||||||
}
|
}
|
||||||
/deep/ .pagination{
|
/deep/ .pagination{
|
||||||
padding-top: 0;
|
padding-top: 0;
|
||||||
|
|||||||
@@ -130,7 +130,7 @@
|
|||||||
</div>
|
</div>
|
||||||
</el-collapse-item>
|
</el-collapse-item>
|
||||||
<el-collapse-item v-if="showTab.indexOf('2') > -1" name="2" title="Logs">
|
<el-collapse-item v-if="showTab.indexOf('2') > -1" name="2" title="Logs">
|
||||||
<log-tab ref="logDetail" :log-data="logData" :unit="chartUnit" @exportLog="exportLog" @limitChange="limitChange"></log-tab>
|
<log-tab ref="logDetail" :log-data="logData" @exportLog="exportLog" @limitChange="queryLogData"></log-tab>
|
||||||
</el-collapse-item>
|
</el-collapse-item>
|
||||||
</template>
|
</template>
|
||||||
</el-collapse>
|
</el-collapse>
|
||||||
@@ -528,9 +528,6 @@ export default {
|
|||||||
reader.readAsText(error.response.data)
|
reader.readAsText(error.response.data)
|
||||||
})
|
})
|
||||||
},
|
},
|
||||||
limitChange (limit) {
|
|
||||||
this.queryLogData(limit)
|
|
||||||
},
|
|
||||||
queryLogData (limit) { // log的chart和table是一个请求
|
queryLogData (limit) { // log的chart和table是一个请求
|
||||||
if (!limit) {
|
if (!limit) {
|
||||||
limit = this.$refs.logDetail.getLimit()
|
limit = this.$refs.logDetail.getLimit()
|
||||||
|
|||||||
@@ -1,8 +1,6 @@
|
|||||||
<template>
|
<template>
|
||||||
<div class="log-detail">
|
<div class="log-detail">
|
||||||
<div id="logChart" class="log-chart">
|
<div id="logChart" class="log-chart"></div>
|
||||||
<!-- <chart ref="logChart" :unit="unit" chart-type="logBar" :show-toolbox="false"></chart>-->
|
|
||||||
</div>
|
|
||||||
<div class="log-operations">
|
<div class="log-operations">
|
||||||
<div class="log-operation">
|
<div class="log-operation">
|
||||||
<span class="operation-label">{{$t('overall.time')}}</span>
|
<span class="operation-label">{{$t('overall.time')}}</span>
|
||||||
@@ -79,7 +77,6 @@ import * as echarts from 'echarts'
|
|||||||
export default {
|
export default {
|
||||||
name: 'logTab',
|
name: 'logTab',
|
||||||
props: {
|
props: {
|
||||||
unit: Number,
|
|
||||||
logData: Array
|
logData: Array
|
||||||
},
|
},
|
||||||
computed: {
|
computed: {
|
||||||
@@ -296,6 +293,9 @@ export default {
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
applyFilter (allTableData, filter) {
|
applyFilter (allTableData, filter) {
|
||||||
|
if (allTableData || allTableData.length === 0) {
|
||||||
|
return { tableData: [], tableChartData: [] }
|
||||||
|
}
|
||||||
let data = [...allTableData]
|
let data = [...allTableData]
|
||||||
// 过滤level
|
// 过滤level
|
||||||
data = data.filter(d => {
|
data = data.filter(d => {
|
||||||
|
|||||||
Reference in New Issue
Block a user