2025-02-20 16:28:28 +08:00
|
|
|
|
<template>
|
|
|
|
|
<!-- 流程列表 -->
|
|
|
|
|
<y9Table :config="y9TableConfig" :filterConfig="filterOperaConfig" @on-curr-page-change="handlerCurrPage"
|
|
|
|
|
ref="filterRef" @on-page-size-change="handlerPageSize">
|
|
|
|
|
<template v-slot:slotSearch>
|
|
|
|
|
<el-button :size="fontSizeObj.buttonSize" :style="{ fontSize: fontSizeObj.baseFontSize }"
|
|
|
|
|
class="global-btn-main" type="primary" @click="search">
|
|
|
|
|
<i class="ri-search-line"></i>
|
|
|
|
|
<span>{{ $t('查询') }}</span>
|
|
|
|
|
</el-button>
|
|
|
|
|
<el-button :size="fontSizeObj.buttonSize" :style="{ fontSize: fontSizeObj.baseFontSize }"
|
|
|
|
|
class="el-button el-button--default global-btn-third" @click="reset">
|
|
|
|
|
<i class="ri-refresh-line"></i>
|
|
|
|
|
<span>{{ $t('重置') }}</span>
|
|
|
|
|
</el-button>
|
|
|
|
|
</template>
|
|
|
|
|
</y9Table>
|
|
|
|
|
<!-- 增加页面 -->
|
|
|
|
|
<y9Dialog v-model:config="addDialogConfig">
|
|
|
|
|
<template v-slot>
|
|
|
|
|
<y9Form ref="ruleFormRef" :config="ruleFormConfig">
|
|
|
|
|
</y9Form>
|
|
|
|
|
</template>
|
|
|
|
|
</y9Dialog>
|
|
|
|
|
|
2025-02-20 18:08:57 +08:00
|
|
|
|
<y9Dialog v-model:config="openChartConfig">
|
|
|
|
|
<template v-slot>
|
|
|
|
|
<el-card class="box-card">
|
|
|
|
|
<div class="card-header">
|
|
|
|
|
<span>{{ $t('服务使用概况') }}</span>
|
|
|
|
|
<div style="height: 32px;"></div>
|
|
|
|
|
</div>
|
|
|
|
|
<div class="overview-content">
|
|
|
|
|
<!-- 数值展示 -->
|
|
|
|
|
<div class="stat-items">
|
|
|
|
|
<div class="stat-item">
|
|
|
|
|
<h2>{{ serviceUsedData.registerCount }}</h2>
|
|
|
|
|
<span>{{ $t('注册接口数') }}</span>
|
|
|
|
|
</div>
|
|
|
|
|
<div class="stat-item">
|
|
|
|
|
<h2>{{ serviceUsedData.aveCount }}</h2>
|
|
|
|
|
<span>{{ $t('平均每天请求次数') }}</span>
|
|
|
|
|
</div>
|
|
|
|
|
<div class="stat-item">
|
|
|
|
|
<h2>{{ serviceUsedData.todayCount }}</h2>
|
|
|
|
|
<span>{{ $t('今日调用次数') }}</span>
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
</el-card>
|
|
|
|
|
<el-row class="bottomArea">
|
|
|
|
|
<el-col :span="4"></el-col>
|
|
|
|
|
<el-col :span="8">
|
|
|
|
|
<el-row class="labelTitle"><span>系统CPU使用率</span></el-row>
|
|
|
|
|
<el-row>
|
|
|
|
|
<div ref="cpuChartContainerRef" style="width: 100%; height: 200px;"></div>
|
|
|
|
|
</el-row>
|
|
|
|
|
</el-col>
|
|
|
|
|
<el-col :span="8">
|
|
|
|
|
<el-row class="labelTitle"><span>内存使用率</span></el-row>
|
|
|
|
|
<el-row>
|
|
|
|
|
<div ref="memoryChartContainerRef" style="width: 100%; height: 200px;"></div>
|
|
|
|
|
</el-row>
|
|
|
|
|
</el-col>
|
|
|
|
|
<el-col :span="4"></el-col>
|
|
|
|
|
</el-row>
|
|
|
|
|
</template>
|
|
|
|
|
</y9Dialog>
|
2025-02-20 16:28:28 +08:00
|
|
|
|
</template>
|
|
|
|
|
<script lang="ts" setup>
|
2025-02-20 18:08:57 +08:00
|
|
|
|
import { computed, h, ref, inject, nextTick } from 'vue';
|
2025-02-20 16:28:28 +08:00
|
|
|
|
import { useSettingStore } from '@/store/modules/settingStore';
|
|
|
|
|
import { useI18n } from 'vue-i18n';
|
|
|
|
|
import { $validCheck } from '@/utils/validate'
|
2025-02-20 18:08:57 +08:00
|
|
|
|
import { getPage, saveInfo, delInfoById } from '@/api/execute/execute'
|
|
|
|
|
import {getRegisterNum} from '@/api/interface/interface'
|
|
|
|
|
import { getCpuUsed, getMaxMemory, getUsedMemory,getHttpServerReqs } from '@/api/serviceResourceIndicators/serviceResourceIndicators'
|
|
|
|
|
import {getInvokeNumToday} from "@/api/home/home";
|
2025-02-20 16:28:28 +08:00
|
|
|
|
import { ElMessage, ElMessageBox, ElSwitch } from 'element-plus';
|
|
|
|
|
import '@/assets/css/tablestatusfontcolor.css';
|
2025-02-20 18:08:57 +08:00
|
|
|
|
import * as echarts from 'echarts';
|
|
|
|
|
import 'echarts-liquidfill';
|
2025-02-20 16:28:28 +08:00
|
|
|
|
|
|
|
|
|
// 注入 字体对象
|
|
|
|
|
const fontSizeObj: any = inject('sizeObjInfo');
|
|
|
|
|
const settingStore = useSettingStore();
|
|
|
|
|
const { t } = useI18n();
|
|
|
|
|
const selectedDate = ref();
|
|
|
|
|
const query: any = ref({});
|
|
|
|
|
const filterRef = ref();
|
|
|
|
|
const interfaceList = ref([]);
|
2025-02-20 18:08:57 +08:00
|
|
|
|
const cpuChartContainerRef = ref();
|
|
|
|
|
const memoryChartContainerRef = ref();
|
|
|
|
|
const serviceUsedData = ref({
|
|
|
|
|
registerCount:0,
|
|
|
|
|
todayCount:0,
|
|
|
|
|
aveCount:0
|
|
|
|
|
})
|
2025-02-20 16:28:28 +08:00
|
|
|
|
|
|
|
|
|
const limitInfo = ref(false)
|
|
|
|
|
const sameId = ref()
|
2025-02-20 18:08:57 +08:00
|
|
|
|
const qbaseUrl = ref()
|
2025-02-20 16:28:28 +08:00
|
|
|
|
|
|
|
|
|
//表格配置
|
|
|
|
|
let y9TableConfig = ref({
|
|
|
|
|
headerBackground: true,
|
|
|
|
|
pageConfig: false,
|
|
|
|
|
columns: [
|
|
|
|
|
{
|
|
|
|
|
type: 'index',
|
|
|
|
|
title: computed(() => t('序号')),
|
|
|
|
|
width: 80,
|
|
|
|
|
fixed: 'left'
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
title: computed(() => t('应用名称')),
|
|
|
|
|
key: 'name'
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
title: computed(() => t('实例ID')),
|
|
|
|
|
key: 'instanceId'
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
title: computed(() => t('域名')),
|
|
|
|
|
key: 'hostName'
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
title: computed(() => t('ip')),
|
|
|
|
|
key: 'ip'
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
title: computed(() => t('端口')),
|
|
|
|
|
key: 'port'
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
title: computed(() => t('状态')),
|
|
|
|
|
key: 'status',
|
|
|
|
|
render: (row) => {
|
|
|
|
|
let statusText = "-"
|
|
|
|
|
let textClass = ""
|
|
|
|
|
if(row.status.toUpperCase() == "UP"){
|
|
|
|
|
statusText="健康"
|
|
|
|
|
textClass = "successText"
|
|
|
|
|
}
|
|
|
|
|
if(row.status.toUpperCase() == "DOWN"){
|
|
|
|
|
statusText="宕机"
|
|
|
|
|
textClass = "stopText"
|
|
|
|
|
}
|
|
|
|
|
if(row.status.toUpperCase() == "STARTING"){
|
|
|
|
|
statusText="启动中"
|
|
|
|
|
}
|
|
|
|
|
if(row.status.toUpperCase() == "OUT_OF_SERVICE"){
|
|
|
|
|
statusText="离线"
|
|
|
|
|
textClass = "stopText"
|
|
|
|
|
}
|
|
|
|
|
if(row.status.toUpperCase() == "UNKNOWN"){
|
|
|
|
|
statusText="未知"
|
|
|
|
|
}
|
|
|
|
|
return h('span', { class: textClass }, t(statusText)); // 输出格式化后的日期和时间
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
title: computed(() => t('注册时间')),
|
|
|
|
|
key: 'createTime',
|
|
|
|
|
render: (row) => {
|
|
|
|
|
let date = new Date(row.createTime);
|
|
|
|
|
let year = date.getFullYear();
|
|
|
|
|
let month = (date.getMonth() + 1).toString().padStart(2, '0'); // 月份从0开始,需要加1并补零
|
|
|
|
|
let day = date.getDate().toString().padStart(2, '0'); // 日期补零
|
|
|
|
|
let hours = date.getHours().toString().padStart(2, '0'); // 小时补零
|
|
|
|
|
let minutes = date.getMinutes().toString().padStart(2, '0'); // 分钟补零
|
|
|
|
|
let seconds = date.getSeconds().toString().padStart(2, '0'); // 秒补零
|
|
|
|
|
|
|
|
|
|
let formattedDateTime = year + '-' + month + '-' + day + ' ' + hours + ':' + minutes + ':' + seconds;
|
|
|
|
|
return formattedDateTime; // 输出格式化后的日期和时间
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
title: computed(() => t('操作')),
|
|
|
|
|
fixed: 'right',
|
2025-02-20 18:08:57 +08:00
|
|
|
|
width: 210,
|
2025-02-20 16:28:28 +08:00
|
|
|
|
render: (row) => {
|
|
|
|
|
return h('div', [h('span', { onClick: () => { view(row) } }, t("详情")),
|
|
|
|
|
h('span', { class: 'leftMargin', onClick: () => { edit(row,"OUT_OF_SERVICE") } }, t('下线')),
|
|
|
|
|
h('span', { class: 'leftMargin', onClick: () => { edit(row,"UP") } }, t('上线')),
|
2025-02-20 18:08:57 +08:00
|
|
|
|
h('span', { class: 'leftMargin', onClick: () => { openChartView(row) } }, t('服务使用情况')),
|
2025-02-20 16:28:28 +08:00
|
|
|
|
]);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
],
|
|
|
|
|
tableData: [],
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
// 过滤条件
|
|
|
|
|
const filterOperaConfig = ref({
|
|
|
|
|
filtersValueCallBack: (filter) => {
|
|
|
|
|
query.value = filter;
|
|
|
|
|
},
|
|
|
|
|
itemList: [
|
|
|
|
|
{
|
|
|
|
|
type: 'input',
|
|
|
|
|
value: '',
|
|
|
|
|
key: 'name',
|
|
|
|
|
label: computed(() => t('名称')),
|
2025-02-20 18:08:57 +08:00
|
|
|
|
labelWidth: '42px',
|
|
|
|
|
span: settingStore.device === 'mobile' ? 24 : 6
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
type: 'input',
|
|
|
|
|
value: '',
|
|
|
|
|
key: 'ip',
|
|
|
|
|
label: computed(() => t('IP')),
|
|
|
|
|
labelWidth: '42px',
|
2025-02-20 16:28:28 +08:00
|
|
|
|
span: settingStore.device === 'mobile' ? 24 : 6
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
type: 'slot',
|
|
|
|
|
slotName: 'slotSearch',
|
|
|
|
|
span: 6
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
type: 'slot',
|
|
|
|
|
slotName: 'slotBtns',
|
|
|
|
|
span: settingStore.device === 'mobile' ? 24 : 12,
|
|
|
|
|
justify: 'flex-end'
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
],
|
|
|
|
|
showBorder: true,
|
|
|
|
|
borderRadio: '4px'
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
//获取已接入接口列表
|
|
|
|
|
async function getDataList(type) {
|
|
|
|
|
y9TableConfig.value.loading = true;
|
|
|
|
|
let res = await getPage(
|
|
|
|
|
query.value
|
|
|
|
|
)
|
|
|
|
|
let tableData = analysisData(res)
|
|
|
|
|
//遍历数组
|
|
|
|
|
let traverseData = tableData
|
|
|
|
|
//赋值数组
|
|
|
|
|
let fzData = []
|
|
|
|
|
if(type!=null){
|
|
|
|
|
for(let key in query.value){
|
|
|
|
|
if(fzData.length!=0){
|
|
|
|
|
traverseData = fzData
|
|
|
|
|
fzData = []
|
|
|
|
|
}
|
|
|
|
|
for(let row of traverseData){
|
|
|
|
|
if(row[key].indexOf(query.value[key])!=-1){
|
|
|
|
|
fzData.push(row)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if(fzData.length==0){
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
y9TableConfig.value.tableData = fzData || []
|
|
|
|
|
y9TableConfig.value.loading = false;
|
|
|
|
|
}else{
|
|
|
|
|
y9TableConfig.value.tableData = tableData || []
|
|
|
|
|
y9TableConfig.value.loading = false;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
//解析json数据
|
|
|
|
|
function analysisData(data){
|
|
|
|
|
//存储需要的列表数据
|
|
|
|
|
let tableData = []
|
|
|
|
|
//获取注册应用数据列表
|
|
|
|
|
let applications = data.applications.application
|
|
|
|
|
//解析转化为table数据
|
|
|
|
|
for(let application of applications){
|
|
|
|
|
let appName = application.name
|
|
|
|
|
//解析实例
|
|
|
|
|
for(let instance of application.instance){
|
|
|
|
|
let instanceData = {
|
|
|
|
|
name: appName,
|
|
|
|
|
instanceId: instance.instanceId,
|
|
|
|
|
hostName:instance.hostName,
|
|
|
|
|
ip: instance.ipAddr,
|
|
|
|
|
port: instance.port["$"],
|
|
|
|
|
status: instance.status,
|
2025-02-20 18:08:57 +08:00
|
|
|
|
createTime: instance.leaseInfo.registrationTimestamp,
|
|
|
|
|
baseUrl: instance.healthCheckUrl.slice(0,-6)
|
2025-02-20 16:28:28 +08:00
|
|
|
|
}
|
|
|
|
|
tableData.push(instanceData)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return tableData
|
|
|
|
|
}
|
|
|
|
|
//查询按钮
|
|
|
|
|
function search() {
|
|
|
|
|
y9TableConfig.value.tableData = []
|
|
|
|
|
getDataList(true)
|
|
|
|
|
}
|
|
|
|
|
//重置按钮
|
|
|
|
|
function reset() {
|
|
|
|
|
filterRef.value.elTableFilterRef.onReset()
|
|
|
|
|
y9TableConfig.value.tableData = []
|
|
|
|
|
query.value = {}
|
|
|
|
|
getDataList(null)
|
|
|
|
|
}
|
|
|
|
|
//初始化table数据
|
|
|
|
|
function initTableData() {
|
|
|
|
|
y9TableConfig.value.tableData = []
|
|
|
|
|
getDataList(null)
|
|
|
|
|
}
|
|
|
|
|
initTableData()
|
|
|
|
|
// 应用 添加 修改表单ref
|
|
|
|
|
const ruleFormRef = ref();
|
|
|
|
|
const validateNumber = (rule: any, value: any, callback: any) => {
|
|
|
|
|
let result = $validCheck('number', value, true);
|
|
|
|
|
if (!result.valid) {
|
|
|
|
|
callback(new Error(result.msg));
|
|
|
|
|
} else {
|
|
|
|
|
callback();
|
|
|
|
|
}
|
|
|
|
|
};
|
2025-02-20 18:08:57 +08:00
|
|
|
|
// 服务器使用情况
|
|
|
|
|
let openChartConfig = ref({
|
|
|
|
|
show: false,
|
|
|
|
|
title: computed(() => t('')),
|
|
|
|
|
showFooter: true,
|
|
|
|
|
onOkLoading: true,
|
|
|
|
|
onOk:(newConfig) =>{
|
|
|
|
|
openChartView(qbaseUrl.value)
|
|
|
|
|
}
|
|
|
|
|
});
|
2025-02-20 16:28:28 +08:00
|
|
|
|
|
|
|
|
|
// 增加 修改应用 弹框的变量配置 控制
|
|
|
|
|
let addDialogConfig = ref({
|
|
|
|
|
show: false,
|
|
|
|
|
title: computed(() => t('新增权限配置')),
|
|
|
|
|
showFooter: true,
|
|
|
|
|
onOkLoading: true,
|
|
|
|
|
onOk: (newConfig) => {
|
|
|
|
|
return new Promise(async (resolve, reject) => {
|
|
|
|
|
const y9RuleFormInstance = ruleFormRef.value?.elFormRef;
|
|
|
|
|
await y9RuleFormInstance.validate(async (valid) => {
|
|
|
|
|
if (valid) {
|
|
|
|
|
let data = ruleFormRef.value.model;
|
|
|
|
|
let formData = new FormData();
|
|
|
|
|
for (let key in data) {
|
|
|
|
|
if (data[key] != null && key != "createTime" && key != "updateTime")
|
|
|
|
|
formData.append(key, data[key])
|
|
|
|
|
}
|
|
|
|
|
let res = await saveInfo(
|
|
|
|
|
formData
|
|
|
|
|
)
|
|
|
|
|
if (res.code == 0) {
|
|
|
|
|
ElMessage({
|
|
|
|
|
message: '数据保存成功',
|
|
|
|
|
type: 'success'
|
|
|
|
|
})
|
|
|
|
|
resolve();
|
|
|
|
|
getDataList(null);
|
|
|
|
|
}
|
|
|
|
|
reject();
|
|
|
|
|
} else {
|
|
|
|
|
reject();
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
//校验ip格式
|
|
|
|
|
const validateIP = (rule: any, value: any, callback: any) => {
|
|
|
|
|
let regular = /^[0-9.,]+$/;
|
|
|
|
|
if(!regular.test(value)){
|
|
|
|
|
callback(new Error("输入了非法字符请检查"));
|
|
|
|
|
}else{
|
|
|
|
|
let ipList = value.split(",")
|
|
|
|
|
for(let it of ipList){
|
|
|
|
|
let ipItems = it.split(".")
|
|
|
|
|
if(ipItems.length!=4){
|
|
|
|
|
callback(new Error("IP"+it+"输入不合法请检查"));
|
|
|
|
|
}else{
|
|
|
|
|
for(let ipItem of ipItems){
|
|
|
|
|
if(ipItem!=null && ipItem!="" && ipItem.length!=0){
|
|
|
|
|
if(Number(ipItem)<0||Number(ipItem)>255){
|
|
|
|
|
callback(new Error("IP "+it+" 输入不合法请检查"));
|
|
|
|
|
}
|
|
|
|
|
}else{
|
|
|
|
|
callback(new Error("IP "+it+" 输入不合法请检查"));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
callback();
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
// 接口表单
|
|
|
|
|
let ruleFormConfig = ref({
|
|
|
|
|
//表单配置
|
|
|
|
|
model: {
|
|
|
|
|
isPrimary: 'Y',
|
|
|
|
|
isEnable: false
|
|
|
|
|
},
|
|
|
|
|
itemList: [
|
|
|
|
|
{
|
|
|
|
|
type: 'textarea',
|
|
|
|
|
label: computed(() => t('应用名称')),
|
|
|
|
|
prop: 'name',
|
|
|
|
|
props:{
|
|
|
|
|
rows:2
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
type: 'textarea',
|
|
|
|
|
label: computed(() => t('实例ID')),
|
|
|
|
|
prop: 'instanceId',
|
|
|
|
|
props:{
|
|
|
|
|
rows:2
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
type: 'textarea',
|
|
|
|
|
label: computed(() => t('域名')),
|
|
|
|
|
prop: 'hostName',
|
|
|
|
|
props:{
|
|
|
|
|
rows:2
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
type: 'input',
|
|
|
|
|
label: computed(() => t('IP')),
|
|
|
|
|
prop: 'ip',
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
type: 'input',
|
|
|
|
|
label: computed(() => t('端口')),
|
|
|
|
|
prop: 'port',
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
type: 'input',
|
|
|
|
|
label: computed(() => t('状态')),
|
|
|
|
|
prop: 'status'
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
type: 'input',
|
|
|
|
|
label: computed(() => t('注册时间')),
|
|
|
|
|
prop: 'createTime',
|
|
|
|
|
}
|
|
|
|
|
],
|
|
|
|
|
descriptionsFormConfig: {
|
|
|
|
|
labelWidth: '200px',
|
|
|
|
|
labelAlign: 'center',
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
//编辑
|
|
|
|
|
async function edit(row,type) {
|
|
|
|
|
let alertText = "下线"
|
|
|
|
|
if (type == "UP") {
|
|
|
|
|
alertText = "上线"
|
|
|
|
|
}
|
|
|
|
|
ElMessageBox.confirm(
|
|
|
|
|
'是否确认' + alertText + "该服务实例",
|
|
|
|
|
alertText+'确认',
|
|
|
|
|
{
|
|
|
|
|
confirmButtonText: '确定',
|
|
|
|
|
cancelButtonText: '取消',
|
|
|
|
|
type: 'info',
|
|
|
|
|
draggable: true
|
|
|
|
|
}
|
|
|
|
|
).then(() => {
|
|
|
|
|
let data = {
|
|
|
|
|
url: "/" + row.name + "/" + row.instanceId + "/status?value=" + type,
|
|
|
|
|
data: { "status": type }
|
|
|
|
|
}
|
|
|
|
|
saveInfo(data).then((res) => {
|
|
|
|
|
getDataList(null)
|
|
|
|
|
ElMessageBox.alert("实例正在" + alertText + ",请稍后刷新页面!", "实例" + alertText + "确认", {
|
|
|
|
|
confirmButtonText: '确认'
|
|
|
|
|
})
|
|
|
|
|
})
|
|
|
|
|
}).catch(() => {
|
|
|
|
|
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
//详情
|
|
|
|
|
async function view(data) {
|
|
|
|
|
let row = JSON.parse(JSON.stringify(data))
|
|
|
|
|
let statusText = "-"
|
|
|
|
|
if (row.status.toUpperCase() == "UP") {
|
|
|
|
|
statusText = "健康"
|
|
|
|
|
}
|
|
|
|
|
if (row.status.toUpperCase() == "DOWN") {
|
|
|
|
|
statusText = "宕机"
|
|
|
|
|
}
|
|
|
|
|
if (row.status.toUpperCase() == "STARTING") {
|
|
|
|
|
statusText = "启动中"
|
|
|
|
|
}
|
|
|
|
|
if (row.status.toUpperCase() == "OUT_OF_SERVICE") {
|
|
|
|
|
statusText = "离线"
|
|
|
|
|
}
|
|
|
|
|
if (row.status.toUpperCase() == "UNKNOWN") {
|
|
|
|
|
statusText = "未知"
|
|
|
|
|
}
|
|
|
|
|
row.status = statusText
|
|
|
|
|
let date = new Date(row.createTime);
|
|
|
|
|
let year = date.getFullYear();
|
|
|
|
|
let month = (date.getMonth() + 1).toString().padStart(2, '0'); // 月份从0开始,需要加1并补零
|
|
|
|
|
let day = date.getDate().toString().padStart(2, '0'); // 日期补零
|
|
|
|
|
let hours = date.getHours().toString().padStart(2, '0'); // 小时补零
|
|
|
|
|
let minutes = date.getMinutes().toString().padStart(2, '0'); // 分钟补零
|
|
|
|
|
let seconds = date.getSeconds().toString().padStart(2, '0'); // 秒补零
|
|
|
|
|
|
|
|
|
|
let formattedDateTime = year + '-' + month + '-' + day + ' ' + hours + ':' + minutes + ':' + seconds;
|
|
|
|
|
console.log(formattedDateTime)
|
|
|
|
|
row.createTime = formattedDateTime
|
|
|
|
|
ruleFormConfig.value.model = row
|
|
|
|
|
for (let it of ruleFormConfig.value.itemList) {
|
|
|
|
|
if (it.props == undefined) {
|
|
|
|
|
it.props = {
|
|
|
|
|
disabled: true
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
it.props.disabled = true
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
addDialogConfig.value.okText = false
|
|
|
|
|
addDialogConfig.value.title = computed(() => t('查看实例信息'))
|
|
|
|
|
addDialogConfig.value.show = true
|
|
|
|
|
}
|
2025-02-20 18:08:57 +08:00
|
|
|
|
const openChartView = (data)=>{
|
|
|
|
|
qbaseUrl.value = data
|
|
|
|
|
openChartConfig.value.okText = "刷新"
|
|
|
|
|
openChartConfig.value.title = computed(() => t('查看服务使用情况'))
|
|
|
|
|
openChartConfig.value.show = true
|
|
|
|
|
let baseData = {
|
|
|
|
|
baseUrl:data.baseUrl
|
|
|
|
|
}
|
|
|
|
|
getCpuUsed(baseData).then((res)=>{
|
|
|
|
|
console.log(res)
|
|
|
|
|
nextTick(() => {
|
|
|
|
|
const myChart = echarts.init(cpuChartContainerRef.value)
|
|
|
|
|
myChart.setOption(buildOptions(parseFloat(res.measurements[0].value.toFixed(4))))
|
2025-02-20 16:28:28 +08:00
|
|
|
|
})
|
2025-02-20 18:08:57 +08:00
|
|
|
|
})
|
|
|
|
|
getMaxMemory(baseData).then((res) => {
|
|
|
|
|
let maxMemory = res.measurements[0].value
|
|
|
|
|
getUsedMemory(baseData).then((res) => {
|
|
|
|
|
let usedMemory = res.measurements[0].value;
|
|
|
|
|
let used = (usedMemory/maxMemory).toFixed(4);
|
|
|
|
|
nextTick(() => {
|
|
|
|
|
const memoryChart = echarts.init(memoryChartContainerRef.value)
|
|
|
|
|
memoryChart.setOption(buildOptions(parseFloat(used)))
|
|
|
|
|
})
|
|
|
|
|
})
|
|
|
|
|
})
|
2025-02-20 16:28:28 +08:00
|
|
|
|
|
2025-02-20 18:08:57 +08:00
|
|
|
|
let param = {
|
|
|
|
|
id:data.instanceId
|
|
|
|
|
}
|
|
|
|
|
getRegisterNum(param).then((res)=>{
|
|
|
|
|
serviceUsedData.value.registerCount = res.data[0].num
|
2025-02-20 16:28:28 +08:00
|
|
|
|
})
|
2025-02-20 18:08:57 +08:00
|
|
|
|
getInvokeNumToday(param).then((res)=>{
|
|
|
|
|
serviceUsedData.value.todayCount = res.data
|
|
|
|
|
})
|
|
|
|
|
getHttpServerReqs(baseData).then((res)=>{
|
|
|
|
|
let measurements = res.measurements;
|
|
|
|
|
for(let it of measurements){
|
|
|
|
|
if(it.statistic.toUpperCase()=="COUNT"){
|
|
|
|
|
serviceUsedData.value.aveCount = calculateAverageDailyCallCount(data.createTime,it.value)
|
|
|
|
|
break;
|
|
|
|
|
}
|
2025-02-20 16:28:28 +08:00
|
|
|
|
}
|
2025-02-20 18:08:57 +08:00
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
// 计算平均每天调用次数
|
|
|
|
|
const calculateAverageDailyCallCount = (registrationDate,totalCallCount) => {
|
|
|
|
|
const now = new Date();
|
|
|
|
|
const registrationDateObj = new Date(registrationDate);
|
|
|
|
|
const daysSinceRegistration = (now - registrationDateObj) / (1000 * 60 * 60 * 24);
|
|
|
|
|
if(daysSinceRegistration<1){
|
|
|
|
|
return totalCallCount;
|
2025-02-20 16:28:28 +08:00
|
|
|
|
}
|
2025-02-20 18:08:57 +08:00
|
|
|
|
const averageDailyCallCount = totalCallCount / daysSinceRegistration;
|
|
|
|
|
return averageDailyCallCount.toFixed(2); // 保留两位小数
|
|
|
|
|
};
|
2025-02-20 16:28:28 +08:00
|
|
|
|
|
2025-02-20 18:08:57 +08:00
|
|
|
|
function Pie() {
|
|
|
|
|
let dataArr = [];
|
|
|
|
|
for (var i = 0; i < 150; i++) {
|
|
|
|
|
if (i % 2 === 0) {
|
|
|
|
|
dataArr.push({
|
|
|
|
|
name: (i + 1).toString(),
|
|
|
|
|
value: 50,
|
|
|
|
|
itemStyle: {
|
|
|
|
|
normal: {
|
|
|
|
|
color: "#00AFFF",
|
|
|
|
|
borderWidth: 0,
|
|
|
|
|
borderColor: "rgba(0,0,0,0)",
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
})
|
|
|
|
|
} else {
|
|
|
|
|
dataArr.push({
|
|
|
|
|
name: (i + 1).toString(),
|
|
|
|
|
value: 100,
|
|
|
|
|
itemStyle: {
|
|
|
|
|
normal: {
|
|
|
|
|
color: "rgba(0,0,0,0)",
|
|
|
|
|
borderWidth: 0,
|
|
|
|
|
borderColor: "rgba(0,0,0,0)"
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return dataArr
|
|
|
|
|
}
|
|
|
|
|
function buildOptions(usedData){
|
|
|
|
|
let options = {
|
|
|
|
|
backgroundColor: 'transparent', // 画布背景色
|
|
|
|
|
series: [
|
|
|
|
|
{
|
|
|
|
|
// value: 50, // 内容 配合formatter
|
|
|
|
|
type: 'liquidFill',
|
|
|
|
|
radius: '70%', // 控制中间圆球的尺寸(此处可以理解为距离外圈圆的距离控制)
|
|
|
|
|
center: ['50%', '50%'],
|
|
|
|
|
data: [usedData, {
|
|
|
|
|
value: usedData,
|
|
|
|
|
direction: 'left', //波浪方向
|
|
|
|
|
}], // data个数代表波浪数
|
|
|
|
|
backgroundStyle: {
|
|
|
|
|
borderWidth: 1,
|
|
|
|
|
color: 'rgba(62, 208, 255, 1)' // 球体本景色
|
|
|
|
|
},
|
|
|
|
|
amplitude: '6 %',//波浪的振幅
|
|
|
|
|
// 修改波浪颜色
|
|
|
|
|
// color: ['#0286ea', 'l#0b99ff'], // 每个波浪不同颜色,颜色数组长度为对应的波浪个数
|
|
|
|
|
color: [{
|
|
|
|
|
type: 'linear',
|
|
|
|
|
x: 0,
|
|
|
|
|
y: 0,
|
|
|
|
|
x2: 0,
|
|
|
|
|
y2: 1,
|
|
|
|
|
colorStops: [
|
|
|
|
|
{
|
|
|
|
|
offset: 1,
|
|
|
|
|
color: '#6CDEFC',
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
offset: 0,
|
|
|
|
|
color: '#429BF7',
|
|
|
|
|
},
|
|
|
|
|
],
|
|
|
|
|
globalCoord: false,
|
|
|
|
|
},
|
|
|
|
|
],
|
|
|
|
|
label: {
|
|
|
|
|
normal: {
|
|
|
|
|
formatter: (usedData * 100).toFixed(2)+'%',
|
|
|
|
|
// formatter: function(params){
|
|
|
|
|
// return params.value* 100 + " \n%";
|
|
|
|
|
// },
|
|
|
|
|
rich: {
|
|
|
|
|
d: {
|
|
|
|
|
fontSize: 20,
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
textStyle: {
|
|
|
|
|
fontSize: 32,
|
|
|
|
|
color: '#fff'
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
},
|
|
|
|
|
outline: {
|
|
|
|
|
show: false
|
2025-02-20 16:28:28 +08:00
|
|
|
|
}
|
2025-02-20 18:08:57 +08:00
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
type: 'pie',
|
|
|
|
|
z: 1,
|
|
|
|
|
center: ['50%', '50%'],
|
|
|
|
|
radius: ['72%', '73.5%'], // 控制外圈圆的粗细
|
|
|
|
|
hoverAnimation: false,
|
|
|
|
|
data: [
|
|
|
|
|
{
|
|
|
|
|
name: '',
|
|
|
|
|
value: 500,
|
|
|
|
|
labelLine: {
|
|
|
|
|
show: false
|
|
|
|
|
},
|
|
|
|
|
itemStyle: {
|
|
|
|
|
color: '#00AFFF'
|
|
|
|
|
},
|
|
|
|
|
emphasis: {
|
|
|
|
|
labelLine: {
|
|
|
|
|
show: false
|
|
|
|
|
},
|
|
|
|
|
itemStyle: {
|
|
|
|
|
color: '#00AFFF'
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
]
|
|
|
|
|
},
|
|
|
|
|
{ //外发光
|
|
|
|
|
type: 'pie',
|
|
|
|
|
z: 1,
|
|
|
|
|
zlevel: -1,
|
|
|
|
|
radius: ['70%', '90.5%'],
|
|
|
|
|
center: ["50%", "50%"],
|
|
|
|
|
hoverAnimation: false,
|
|
|
|
|
clockWise: false,
|
|
|
|
|
itemStyle: {
|
|
|
|
|
normal: {
|
|
|
|
|
borderWidth: 20,
|
|
|
|
|
color: 'rgba(224,242,255,1)',
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
label: {
|
|
|
|
|
show: false
|
|
|
|
|
},
|
|
|
|
|
data: [100]
|
|
|
|
|
},
|
|
|
|
|
{ //底层外发光
|
|
|
|
|
type: 'pie',
|
|
|
|
|
z:1,
|
|
|
|
|
zlevel: -2,
|
|
|
|
|
radius: ['70%', '100%'],
|
|
|
|
|
center: ["50%", "50%"],
|
|
|
|
|
hoverAnimation: false,
|
|
|
|
|
clockWise: false,
|
|
|
|
|
itemStyle: {
|
|
|
|
|
normal: {
|
|
|
|
|
borderWidth: 20,
|
|
|
|
|
color: 'rgba(224,242,255,.4)',
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
label: {
|
|
|
|
|
show: false
|
|
|
|
|
},
|
|
|
|
|
data: [100]
|
|
|
|
|
},
|
|
|
|
|
// 虚点
|
|
|
|
|
{
|
|
|
|
|
type: 'pie',
|
|
|
|
|
zlevel: 0,
|
|
|
|
|
silent: true,
|
|
|
|
|
radius: ['78%', '80%'],
|
|
|
|
|
z: 1,
|
|
|
|
|
label: {
|
|
|
|
|
normal: {
|
|
|
|
|
show: false
|
|
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
labelLine: {
|
|
|
|
|
normal: {
|
|
|
|
|
show: false
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
data: Pie()
|
|
|
|
|
},
|
|
|
|
|
]
|
|
|
|
|
}
|
|
|
|
|
return options
|
2025-02-20 16:28:28 +08:00
|
|
|
|
}
|
|
|
|
|
</script>
|
|
|
|
|
<style>
|
|
|
|
|
.leftMargin {
|
|
|
|
|
margin-left: 5px;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.operate {
|
|
|
|
|
cursor: pointer;
|
|
|
|
|
}
|
2025-02-20 18:08:57 +08:00
|
|
|
|
</style>
|
|
|
|
|
<style scoped>
|
|
|
|
|
.box-card {
|
|
|
|
|
border-radius: 4px;
|
|
|
|
|
border: 1px solid var(--el-border-color-lighter);
|
|
|
|
|
}
|
|
|
|
|
:deep(.el-card__header) {
|
|
|
|
|
padding: 12px 20px;
|
|
|
|
|
border-bottom: 1px solid var(--el-border-color-lighter);
|
|
|
|
|
background-color: var(--el-bg-color);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
:deep(.el-card__body) {
|
|
|
|
|
height: 130px !important;
|
|
|
|
|
padding: 0px 15px 15px 15px !important;
|
|
|
|
|
}
|
|
|
|
|
.overview-content {
|
|
|
|
|
height: calc(100% - 21px);
|
|
|
|
|
display: flex;
|
|
|
|
|
padding: 0px 20px;
|
|
|
|
|
|
|
|
|
|
.stat-items {
|
|
|
|
|
width: 100%;
|
|
|
|
|
display: flex;
|
|
|
|
|
justify-content: space-around;
|
|
|
|
|
align-items: center;
|
|
|
|
|
|
|
|
|
|
.stat-item {
|
|
|
|
|
text-align: center;
|
|
|
|
|
min-width: 80px;
|
|
|
|
|
|
|
|
|
|
h2 {
|
|
|
|
|
font-size: 45px;
|
|
|
|
|
color: var(--el-color-primary);
|
|
|
|
|
margin-bottom: 6px;
|
|
|
|
|
font-family: yjsz;
|
|
|
|
|
font-weight: 400;
|
|
|
|
|
margin-top: 0px;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
span {
|
|
|
|
|
font-size: 14px;
|
|
|
|
|
color: #666;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.card-header {
|
|
|
|
|
display: flex;
|
|
|
|
|
justify-content: space-between;
|
|
|
|
|
align-items: center;
|
|
|
|
|
padding: 8px 20px;
|
|
|
|
|
border-bottom: 1px solid #ebeef5;
|
|
|
|
|
white-space: nowrap;
|
|
|
|
|
|
|
|
|
|
span {
|
|
|
|
|
font-size: 15px;
|
|
|
|
|
font-weight: 500;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
:deep(.el-pagination) {
|
|
|
|
|
margin: 0;
|
|
|
|
|
padding: 0;
|
|
|
|
|
|
|
|
|
|
.el-pagination__jump {
|
|
|
|
|
display: none;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.btn-prev,
|
|
|
|
|
.btn-next {
|
|
|
|
|
min-width: 22px;
|
|
|
|
|
height: 22px;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.el-pager {
|
|
|
|
|
li {
|
|
|
|
|
min-width: 22px;
|
|
|
|
|
height: 22px;
|
|
|
|
|
line-height: 22px;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
.labelTitle{
|
|
|
|
|
text-align: center;
|
|
|
|
|
display: flex; justify-content: center;
|
|
|
|
|
span {
|
|
|
|
|
font-size: 16px;
|
|
|
|
|
color: #666;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
.bottomArea{
|
|
|
|
|
margin-top: 10px;
|
|
|
|
|
}
|
2025-02-20 16:28:28 +08:00
|
|
|
|
</style>
|