wuqun 2 years ago
parent
commit
cce3bf506f
  1. 4
      api/.vscode/launch.json
  2. 89
      api/app/lib/controllers/control/analysis.js
  3. 5
      api/app/lib/routes/control/index.js
  4. 14
      api/app/lib/schedule/alarms_push.js
  5. 48
      web/client/src/sections/control/actions/control.js
  6. 264
      web/client/src/sections/control/containers/control.jsx
  7. 5
      web/client/src/utils/webapi.js

4
api/.vscode/launch.json

@ -16,9 +16,9 @@
"-p 4600", "-p 4600",
"-f http://localhost:4600", "-f http://localhost:4600",
// //
"-g postgres://postgres:123@10.8.30.32:5432/orational_service", // "-g postgres://postgres:123@10.8.30.32:5432/orational_service",
// //
// "-g postgres://FashionAdmin:123456@10.8.30.156:5432/POMS", "-g postgres://FashionAdmin:123456@10.8.30.156:5432/POMS",
"-k 10.8.30.72:29092,10.8.30.73:29092,10.8.30.74:29092", "-k 10.8.30.72:29092,10.8.30.73:29092,10.8.30.74:29092",
"--iotaProxy http://10.8.30.157:17007", "--iotaProxy http://10.8.30.157:17007",
"--redisHost 10.8.30.112", "--redisHost 10.8.30.112",

89
api/app/lib/controllers/control/analysis.js

