You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
461 lines
16 KiB
461 lines
16 KiB
'use strict';
|
|
const fs = require('fs');
|
|
const moment = require('moment');
|
|
const { alarmConfirmLog } = require('./alarmConfirmLog');
|
|
async function groupList (ctx) {
|
|
try {
|
|
const { models } = ctx.fs.dc;
|
|
const { clickHouse } = ctx.app.fs
|
|
|
|
const { database: dataAlarm } = clickHouse.dataAlarm.opts.config
|
|
const groupRes = await clickHouse.anxinyun.query(`
|
|
SELECT * FROM t_alarm_group
|
|
`).toPromise();
|
|
|
|
for (let g of groupRes) {
|
|
g.unit = await await clickHouse.anxinyun.query(`
|
|
SELECT DISTINCT
|
|
t_alarm_group_unit.id AS id,t_alarm_group_unit.name AS name,t_alarm_group_unit.group_id AS groupId
|
|
FROM t_alarm_group_unit
|
|
INNER JOIN ${dataAlarm}.alarms
|
|
ON t_alarm_group_unit.id = ${dataAlarm}.alarms.AlarmGroupUnit
|
|
WHERE group_id = ${g.id}
|
|
`).toPromise();
|
|
}
|
|
ctx.status = 200;
|
|
ctx.body = groupRes
|
|
} catch (error) {
|
|
ctx.fs.logger.error(`path: ${ctx.path}, error: ${error}`);
|
|
ctx.status = 400;
|
|
ctx.body = {
|
|
message: typeof error == 'string' ? error : undefined
|
|
}
|
|
}
|
|
}
|
|
|
|
async function list (ctx) {
|
|
try {
|
|
const { models } = ctx.fs.dc;
|
|
const { clickHouse } = ctx.app.fs
|
|
const { utils: { judgeSuper, anxinStrucIdRange } } = ctx.app.fs
|
|
const { database: anxinyun } = clickHouse.anxinyun.opts.config
|
|
|
|
const { pepProjectId, keywordTarget, keyword, groupId, groupUnitId, sustainTimeStart, sustainTimeEnd, limit, page, state, onlineState, toExport } = ctx.query
|
|
|
|
let anxinStruc = await anxinStrucIdRange({
|
|
ctx, pepProjectId, keywordTarget, keyword
|
|
})
|
|
let whereOption = []
|
|
if (anxinStruc.length) {
|
|
const anxinStrucIds = anxinStruc.map(a => a.strucId)
|
|
whereOption.push(`alarms.StructureId IN (${anxinStrucIds.join(",")})`)
|
|
|
|
if (groupId) {
|
|
whereOption.push(`alarms.AlarmGroup IN (${groupId})`)
|
|
}
|
|
if (groupUnitId) {
|
|
whereOption.push(`alarms.AlarmGroupUnit=${groupUnitId}`)
|
|
}
|
|
if (sustainTimeStart && sustainTimeEnd) {
|
|
let momentStart = moment(sustainTimeStart).format('YYYY-MM-DD HH:mm:ss')
|
|
let momentEnd = moment(sustainTimeEnd).format('YYYY-MM-DD HH:mm:ss')
|
|
whereOption.push(`
|
|
(
|
|
alarms."StartTime"
|
|
BETWEEN '${momentStart}' AND '${momentEnd}'
|
|
OR
|
|
"alarms"."EndTime" BETWEEN '${momentStart}' AND '${momentEnd}'
|
|
OR (
|
|
"alarms"."StartTime" <= '${momentStart}'
|
|
AND
|
|
"alarms"."EndTime" >= '${momentEnd}'
|
|
)
|
|
)
|
|
`)
|
|
}
|
|
if (keywordTarget == 'source' && keyword) {
|
|
whereOption.push(`SourceName LIKE '%${keyword}%'`)
|
|
}
|
|
if (state) {
|
|
if (state == 'new') {
|
|
whereOption.push(`alarms.State < 3`)
|
|
} else if (state == 'histroy') {
|
|
whereOption.push(`alarms.State >= 3`)
|
|
}
|
|
}
|
|
if (onlineState) {
|
|
if (onlineState == 'online') {
|
|
whereOption.push(`DeviceStatus.Status = ${1}`)
|
|
} else if (onlineState == 'offline') {
|
|
whereOption.push(`DeviceStatus.Status = ${0}`)
|
|
}
|
|
}
|
|
|
|
let alarmQueryOptionStr = `
|
|
FROM
|
|
alarms
|
|
LEFT JOIN (
|
|
SELECT DeviceId, any(Status) AS Status,max(Time) FROM DeviceStatus GROUP BY DeviceId
|
|
) AS DeviceStatus
|
|
ON DeviceStatus.DeviceId = alarms.SourceId
|
|
LEFT JOIN ${anxinyun}.t_structure
|
|
ON ${anxinyun}.t_structure.id = alarms.StructureId
|
|
LEFT JOIN ${anxinyun}.t_alarm_code
|
|
ON ${anxinyun}.t_alarm_code.code = alarms.AlarmTypeCode
|
|
LEFT JOIN ${anxinyun}.t_alarm_type
|
|
ON ${anxinyun}.t_alarm_type.id = alarms.AlarmTypeId
|
|
${whereOption.length ? 'WHERE ' + whereOption.join(' AND ') : ''}
|
|
`
|
|
const alarmRes = await clickHouse.dataAlarm.query(`
|
|
SELECT
|
|
alarms.AlarmId AS AlarmId,
|
|
alarms.State AS State,
|
|
SourceName, StartTime, EndTime,
|
|
alarms.CurrentLevel AS CurrentLevel,
|
|
SourceTypeId,
|
|
AlarmAdviceProblem, AlarmGroup, AlarmGroupUnit, AlarmAdviceProblem,
|
|
alarms.StructureId AS StructureId,
|
|
${`DeviceStatus.Status AS DeviceStatus,`}
|
|
${anxinyun}.t_structure.name AS StructureName,
|
|
${anxinyun}.t_structure.longitude AS StructureLongitude,
|
|
${anxinyun}.t_structure.latitude AS StructureLatitude,
|
|
${anxinyun}.t_alarm_code.name AS AlarmCodeName,
|
|
AlarmContent
|
|
${alarmQueryOptionStr}
|
|
ORDER BY alarms.StartTime DESC
|
|
${!toExport && limit ? 'LIMIT ' + limit : ''}
|
|
${!toExport && limit && page ? 'OFFSET ' + parseInt(limit) * parseInt(page) : ''}
|
|
`).toPromise();
|
|
|
|
// TODO lukai 说这里要加也不知道啥时候加
|
|
// ,
|
|
// ${anxinyun}.t_alarm_type.old_name AS alarmTypeOldName
|
|
|
|
// State = 3 是 自动恢复 / 4 是 人工恢复 / 其他数字 是 需要恢复
|
|
// state = 2 是 等级提升 / 1 是持续产生 / 0 是首次产生
|
|
|
|
// SourceType 0: 'DTU' / 1: '传感器' / 2: '测点'
|
|
|
|
const countAlarm = await clickHouse.dataAlarm.query(`
|
|
SELECT
|
|
count(alarms.AlarmId) AS count
|
|
${alarmQueryOptionStr}
|
|
`).toPromise();
|
|
|
|
const confirmedAlarm = alarmRes
|
|
// TODO: 开发临时注释
|
|
.filter(ar => ar.State && ar.State > 2)
|
|
.map(ar => "'" + ar.AlarmId + "'")
|
|
const confirmedAlarmDetailMax = confirmedAlarm.length ?
|
|
await clickHouse.dataAlarm.query(`
|
|
SELECT
|
|
max(Time) AS Time, AlarmId , max(Content) AS Content
|
|
FROM
|
|
alarm_details
|
|
WHERE
|
|
AlarmId IN (${confirmedAlarm.join(',')})
|
|
GROUP BY AlarmId
|
|
`).toPromise() :
|
|
[];
|
|
|
|
const detailCountAlarm = alarmRes
|
|
.map(ar => "'" + ar.AlarmId + "'")
|
|
const alarmDetailCount = detailCountAlarm.length ? await clickHouse.dataAlarm.query(`
|
|
SELECT
|
|
count(Time) AS count, AlarmId
|
|
FROM
|
|
alarm_details
|
|
WHERE
|
|
AlarmId IN (${detailCountAlarm.join(',')})
|
|
AND AlarmState < 3
|
|
GROUP BY AlarmId
|
|
`).toPromise() :
|
|
[]
|
|
|
|
|
|
alarmRes.forEach(ar => {
|
|
ar.pomsProject = (
|
|
anxinStruc.find(as => as.strucId == ar.StructureId) ||
|
|
{
|
|
pomsProject: [
|
|
// TODO: 开发临时添加
|
|
]
|
|
}
|
|
).pomsProject
|
|
|
|
// 最新告警详情 - 确认信息
|
|
let corConfirmedData = (confirmedAlarmDetailMax.find(cdm => cdm.AlarmId == ar.AlarmId) || {});
|
|
|
|
ar.confirmedContent = corConfirmedData.Content || null
|
|
ar.confirmedTime = corConfirmedData.Time || null
|
|
|
|
// 告警详情的数量
|
|
ar.detailCount = (alarmDetailCount.find(adc => adc.AlarmId == ar.AlarmId) || { count: 0 }).count
|
|
})
|
|
|
|
if (toExport) {
|
|
await exportDataAlarms(ctx, alarmRes, groupId);
|
|
} else {
|
|
ctx.status = 200;
|
|
ctx.body = {
|
|
count: countAlarm[0].count,
|
|
rows: alarmRes
|
|
}
|
|
}
|
|
} else {
|
|
ctx.body = {
|
|
count: 0,
|
|
rows: []
|
|
}
|
|
}
|
|
ctx.status = 200;
|
|
} catch (error) {
|
|
ctx.fs.logger.error(`path: ${ctx.path}, error: ${error}`);
|
|
ctx.status = 400;
|
|
ctx.body = {
|
|
message: typeof error == 'string' ? error : undefined
|
|
}
|
|
}
|
|
}
|
|
|
|
async function getAlarmGroups(ctx) {
|
|
try {
|
|
const { clickHouse } = ctx.app.fs
|
|
const groupRes = await clickHouse.anxinyun.query(`
|
|
SELECT * FROM t_alarm_group
|
|
`).toPromise();
|
|
|
|
for (let g of groupRes) {
|
|
g.unit = await await clickHouse.anxinyun.query(`
|
|
SELECT * FROM t_alarm_group_unit WHERE group_id = ${g.id}
|
|
`).toPromise();
|
|
}
|
|
return groupRes
|
|
} catch (error) {
|
|
ctx.fs.logger.error(`path: ${ctx.path}, error: ${error}`);
|
|
ctx.status = 400;
|
|
ctx.body = {
|
|
message: typeof error == 'string' ? error : undefined
|
|
}
|
|
}
|
|
}
|
|
async function exportDataAlarms(ctx, alarmList, groupId) {
|
|
try {
|
|
const { utils: { simpleExcelDown, getExportAlarmHeader } } = ctx.app.fs;
|
|
let header = await getExportAlarmHeader(ctx, groupId);
|
|
let alarmGroups = await getAlarmGroups(ctx);
|
|
|
|
let ggroups = []
|
|
alarmGroups.filter(ag => groupId.split(',').indexOf(ag.id + '') != -1).map(g => {
|
|
g.unit.map(u => {
|
|
ggroups.push(u)
|
|
})
|
|
})
|
|
|
|
let exportData = []
|
|
for (let item of alarmList.slice(0, 1000000)) {//最多百万条
|
|
let projectNames = item.pomsProject.map(p => {
|
|
return p.name || p.pepProject.projectName
|
|
})
|
|
item.projectName = projectNames.join('\r\n') || '无';//项目名称
|
|
|
|
let ycType = ggroups.find(r => r.id == item.AlarmGroupUnit) ? ggroups.find(r => r.id == item.AlarmGroupUnit).name : '无';//异常类型
|
|
//AlarmGroupUnit中断类型
|
|
item.AlarmGroupUnit = ycType;
|
|
|
|
//type异常类型
|
|
item.type = groupId == "4,5" ? item.DeviceStatus == 0 ? "离线" : '' : ycType;
|
|
|
|
//Strategy策略类型
|
|
item.Strategy = ycType;
|
|
//State命中状态
|
|
item.State = item.State == 3 || item.State == 4 ? "历史" : "当前"
|
|
|
|
//station位置信息
|
|
item.station = item.StructureLongitude && item.StructureLatitude ? item.StructureLongitude + '. ' + item.StructureLatitude : "无";
|
|
|
|
//venderName设备厂家
|
|
item.venderName = item.platform ? '未知' : item.venderName ? item.venderName : "无";//存疑 参考前端得到
|
|
|
|
let time = moment(item.confirmedTime || item.EndTime || moment().format("YYYY-MM-DD HH:mm:ss")).diff(moment(item.StartTime), 'seconds')
|
|
item.sustainTime = time < 60 ? '< 1分钟' : time > 3600 ? Math.floor(time / 3600) + '小时' + Math.floor((time - Math.floor(time / 3600) * 3600) / 60) + '分钟' : Math.floor((time - Math.floor(time / 3600) * 3600) / 60) + '分钟';
|
|
|
|
item.AlarmCodeName = item.AlarmCodeName || '无';
|
|
item.EndTime = item.EndTime || '无';
|
|
item.confirmedContent = item.confirmedContent || '无';
|
|
item.confirmedTime = item.confirmedTime || '无';
|
|
|
|
exportData.push(item)
|
|
}
|
|
let ttype = groupId == '1' ? '数据中断' : groupId == '2' ? '数据异常' : groupId == '3' ? '策略命中' : '设备异常';//数据类告警类别
|
|
const fileName = `数据异常列表_${ttype}_${moment().format('YYYYMMDDHHmmss')}` + '.csv'
|
|
const filePath = await simpleExcelDown({ data: exportData, header, fileName: fileName })
|
|
const fileData = fs.readFileSync(filePath);
|
|
ctx.status = 200;
|
|
ctx.set('Content-Type', 'application/x-xls');
|
|
ctx.set('Content-disposition', 'attachment; filename=' + encodeURI(fileName));
|
|
ctx.body = fileData;
|
|
} catch (error) {
|
|
ctx.fs.logger.error(`path: ${ctx.path}, error: ${error}`);
|
|
ctx.status = 400;
|
|
ctx.body = {
|
|
message: typeof error == 'string' ? error : undefined
|
|
}
|
|
}
|
|
}
|
|
async function detail(ctx) {
|
|
try {
|
|
const { models } = ctx.fs.dc;
|
|
const { clickHouse } = ctx.app.fs
|
|
const { alarmId, limit, page } = ctx.query
|
|
|
|
const detailRes = await clickHouse.dataAlarm.query(`
|
|
SELECT * FROM alarm_details
|
|
WHERE AlarmId = '${alarmId}'
|
|
ORDER BY Time ASC
|
|
${limit ? 'LIMIT ' + limit : ''}
|
|
${limit && page ? 'OFFSET ' + parseInt(limit) * parseInt(page) : ''}
|
|
`).toPromise()
|
|
const count = await clickHouse.dataAlarm.query(`
|
|
SELECT count(*) AS count FROM alarm_details
|
|
WHERE AlarmId = '${alarmId}'
|
|
`).toPromise()
|
|
ctx.status = 200;
|
|
// ctx.body = {
|
|
// count: count[0].count,
|
|
// rows: detailRes
|
|
// }
|
|
ctx.body = detailRes
|
|
} catch (error) {
|
|
ctx.fs.logger.error(`path: ${ctx.path}, error: ${error}`);
|
|
ctx.status = 400;
|
|
ctx.body = {
|
|
message: typeof error == 'string' ? error : undefined
|
|
}
|
|
}
|
|
}
|
|
|
|
function confirm (opts) {
|
|
return async function (ctx) {
|
|
try {
|
|
const { models } = ctx.fs.dc;
|
|
const { utils: { kfkSendAsync } } = ctx.app.fs
|
|
const { clickHouse } = ctx.app.fs
|
|
const { content = '', alarmId, confirmPost } = ctx.request.body;
|
|
// 发送告警恢复通知
|
|
// Topic: alarm
|
|
/*
|
|
* {
|
|
* messageMode: "AlarmManElimination",
|
|
* sourceId: "",
|
|
* alarmTypeCode: "",
|
|
* sponsor: userId,
|
|
* content: "确认消息",
|
|
* time: "YYYY-MM-DDTHH:mm:ss.SSSZ"
|
|
* }
|
|
*/
|
|
|
|
const alarmRes = await clickHouse.dataAlarm.query(`
|
|
SELECT * FROM alarms WHERE AlarmId IN (${alarmId.map(a => `'${a}'`).join(',')})
|
|
`).toPromise();
|
|
|
|
if (!alarmRes.length) {
|
|
throw '没有查询到对应的告警信息'
|
|
}
|
|
|
|
const [corAlarm] = alarmRes
|
|
if ([3, 4].some(s => s == corAlarm.State)) {
|
|
throw '告警信息已确认'
|
|
}
|
|
for (let corAlarm of alarmRes) {
|
|
if ([3, 4].some(s => s == corAlarm.State)) {
|
|
continue
|
|
}
|
|
const message = {
|
|
messageMode: "AlarmManElimination",
|
|
sourceId: corAlarm.SourceId,
|
|
alarmTypeCode: corAlarm.AlarmTypeCode,
|
|
sponsor: opts.anxinCloud.confirmAlarmAnxinUserId,
|
|
content: content,
|
|
time: moment().toISOString()
|
|
};
|
|
|
|
const payloads = [{
|
|
topic: `${opts.kafka.topicPrefix}_alarm`,
|
|
messages: [JSON.stringify(message)],
|
|
partition: 0
|
|
}];
|
|
|
|
await kfkSendAsync(payloads)
|
|
}
|
|
|
|
await alarmConfirmLog(ctx, confirmPost, content);//告警确认日志
|
|
|
|
ctx.status = 204;
|
|
} catch (error) {
|
|
ctx.fs.logger.error(`path: ${ctx.path}, error: ${error}`);
|
|
ctx.status = 400;
|
|
ctx.body = {
|
|
message: typeof error == 'string' ? error : undefined
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
async function detailAggregation (ctx) {
|
|
try {
|
|
const { models } = ctx.fs.dc;
|
|
const { alarmId } = ctx.query
|
|
const { clickHouse } = ctx.app.fs
|
|
|
|
const alarmDetailAggRes = await clickHouse.dataAlarm.query(`
|
|
SELECT
|
|
formatDateTime(Time,'%F %H') hours, count(AlarmId) count
|
|
FROM
|
|
alarm_details
|
|
WHERE
|
|
AlarmId='${alarmId}'
|
|
GROUP BY
|
|
hours;
|
|
`).toPromise();
|
|
|
|
ctx.status = 200;
|
|
ctx.body = alarmDetailAggRes
|
|
} catch (error) {
|
|
ctx.fs.logger.error(`path: ${ctx.path}, error: ${error}`);
|
|
ctx.status = 400;
|
|
ctx.body = {
|
|
message: typeof error == 'string' ? error : undefined
|
|
}
|
|
}
|
|
}
|
|
|
|
async function alarmCount (ctx) {
|
|
try {
|
|
const { models } = ctx.fs.dc;
|
|
const { clickHouse } = ctx.app.fs
|
|
|
|
const alarmUnconfirmedAggRes = await clickHouse.dataAlarm.query(`
|
|
SELECT count(AlarmId) count, AlarmGroup from alarms GROUP BY AlarmGroup;
|
|
`).toPromise();
|
|
|
|
ctx.status = 200;
|
|
ctx.body = alarmUnconfirmedAggRes
|
|
} catch (error) {
|
|
ctx.fs.logger.error(`path: ${ctx.path}, error: ${error}`);
|
|
ctx.status = 400;
|
|
ctx.body = {
|
|
message: typeof error == 'string' ? error : undefined
|
|
}
|
|
}
|
|
}
|
|
|
|
module.exports = {
|
|
list,
|
|
detail,
|
|
groupList,
|
|
confirm,
|
|
detailAggregation,
|
|
alarmCount,
|
|
};
|