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.
1281 lines
39 KiB
1281 lines
39 KiB
'use strict';
|
|
const { QueryTypes } = require('sequelize');
|
|
const moment = require('moment');
|
|
const xlsxDownload = require('../../../../utils/xlsxDownload.js');
|
|
const fs = require('fs');
|
|
async function reportList (ctx) {
|
|
try {
|
|
const models = ctx.fs.dc.models;
|
|
const { limit, page, startTime, endTime, keyword, userId, reportType, isTop, asc, projectType, handleState = '', performerId = '', codeRoad } = ctx.query
|
|
const { userInfo } = ctx.fs.api
|
|
const sequelize = ctx.fs.dc.orm;
|
|
|
|
let findUsers = []
|
|
if (
|
|
userInfo.loginSource == 'wx'
|
|
&& userInfo.isAdmin
|
|
&& userInfo.phone != 'SuperAdmin'
|
|
&& (!performerId && !handleState)
|
|
) {
|
|
// 是管理员但不是超管 查自己部门及下级部门的所有用户的信息
|
|
const sqlStr = `
|
|
WITH RECURSIVE sub_departments AS (
|
|
SELECT id, dependence
|
|
FROM department
|
|
WHERE id = ${userInfo.departmentId}
|
|
UNION ALL
|
|
SELECT d.id, d.dependence
|
|
FROM sub_departments sd
|
|
JOIN department d ON sd.id = d.dependence
|
|
)
|
|
SELECT u.id
|
|
FROM "user" AS u
|
|
JOIN sub_departments sd ON u.department_id = sd.id
|
|
WHERE u.delete = false;
|
|
`
|
|
|
|
const res = await sequelize.query(sqlStr, { type: QueryTypes.SELECT })
|
|
findUsers = res.map(item => {
|
|
return item.id
|
|
})
|
|
findUsers.push(-1)
|
|
}
|
|
|
|
let findOption = {
|
|
where: {
|
|
|
|
},
|
|
attributes: ['id', 'road', 'time', 'projectType', 'roadSectionStart', 'performerId', 'roadSectionEnd', 'reportType', 'address',
|
|
'content', 'longitude', 'latitude', 'projectName', 'handleState', 'codeRoad', 'handleContent', 'handlePic', 'videoUrl',
|
|
'scenePic'],
|
|
include: [{
|
|
model: models.User,
|
|
attributes: ['name']
|
|
}],
|
|
//order: [['time', asc ? 'ASC' : 'DESC']],
|
|
order: [['time', 'DESC']],
|
|
|
|
}
|
|
if (limit) {
|
|
findOption.limit = limit
|
|
}
|
|
if (page && limit) {
|
|
findOption.offset = page * limit
|
|
}
|
|
if (startTime && endTime) {
|
|
findOption.where = {
|
|
time: {
|
|
'$between': [startTime, endTime]
|
|
}
|
|
}
|
|
}
|
|
if (keyword) {
|
|
if (reportType == 'road') {
|
|
findOption.where.projectName = {
|
|
'$like': `%${keyword}%`
|
|
}
|
|
} else {
|
|
findOption.where.road = {
|
|
'$like': `%${keyword}%`
|
|
}
|
|
}
|
|
}
|
|
if (userId) {
|
|
findOption.where.userId = { $in: userId.split(',').map(Number) }
|
|
}
|
|
if (findUsers.length) {
|
|
findOption.where.userId = { $in: findUsers }
|
|
}
|
|
if (reportType) {
|
|
findOption.where.reportType = reportType
|
|
}
|
|
if (projectType) {
|
|
findOption.where.projectType = projectType
|
|
}
|
|
if (performerId) {
|
|
let performerIds = performerId.split(',')
|
|
findOption.where.performerId = { $in: performerIds }
|
|
}
|
|
if (handleState) {
|
|
let handleStates = handleState.split(',')
|
|
findOption.where.handleState = { $in: handleStates }
|
|
}
|
|
if (codeRoad) {
|
|
findOption.where.codeRoad = codeRoad
|
|
}
|
|
|
|
let reportRes = null;
|
|
|
|
if (isTop) {
|
|
const sqlStr = `
|
|
select NR.*, "user".name as user_name
|
|
from (SELECT R.*, "row_number"() OVER (PARTITION BY R.user_id ORDER BY R."time" DESC) AS NEWINDEX
|
|
FROM report AS R
|
|
${reportType ? `
|
|
where report_type = '${reportType}'
|
|
`: ''
|
|
}
|
|
) AS NR
|
|
left join "user" on "user".id = NR.user_id
|
|
WHERE NEWINDEX = 1 order by id desc;
|
|
`
|
|
reportRes = await sequelize.query(sqlStr, { type: QueryTypes.SELECT });
|
|
if (reportType == 'road') {
|
|
const projectNameArr = reportRes.map(item => item.project_name).filter(item => item)
|
|
const projectRes = projectNameArr.length ? await models.Project.findAll({
|
|
where: {
|
|
entryName: { $in: projectNameArr }
|
|
}
|
|
}) : []
|
|
for (let r of reportRes) {
|
|
let corProject = projectRes.find(item => item.entryName == r.project_name)
|
|
if (corProject) {
|
|
r.project = corProject.dataValues
|
|
}
|
|
}
|
|
}
|
|
} else {
|
|
reportRes = await models.Report.findAll(findOption)
|
|
}
|
|
|
|
ctx.status = 200;
|
|
ctx.body = reportRes
|
|
} catch (error) {
|
|
ctx.fs.logger.error(`path: ${ctx.path}, error: ${error}`);
|
|
ctx.status = 400;
|
|
ctx.body = {
|
|
message: typeof error == 'string' ? error : undefined
|
|
}
|
|
}
|
|
}
|
|
|
|
async function reportPosition (ctx) {
|
|
try {
|
|
const models = ctx.fs.dc.models;
|
|
const { startTime, endTime, userId, reportType } = ctx.query
|
|
const sequelize = ctx.fs.dc.ORM;
|
|
|
|
let findMxTimeOption = {
|
|
attributes: [
|
|
'userId',
|
|
[sequelize.fn('MAX', sequelize.col('time')), 'maxTime'],
|
|
],
|
|
where: {
|
|
|
|
},
|
|
group: ['report.user_id'],
|
|
}
|
|
|
|
if (startTime && endTime) {
|
|
findMxTimeOption.where = {
|
|
time: {
|
|
'$between': [startTime, endTime]
|
|
}
|
|
}
|
|
}
|
|
|
|
if (userId) {
|
|
findMxTimeOption.where.userId = userId
|
|
}
|
|
if (reportType) {
|
|
findMxTimeOption.where.reportType = reportType
|
|
}
|
|
|
|
const reportMaxTimeRes = await models.Report.findAll(findMxTimeOption)
|
|
const timeArr = reportMaxTimeRes.map(item => item.dataValues.maxTime)
|
|
const reportRes = await models.Report.findAll({
|
|
where: {
|
|
time: { '$in': timeArr }
|
|
}
|
|
})
|
|
ctx.status = 200;
|
|
ctx.body = reportRes
|
|
} catch (error) {
|
|
ctx.fs.logger.error(`path: ${ctx.path}, error: ${error}`);
|
|
ctx.status = 400;
|
|
ctx.body = {
|
|
message: typeof error == 'string' ? error : undefined
|
|
}
|
|
}
|
|
}
|
|
|
|
async function reportDetail (ctx) {
|
|
try {
|
|
const models = ctx.fs.dc.models;
|
|
const { reportId } = ctx.params
|
|
|
|
const reportRes = await models.Report.findOne({
|
|
where: {
|
|
id: reportId
|
|
},
|
|
include: [{
|
|
model: models.Road,
|
|
// where: {
|
|
// del: false
|
|
// },
|
|
as: 'road_',
|
|
}],
|
|
})
|
|
|
|
ctx.status = 200;
|
|
ctx.body = reportRes
|
|
} catch (error) {
|
|
ctx.fs.logger.error(`path: ${ctx.path}, error: ${error}`);
|
|
ctx.status = 400;
|
|
ctx.body = {
|
|
message: typeof error == 'string' ? error : undefined
|
|
}
|
|
}
|
|
}
|
|
|
|
async function reportHandle (ctx) {
|
|
try {
|
|
const { models } = ctx.fs.dc;
|
|
|
|
const { reportId } = ctx.params
|
|
const { handleState } = ctx.request.body
|
|
const data = ctx.request.body
|
|
/**
|
|
* data = {
|
|
* handleState,
|
|
* handleContent,
|
|
* handlePic
|
|
* }
|
|
*/
|
|
await models.Report.update(data, {
|
|
where: {
|
|
id: reportId
|
|
}
|
|
})
|
|
|
|
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 createReport (ctx) {
|
|
try {
|
|
const { userId } = ctx.fs.api
|
|
const models = ctx.fs.dc.models;
|
|
const data = ctx.request.body;
|
|
|
|
await models.Report.create({
|
|
...data,
|
|
userId,
|
|
time: new Date(),
|
|
})
|
|
|
|
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 deleteReport (ctx) {
|
|
try {
|
|
const models = ctx.fs.dc.models;
|
|
const { reportId } = ctx.params;
|
|
|
|
await models.Report.destroy({
|
|
where: {
|
|
id: reportId
|
|
}
|
|
})
|
|
|
|
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
|
|
}
|
|
}
|
|
}
|
|
|
|
// TODO 小程序填写道路名称的时候的道路筛选 是一起都返回 还是不断传关键字搜索返回
|
|
async function spotPrepare (ctx) {
|
|
try {
|
|
const { models } = ctx.fs.dc;
|
|
const sequelize = ctx.fs.dc.orm;
|
|
const { percentage } = ctx.request.body;
|
|
const departmentIdRes = await models.Department.findAll({
|
|
attributes: ['id', 'name'],
|
|
where: {
|
|
dependence: null,
|
|
delete: false,
|
|
}
|
|
})
|
|
let lukyDepartment = ''
|
|
if (departmentIdRes.length) {
|
|
lukyDepartment = departmentIdRes[(Math.round(Math.random() * departmentIdRes.length) || 1) - 1]
|
|
} else {
|
|
throw `暂无乡镇信息`
|
|
}
|
|
|
|
const sqlStr = `
|
|
WITH RECURSIVE sub_departments AS (
|
|
SELECT id, dependence
|
|
FROM department
|
|
WHERE id = ${lukyDepartment.id}
|
|
UNION ALL
|
|
SELECT d.id, d.dependence
|
|
FROM sub_departments sd
|
|
JOIN department d ON sd.id = d.dependence
|
|
)
|
|
SELECT u.id
|
|
FROM "user" AS u
|
|
JOIN sub_departments sd ON u.department_id = sd.id
|
|
WHERE u.delete = false;
|
|
`
|
|
|
|
const userRes = await sequelize.query(sqlStr, { type: QueryTypes.SELECT })
|
|
let findUsers = []
|
|
findUsers = userRes.map(item => {
|
|
return item.id
|
|
})
|
|
|
|
const reportCount = findUsers.length ? await models.Report.count({
|
|
where: {
|
|
reportType: 'conserve',
|
|
userId: { $in: findUsers }
|
|
}
|
|
}) : 0
|
|
|
|
const previewRes = await models.ReportSpotCheckPreview.create({
|
|
percentage: percentage,
|
|
departmentId: lukyDepartment.id,
|
|
date: moment().format(),
|
|
reportCount: reportCount,
|
|
checked: false,
|
|
})
|
|
|
|
|
|
ctx.status = 200;
|
|
ctx.body = {
|
|
lukyDepartment,
|
|
reportCount,
|
|
previewId: previewRes.id
|
|
}
|
|
} catch (error) {
|
|
ctx.fs.logger.error(`path: ${ctx.path}, error: ${error}`);
|
|
ctx.status = 400;
|
|
ctx.body = {
|
|
message: typeof error == 'string' ? error : undefined
|
|
}
|
|
}
|
|
}
|
|
|
|
async function spotCheck (ctx) {
|
|
const transaction = await ctx.fs.dc.orm.transaction();
|
|
try {
|
|
const { models } = ctx.fs.dc;
|
|
const sequelize = ctx.fs.dc.orm;
|
|
const { previewId } = ctx.query
|
|
if (!previewId) {
|
|
throw '参数错误'
|
|
}
|
|
|
|
const prepareRes = await models.ReportSpotCheckPreview.findOne({
|
|
where: {
|
|
id: previewId
|
|
}
|
|
})
|
|
|
|
const sqlStr = `
|
|
WITH RECURSIVE sub_departments AS (
|
|
SELECT id, dependence
|
|
FROM department
|
|
WHERE id = ${prepareRes.departmentId}
|
|
UNION ALL
|
|
SELECT d.id, d.dependence
|
|
FROM sub_departments sd
|
|
JOIN department d ON sd.id = d.dependence
|
|
)
|
|
SELECT u.id
|
|
FROM "user" AS u
|
|
JOIN sub_departments sd ON u.department_id = sd.id
|
|
WHERE u.delete = false;
|
|
`
|
|
|
|
const userRes = await sequelize.query(sqlStr, { type: QueryTypes.SELECT })
|
|
let findUsers = []
|
|
findUsers = userRes.map(item => {
|
|
return item.id
|
|
})
|
|
let checkCount = Math.ceil(prepareRes.reportCount * (prepareRes.percentage / 100))
|
|
|
|
const reportRes = await findUsers.length && checkCount ? await models.Report.findAll({
|
|
where: {
|
|
reportType: 'conserve',
|
|
userId: { $in: findUsers },
|
|
},
|
|
include: [{
|
|
model: models.User,
|
|
attributes: ['name']
|
|
}],
|
|
order: sequelize.random(), // 随机排序
|
|
limit: checkCount, // 限制返回的记录数
|
|
}) : []
|
|
|
|
await models.ReportSpotCheckPreview.update({
|
|
checked: true
|
|
}, {
|
|
where: {
|
|
id: previewId
|
|
}
|
|
})
|
|
if (reportRes.length) {
|
|
let spotDate = moment().format('YYYY-MM-DD')
|
|
await models.ReportSpotCheck.bulkCreate(reportRes.map(r => {
|
|
return {
|
|
reportId: r.id,
|
|
spotDate: spotDate,
|
|
prepareId: previewId
|
|
}
|
|
}))
|
|
}
|
|
const rslt = findUsers.length && checkCount ? await models.ReportSpotCheckPreview.findAll({
|
|
where: {
|
|
checked: 'true',
|
|
id: previewId
|
|
},
|
|
include: [{
|
|
model: models.Department,
|
|
attributes: ['name']
|
|
}, {
|
|
model: models.ReportSpotCheck,
|
|
include: [{
|
|
model: models.Report,
|
|
include: [{
|
|
model: models.User,
|
|
attributes: ['name']
|
|
}],
|
|
}]
|
|
}
|
|
],
|
|
}) : []
|
|
|
|
await transaction.commit();
|
|
ctx.status = 200;
|
|
ctx.body = rslt
|
|
} catch (error) {
|
|
await transaction.rollback();
|
|
ctx.fs.logger.error(`path: ${ctx.path}, error: ${error}`);
|
|
ctx.status = 400;
|
|
ctx.body = {
|
|
message: typeof error == 'string' ? error : undefined
|
|
}
|
|
}
|
|
}
|
|
async function spotCheckDetail (ctx) {
|
|
const { models } = ctx.fs.dc
|
|
const { startTime, endTime } = ctx.query
|
|
try {
|
|
const res = await models.ReportSpotCheckPreview.findAll({
|
|
where: {
|
|
checked: 'true'
|
|
},
|
|
order: [['date', 'DESC']],
|
|
include: [{
|
|
model: models.Department,
|
|
attributes: ['name']
|
|
}, {
|
|
model: models.ReportSpotCheck,
|
|
where: { spotDate: { $between: [moment(startTime).format('YYYY-MM-DD'), moment(endTime).format('YYYY-MM-DD')] } },
|
|
order: [['spot_date', 'DESC']],
|
|
include: [{
|
|
model: models.Report,
|
|
order: [['date', 'DESC']],
|
|
include: [{
|
|
model: models.User,
|
|
attributes: ['name']
|
|
}],
|
|
}]
|
|
}
|
|
],
|
|
})
|
|
ctx.body = res
|
|
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 roadSpotPrepare (ctx) {
|
|
try {
|
|
const { models } = ctx.fs.dc;
|
|
const sequelize = ctx.fs.dc.orm;
|
|
const { countyPercentage } = ctx.request.body;
|
|
|
|
const lastSpotRes = await models.RoadSpotCheckPreview.findOne({
|
|
where: {
|
|
checked: true
|
|
},
|
|
// attributes: ['id', 'countyPercentage', 'date',],
|
|
order: [['date', 'DESC']],
|
|
// include: [{
|
|
// model: models.RoadSpotCheck,
|
|
// include: [{
|
|
// model: models.Road,
|
|
// // attributes: ['id', 'name']
|
|
// }]
|
|
// }]
|
|
})
|
|
|
|
let lastCountyPercentage = 0 // 最后一次的县道百分比
|
|
let lastCountyRoadIds = [] // 上次查得的县道id
|
|
let lastTownRoadIds = [] // 上次查得的乡镇道id
|
|
let lastVillageRoadRoadIds = [] // 上次查得的村道id
|
|
let lastVillageIds = []
|
|
let gather = [] //汇总
|
|
|
|
if (lastSpotRes) {
|
|
lastCountyPercentage = lastSpotRes.countyPercentage
|
|
lastCountyRoadIds = lastSpotRes.countyRoadId || []
|
|
|
|
let lastCounty = await models.RoadSpotCheckPreview.findAll({
|
|
where: {
|
|
checked: true,
|
|
id: { $lt: lastSpotRes.id }
|
|
},
|
|
limit: 3,
|
|
order: [['date', 'DESC']],
|
|
})
|
|
|
|
if (lastCounty) {
|
|
lastCounty.forEach(d => {
|
|
lastTownRoadIds = lastTownRoadIds.concat(d.townshipRoadId) || []
|
|
|
|
})
|
|
}
|
|
|
|
let lastVillage = await models.RoadSpotCheckPreview.findAll({
|
|
where: {
|
|
checked: true,
|
|
id: { $lt: lastSpotRes.id }
|
|
},
|
|
limit: 9,
|
|
order: [['date', 'DESC']],
|
|
})
|
|
|
|
if (lastVillage) {
|
|
lastVillage.forEach(d => {
|
|
lastVillageRoadRoadIds = lastVillageRoadRoadIds.concat(d.villageRoadId) || []
|
|
lastVillageIds = lastVillageIds.concat(d.villageId) || []
|
|
|
|
})
|
|
}
|
|
|
|
|
|
// lastTownRoadIds = lastSpotRes.townshipRoadId || []
|
|
// lastVillageRoadRoadIds = lastSpotRes.villageRoadId || []
|
|
// lastVillageIds = lastSpotRes.villageId || []
|
|
}
|
|
|
|
// 先查上次没查的范围内的 然后比较百分比 如果重叠 再查上次查过的
|
|
let keyMap = {
|
|
'县': `route_code LIKE 'X%'`,
|
|
'乡': `route_code LIKE 'Y%'`,
|
|
'村': `route_code LIKE 'C%'`
|
|
}
|
|
async function getRoadTotalMileage (key, otherWhere = [], spot) {
|
|
let res = await sequelize.query(`
|
|
SELECT
|
|
SUM(COALESCE(CAST(chainage_mileage AS DOUBLE PRECISION), 0)) AS total_mileage
|
|
FROM road
|
|
WHERE del = false
|
|
AND spot = ${spot}
|
|
AND ${keyMap[key]} ${otherWhere.length ? `AND ${otherWhere.join(' AND ')}` : ''}
|
|
`)
|
|
return res[0][0].total_mileage || 0
|
|
}
|
|
|
|
async function getRoadSpot (key, lastRoadIds = [], inOrNot, otherWhere = []) {
|
|
if (!lastRoadIds.length && !inOrNot) {
|
|
return []
|
|
}
|
|
return await sequelize.query(`
|
|
SELECT id, chainage_mileage FROM road
|
|
WHERE del = false
|
|
AND spot = true
|
|
AND ${keyMap[key]}
|
|
${lastRoadIds.length ?
|
|
`AND id ${inOrNot ? 'IN' : 'NOT IN'} (
|
|
${lastRoadIds.map(item => `'${item}'`).join(',')},-1
|
|
)` : ''
|
|
}
|
|
AND chainage_mileage IS NOT NULL ${otherWhere.length ? `AND ${otherWhere.join(' AND ')}` : ''}
|
|
ORDER BY RANDOM()
|
|
`, { type: QueryTypes.SELECT });
|
|
}
|
|
|
|
async function spotRoadId (key, lastRoadIds, targetMileage, otherWhere = [], villageIdList = []) {
|
|
let spotRoadIds = []
|
|
let accumulationMileage = 0
|
|
|
|
async function filterRoad (otherWhere, getRepeat = true) {
|
|
|
|
if (key == '村' && getRepeat == true) {
|
|
|
|
} else {
|
|
let roadUnSpotedRes = await getRoadSpot(key, lastRoadIds, false, otherWhere)
|
|
|
|
for (let r of roadUnSpotedRes) {
|
|
spotRoadIds.push(r.id)
|
|
accumulationMileage += parseFloat(r.chainage_mileage)
|
|
if (accumulationMileage >= targetMileage) {
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (accumulationMileage < targetMileage && getRepeat) {
|
|
// 还小于 说明没取够
|
|
let roadUnSpotedRepeatRes = await getRoadSpot(
|
|
key,
|
|
lastRoadIds,
|
|
true,
|
|
otherWhere
|
|
)
|
|
for (let r of roadUnSpotedRepeatRes) {
|
|
spotRoadIds.push(r.id)
|
|
accumulationMileage += parseFloat(r.chainage_mileage)
|
|
if (accumulationMileage >= targetMileage) {
|
|
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (key == '村') {
|
|
for await (let villageId of villageIdList) {
|
|
await filterRoad([
|
|
...otherWhere,
|
|
`village_id=${villageId}`
|
|
], false)
|
|
|
|
spotVillageIds.push(villageId)
|
|
|
|
if (accumulationMileage >= targetMileage) {
|
|
break;
|
|
}
|
|
}
|
|
if (accumulationMileage < targetMileage) {
|
|
// 还小于 说明没取够
|
|
await filterRoad(otherWhere, true)
|
|
}
|
|
} else {
|
|
await filterRoad(otherWhere, true)
|
|
}
|
|
|
|
return [spotRoadIds, accumulationMileage]
|
|
}
|
|
let villageMil = 0, townMil = 0, countryMil = 0;
|
|
|
|
// 抽县
|
|
const countryRoadTotalMileage = await getRoadTotalMileage('县', '', true)
|
|
const countryRoadHide = await getRoadTotalMileage('县', '', false)
|
|
const countryRoadNeedMileage = countryRoadTotalMileage * countyPercentage / 100
|
|
let spotCountyRoadIdsArr = await spotRoadId('县', lastCountyRoadIds, countryRoadNeedMileage, [])
|
|
let spotCountyRoadIds = spotCountyRoadIdsArr[0]
|
|
let countryMil_ = spotCountyRoadIdsArr[1] || 0
|
|
if (countryMil_) countryMil += countryMil_;
|
|
|
|
gather.push({
|
|
name: '南昌县交通运输局',
|
|
county: countryRoadTotalMileage + countryRoadHide,
|
|
countyParticipate: countryRoadTotalMileage,
|
|
countyPresent: countryMil_,
|
|
countyDifferenceValue: countryMil_ - countryRoadNeedMileage
|
|
})
|
|
|
|
let town = await models.Town.findAll() || []
|
|
|
|
|
|
// 抽乡
|
|
const allTownCodeRes = await sequelize.query(`
|
|
SELECT
|
|
DISTINCT township_code
|
|
FROM road
|
|
WHERE del = false
|
|
AND spot = true
|
|
AND township_code IS NOT NULL
|
|
`, { type: QueryTypes.SELECT });
|
|
|
|
let spotTownRoadIds = []
|
|
let spotVillageRoadIds = []
|
|
let spotVillageIds = []
|
|
|
|
let allTownCodeResTown = []
|
|
for await (let t of allTownCodeRes) {
|
|
// 遍历每个乡镇并抽取
|
|
let otherWhere = [`township_code='${t.township_code}'`]
|
|
const townRoadTotalMileage = await getRoadTotalMileage('乡', otherWhere, true)
|
|
const townRoadHide = await getRoadTotalMileage('乡', otherWhere, false)
|
|
const townRoadNeedMileage = townRoadTotalMileage * 25 / 100
|
|
|
|
let spotTownRoadIdsArr = await spotRoadId('乡', lastTownRoadIds, townRoadNeedMileage, otherWhere)
|
|
let spotTownRoadIds_ = spotTownRoadIdsArr[0]
|
|
let townMil_ = spotTownRoadIdsArr[1]
|
|
spotTownRoadIds = spotTownRoadIds.concat(spotTownRoadIds_)
|
|
if (townMil_) townMil += townMil_
|
|
|
|
|
|
// 抽村
|
|
const villageRoadTotalMileage = await getRoadTotalMileage('村', otherWhere, true)
|
|
const villageRoadTotalHide = await getRoadTotalMileage('村', otherWhere, false)
|
|
const villageRoadNeedMileage = villageRoadTotalMileage * 10 / 100
|
|
|
|
let spotFirstVillageId = -1
|
|
// 随机选取一个不在上次查过的村
|
|
let villageRes = await sequelize.query(`
|
|
SELECT id FROM village
|
|
WHERE township_code = '${t.township_code}'
|
|
${lastVillageIds.length ? `AND id NOT IN (
|
|
${lastVillageIds.map(item => `'${item}'`).join(',')},-1
|
|
)`: ''}
|
|
ORDER BY RANDOM()
|
|
LIMIT 1
|
|
`, { type: QueryTypes.SELECT });
|
|
if (!villageRes.length) {
|
|
// 没有村了,随机选一个
|
|
villageRes = await sequelize.query(`
|
|
SELECT id FROM village
|
|
WHERE township_code = '${t.township_code}'
|
|
ORDER BY RANDOM()
|
|
LIMIT 1
|
|
`, { type: QueryTypes.SELECT });
|
|
}
|
|
if (villageRes.length) {
|
|
spotVillageIds.push(villageRes[0].id)
|
|
spotFirstVillageId = villageRes[0].id
|
|
} else {
|
|
allTownCodeResTown.push({
|
|
code: t.township_code,
|
|
township: townRoadTotalMileage + townRoadHide,
|
|
townshipParticipate: townRoadTotalMileage,
|
|
townshipPresent: townMil_,
|
|
townshipDifferenceValue: townMil_ - townRoadNeedMileage,
|
|
village: villageRoadTotalMileage + villageRoadTotalHide,
|
|
villageParticipate: villageRoadTotalMileage,
|
|
villagePresent: 0,
|
|
villageDifferenceValue: 0
|
|
})
|
|
continue;
|
|
}
|
|
|
|
const villageNearRes = await sequelize.query(`
|
|
SELECT id,calc_village,distance FROM village_distance
|
|
WHERE origin_village = ${spotFirstVillageId}
|
|
ORDER BY distance ASC
|
|
`, { type: QueryTypes.SELECT })
|
|
|
|
let villageCheckIdList = villageNearRes.map(item => item.calc_village)
|
|
villageCheckIdList.unshift(spotFirstVillageId)
|
|
villageCheckIdList = [...(new Set(villageCheckIdList))]
|
|
|
|
let spotVillageRoadIdsArr = await spotRoadId('村', lastVillageRoadRoadIds, villageRoadNeedMileage, otherWhere, villageCheckIdList)
|
|
let spotVillageRoadIds_ = spotVillageRoadIdsArr[0]
|
|
let villageMil_ = spotVillageRoadIdsArr[1]
|
|
spotVillageRoadIds = spotVillageRoadIds.concat(spotVillageRoadIds_)
|
|
if (villageMil_) villageMil += villageMil_
|
|
|
|
allTownCodeResTown.push({
|
|
code: t.township_code,
|
|
township: townRoadTotalMileage + townRoadHide,
|
|
townshipParticipate: townRoadTotalMileage,
|
|
townshipPresent: townMil_,
|
|
townshipDifferenceValue: townMil_ - townRoadNeedMileage,
|
|
village: villageRoadTotalMileage + villageRoadTotalHide,
|
|
villageParticipate: villageRoadTotalMileage,
|
|
villagePresent: villageMil_,
|
|
villageDifferenceValue: villageMil_ - villageRoadNeedMileage
|
|
})
|
|
|
|
}
|
|
|
|
town.forEach(async d => {
|
|
if (allTownCodeResTown.find(c => c.code == d.code)) {
|
|
gather.push({ ...(allTownCodeResTown.find(c => c.code == d.code) || {}), name: d.name })
|
|
} else {
|
|
gather.push({
|
|
name: d.name,
|
|
township: await getRoadTotalMileage('乡', [`township_code='${d.code}'`], false),
|
|
village: await getRoadTotalMileage('村', [`township_code='${d.code}'`], false),
|
|
})
|
|
}
|
|
})
|
|
|
|
|
|
|
|
const previewRes = await models.RoadSpotCheckPreview.create({
|
|
countyPercentage: countyPercentage,
|
|
date: moment().format(),
|
|
countyRoadId: spotCountyRoadIds,
|
|
townshipRoadId: spotTownRoadIds,
|
|
villageRoadId: spotVillageRoadIds,
|
|
villageId: spotVillageIds,
|
|
checked: false,
|
|
villageMil,
|
|
townMil,
|
|
countryMil,
|
|
gather
|
|
})
|
|
|
|
|
|
|
|
ctx.status = 200;
|
|
ctx.body = {
|
|
previewId: previewRes.id,
|
|
spotCountyRoadCount: spotCountyRoadIds.length,
|
|
spotTownRoadCount: spotTownRoadIds.length,
|
|
spotVillageRoadCount: spotVillageRoadIds.length,
|
|
villageMil,
|
|
townMil,
|
|
countryMil
|
|
|
|
}
|
|
} catch (error) {
|
|
ctx.fs.logger.error(`path: ${ctx.path}, error: ${error}`);
|
|
ctx.status = 400;
|
|
ctx.body = {
|
|
message: typeof error == 'string' ? error : undefined
|
|
}
|
|
}
|
|
}
|
|
|
|
async function confirmRoadSpot (ctx) {
|
|
const transaction = await ctx.fs.dc.orm.transaction();
|
|
try {
|
|
const { models } = ctx.fs.dc;
|
|
const sequelize = ctx.fs.dc.orm;
|
|
const { previewId } = ctx.request.body
|
|
|
|
const prepareRes = await models.RoadSpotCheckPreview.findOne({
|
|
where: {
|
|
id: previewId
|
|
}
|
|
})
|
|
let spotedRoadIds = []
|
|
spotedRoadIds = spotedRoadIds.concat(prepareRes.dataValues.countyRoadId)
|
|
spotedRoadIds = spotedRoadIds.concat(prepareRes.dataValues.townshipRoadId)
|
|
spotedRoadIds = spotedRoadIds.concat(prepareRes.dataValues.villageRoadId)
|
|
|
|
const conserveCountRes = await sequelize.query(`
|
|
SELECT road_id, COUNT(road_id) as count
|
|
FROM report
|
|
WHERE road_id IN (
|
|
${spotedRoadIds.map(item => item).join(',')}
|
|
) AND report_type = 'conserve'
|
|
GROUP BY road_id
|
|
`, { type: QueryTypes.SELECT })
|
|
|
|
let spotRslt = []
|
|
for await (let item of spotedRoadIds) {
|
|
let corConserveCount = conserveCountRes.find(cc => cc.road_id == item)
|
|
spotRslt.push({
|
|
roadId: item,
|
|
maintenanceCount: corConserveCount ? corConserveCount.count : 0,
|
|
prepareId: previewId
|
|
})
|
|
}
|
|
|
|
await models.RoadSpotCheck.bulkCreate(spotRslt, {
|
|
transaction
|
|
})
|
|
await models.RoadSpotCheckPreview.update({
|
|
checked: true
|
|
}, {
|
|
where: {
|
|
id: previewId
|
|
},
|
|
transaction
|
|
})
|
|
|
|
await transaction.commit();
|
|
ctx.status = 204;
|
|
} catch (error) {
|
|
await transaction.rollback();
|
|
ctx.fs.logger.error(`path: ${ctx.path}, error: ${error}`);
|
|
ctx.status = 400;
|
|
ctx.body = {
|
|
message: typeof error == 'string' ? error : undefined
|
|
}
|
|
}
|
|
}
|
|
|
|
async function roadSpotList (ctx) {
|
|
try {
|
|
const { models } = ctx.fs.dc;
|
|
const { startTime, endTime, page, limit } = ctx.query
|
|
|
|
let findOpt = {
|
|
order: [['id', 'DESC']],
|
|
where: {
|
|
checked: true
|
|
}
|
|
}
|
|
|
|
if (startTime && endTime) {
|
|
findOpt.where.date = {
|
|
$between: [moment(startTime).startOf('day').format(), moment(endTime).endOf('day').format()]
|
|
}
|
|
|
|
}
|
|
|
|
if (page && limit) {
|
|
findOpt.offset = (page - 1) * limit
|
|
findOpt.limit = limit
|
|
}
|
|
|
|
const listRes = await models.RoadSpotCheckPreview.findAll(findOpt)
|
|
|
|
|
|
|
|
ctx.status = 200;
|
|
ctx.body = listRes.map(item => {
|
|
return {
|
|
id: item.id,
|
|
date: item.date,
|
|
countyPercentage: item.countyPercentage,
|
|
spotCountyRoadCount: item.countyRoadId ? item.countyRoadId.length : 0,
|
|
spotTownRoadCount: item.townshipRoadId ? item.townshipRoadId.length : 0,
|
|
spotVillageRoadCount: item.villageRoadId ? item.villageRoadId.length : 0,
|
|
countyRoadId: item.countyRoadId,
|
|
townshipRoadId: item.townshipRoadId,
|
|
villageRoadId: item.villageRoadId,
|
|
villageMil: item.villageMil,
|
|
townMil: item.townMil,
|
|
countryMil: item.countryMil
|
|
}
|
|
})
|
|
} catch (error) {
|
|
ctx.fs.logger.error(`path: ${ctx.path}, error: ${error}`);
|
|
ctx.status = 400;
|
|
ctx.body = {
|
|
message: typeof error == 'string' ? error : undefined
|
|
}
|
|
}
|
|
}
|
|
|
|
async function roadSpotDetail (ctx) {
|
|
try {
|
|
const { models } = ctx.fs.dc;
|
|
const { previewId, keyword } = ctx.query
|
|
|
|
const listRes = await models.RoadSpotCheck.findAll({
|
|
where: {
|
|
prepareId: previewId,
|
|
},
|
|
include: [{
|
|
model: models.Road,
|
|
where: {
|
|
...(keyword ? { routeName: { $ilike: `%${keyword}%` } } : {}),
|
|
del: false,
|
|
},
|
|
}, {
|
|
model: models.RoadSpotCheckPreview,
|
|
attributes: ['id'],
|
|
include: [{
|
|
model: models.RoadSpotCheckChangeLog,
|
|
attributes: ['id', 'changeRoadId'],
|
|
}]
|
|
}]
|
|
})
|
|
|
|
|
|
ctx.status = 200;
|
|
ctx.body = listRes
|
|
} catch (error) {
|
|
ctx.fs.logger.error(`path: ${ctx.path}, error: ${error}`);
|
|
ctx.status = 400;
|
|
ctx.body = {
|
|
message: typeof error == 'string' ? error : undefined
|
|
}
|
|
}
|
|
}
|
|
|
|
async function roadSpotChange (ctx) {
|
|
const transaction = await ctx.fs.dc.orm.transaction();
|
|
try {
|
|
const { models } = ctx.fs.dc;
|
|
const sequelize = ctx.fs.dc.orm;
|
|
const { originRoadId, changeRoadId, previewId } = ctx.request.body
|
|
|
|
if (originRoadId == changeRoadId) {
|
|
throw '原路段与目标路段不能相同'
|
|
}
|
|
|
|
const previewRes = await models.RoadSpotCheckPreview.findOne({
|
|
where: {
|
|
id: previewId
|
|
}
|
|
})
|
|
|
|
if (!previewRes) {
|
|
throw '预览数据不存在'
|
|
}
|
|
|
|
let previewUpdated = false
|
|
async function updatePreview (key) {
|
|
if (previewUpdated) return
|
|
|
|
if (previewRes[key] && previewRes[key].includes(originRoadId)) {
|
|
let originRoadIds = previewRes.dataValues[key]
|
|
let originRoadIdIndex = originRoadIds.indexOf(originRoadId)
|
|
|
|
originRoadIds.splice(originRoadIdIndex, 1, changeRoadId)
|
|
|
|
await models.RoadSpotCheckPreview.update({
|
|
[key]: originRoadIds
|
|
}, {
|
|
where: {
|
|
id: previewId
|
|
},
|
|
transaction
|
|
})
|
|
previewUpdated = true
|
|
}
|
|
}
|
|
await updatePreview('countyRoadId')
|
|
await updatePreview('townshipRoadId')
|
|
await updatePreview('villageRoadId')
|
|
|
|
if (!previewUpdated) {
|
|
throw '更改路段不在抽查范围内'
|
|
}
|
|
|
|
const roadRes = await models.Road.findAll({
|
|
where: {
|
|
id: { $in: [originRoadId, changeRoadId] },
|
|
del: false,
|
|
}
|
|
})
|
|
|
|
if (roadRes.length != 2) {
|
|
throw '路段不存在'
|
|
}
|
|
|
|
let content = ''
|
|
let curOriginRoad = roadRes.find(item => item.id == originRoadId)
|
|
let curChangeRoad = roadRes.find(item => item.id == changeRoadId)
|
|
|
|
content = `将${curOriginRoad.routeName}从${curOriginRoad.startingPlaceName}到${curOriginRoad.stopPlaceName}改为${curChangeRoad.routeName}从${curChangeRoad.startingPlaceName}到${curChangeRoad.stopPlaceName}`
|
|
|
|
const maintenanceCount = await sequelize.query(`
|
|
SELECT COUNT(id) as count
|
|
FROM report
|
|
WHERE report_type = 'conserve' AND road_id=${changeRoadId}
|
|
`, { type: QueryTypes.SELECT })
|
|
|
|
await models.RoadSpotCheck.update({
|
|
roadId: changeRoadId,
|
|
maintenanceCount: maintenanceCount[0].count
|
|
}, {
|
|
where: {
|
|
roadId: originRoadId,
|
|
prepareId: previewId
|
|
},
|
|
transaction
|
|
})
|
|
await models.RoadSpotCheckChangeLog.create({
|
|
userId: ctx.fs.api.userId,
|
|
time: moment().format(),
|
|
originRoadId: originRoadId,
|
|
changeRoadId: changeRoadId,
|
|
content: content,
|
|
prepareId: previewId
|
|
}, {
|
|
transaction
|
|
})
|
|
|
|
|
|
await transaction.commit();
|
|
ctx.status = 204;
|
|
} catch (error) {
|
|
await transaction.rollback();
|
|
ctx.fs.logger.error(`path: ${ctx.path}, error: ${error}`);
|
|
ctx.status = 400;
|
|
ctx.body = {
|
|
message: typeof error == 'string' ? error : undefined
|
|
}
|
|
}
|
|
}
|
|
|
|
async function roadSpotChangList (ctx) {
|
|
try {
|
|
const { models } = ctx.fs.dc;
|
|
const { startTime, endTime, page, limit } = ctx.query
|
|
|
|
let findOptPreview = {
|
|
where: {
|
|
checked: true,
|
|
},
|
|
attributes: ['id', 'date']
|
|
}
|
|
|
|
if (startTime && endTime) {
|
|
findOptPreview.where = {
|
|
date: {
|
|
$between: [moment(startTime).startOf('day').format(), moment(endTime).endOf('day').format()]
|
|
}
|
|
}
|
|
}
|
|
|
|
let findOpt = {
|
|
order: [['id', 'DESC']],
|
|
where: {
|
|
|
|
},
|
|
include: [{
|
|
model: models.RoadSpotCheckPreview,
|
|
...findOptPreview,
|
|
required: true,
|
|
}, {
|
|
model: models.User,
|
|
attributes: ['name']
|
|
}]
|
|
}
|
|
|
|
if (page && limit) {
|
|
findOpt.offset = (page - 1) * limit
|
|
findOpt.limit = limit
|
|
}
|
|
|
|
const listRes = await models.RoadSpotCheckChangeLog.findAll(findOpt)
|
|
|
|
ctx.status = 200;
|
|
ctx.body = listRes
|
|
} catch (error) {
|
|
ctx.fs.logger.error(`path: ${ctx.path}, error: ${error}`);
|
|
ctx.status = 400;
|
|
ctx.body = {
|
|
message: typeof error == 'string' ? error : undefined
|
|
}
|
|
}
|
|
}
|
|
|
|
async function exportSpotRode (ctx) {
|
|
try {
|
|
const { models } = ctx.fs.dc;
|
|
const { previewId } = ctx.query
|
|
|
|
const previewRes = await models.RoadSpotCheckPreview.findOne({
|
|
where: {
|
|
id: previewId
|
|
}
|
|
})
|
|
|
|
const listRes = await models.RoadSpotCheck.findAll({
|
|
where: {
|
|
prepareId: previewId
|
|
},
|
|
include: [{
|
|
model: models.Road,
|
|
// required: false,
|
|
where: {
|
|
del: false,
|
|
}
|
|
}]
|
|
})
|
|
|
|
const header = [{
|
|
key: 'level',
|
|
title: '道路类型',
|
|
}, {
|
|
key: 'routeName',
|
|
title: '路线名称',
|
|
}, {
|
|
key: 'routeCode',
|
|
title: '路线代码',
|
|
}, {
|
|
key: 'sectionNo',
|
|
title: '路段序号',
|
|
}, {
|
|
key: 'startingPlaceName',
|
|
title: '起点名称',
|
|
}, {
|
|
key: 'stopPlaceName',
|
|
title: '止点名称',
|
|
}, {
|
|
key: 'chainageMileage',
|
|
title: '里程',
|
|
}, {
|
|
key: 'maintenanceCount',
|
|
title: '养护次数(次)',
|
|
},]
|
|
|
|
function judgeLevel (routeCode) {
|
|
if (routeCode) {
|
|
if (routeCode.startsWith('X')) {
|
|
return '县道'
|
|
} else if (routeCode.startsWith('Y')) {
|
|
return '乡道'
|
|
} else if (routeCode.startsWith('C')) {
|
|
return '村道'
|
|
}
|
|
return ''
|
|
} else {
|
|
return ''
|
|
}
|
|
}
|
|
|
|
let exportData = listRes.map(({ dataValues: item }) => {
|
|
let road = item.road && item.road.dataValues || {}
|
|
return {
|
|
level:
|
|
judgeLevel(road.routeCode)
|
|
,
|
|
routeName: road.routeName,
|
|
routeCode: road.routeCode,
|
|
sectionNo: road.sectionNo,
|
|
startingPlaceName: road.startingPlaceName,
|
|
stopPlaceName: road.stopPlaceName,
|
|
chainageMileage: road.chainageMileage,
|
|
maintenanceCount: item.maintenanceCount,
|
|
}
|
|
})
|
|
|
|
const fileName = `${moment(previewRes.date).format('YYYY年MM月DD日HH时mm分')}道路抽查记录` + '.csv'
|
|
const filePath = await xlsxDownload.simpleExcelDown({
|
|
// data: exportData, header, fileName: fileName, gather: { data: previewRes.gather || [] }
|
|
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
|
|
}
|
|
}
|
|
}
|
|
|
|
module.exports = {
|
|
reportList,
|
|
reportPosition,
|
|
reportDetail, createReport, deleteReport, reportHandle,
|
|
spotPrepare, spotCheck, spotCheckDetail,
|
|
roadSpotPrepare, confirmRoadSpot, roadSpotList, roadSpotDetail, roadSpotChange, roadSpotChangList, exportSpotRode
|
|
};
|