@ -1,86 +1,6 @@
'use strict'; 'use strict';
const moment = require('moment'); const moment = require('moment');
async function dataList (ctx) {
try {
const { models } = ctx.fs.dc;
const { userId, pepUserId, userInfo = {}, pepUserInfo } = ctx.fs.api
const { clickHouse } = ctx.app.fs
const { utils: { judgeSuper, anxinStrucIdRange, pomsProjectRange } } = ctx.app.fs
const { database: anxinyun } = clickHouse.anxinyun.opts.config
const { pepProjectId } = ctx.request.query
let anxinStruc = await anxinStrucIdRange({
ctx, pepProjectId
})
let pomsProject = await pomsProjectRange({
ctx, pepProjectId,
})
const pomsProjectIds = pomsProject.map(p => p.id)
if (anxinStruc.length) {
const anxinStrucIds = anxinStruc.map(a => a.strucId) || []
const dataAlarm = await clickHouse.dataAlarm.query(`
SELECT
formatDateTime(alarmData.StartTime,'%F %H') hours, count(AlarmId) count
FROM
( SELECT
AlarmId,State,StartTime
FROM
alarms
WHERE
alarms.StructureId IN (${anxinStrucIds.join(",")})
AND
AlarmGroup = 3) AS alarmData
GROUP BY hours
`).toPromise();
// const confirmedAlarm = dataAlarm
// // TODO: 开发临时注释
// .filter(ar => ar.State && ar.State > 2)
// .map(ar => "'" + ar.AlarmId + "'")
// // formatDateTime(Time,'%F %H') hours, count(AlarmId) count
// const dataConfirme = confirmedAlarm.length ?
// await clickHouse.dataAlarm.query(`
// SELECT
// max(Time) AS Time, AlarmId
// FROM
// alarm_details
// WHERE
// AlarmId IN (${confirmedAlarm.join(',')})
// GROUP BY AlarmId
// `).toPromise() :
// [];
// dataAlarm.forEach(ar => {
// ar.confirme =
// dataConfirme.find(as => as.AlarmId == ar.AlarmId) || {}
// })
ctx.status = 200
ctx.body = dataAlarm
} else {
ctx.status = 200
ctx.body = {
dataAlarm: 0,
}
}
} catch (error) {
ctx.fs.logger.error(`path: ${ctx.path}, error: ${error}`);
ctx.status = 400;
ctx.body = {
message: typeof error == 'string' ? error : undefined
}
}
}
async function personnelApp (ctx) { async function personnelApp (ctx) {
try { try {
const models = ctx.fs.dc.models; const models = ctx.fs.dc.models;
@ -219,7 +139,7 @@ async function personnelApp (ctx) {
findOptions.where.id = { $in: userInfo.correlationProject } findOptions.where.id = { $in: userInfo.correlationProject }
} }
if (pepId) { if (pepId) {
findOption.where.id = pepId findOptions.where.id = pepId
} }
const proRes = await models.ProjectCorrelation.findAndCountAll(findOptions) const proRes = await models.ProjectCorrelation.findAndCountAll(findOptions)
@ -267,7 +187,7 @@ async function personnelApp (ctx) {
let appproRes = proRes.rows.filter(v => v.dataValues.pepProjectIsDelete != 1).map(r => { let appproRes = proRes.rows.filter(v => v.dataValues.pepProjectIsDelete != 1).map(r => {
if (r.dataValues.apps.length > 0) { if (r.dataValues.apps.length > 0) {
r.dataValues.apps.map(vv => { r.dataValues.apps.map(vv => {
if (webApp.map(n => n.name).indexOf(vv.dataValues.name)) { if (!webApp.map(n => n.name).includes(vv.dataValues.name)) {
webApp.push({ name: vv.dataValues.name, url: vv.dataValues.url }) webApp.push({ name: vv.dataValues.name, url: vv.dataValues.url })
} }
}) })
@ -283,7 +203,7 @@ async function personnelApp (ctx) {
ctx.status = 200 ctx.status = 200
ctx.body = { ctx.body = {
personnel: personnel.map(v => v.dataValues.name), personnel: personnel.map(v => ({ name: v.dataValues.name, department: v.dataValues.departments.map(r => r.name) })),
webApp: webApp webApp: webApp
} }
} catch (error) { } catch (error) {
@ -471,7 +391,7 @@ async function problem (ctx) {
ctx.status = 200; ctx.status = 200;
ctx.body = sum ctx.body = sum
} else { } else {
ctx.body =[] ctx.body = []
} }
ctx.status = 200; ctx.status = 200;
} catch (error) { } catch (error) {
@ -485,7 +405,6 @@ async function problem (ctx) {
module.exports = { module.exports = {
dataList,
personnelApp, personnelApp,
problem problem
} }

5
api/app/lib/routes/control/index.js

@ -14,12 +14,9 @@ module.exports = function (app, router, opts) {
app.fs.api.logAttr['DEL/console/toollink'] = { content: '删除常用工具', visible: true }; app.fs.api.logAttr['DEL/console/toollink'] = { content: '删除常用工具', visible: true };
router.del('/console/toollink/:linkId', toolLink.del); router.del('/console/toollink/:linkId', toolLink.del);
app.fs.api.logAttr['GET/console/count'] = { content: '查询告警数量', visible: true }; app.fs.api.logAttr['GET/console/count'] = { content: '工作台查询告警数量', visible: true };
router.get('/console/count', toolLink.count); router.get('/console/count', toolLink.count);
app.fs.api.logAttr['GET/analysis/dataList'] = { content: '查询数据告警产生,确认数量', visible: true };
router.get('/analysis/dataList', analysis.dataList);
app.fs.api.logAttr['GET/analysis/userlist'] = { content: '查询关联人员,web应用', visible: true }; app.fs.api.logAttr['GET/analysis/userlist'] = { content: '查询关联人员,web应用', visible: true };
router.get('/analysis/userlist', analysis.personnelApp); router.get('/analysis/userlist', analysis.personnelApp);

14
api/app/lib/schedule/alarms_push.js

@ -120,7 +120,8 @@ module.exports = function (app, opts) {
return s.id return s.id
}) })
searchStrucIds = searchStrucIds.concat([991, 1052, 700]) // 开发测试用的数据
// searchStrucIds = searchStrucIds.concat([991, 1052, 700])
if (searchStrucIds.length) { if (searchStrucIds.length) {
searchStrucIds.unshift(-1) searchStrucIds.unshift(-1)
@ -377,7 +378,7 @@ module.exports = function (app, opts) {
dataAlarms = await clickHouse.dataAlarm.query(` dataAlarms = await clickHouse.dataAlarm.query(`
SELECT * FROM alarms SELECT * FROM alarms
WHERE WHERE
${`'State NOT IN (3, 4) AND '`} ${`State NOT IN (3, 4) AND `}
StructureId IN (${searchStrucIds.join(',')}) StructureId IN (${searchStrucIds.join(',')})
${dataAlarmOption.length ? ' AND ' + dataAlarmOption.join(' AND ') : ''} ${dataAlarmOption.length ? ' AND ' + dataAlarmOption.join(' AND ') : ''}
ORDER BY StartTime DESC ORDER BY StartTime DESC
@ -602,8 +603,13 @@ module.exports = function (app, opts) {
deviceStatistic.add(d.SourceId) deviceStatistic.add(d.SourceId)
} }
if (c.tactics == 'abnormal_rate') { if (c.tactics == 'abnormal_rate') {
let a = ((deviceStatistic.size + videoAlarms.length) / (parseInt(deviceCount) + parseInt(cameraCount))).toFixed(1) + '%' let rate = ((deviceStatistic.size + videoAlarms.length) / (parseInt(deviceCount) + parseInt(cameraCount)));
emailSubTitle = emailSubTitle.replace('--%', ((deviceStatistic.size + videoAlarms.length) / (parseInt(deviceCount) + parseInt(cameraCount))).toFixed(1) + '%')
if (rate < parseFloat(deviceProportion)) {
continue
}
emailSubTitle = emailSubTitle.replace('--%', rate.toFixed(1) + '%')
} }
let html = ` let html = `
<html> <html>

48
web/client/src/sections/control/actions/control.js

@ -47,6 +47,18 @@ export function getConsoleCount (query) { //工作台数量查询
}); });
} }
export function getConsoleUser (query) { //查询关联人员,web应用
return dispatch => basicAction({
type: 'get',
dispatch: dispatch,
query,
actionType: 'GET_CONSLE_USER',
url: `${ApiTable.getConsoleUser}`,
msg: { option: '查询关联人员,web应用' },
reducer: { name: '' }
});
}
export function getConsoleAbnormal (query) { //项目概览异常查询 export function getConsoleAbnormal (query) { //项目概览异常查询
return dispatch => basicAction({ return dispatch => basicAction({
type: 'get', type: 'get',
@ -58,3 +70,39 @@ export function getConsoleAbnormal (query) { //项目概览异常查询
reducer: { name: '' } reducer: { name: '' }
}); });
} }
export function getDataAlarmsAggDay (query) { //查询BI分析数据-数据
return dispatch => basicAction({
type: 'get',
dispatch: dispatch,
query,
actionType: 'GET_DATA_ALARMS_AGG_DAY',
url: `${ApiTable.getDataAlarmsAggDay}`,
msg: { option: '查询BI分析数据' },
reducer: { name: '' }
});
}
export function getVideoAlarmsAggDay (query) { //查询BI分析数据-视频异常
return dispatch => basicAction({
type: 'get',
dispatch: dispatch,
query,
actionType: 'GET_VIDEO_ALARMS_AGG_DAY',
url: `${ApiTable.getVideoAlarmsAggDay}`,
msg: { option: '查询BI分析视频数据' },
reducer: { name: '' }
});
}
export function getAppAlarmsAggDay (query) { //查询BI分析数据-应用
return dispatch => basicAction({
type: 'get',
dispatch: dispatch,
query,
actionType: 'GET_APP_ALARMS_AGG_DAY',
url: `${ApiTable.getAppAlarmsAggDay}`,
msg: { option: '查询BI分析应用数据' },
reducer: { name: '' }
});
}

264
web/client/src/sections/control/containers/control.jsx

@ -7,7 +7,7 @@ import PerfectScrollbar from "perfect-scrollbar";
import repairFQA from '../../means/containers/repairFQA'; import repairFQA from '../../means/containers/repairFQA';
import { Setup, OutHidden } from "$components"; import { Setup, OutHidden } from "$components";
import ReactECharts from 'echarts-for-react'; import ReactECharts from 'echarts-for-react';
const { Meta } = Card; import moment from "moment";
let newScrollbar; let newScrollbar;
let overviewScrollbar; let overviewScrollbar;
@ -21,21 +21,15 @@ let alarmScrollbar;
const Control = (props) => { const Control = (props) => {
const { dispatch, actions, user, loading, socket, pepProjectId } = props const { dispatch, actions, user, loading, socket, pepProjectId } = props
const { control } = actions const { control } = actions
const stationList = [
'url(/assets/images/console/lan_1.png)',
'url(/assets/images/console/lv_1.png)',
'url(/assets/images/console/huang_1.png)',
'url(/assets/images/console/hong_1.png)',
]
const [timelineList, setTimelineList] = useState(['', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', ''])// const [timelineList, setTimelineList] = useState(['', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', ''])//
const [memberList, setMemberList] = useState(['', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', ''])// const [memberList, setMemberList] = useState([])//
const [equipmentList, setEquipmentList] = useState(['', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', ''])// const [equipmentList, setEquipmentList] = useState(['', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', ''])//
const [webList, setWebList] = useState(['', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', ''])//web const [webList, setWebList] = useState([])//web
const [problemsList, setProblemsList] = useState(['', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', ''])//& const [problemsList, setProblemsList] = useState([])//&
const [setup, setSetup] = useState(false); // const [setup, setSetup] = useState(false); //
const [tableType, setTableType] = useState(''); //localStorage const [tableType, setTableType] = useState(''); //localStorage
const [tool, setTool] = useState(false); // const [tool, setTool] = useState(false); //
@ -43,7 +37,10 @@ const Control = (props) => {
const [compile, setCompile] = useState({}); // const [compile, setCompile] = useState({}); //
const [toolShow, setToolShow] = useState([]); // const [toolShow, setToolShow] = useState([]); //
const [tableSetup, setTableSetup] = useState([]); // const [tableSetup, setTableSetup] = useState([]); //
const [workData, setWorkData] = useState(); // const [workData, setWorkData] = useState({}); //
const [dataBI, setDataBI] = useState({}); //BI-
const [videoBI, setVideoBI] = useState([]); //BI-
const [appBI, setAppBI] = useState([]); //BI-
const exhibition = useRef({ workbench: [], statistical: [] }) // const exhibition = useRef({ workbench: [], statistical: [] }) //
const FormApi = useRef() const FormApi = useRef()
@ -69,22 +66,60 @@ const Control = (props) => {
data.map(v => { data.map(v => {
localStorage.getItem(v) == null localStorage.getItem(v) == null
? localStorage.setItem(v, JSON.stringify(show[v])) ? localStorage.setItem(v, JSON.stringify(show[v]))
: ""; : ""
attribute(v) attribute(v)
}) })
}, []) }, [])
useEffect(() => { useEffect(() => {
//
dispatch(control.getConsoleCount({ pepProjectId: pepProjectId })).then(res => { dispatch(control.getConsoleCount({ pepProjectId: pepProjectId })).then(res => {
if (res.success) setWorkData(res.payload.data) if (res.success) setWorkData(res.payload.data)
}) })
// --&
dispatch(control.getConsoleAbnormal({ pepProjectId: pepProjectId })).then(res => { dispatch(control.getConsoleAbnormal({ pepProjectId: pepProjectId })).then(res => {
console.log(res.payload.data); if (res.success) {
if (res.success) setProblemsList([...res.payload.data,...res.payload.data]) if (res.payload.data?.length > 4) {
setProblemsList([...res.payload.data, ...res.payload.data])
startmarquee(500, 2000, 'problems')
} else {
setProblemsList(res.payload.data)
}
}
})
// --web
dispatch(control.getConsoleUser({ pepId: pepProjectId })).then(res => {
if (res.success) {
if (res.payload.data?.personnel?.length > 5) {
setMemberList([...res.payload.data?.personnel, ...res.payload.data?.personnel])
startmarquee(600, 2000, 'member')
} else {
setMemberList(res.payload.data?.personnel)
}
if (res.payload.data?.webApp?.length > 3) {
setWebList([...res.payload.data?.webApp, ...res.payload.data?.webApp])
startmarquee(600, 2000, 'web')
} else {
setWebList(res.payload.data?.webApp)
}
}
})
// BI-
dispatch(control.getDataAlarmsAggDay({ pepProjectId: pepProjectId })).then(res => {
if (res.success) setDataBI(res.payload.data)
})
//BI-
dispatch(control.getVideoAlarmsAggDay({ pepProjectId: pepProjectId })).then(res => {
if (res.success) setVideoBI(res.payload.data)
})
//BI-
dispatch(control.getAppAlarmsAggDay({ pepProjectId: pepProjectId })).then(res => {
// console.log(res.payload.data);
if (res.success) setAppBI(res.payload.data)
}) })
}, [pepProjectId])
}, [pepProjectId])
useEffect(() => { useEffect(() => {
const domProject = document.getElementById("news"); const domProject = document.getElementById("news");
@ -167,25 +202,55 @@ const Control = (props) => {
if (res.success) setToolShow(res.payload.data) if (res.success) setToolShow(res.payload.data)
}) })
} }
function startmarquee (speed, delay, name) {
/*
函数startmarquee的参数
lh文字一次向上滚动的距离或高度
speed滚动速度
delay滚动停顿的时间间隔
index可以使封装后的函数应用于页面当中不同的元素
*/
var t;
var p = false;
let top = 0
var o = document.getElementById(name);
if (o) {
o.onmouseover = () => p = true
o.onmouseout = () => p = false
o.scrollTop = 0;
const start = () => {
t = setInterval(() => {
if (!p) (top += 10, o.scrollTop = top)
if (p) (clearInterval(t), setTimeout(start, delay))
if (o.scrollTop >= o.scrollHeight / 2) (top = 0, o.scrollTop = 0)
}, speed);
}
setTimeout(start, 1000);
}
}
let Select = { let Select = {
workbench: ['project', 'data', 'app', 'device'], workbench: ['project', 'data', 'app', 'device'],
statistical: ['milestone', 'personnel', 'DeviceAccess', 'web', 'problem'], statistical: ['milestone', 'personnel', 'DeviceAccess', 'web', 'problem'],
analyse: ['dataInterrupt', 'dataAnomaly', 'strategyHit', 'videoException', 'appAbnormal', 'unitException', 'problemAnalysis'], analyse: ['dataInterrupt', 'dataAbnormal', 'policyHit', 'videoException', 'appAbnormal', 'deviceAbnormal', 'problemAnalysis'],
dynamic: [], dynamic: [],
} }
let show = { let show = {
workbench: ['project', 'data', 'app', 'device'], workbench: ['project', 'data', 'app', 'device'],
statistical: ['milestone', 'personnel', 'DeviceAccess', 'web', 'problem'], statistical: ['milestone', 'personnel', 'DeviceAccess', 'web', 'problem'],
analyse: ['dataInterrupt', 'dataAnomaly', 'strategyHit', 'videoException', 'appAbnormal', 'unitException', 'problemAnalysis'], analyse: ['dataInterrupt', 'dataAbnormal', 'policyHit', 'videoException', 'appAbnormal', 'deviceAbnormal', 'problemAnalysis'],
dynamic: [], dynamic: [],
} }
let listAll = [ let listAll = [
{ name: '关注的项目', sort: 1, key: 'project', data: workData?.projects || 0, img: 'url(/assets/images/console/lan_1.png)' }, { name: '关注的项目', sort: 1, key: 'project', data: workData?.projects || 0, img: 'url(/assets/images/console/lan_1.png)' },
{ name: '数据告警', sort: 2, key: 'data', data: workData?.dataSurplus || 0, img: 'url(/assets/images/console/lv_1.png)' }, { name: '数据告警', sort: 2, key: 'data', data: workData?.dataSurplus || 0, img: 'url(/assets/images/console/lv_1.png)', url: '/problem/dataAlarm/dataLnterrupt' },
{ name: '应用告警', sort: 2, key: 'app', data: workData?.appSurplus || 0, img: 'url(/assets/images/console/hong_1.png)' }, { name: '应用告警', sort: 2, key: 'app', data: workData?.appSurplus || 0, img: 'url(/assets/images/console/hong_1.png)', url: '/problem/useAlarm/useAbnormal' },
{ name: '设备告警', sort: 2, key: 'device', data: workData?.toolSurplus || 0, img: 'url(/assets/images/console/hong_1.png)' }, { name: '设备告警', sort: 2, key: 'device', data: workData?.toolSurplus || 0, img: 'url(/assets/images/console/hong_1.png)', url: '/problem/deviceAlarm/deviceAbnormal' },
{ name: '项目里程碑', sort: 1, key: 'milestone', }, { name: '项目里程碑', sort: 1, key: 'milestone', },
{ name: '相关成员', sort: 2, key: 'personnel', }, { name: '相关成员', sort: 2, key: 'personnel', },
@ -194,11 +259,11 @@ const Control = (props) => {
{ name: '异常&问题', sort: 5, key: 'problem', }, { name: '异常&问题', sort: 5, key: 'problem', },
{ name: '数据中断', sort: 1, key: 'dataInterrupt', }, { name: '数据中断', sort: 1, key: 'dataInterrupt', },
{ name: '数据异常', sort: 2, key: 'dataAnomaly', }, { name: '数据异常', sort: 2, key: 'dataAbnormal', },
{ name: '策略命中', sort: 3, key: 'strategyHit', }, { name: '策略命中', sort: 3, key: 'policyHit', },
{ name: '视频异常', sort: 4, key: 'videoException', }, { name: '视频异常', sort: 4, key: 'videoException', },
{ name: '应用异常', sort: 5, key: 'appAbnormal', }, { name: '应用异常', sort: 5, key: 'appAbnormal', },
{ name: '设备异常', sort: 6, key: 'unitException', }, { name: '设备异常', sort: 6, key: 'deviceAbnormal', },
{ name: '问题处置效率分析', sort: 7, key: 'problemAnalysis', }, { name: '问题处置效率分析', sort: 7, key: 'problemAnalysis', },
] ]
@ -222,7 +287,6 @@ const Control = (props) => {
return ( return (
// 11 ? <img src="/assets/images/install/watting.png" alt="" style={{ width: 'calc(100% + 16px)', position: "relative", top: -12, left: -8, }} /> :
<> <>
<div style={{ padding: '0px 40px', width: '100%' }} className='console'> <div style={{ padding: '0px 40px', width: '100%' }} className='console'>
{/* 头部 */} {/* 头部 */}
@ -303,7 +367,7 @@ const Control = (props) => {
<span style={{ fontSize: 12, color: '#4A4A4A' }}>{item.name == '关注的项目' ? ' ( 个 )' : ' ( 条 )'}</span> <span style={{ fontSize: 12, color: '#4A4A4A' }}>{item.name == '关注的项目' ? ' ( 个 )' : ' ( 条 )'}</span>
</div> </div>
<div style={{ marginTop: 15, display: 'flex', alignItems: 'center' }}> <div style={{ marginTop: 15, display: 'flex', alignItems: 'center' }}>
<div style={{ fontSize: 32, color: index == 0 ? '#0F7EFB' : index == 1 ? '#0091A1' : index == 2 ? '#FE9812' : '#FF7575', fontFamily: 'YouSheBiaoTiHei' }}>{item.data}</div> <div onClick={() => dispatch(push(item.url))} style={{ fontSize: 32, color: index == 0 ? '#0F7EFB' : index == 1 ? '#0091A1' : index == 2 ? '#FE9812' : '#FF7575', fontFamily: 'YouSheBiaoTiHei' }}>{item.data}</div>
{item.name == '关注的项目' ? '' : <div style={{ fontSize: 12, color: '#969799', marginLeft: 4 }}>待处理</div>} {item.name == '关注的项目' ? '' : <div style={{ fontSize: 12, color: '#969799', marginLeft: 4 }}>待处理</div>}
</div> </div>
</div> </div>
@ -400,7 +464,7 @@ const Control = (props) => {
相关成员 相关成员
</div> </div>
<div id='member' style={{ position: 'relative', height: 161 }}> <div id='member' style={{ position: 'relative', height: 161 }}>
{memberList.map((item, index) => { {memberList?.map((item, index) => {
return ( return (
<div key={index + 'member'} style={{ display: 'flex', justifyContent: 'space-between', marginBottom: 15 }}> <div key={index + 'member'} style={{ display: 'flex', justifyContent: 'space-between', marginBottom: 15 }}>
<div style={{ display: 'flex', alignItems: 'center' }}> <div style={{ display: 'flex', alignItems: 'center' }}>
@ -408,7 +472,7 @@ const Control = (props) => {
<img src="/assets/images/console/member.png" alt="成员" style={{ width: '100%', height: '100%' }} /> <img src="/assets/images/console/member.png" alt="成员" style={{ width: '100%', height: '100%' }} />
</div> </div>
<div style={{ marginLeft: 8, fontSize: 14, color: '#4A4A4A' }}> <div style={{ marginLeft: 8, fontSize: 14, color: '#4A4A4A' }}>
刘昊然 {item.name}
</div> </div>
<div style={{ fontSize: 12, color: '#969799' }}> <div style={{ fontSize: 12, color: '#969799' }}>
负责人 负责人
@ -467,17 +531,19 @@ const Control = (props) => {
{ {
webList.map((item, index) => { webList.map((item, index) => {
return ( return (
<div key={index + 'web'} style={{ marginBottom: 15, display: 'flex', alignItems: 'center', justifyContent: 'space-between' }}> <div key={index + 'webb'} style={{ marginBottom: 15, }}>
<div style={{ display: 'flex' }}> <div style={{ display: 'flex' }}>
<div style={{ width: 18, height: 18, marginRight: 8 }}> <div style={{ width: 18, height: 18, marginRight: 8 }}>
<img src="/assets/images/console/icon_webpage.png" alt="web应用" style={{ width: '100%', height: '100%' }} /> <img src="/assets/images/console/icon_webpage.png" alt="web应用" style={{ width: '100%', height: '100%' }} />
</div> </div>
<div style={{ color: '#646566', fontSize: 14, borderBottom: '1px dotted #0F7EFB' }}> <div style={{ color: '#646566', fontSize: 14, borderBottom: '1px dotted #0F7EFB' }}>
superchangnan.anxiny <a href={item.url} target="_blank">
{item.url}
</a>
</div> </div>
</div> </div>
<div style={{ color: '#4A4A4A', fontSize: 14, marginRight: 40 }}> <div style={{ color: '#4A4A4A', fontSize: 14, marginLeft: 26 }}>
第三方 {item.name}
</div> </div>
</div> </div>
) )
@ -494,14 +560,16 @@ const Control = (props) => {
</div> </div>
<div id='problems' style={{ position: 'relative', height: 161 }}> <div id='problems' style={{ position: 'relative', height: 161 }}>
{ {
problemsList.map((item, index) => { problemsList?.map((v, index) => {
return ( return (
<div style={{ marginBottom: 15, paddingRight: 30, }}> <div style={{ marginBottom: 15, paddingRight: 30, }} key={'problems' + index} onClick={() => {
<div style={{ fontSize: 14, color: '#646566' }}> dispatch(push(v.url))
告警源A数据信息中断诊断为 <span style={{ color: '#0F7EFB', borderBottom: '1px dotted #0F7EFB', cursor: "pointer" }}>服务异常请前往确认</span> }}>
<div style={{ fontSize: 14, color: '#646566' }} >
{v.SourceName}{v.groupName}{v.groupName == '视频异常' ? "" : ',诊断为 '} <span style={{ color: '#0F7EFB', borderBottom: '1px dotted #0F7EFB', cursor: "pointer" }} >{v.typeName}请前往确认</span>
</div> </div>
<div style={{ color: '#969799', fontSize: 12, marginRight: 40, marginTop: 4 }}> <div style={{ color: '#969799', fontSize: 12, marginRight: 40, marginTop: 4 }}>
2022-05-21 15:23:41 {moment(v.StartTime).format("YYYY-MM-DD HH:mm:ss")}
</div> </div>
</div> </div>
) )
@ -531,7 +599,37 @@ const Control = (props) => {
</div> </div>
<div style={{ width: '100%', height: '100%' }}> <div style={{ width: '100%', height: '100%' }}>
{exhibition.current?.analyse?.map((v, index) => { {exhibition.current?.analyse?.map((v, index) => {
return <div id={'ReactECharts' + index} style={{ marginTop: 20, padding: 10, width: '50%', display: "inline-block" }}> let startValue = ''
if (v.key !== 'problemAnalysis') {
switch (v.key) {
case 'videoException':
let videos = videoBI?.filter(u => moment(moment().day(moment().day() - 30).format('YYYY-MM-DD')).isBefore(u.day)) || []
if (videos.length) {
startValue = videos[0]?.day
} else {
startValue = videoBI?.slice(-1)[0]?.day
}
break;
case 'appAbnormal':
let apps = appBI?.filter(u => moment(moment().day(moment().day() - 30).format('YYYY-MM-DD')).isBefore(u.day)) || []
if (apps.length) {
startValue = apps[0]?.day
} else {
startValue = appBI?.slice(-1)[0]?.day
}
break;
default:
let datas = dataBI[v.key]?.filter(u => moment(moment().day(moment().day() - 30).format('YYYY-MM-DD')).isBefore(u.day)) || []
if (datas.length) {
startValue = datas[0]?.day
} else {
startValue = dataBI[v.key]?.slice(-1)[0]?.day
}
break;
}
}
// console.log(startValue);
return v.key == 'problemAnalysis' ? <div id={'ReactECharts' + index} style={{ marginTop: 20, padding: 10, width: '50%', display: "inline-block" }}>
<ReactECharts <ReactECharts
option={{ option={{
title: { title: {
@ -545,23 +643,18 @@ const Control = (props) => {
}, },
dataZoom: [ dataZoom: [
{ {
show: true, type: 'slider',
type: 'inside', // startValue:
filterMode: 'none',
xAxisIndex: [0],
}, },
{ {
show: true,
type: 'inside', type: 'inside',
filterMode: 'none', },
yAxisIndex: [0],
}
], ],
tooltip: { tooltip: {
trigger: 'axis' trigger: 'axis'
}, },
legend: { legend: {
data: ['次数'], data: [v.name, '已处理(含自动恢复)'],
right: '10%', right: '10%',
}, },
xAxis: { xAxis: {
@ -572,21 +665,86 @@ const Control = (props) => {
}, },
yAxis: { yAxis: {
type: 'value', type: 'value',
name: "数", name: "数",
}, },
series: [ series: [
{ {
data: [],
type: 'line', type: 'line',
name: '次数', name: v.name,
smooth: true, smooth: true,
areaStyle: { areaStyle: {
color: '#0e9cff26', color: '#0e9cff26',
}, },
markLine: { data: []
data: [{ type: 'average', name: 'Avg' }] },
} ]
} }}
notMerge={true}
lazyUpdate={true}
theme={'ReactEChart' + index}
// onChartReady={this.onChartReadyCallback}
// onEvents={EventsDict}
// opts={}
/>
</div>
: <div id={'ReactECharts' + index} style={{ marginTop: 20, padding: 10, width: '50%', display: "inline-block" }}>
<ReactECharts
option={{
title: {
text: v.name,
},
grid: {
left: '5%',
// right: '4%',
// bottom: '3%',
// containLabel: true
},
dataZoom: [
{
type: 'slider',
startValue: startValue
},
{
type: 'inside',
},
],
tooltip: {
trigger: 'axis'
},
legend: {
data: [v.name, '已处理(含自动恢复)'],
right: '10%',
},
xAxis: {
type: 'category',
name: "时间",
boundaryGap: false,
data: v.key == 'videoException' ? videoBI?.map(u => u.day) : v.key == 'appAbnormal' ? appBI?.map(u => u.day) : dataBI[v.key]?.map(u => u.day) || []
},
yAxis: {
type: 'value',
name: "条数",
},
series: [
{
type: 'line',
name: v.name,
smooth: true,
areaStyle: {
color: '#0e9cff26',
},
data: v.key == 'videoException' ? videoBI?.map(u => u.total) : v.key == 'appAbnormal' ? appBI?.map(u => u.total) : dataBI[v.key]?.map(u => u.total) || []
},
{
type: 'line',
name: '已处理(含自动恢复)',
smooth: true,
areaStyle: {
color: '#0e9cff26',
},
data: v.key == 'videoException' ? videoBI?.map(u => u.done) : v.key == 'appAbnormal' ? appBI?.map(u => u.done) : dataBI[v.key]?.map(u => u.done) || []
},
] ]
}} }}
notMerge={true} notMerge={true}

5
web/client/src/utils/webapi.js

@ -60,8 +60,11 @@ export const ApiTable = {
consoleToollink: 'console/toollink', //常用工具 consoleToollink: 'console/toollink', //常用工具
deleteConsoleToollink: 'console/toollink/{linkId}', //删除常用工具 deleteConsoleToollink: 'console/toollink/{linkId}', //删除常用工具
getConsoleCount: 'console/count', //工作台数量查询 getConsoleCount: 'console/count', //工作台数量查询
getConsoleUser: 'analysis/userlist', //查询关联人员,web应用
getConsoleAbnormal: 'analysis/problem', //项目概览异常查询 getConsoleAbnormal: 'analysis/problem', //项目概览异常查询
getDataAlarmsAggDay: 'data/alarms/agg/day', //查询BI分析数据-数据
getVideoAlarmsAggDay: 'video/alarms/agg/day', //查询BI分析数据-视频异常
getAppAlarmsAggDay: 'app/alarms/agg/day', //查询BI分析数据-应用
}; };
export const RouteTable = { export const RouteTable = {
apiRoot: "/api/root", apiRoot: "/api/root",

Loading…
Cancel
Save