settle.js 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323
  1. 'use strict';
  2. /**
  3. *
  4. *
  5. * @author Mai
  6. * @date
  7. * @version
  8. */
  9. const auditConst = require('../const/audit');
  10. const projectLogConst = require('../const/project_log');
  11. module.exports = app => {
  12. class Settle extends app.BaseService {
  13. /**
  14. * 构造函数
  15. *
  16. * @param {Object} ctx - egg全局变量
  17. * @return {void}
  18. */
  19. constructor(ctx) {
  20. super(ctx);
  21. this.tableName = 'settle';
  22. }
  23. /**
  24. * 获取标段下的全部结算期,按倒序
  25. * @param tenderId
  26. * @return {Promise<void>}
  27. */
  28. async getValidSettles(tenderId) {
  29. const settles = await this.db.select(this.tableName, {
  30. where: { tid: tenderId },
  31. orders: [['settle_order', 'desc']],
  32. });
  33. if (settles.length !== 0 && !this.ctx.session.sessionUser.is_admin) {
  34. const last = settles[0];
  35. if (last.status === auditConst.settle.status.uncheck && !this.ctx.tender.isTourist) {
  36. if (last.user_id !== this.ctx.session.sessionUser.accountId) {
  37. settles.splice(0, 1);
  38. }
  39. }
  40. }
  41. return settles;
  42. }
  43. async getLatestCompleteSettle(tenderId) {
  44. const settles = await this.getAllDataByCondition({
  45. where: { tid: tenderId, status: auditConst.settle.status.checked },
  46. order: [['settle_order', 'desc']],
  47. limit: 1, offset: 0
  48. });
  49. return settles[0];
  50. }
  51. /**
  52. * 新增结算期
  53. * @param tenderId - 标段id
  54. * @param date - 结算年月
  55. * @param period - 结算周期
  56. * @return {Promise<void>}
  57. */
  58. async addSettle(tenderId, date, period) {
  59. const settles = await this.getAllDataByCondition({
  60. where: { tid: tenderId },
  61. order: ['order'],
  62. });
  63. const pre = settles[settles.length - 1];
  64. if (settles.length > 0 && pre.status !== auditConst.settle.status.checked) {
  65. throw '上一期未审批通过,请等待上一期审批通过后,再新增数据';
  66. }
  67. const checkedStage = await this.ctx.service.stage.getLastestCompleteStage(tenderId);
  68. if (!checkedStage) throw '不存在审批通过的计量期,请先进行期计量,再结算';
  69. const newSettle = {
  70. tid: tenderId,
  71. add_sid: checkedStage.order, add_sorder: checkedStage.order,
  72. user_id: this.ctx.session.sessionUser.accountId,
  73. settle_order: settles.length + 1, settle_time: date, settle_period: period,
  74. audit_times: 1, audit_status: auditConst.settle.status.uncheck,
  75. };
  76. if (pre) {
  77. newSettle.pre_contract_tp = this.ctx.helper.add(pre.pre_contract_tp, pre.contract_tp);
  78. newSettle.pre_positive_qc_tp = this.ctx.helper.add(pre.pre_positive_qc_tp, pre.positive_qc_tp);
  79. newSettle.pre_negative_qc_tp = this.ctx.helper.add(pre.pre_negative_qc_tp, pre.negative_qc_tp);
  80. newSettle.pre_qc_tp = this.ctx.helper.add(pre.pre_qc_tp, pre.qc_tp);
  81. newSettle.pre_tp = this.ctx.helper.add(pre.tp, pre.pre_tp);
  82. }
  83. const transaction = await this.db.beginTransaction();
  84. try {
  85. // 新增期记录
  86. const result = await transaction.insert(this.tableName, newSettle);
  87. if (result.affectedRows === 1) {
  88. newSettle.id = result.insertId;
  89. } else {
  90. throw '新增期数据失败';
  91. }
  92. await this.ctx.service.settleAudit.copyPreAuditors(transaction, pre, newSettle);
  93. // 存在上一期时
  94. if (pre) {
  95. // todo 复制上一期其他数据
  96. }
  97. await transaction.commit();
  98. } catch (err) {
  99. await transaction.rollback();
  100. throw err;
  101. }
  102. return newSettle;
  103. }
  104. /**
  105. * 编辑结算期
  106. *
  107. * @param {Number} tenderId - 标段Id
  108. * @param {Number} order - 第N期
  109. * @param {String} date - 结算年月
  110. * @param {String} period - 结算周期
  111. * @return {Promise<void>}
  112. */
  113. async saveSettle(tenderId, order, date, period) {
  114. await this.db.update(this.tableName, {
  115. settle_time: date,
  116. settle_period: period,
  117. }, { where: { tid: tenderId, settle_order: order } });
  118. }
  119. /**
  120. * 删除结算期
  121. * @param id
  122. * @returns {Promise<boolean>}
  123. */
  124. async deleteSettle(id) {
  125. const transaction = await this.db.beginTransaction();
  126. try {
  127. const settleInfo = await this.getDataById(id);
  128. await transaction.delete(this.tableName, { id });
  129. await transaction.delete(this.ctx.service.settleAudit.tableName, { sid: id });
  130. // await transaction.delete(this.ctx.service.settleAuditAss.tableName, { sid: id });
  131. // 记录删除日志
  132. await this.ctx.service.projectLog.addProjectLog(transaction, projectLogConst.type.settle, projectLogConst.status.delete, `第${settleInfo.order}期`);
  133. await transaction.commit();
  134. return true;
  135. } catch (err) {
  136. await transaction.rollback();
  137. throw err;
  138. }
  139. }
  140. async loadRelaUser(settle) {
  141. const status = auditConst.settle.status;
  142. const accountId = this.ctx.session.sessionUser.accountId;
  143. settle.user = await this.ctx.service.projectAccount.getAccountInfoById(settle.user_id);
  144. settle.auditors = await this.ctx.service.settleAudit.getAuditors(settle.id, settle.audit_times); // 全部参与的审批人
  145. settle.auditorIds = this._.map(settle.auditors, 'aid');
  146. settle.curAuditors = settle.auditors.filter(x => { return x.status === status.checking; }); // 当前流程中审批中的审批人
  147. settle.curAuditorIds = this._.map(settle.curAuditors, 'aid');
  148. settle.flowAuditors = settle.curAuditors.length > 0 ? settle.auditors.filter(x => { return x.order === settle.curAuditors[0].order; }) : []; // 当前流程中参与的审批人(包含会签时,审批通过的人)
  149. settle.flowAuditorIds = this._.map(settle.flowAuditors, 'aid');
  150. settle.auditorGroups = this.ctx.helper.groupAuditors(settle.auditors);
  151. settle.userGroups = this.ctx.helper.groupAuditorsUniq(settle.auditorGroups);
  152. settle.finalAuditorIds = settle.userGroups[settle.userGroups.length - 1].map(x => { return x.aid; });
  153. // 协作相关
  154. settle.assists = []; //await this.service.settleAuditAss.getData(settle); // 全部协同人
  155. settle.assists = settle.assists.filter(x => {
  156. return x.user_id === settle.user_id || settle.auditorIds.indexOf(x.user_id) >= 0;
  157. }); // 过滤无效协同人
  158. settle.userAssists = settle.assists.filter(x => { return x.user_id === settle.user_id; }); // 原报协同人
  159. settle.userAssistIds = this._.map(settle.userAssists, 'ass_user_id');
  160. settle.auditAssists = settle.assists.filter(x => { return x.user_id !== settle.user_id; }); // 审批协同人
  161. settle.auditAssistIds = this._.map(settle.auditAssists, 'ass_user_id');
  162. settle.curAssists = settle.assists.filter(x => { return settle.curAuditorIds.indexOf(x.user_id) >= 0; }); // 当前审批人的协同人
  163. settle.curAssistsIds = this._.map(settle.curAssists, 'ass_user_id');
  164. settle.relaAssists = settle.assists.filter(x => { return x.user_id === accountId }); // 登录人的协同人
  165. // 当前参与人Id
  166. settle.userIds = settle.status === settle.uncheck // 当前流程下全部参与人id
  167. ? [settle.user_id, ...settle.userAssistIds]
  168. : [settle.user_id, ...settle.userAssistIds, ...settle.auditorIds, ...settle.auditAssistIds];
  169. }
  170. async loadAuditViewData(settle) {
  171. const times = settle.audit_status === auditConst.settle.status.checkNo ? settle.audit_times - 1 : settle.audit_times;
  172. if (!settle.user) settle.user = await this.ctx.service.projectAccount.getAccountInfoById(settle.user_id);
  173. settle.auditHistory = await this.ctx.service.settleAudit.getAuditorHistory(settle.id, times);
  174. // 获取审批流程中左边列表
  175. if (settle.status === auditConst.settle.status.checkNo && settle.user_id !== this.ctx.session.sessionUser.accountId) {
  176. const auditors = await this.ctx.service.settleAudit.getAuditors(settle.id, settle.audit_times - 1); // 全部参与的审批人
  177. const auditorGroups = this.ctx.helper.groupAuditors(auditors);
  178. settle.hisUserGroup = this.ctx.helper.groupAuditorsUniq(auditorGroups);
  179. } else {
  180. settle.hisUserGroup = settle.userGroups;
  181. }
  182. }
  183. /**
  184. * cancancel = 0 不可撤回
  185. * cancancel = 1 原报撤回
  186. * cancancel = 2 审批人撤回 审批通过
  187. * cancancel = 3 审批人撤回 审批退回上一人
  188. * cancancel = 4 审批人撤回 退回原报
  189. * cancancel = 5 会签未全部审批通过时,审批人撤回 审批通过
  190. *
  191. * @param settle
  192. * @returns {Promise<void>}
  193. */
  194. async _doCheckSettleCanCancel(settle) {
  195. // 默认不可撤回
  196. settle.cancancel = 0;
  197. // 获取当前审批人的上一个审批人,判断是否是当前登录人,并赋予撤回功能,(当审批人存在有审批过时,上一人不允许再撤回)
  198. const status = auditConst.settle.status;
  199. if (settle.status === status.checked || settle.status === status.uncheck) return;
  200. const accountId = this.ctx.session.sessionUser.accountId;
  201. if (settle.status !== status.checkNo) {
  202. // 找出当前操作人上一个审批人,包括审批完成的和退回上一个审批人的,同时当前操作人为第一人时,就是则为原报
  203. if (settle.flowAuditors.find(x => { return x.status !== status.checking}) && settle.flowAuditorIds.indexOf(accountId) < 0) return; // 当前流程存在审批人审批通过时,不可撤回
  204. const flowAssists = settle.auditAssists.filter(x => { return settle.flowAuditorIds.indexOf(x.user_id) >= 0; });
  205. if (flowAssists.find(x => { return x.confirm; })) return; //当前流程存在协审人确认时,不可撤回
  206. if (settle.curAuditorIds.indexOf(accountId) < 0 && settle.flowAuditorIds.indexOf(accountId) >= 0) {
  207. settle.cancancel = 5; // 会签未全部审批通过时,审批人撤回审批通过
  208. return;
  209. }
  210. const preAuditors = settle.curAuditors[0] && settle.curAuditors[0].order !== 1 ? settle.auditors.filter(x => { return x.order === settle.curAuditors[0].order - 1; }) : [];
  211. const preAuditorCheckAgain = preAuditors.find(pa => { return pa.status === status.checkAgain; });
  212. const preAuditorCheckCancel = preAuditors.find(pa => { return pa.status === status.checkCancel; });
  213. const preAuditorHasOld = preAuditors.find(pa => { return pa.is_old === 1; });
  214. const preAuditorIds = (preAuditorCheckAgain ? [] : preAuditors.map(x => { return x.aid })); // 重审不可撤回
  215. if ((this._.isEqual(settle.flowAuditorIds, preAuditorIds) && preAuditorCheckCancel) || preAuditorHasOld) {
  216. return; // 不可以多次撤回
  217. }
  218. const preAuditChecked = preAuditors.find(pa => { return pa.status === status.checked && pa.aid === accountId; });
  219. const preAuditCheckNoPre = preAuditors.find(pa => { return pa.status === status.checkNoPre && pa.aid === accountId; });
  220. if (preAuditorIds.indexOf(accountId) >= 0) {
  221. if (preAuditChecked) {
  222. settle.cancancel = 2;// 审批人撤回审批通过
  223. } else if (preAuditCheckNoPre) {
  224. settle.cancancel = 3;// 审批人撤回审批退回上一人
  225. }
  226. settle.preAuditors = preAuditors;
  227. } else if (preAuditors.length === 0 && accountId === settle.user_id) {
  228. settle.cancancel = 1;// 原报撤回
  229. }
  230. } else {
  231. const lastAuditors = await this.service.settleAudit.getAuditors(settle.id, settle.times - 1);
  232. const onAuditor = _.findLast(lastAuditors, { status: status.checkNo });
  233. if (onAuditor.aid === accountId) {
  234. settle.cancancel = 4;// 审批人撤回退回原报
  235. settle.preAuditors = lastAuditors.filter(x => { return x.order === onAuditor.order });
  236. }
  237. }
  238. }
  239. async _doCheckSettleReadOnly(settle) {
  240. const status = auditConst.settle.status;
  241. // 校验权限(参与人、分享、游客)
  242. const accountId = this.ctx.session.sessionUser.accountId;
  243. const shareIds = [];
  244. // 是否只读
  245. if (settle.status === status.uncheck || settle.status === status.checkNo) {
  246. settle.readOnly = accountId !== settle.user_id && settle.userAssistIds.indexOf(accountId) < 0;
  247. } else {
  248. settle.readOnly = true;
  249. }
  250. // 读取数据相关
  251. settle.curTimes = settle.status === status.checkNo && settle.readOnly ? settle.times - 1 : settle.times;
  252. // 协作人相关
  253. if (settle.status === status.uncheck) {
  254. if (!settle.readOnly) {
  255. settle.assist = settle.userAssists.find(x => { return x.ass_user_id === accountId; });
  256. }
  257. settle.curTimes = settle.times;
  258. } else if (settle.status === status.checkNo) {
  259. if (!settle.readOnly) {
  260. settle.assist = settle.userAssists.find(x => { return x.ass_user_id === accountId; });
  261. settle.curTimes = settle.times;
  262. }
  263. } else {
  264. const ass = settle.auditAssists.find(x => { return settle.flowAuditorIds.indexOf(x.user_id) >= 0 && x.ass_user_id === accountId; });
  265. if (!settle.readOnly) {
  266. settle.assist = ass;
  267. }
  268. if (!settle.readOnly) {
  269. settle.readOnly = !_.isEqual(settle.flowAuditorIds, settle.curAuditorIds);
  270. settle.canCheck = true;
  271. }
  272. }
  273. if (settle.readOnly) {
  274. settle.assist = accountId === settle.user_id || settle.auditorIds.indexOf(accountId) >= 0 ? null : settle.auditAssists.find(x => { return x.ass_user_id === accountId});
  275. }
  276. // 上传文件权限
  277. const permission = this.ctx.session.sessionUser.permission;
  278. if (settle.userIds.indexOf(accountId) >= 0 || this.ctx.session.sessionUser.is_admin) {
  279. settle.filePermission = true;
  280. } else {
  281. if (shareIds.indexOf(accountId) !== -1 || (permission !== null && permission.tender !== undefined && permission.tender.indexOf('2') !== -1)) {// 分享人
  282. if (settle.status === status.uncheck) throw '您无权查看该数据';
  283. settle.filePermission = false;
  284. } else if (this.ctx.tender.isTourist || this.ctx.session.sessionUser.is_admin) {
  285. settle.filePermission = this.ctx.tender.touristPermission.file || settle.auditorIds.indexOf(accountId) !== -1;
  286. } else {
  287. throw '您无权查看该数据';
  288. }
  289. }
  290. }
  291. async doCheckSettle(settle) {
  292. // 读取原报、审核人等参与人数据
  293. await this.loadRelaUser(settle);
  294. // 是否只读等权限
  295. await this._doCheckSettleReadOnly(settle);
  296. // 可否撤回,是哪一种撤回
  297. await this._doCheckSettleCanCancel(settle);
  298. }
  299. }
  300. return Settle;
  301. };