diff --git a/api/app/lib/controllers/project/group.js b/api/app/lib/controllers/project/group.js index 492b87b..016dd04 100644 --- a/api/app/lib/controllers/project/group.js +++ b/api/app/lib/controllers/project/group.js @@ -586,6 +586,90 @@ async function groupProject (ctx) { } } + +async function getProjectWorkOrders(){ + try{ + // 计算一个月前的日期 + const oneMonthAgo = moment().subtract(1, 'months').toDate() + const { models } = ctx.fs.dc + const {Op, fn, col} = ctx.fs.dc.ORM.sequelize + const { projectIds } = ctx.query + if(projectIds&&projectIds.length){ + const projectIdsArr=projectIds.map(Number) + const res= await models.FormDataTable.findAll({ + attributes: [ + 'projectId', + [fn('COUNT', col('id')), 'count'], + ], + where: { + projectIds: { + [Op.in]: projectIdsArr, + }, + startTime: { + [Op.gte]: oneMonthAgo, + }, + }, + group: ['projectId'], + order: [[fn('COUNT', col('id')), 'DESC']], + }) + ctx.body=res + ctx.status=200 + }else{ + ctx.body='没有查询到数据' + 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 getWorkOrdersRepairRank(){ + try{ + const oneMonthAgo = moment().subtract(1, 'months').toDate() + const { models } = ctx.fs.dc + const {Op, fn, col} = ctx.fs.dc.ORM.sequelize + const { projectIds } = ctx.query + if(projectIds&&projectIds.length){ + const res= await models.FormDataTable.findAll({ + where: { + projectIds: { + [Op.in]: projectIdsArr, + }, + startTime: { + [Op.gte]: oneMonthAgo, + } + }, + order: [ + [fn('TIMESTAMPDIFF', literal('SECOND'), col('endTime'), col('startTime')), 'DESC'] + ] + }) + ctx.body=res + ctx.status=200 + }else{ + ctx.body='没有查询到信息' + 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 + } + } +} + + module.exports = { groupList, groupDetail, @@ -594,5 +678,7 @@ module.exports = { groupStatistic, groupStatisticOnline, groupStatisticAlarm, - groupProject + groupProject, + getProjectWorkOrders, + getWorkOrdersRepairRank }; \ No newline at end of file diff --git a/api/app/lib/middlewares/camunda-rest.js b/api/app/lib/middlewares/camunda-rest.js index 41f2f32..b0ab03a 100644 --- a/api/app/lib/middlewares/camunda-rest.js +++ b/api/app/lib/middlewares/camunda-rest.js @@ -10,7 +10,7 @@ function factory(app, router, opts) { const password = ctx.fs.api.password ? ctx.fs.api.password : "fs-workflow"; // console.log('[camunda-rest] ctx.fs.api:', JSON.stringify(ctx.fs.api)) - console.log('[camunda-rest] ctx.fs.api.camundaUserId:', ctx.fs.api.camundaUserId) + // console.log('[camunda-rest] ctx.fs.api.camundaUserId:', ctx.fs.api.camundaUserId) const req = { get: (url, query) => { return request diff --git a/api/app/lib/models/form_data_table.js b/api/app/lib/models/form_data_table.js new file mode 100644 index 0000000..34613f4 --- /dev/null +++ b/api/app/lib/models/form_data_table.js @@ -0,0 +1,70 @@ +/* eslint-disable*/ + +'use strict'; + +module.exports = dc => { + const DataTypes = dc.ORM; + const sequelize = dc.orm; + const FormDataTable = sequelize.define("formDataTable", { + id: { + type: DataTypes.INTEGER, + allowNull: false, + defaultValue: "nextval(\"formDataTable_id_seq\"::regclass)", + comment: "id", + primaryKey: true, + field: "id", + autoIncrement: false + }, + projectId: { + type: DataTypes.INTEGER, + allowNull: true, + defaultValue: null, + comment: "项目id", + primaryKey: false, + field: "project_id", + autoIncrement: false + }, + formname: { + type: DataTypes.STRING, + allowNull: true, + defaultValue: null, + comment: "工单名称", + primaryKey: false, + field: "formname", + autoIncrement: false + }, + startTime: { + type: DataTypes.DATE, + allowNull: true, + defaultValue: null, + comment: "开始时间", + primaryKey: false, + field: "start_time", + autoIncrement: false + }, + endTime: { + type: DataTypes.DATE, + allowNull: true, + defaultValue: null, + comment: "结束时间", + primaryKey: false, + field: "end_time", + autoIncrement: false + }, + state: { + type: DataTypes.STRING, + allowNull: true, + defaultValue: null, + comment: null, + primaryKey: false, + field: "state", + autoIncrement: false + } + }, { + tableName: "form_data_table", + comment: "", + indexes: [] + }); + dc.models.FormDataTable = FormDataTable; + return FormDataTable; +}; \ No newline at end of file diff --git a/api/app/lib/routes/project/index.js b/api/app/lib/routes/project/index.js index 57da207..2d6e46c 100644 --- a/api/app/lib/routes/project/index.js +++ b/api/app/lib/routes/project/index.js @@ -54,4 +54,12 @@ module.exports = function (app, router, opts) { app.fs.api.logAttr['GET/project/group/list'] = { content: '获取分组项目信息', visible: true }; router.get('/project/group/list', projectGroup.groupProject); + + app.fs.api.logAttr['GET/project/workOrders'] = { content: '获取项目工单信息', visible: true }; + router.get('/project/workOrders', projectGroup.getProjectWorkOrders); + + + app.fs.api.logAttr['GET/project/workOrders/repairRank'] = { content: '获取工单修复信息', visible: true }; + router.get('project/workOrders/repairRank', projectGroup.getWorkOrdersRepairRank); + }; \ No newline at end of file diff --git a/api/app/lib/schedule/workOrder.js b/api/app/lib/schedule/workOrder.js index 7eb38a0..07222b9 100644 --- a/api/app/lib/schedule/workOrder.js +++ b/api/app/lib/schedule/workOrder.js @@ -1,116 +1,94 @@ -// const schedule = require('node-schedule'); -// const moment = require('moment') - -// module.exports = function (app, opts) { -// const updateAttendance = app.fs.scheduleInit( -// { -// interval: '34 21 4 * * *', -// // interval: '34 */15 * * * *', -// immediate: true, -// proRun: true, -// }, -// async()=>{ -// try{ -// const startTime = moment() -// const { models } = app.fs.dc -// const { clickHouse } = app.fs -// const { database: camWorkflow } = clickHouse.camWorkflow.opts.config -// //工单数据 -// let workOrderNeedData = { -// projectName: { -// keyWord: ['项目名称'], -// require: true, -// }, -// applyTime: { -// keyWord: ['申请时间'], -// require: true, -// }, -// completionTime: { -// keyWord: ['处理完成时间'], -// require: true, -// }, -// state:{ -// keyWord:[''], -// require: true, -// }, - - - - - - - - - - - - -// } -// const attendanceRes = await clickHouse.pepEmis.query( -// ` -// SELECT -// story.id AS historyId, -// story.apply_user AS pepUserId, -// story.form_data AS formData, -// story.submit_form_data AS submitFormData, -// fform.form_schema AS formSchema, -// fprocess.name AS processName, -// procin.state_ AS state, -// fform.id AS formId, -// fversion.id AS versionId, -// fgroup.name AS groupName -// FROM -// workflow_process_history AS story -// INNER JOIN workflow_process_version AS fversion -// ON fversion.id = story.version_id -// INNER JOIN workflow_process_form AS fform -// ON fform.id = fversion.form_id -// INNER JOIN workflow_process AS fprocess -// ON fprocess.id = fform.process_id -// INNER JOIN workflow_group AS fgroup -// ON fgroup.id = fprocess.group_id -// AND fgroup.name = '' -// INNER JOIN ${camWorkflow}.act_hi_procinst AS procin -// ON procin.id_ = story.procinst_id` -// +` ${existOvertimeCount || existVacateCount ?`WHERE story.create_at > '2023-03-16 00:00:00'`: ''}` -// ).toPromise() - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -// }catch(error){ - -// } -// } -// ); -// return { -// updateAttendance, -// } -// } \ No newline at end of file +const schedule = require('node-schedule'); +const moment = require('moment') +let isDev = false + +module.exports = function (app, opts) { + const workOrder = app.fs.scheduleInit( + { + interval: '0 * * * *',//一小时执行一次 + immediate: isDev, + proRun: !isDev, + }, + async()=>{ + try{ + //前一次执行时间 + console.log('工单数据抽取开始') + let lastExecutionTime = null; + const { parseProcessData } = app.fs.utils + const startTime = moment() + const { models } = app.fs.dc + const { clickHouse } = app.fs + const { database: camWorkflow } = clickHouse.camWorkflow.opts.config + //新建表是否有数据 + const rescount=await models.FormDataTable.count() + if (lastExecutionTime === null) { + lastExecutionTime = moment().subtract(1, 'hour'); + } + const formRes = await clickHouse.pepEmis.query( + `SELECT + story.id AS historyId, + story.procinst_id as procinstId, + story.apply_user AS pepUserId, + story.form_data AS formData, + story.submit_form_data AS submitFormData, + story.create_at as createTime, + fform.form_schema AS formSchema, + fprocess.name AS processName, + procin.state_ AS state, + procin.end_time_ as endTime, + fform.id AS formId, + fversion.id AS versionId, + fgroup.name AS groupName + FROM + workflow_process_history AS story + INNER JOIN workflow_process_version AS fversion + ON fversion.id = story.version_id + INNER JOIN workflow_process_form AS fform + ON fform.id = fversion.form_id + INNER JOIN workflow_process AS fprocess + ON fprocess.id = fform.process_id + AND fprocess.name = '运维中台售后问题处理工单' + INNER JOIN workflow_group AS fgroup + ON fgroup.id = fprocess.group_id + INNER JOIN ${camWorkflow}.act_hi_procinst AS procin + ON procin.id_ = story.procinst_id + ${rescount ?` WHERE procin.end_time_ is not null + AND procin.end_time_ > '${lastExecutionTime}' + AND procin.end_time_ <='${startTime}'` + : ''}` + ).toPromise() + console.log('formRes1',formRes) + const procinstIds = [...new Set(formRes.map(e => e.procinstId))]; + // 获取流程实例变量 + if(formRes && formRes.length > 0){ + let procinstsVariables = await ctx.app.camunda.request.post(encodeURI(`/engine-rest/history/variable-instance`), { + processInstanceIdIn: procinstIds + }) + for (let f of formRes) { + if(f?.formData?.formData && f?.formData?.workflowProcessVersion?.workflowProcessForm?.formSchema) { + const parseData = parseProcessData({ + formSchema: JSON.parse(f.formSchema), + formData: JSON.parse(f.formData) + }) + await models.Workorder.create({ + projectId:parseData.pomsProjectId.value || null, + formname:procinstsVariables.find(item=>item.id===f.procinstId).name||null, + state: f.state||null, + endTime:f.endTime||null, + startTime:f.createTime||null + }) + console.log('工单数据抽取结束') + } + } + + } + }catch(error){ + console.error(error) + } + } + ); + return { + workOrder, + } +} \ No newline at end of file diff --git a/script/3.2/schema/1.create_form_data_table.sql b/script/3.2/schema/1.create_form_data_table.sql new file mode 100644 index 0000000..72653a7 Binary files /dev/null and b/script/3.2/schema/1.create_form_data_table.sql differ diff --git a/web/client/src/sections/projectGroup/actions/group.js b/web/client/src/sections/projectGroup/actions/group.js index 53149e0..928b428 100644 --- a/web/client/src/sections/projectGroup/actions/group.js +++ b/web/client/src/sections/projectGroup/actions/group.js @@ -103,3 +103,35 @@ export function groupProject (query = {}) { }); } + +export function getProjectWorkOrders (query) { + return (dispatch) => basicAction({ + type: "get", + dispatch: dispatch, + query, + actionType: "GET_PROJECT_WORK_ORDERS", + url: `${ApiTable.projectWordOrders}`, + msg: { error: "获取项目工单信息失败" }, + reducer: { + name: "projectWorkOrders", + params: { noClear: true } + }, + }); +} + + +export function getWorkOrdersRepairRank (query) { + return (dispatch) => basicAction({ + type: "get", + dispatch: dispatch, + query, + actionType: "GET_WORK_ORDERS_REPAIR_RANK", + url: `${ApiTable.wordOrdersRepairRank}`, + msg: { error: "获取工单修复排名信息失败" }, + reducer: { + name: "projectWorkOrders", + params: { noClear: true } + }, + }); +} + diff --git a/web/client/src/sections/projectGroup/containers/bigscreen.jsx b/web/client/src/sections/projectGroup/containers/bigscreen.jsx index 133b9c5..fbfd939 100644 --- a/web/client/src/sections/projectGroup/containers/bigscreen.jsx +++ b/web/client/src/sections/projectGroup/containers/bigscreen.jsx @@ -15,7 +15,7 @@ let interrupt let repair let overviewScrollbar; const Bigscreen = (props) => { - const { dispatch, actions, user, match, history, clientHeight, groupStatisticOnline, ...restProps } = props + const { dispatch, actions, user, match, history, clientHeight, groupStatisticOnline, pomsProjectBasicAll,...restProps } = props const [InterruptRank, setInterruptRank] = useState([]) const [online, setOnline] = useState([]) @@ -56,19 +56,22 @@ const Bigscreen = (props) => { } }) - dispatch(actions.projectGroup.groupProject({ groupId: groupId_ })).then(res => { - if (res.success) { - setGroupProject(res.payload.data?.map(v => ({ ...v, value: (Math.random() * 20).toFixed(0) })) || []) - setProportion([...res.payload.data?.slice(0, 3)?.map(v => ({ name: v.name || v.pepProjectName, value: (Math.random() * 20).toFixed(0) })), { value: 20, name: '其它' }]) - } - }) + // dispatch(actions.projectGroup.groupProject({ groupId: groupId_ })).then(res => { + // if (res.success) { + // // setProportion([...res.payload.data?.slice(0, 3)?.map(v => ({ name: v.name || v.pepProjectName, value: (Math.random() * 20).toFixed(0) })), { value: 20, name: '其它' }]) + // } + // }) dispatch(actions.projectGroup.getProjectGroupDetail(groupId_)).then(res => { if (res.success) { - setGroupDetail(res.payload.data) + setGroupDetail(res.payload.data) } }) + + + + const interruptDom = document.getElementById("interrupt"); if (interruptDom) { interrupt = new PerfectScrollbar("#interrupt", { @@ -132,6 +135,28 @@ const Bigscreen = (props) => { } }, [mockData]) + + + useEffect(() => { + if(groupDetail?.pomsProjectIds&&groupDetail?.pomsProjectIds.length){ + const query=res.payload.data.pomsProjectIds.split(',') + dispatch(actions.projectGroup.getProjectWorkOrders({projectIds:query}).then(res=>{ + if(res.success){ + setProportion([...res.payload.data?.slice(0, 3)?.map(v => ({ name: pomsProjectBasicAll.find(item=>item.valiue===v.projectId)?.label, value: v.count })), + { value: res.payload.data&&res.payload.data.length>3?res.payload.data?.slice(3)?.reduce((p,c)=>{ + return p+c + },0):0, name: '其它' }]) + } + })) + dispatch(actions.projectGroup.getProjectWorkOrders({projectIds:query}).then(res=>{ + if(res.success){ + setGroupProject(res.payload.data?.slice(0, 10).map(v => ({name:v.formname,duration:v?.endTime.diff(v?.startTime,'hours') })) || []) + } + })) + + } + },[groupDetail]) + let statisticOnline = (groupId) => { dispatch(actions.projectGroup.groupStatisticOnline({ groupId })).then(res => { if (res.success) { @@ -292,16 +317,18 @@ const Bigscreen = (props) => {