facades.js 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508
  1. 'use strict';
  2. /**
  3. *
  4. *
  5. * @author Zhong
  6. * @date 2018/6/1
  7. * @version
  8. */
  9. import mongoose from 'mongoose';
  10. import CompilationModel from "../../users/models/compilation_model";
  11. import moment from 'moment';
  12. const uuidV1 = require('uuid/v1');
  13. const billsLibModel = mongoose.model('std_bills_lib_list');
  14. const billsGuideLibModel = mongoose.model('std_billsGuidance_lib');
  15. const billsGuideItemsModel = mongoose.model('std_billsGuidance_items');
  16. const stdBillsLibModel = mongoose.model('std_bills_lib_list');
  17. const stdBillsModel = mongoose.model('std_bills_lib_bills');
  18. const stdBillsJobsModel = mongoose.model('std_bills_lib_jobContent');
  19. const stdRationModel = mongoose.model('std_ration_lib_ration_items');
  20. const engLibModel = mongoose.model('engineering_lib');
  21. const compilationModel = mongoose.model('compilation');
  22. const billMaterialModel = mongoose.model('std_billsGuidance_materials');
  23. const gljModel = mongoose.model('std_glj_lib_gljList');
  24. const gljLibModel = mongoose.model('std_glj_lib_map');
  25. const _ = require('lodash');
  26. const zhLibID = 'cf851660-3534-11ec-9641-2da8021b8e4e';
  27. const cqLibID = '90c51220-a740-11e8-a354-ab5db7d42428';
  28. module.exports = {
  29. handleCopyItems,
  30. getComBillsLibInfo,
  31. getBillsGuideLibs,
  32. initBillsGuideLib,
  33. updateBillsGuideLib,
  34. getLibWithBills,
  35. getItemsBybills,
  36. updateItems,
  37. getBillMaterials,
  38. editBillMaterials,
  39. testItems
  40. };
  41. function setChildren(bill, parentMap) {
  42. let children = parentMap[bill.ID];
  43. if (children) {
  44. for (let c of children) {
  45. setChildren(c, parentMap);
  46. }
  47. bill.children = children;
  48. } else {
  49. bill.children = [];
  50. }
  51. }
  52. function sortChildren(lists) {
  53. let IDMap = {},
  54. nextMap = {},
  55. firstNode = null,
  56. newList = [];
  57. for (let l of lists) {
  58. if (l.children && l.children.length > 0) l.children = sortChildren(l.children); //递规排序
  59. IDMap[l.ID] = l;
  60. if (l.NextSiblingID != -1) nextMap[l.NextSiblingID] = l;
  61. }
  62. for (let t of lists) {
  63. if (!nextMap[t.ID]) { //如果在下一节点映射没找到,则是第一个节点
  64. firstNode = t;
  65. break;
  66. }
  67. }
  68. if (firstNode) {
  69. newList.push(firstNode);
  70. delete IDMap[firstNode.ID];
  71. setNext(firstNode, newList);
  72. }
  73. //容错处理,如果链断了的情况,直接添加到后面
  74. for (let key in IDMap) {
  75. if (IDMap[key]) newList.push(IDMap[key])
  76. }
  77. return newList;
  78. function setNext(node, array) {
  79. if (node.NextSiblingID != -1) {
  80. let next = IDMap[node.NextSiblingID];
  81. if (next) {
  82. array.push(next);
  83. delete IDMap[next.ID];
  84. setNext(next, array);
  85. }
  86. }
  87. }
  88. }
  89. function getItemFromChildren(children, allItems) {
  90. for (let i = 0; i < children.length; i++) {
  91. const cur = children[i];
  92. const next = children[i + 1];
  93. cur.NextSiblingID = next && next.ID || -1;
  94. allItems.push(cur);
  95. if (cur.children && cur.children.length) {
  96. getItemFromChildren(cur.children, allItems);
  97. }
  98. }
  99. }
  100. function resetTreeData(billsList) {
  101. const idMapping = {};
  102. idMapping['-1'] = -1;
  103. // 建立新ID-旧ID映射
  104. billsList.forEach(bills => idMapping[bills.ID] = uuidV1());
  105. billsList.forEach(function (bills) {
  106. bills.ID = idMapping[bills.ID] ? idMapping[bills.ID] : -1;
  107. bills.ParentID = idMapping[bills.ParentID] ? idMapping[bills.ParentID] : -1;
  108. bills.NextSiblingID = idMapping[bills.NextSiblingID] ? idMapping[bills.NextSiblingID] : -1;
  109. });
  110. }
  111. // 从某库中根据清单编号拷贝工艺
  112. async function handleCopyItems(sourceGuideLibID, sourceBillsLibID, targetGuideLibID, targetBillsLibID) {
  113. const sourceBills = await stdBillsModel.find({ billsLibId: sourceBillsLibID, deleted: false }, '-_id ID code').lean();
  114. const targetBills = await stdBillsModel.find({ billsLibId: targetBillsLibID, deleted: false }, '-_id ID code').lean();
  115. const sourceItems = await billsGuideItemsModel.find({ libID: sourceGuideLibID, type: 0, deleted: false }, '-_id').lean(); // 过滤掉定额项
  116. const targetCodeIDMap = {};
  117. targetBills.forEach(bills => targetCodeIDMap[bills.code] = bills.ID);
  118. const toDeleteIDList = [];
  119. const insertBulks = [];
  120. sourceBills.forEach(bills => {
  121. const matchedItems = sourceItems.filter(item => item.billsID === bills.ID);
  122. // 重新将断缺的树结构数据整理好
  123. if (matchedItems.length) {
  124. const parentMap = {};
  125. matchedItems.forEach(item => {
  126. (parentMap[item.ParentID] || (parentMap[item.ParentID] = [])).push(item);
  127. });
  128. const roots = parentMap['-1'];
  129. roots.forEach(root => setChildren(root, parentMap));
  130. const sorted = sortChildren(roots);
  131. const newItems = [];
  132. getItemFromChildren(sorted, newItems);
  133. resetTreeData(newItems);
  134. const targetBillsID = targetCodeIDMap[bills.code];
  135. if (targetBillsID) {
  136. toDeleteIDList.push(targetBillsID);
  137. newItems.forEach(newItem => {
  138. newItem.libID = targetGuideLibID;
  139. newItem.billsID = targetBillsID;
  140. insertBulks.push({
  141. insertOne: { document: newItem }
  142. });
  143. })
  144. }
  145. }
  146. });
  147. if (toDeleteIDList.length) {
  148. await billsGuideItemsModel.remove({ libID: targetGuideLibID, billsID: { $in : toDeleteIDList } });
  149. }
  150. if (insertBulks.length) {
  151. await billsGuideItemsModel.bulkWrite(insertBulks);
  152. }
  153. }
  154. async function getCompilationList() {
  155. let compilationModel = new CompilationModel();
  156. return await compilationModel.getCompilationList();
  157. }
  158. async function getComBillsLibInfo() {
  159. let rst = {compilationList: [], billsLibs: []};
  160. let compilationList = await getCompilationList();
  161. if(compilationList.length <= 0){
  162. throw '没有数据';
  163. }
  164. else{
  165. for(let compilation of compilationList){
  166. rst.compilationList.push({_id: compilation._id, name: compilation.name});
  167. }
  168. rst.billsLibs = await billsLibModel.find({deleted: false}, '-_id billsLibId billsLibName');
  169. return rst;
  170. }
  171. }
  172. async function getBillsGuideLibs(findData, isTemporary) {
  173. if (isTemporary) {
  174. const libs = await billsGuideLibModel.find({ ID: { $in: [zhLibID, cqLibID] } }).lean();
  175. return libs;
  176. }
  177. return await billsGuideLibModel.find(findData);
  178. }
  179. // 如果是“重庆费用定额(2018)”,则默认叶子清单的项目指引窗口只有一条数据,取清单名称。
  180. async function genGuidanceItems_cq18(guidanceLibId, billsLibId) {
  181. let bills = await stdBillsModel.find({billsLibId: billsLibId, deleted: false, 'jobs.0': {$exists: true}});
  182. let insertArr = [];
  183. for(let bill of bills){
  184. let newItem = {
  185. libID: guidanceLibId,
  186. ID: uuidV1(), ParentID: -1,
  187. NextSiblingID: -1,
  188. name: bill.name,
  189. type: 0,
  190. billsID: bill.ID
  191. };
  192. insertArr.push({insertOne: {document: newItem}});
  193. }
  194. await billsGuideItemsModel.bulkWrite(insertArr);
  195. }
  196. //拷贝工作内容并转化为树结构,形成项目指引数据
  197. async function genGuidanceItems(guidanceLibId, billsLibId){
  198. let bills = await stdBillsModel.find({billsLibId: billsLibId, deleted: false, 'jobs.0': {$exists: true}});
  199. //设置工作内容数据
  200. let jobIds = [];
  201. let totalJobs = [];
  202. for(let bill of bills){
  203. for(let job of bill.jobs){
  204. jobIds.push(job.id);
  205. }
  206. }
  207. jobIds = Array.from(new Set(jobIds));
  208. if(jobIds.length > 0){
  209. totalJobs = await stdBillsJobsModel.find({deleted: false, id: {$in: jobIds}});
  210. }
  211. if(totalJobs.length > 0){
  212. let jobIdIndex = {};//id索引
  213. for(let job of totalJobs){
  214. jobIdIndex[job.id] = job;
  215. }
  216. let insertArr = [];
  217. for(let bill of bills){
  218. //排序后根据serialNo转换成NextSiblingID,倒序
  219. bill.jobs.sort(function (a, b) {
  220. let rst = 0;
  221. if(a.serialNo > b.serialNo){
  222. rst = -1;
  223. }
  224. else if(a.serialNo < b.serialNo){
  225. rst = 1;
  226. }
  227. return rst;
  228. });
  229. let jobNoIndex = {};//下标索引
  230. for(let i = 0; i < bill.jobs.length; i++){
  231. let newItem = {libID: guidanceLibId, ID: uuidV1(), ParentID: -1, NextSiblingID: jobNoIndex[i - 1] ? jobNoIndex[i - 1]['ID'] : -1,
  232. name: jobIdIndex[bill.jobs[i]['id']]['content'], type: 0, billsID: bill.ID};
  233. jobNoIndex[i] = newItem;
  234. insertArr.push({insertOne: {document: newItem}});
  235. }
  236. }
  237. await billsGuideItemsModel.bulkWrite(insertArr);
  238. }
  239. }
  240. async function initBillsGuideLib(updateData){
  241. await billsGuideLibModel.create(updateData);
  242. let compilation = await compilationModel.findOne({_id: mongoose.Types.ObjectId(updateData.compilationId)});
  243. if (compilation &&
  244. compilation.overWriteUrl &&
  245. compilation.overWriteUrl === '/web/over_write/js/chongqing_2018.js') {
  246. await genGuidanceItems_cq18(updateData.ID, updateData.billsLibId);
  247. } else {
  248. await genGuidanceItems(updateData.ID, updateData.billsLibId);
  249. }
  250. }
  251. async function updateBillsGuideLib(data) {
  252. if(data.updateType === 'delete'){
  253. //删除所有条目
  254. await billsGuideLibModel.remove(data.findData);
  255. await billsGuideItemsModel.remove({libID: data.findData.ID});
  256. }
  257. else {
  258. await billsGuideLibModel.update(data.findData, {$set: data.updateData});
  259. await engLibModel.update({'billsGuidance_lib.id': data.findData.ID}, {$set: {'billsGuidance_lib.$.name': data.updateData.name}}, {multi: true});
  260. }
  261. }
  262. async function getLibWithBills(libID){
  263. let guidanceLib = await getBillsGuideLibs({ID: libID});
  264. if(guidanceLib.length === 0){
  265. throw '不存在此指引库!';
  266. }
  267. let billsLib = await stdBillsLibModel.findOne({billsLibId: guidanceLib[0].billsLibId});
  268. if(!billsLib){
  269. throw '引用的清单规则库不存在!';
  270. }
  271. let bills = await stdBillsModel.find({billsLibId: billsLib.billsLibId}, '-_id code name ID NextSiblingID ParentID jobs items comment').lean();
  272. const guideItems = await billsGuideItemsModel.find({ libID: guidanceLib[0].ID }, '-_id billsID').lean();
  273. const billsMap = {};
  274. for (const item of guideItems) {
  275. billsMap[item.billsID] = true;
  276. }
  277. for (const item of bills) {
  278. if (billsMap[item.ID]) {
  279. item.hasGuide = true;
  280. }
  281. }
  282. return {guidanceLib: guidanceLib[0], bills};
  283. }
  284. function getAttrs(field, datas){
  285. let rst = [];
  286. for(let data of datas){
  287. if(data[field]){
  288. rst.push(data[field]);
  289. }
  290. }
  291. return rst;
  292. }
  293. //定额项目指所引用定额是否被删除
  294. function rationAllExist(rationItems, stdRationIdx) {
  295. for(let item of rationItems){
  296. if(!stdRationIdx[item.rationID]){
  297. return false;
  298. }
  299. }
  300. return true;
  301. }
  302. //将同层树结构转为顺序数组
  303. function chainToArr(nodes){
  304. let rst = [];
  305. let tempIdx = {};
  306. let nodeIdx = {};
  307. //建索引
  308. for(let node of nodes){
  309. tempIdx[node.ID] = {ID: node.ID, NextSiblingID: node.NextSiblingID, preSibling: null, nextSibling: null};
  310. nodeIdx[node.ID] = node;
  311. }
  312. //建链
  313. for(let i in tempIdx){
  314. let temp = tempIdx[i];
  315. if(temp.NextSiblingID != -1){
  316. let next = tempIdx[temp.NextSiblingID];
  317. temp.nextSibling = next;
  318. next.preSibling = temp;
  319. }
  320. }
  321. let firstNode = null;
  322. for(let i in tempIdx){
  323. if(!tempIdx[i].preSibling){
  324. firstNode = tempIdx[i];
  325. break;
  326. }
  327. }
  328. //获得顺序队列
  329. while(firstNode){
  330. rst.push(nodeIdx[firstNode.ID]);
  331. firstNode = firstNode.nextSibling;
  332. }
  333. return rst;
  334. }
  335. async function getItemsBybills(guidanceLibID, billsID){
  336. const type = {job: 0, ration: 1};
  337. let items = await billsGuideItemsModel.find({libID: guidanceLibID, billsID: billsID, deleted: false});
  338. let rationItems = _.filter(items, {type: type.ration});
  339. let rationIds = getAttrs('rationID', rationItems);
  340. let stdRations = await stdRationModel.find({ID: {$in: rationIds}, $or: [{isDeleted: null}, {isDeleted: false}]});
  341. let stdRationIndex = {};
  342. for(let stdRation of stdRations){
  343. stdRationIndex[stdRation.ID] = stdRation;
  344. }
  345. //判断定额完整性
  346. if(!rationAllExist(rationItems, stdRationIndex)){
  347. //建定额链, 排序后再清除不存在的定额,保证顺序正确性
  348. rationItems = chainToArr(rationItems);
  349. //清除已被删除的定额
  350. let removeIds = [];
  351. _.remove(rationItems, function (item) {
  352. if(!stdRationIndex[item.rationID]){
  353. removeIds.push(item.ID);
  354. return true;
  355. }
  356. return false;
  357. });
  358. _.remove(items, function (item) {
  359. return removeIds.includes(item.ID);
  360. });
  361. await billsGuideItemsModel.remove({ID: {$in: removeIds}});
  362. //重组树结构
  363. let bulkArr = [];
  364. for(let i = 0, len = rationItems.length; i < len; i++){
  365. rationItems[i].NextSiblingID = rationItems[i + 1] ? rationItems[i + 1].ID : -1;
  366. bulkArr.push({updateOne: {filter: {ID: rationItems[i].ID}, update: {$set: {NextSiblingID: rationItems[i].NextSiblingID}}}});
  367. }
  368. await billsGuideItemsModel.bulkWrite(bulkArr);
  369. }
  370. return items;
  371. }
  372. async function updateItems(updateDatas) {
  373. let bulkArr = [];
  374. for(let updateData of updateDatas){
  375. if(updateData.updateType === 'create'){
  376. bulkArr.push({insertOne: {document: updateData.updateData}});
  377. }
  378. else if(updateData.updateType === 'update'){
  379. bulkArr.push({updateOne: {filter: updateData.findData, update: {$set: updateData.updateData}}});
  380. }
  381. else{
  382. bulkArr.push({deleteOne: {filter: updateData.findData}});
  383. }
  384. }
  385. if(bulkArr.length > 0){
  386. await billsGuideItemsModel.bulkWrite(bulkArr);
  387. }
  388. }
  389. // 获取清单材料数据
  390. async function getBillMaterials(libID, billID) {
  391. const billMaterial = await billMaterialModel.findOne({ libID, billID }).lean();
  392. if (!billMaterial || !billMaterial.materials) {
  393. return [];
  394. }
  395. const gljIDs = billMaterial.materials.map(m => m.gljID);
  396. const gljList = await gljModel.find({ ID: { $in: gljIDs } }, '-_id ID code name specs').lean();
  397. return gljList.map(glj => ({ gljID: glj.ID, code: glj.code, name: glj.name, specs: glj.specs }));
  398. }
  399. // 编辑清单材料数据,返回清单材料数据
  400. async function editBillMaterials(libID, billID, gljCodes, compilationID) {
  401. const gljLib = await gljLibModel.findOne({ compilationId: compilationID }, '-_id ID').lean();
  402. const gljList = await gljModel.find({ repositoryId: gljLib.ID, code: { $in: gljCodes }, }, '-_id ID code name specs').lean();
  403. const gljMap = {};
  404. const materials = [];
  405. gljList.forEach(glj => {
  406. materials.push({ gljID: glj.ID });
  407. gljMap[glj.code] = 1;
  408. });
  409. const missCodes = gljCodes.filter(code => !gljMap[code]);
  410. if (missCodes.length) {
  411. throw new Error(`没有找到人材机:<br/>${missCodes.join('<br/>')}`);
  412. }
  413. const billMaterial = await billMaterialModel.findOne({ libID, billID }, '-_id ID').lean();
  414. if (billMaterial) {
  415. await billMaterialModel.update({ ID: billMaterial.ID }, { $set: { materials } });
  416. } else {
  417. await billMaterialModel.create({ libID, billID, ID: uuidV1(), materials });
  418. }
  419. return gljList.map(glj => ({ gljID: glj.ID, code: glj.code, name: glj.name, specs: glj.specs }));
  420. }
  421. async function testItems(libID) {
  422. let items = await billsGuideItemsModel.find({libID: libID});
  423. //删除垃圾数据
  424. let delBulk = [];
  425. let itemsMapping = {};
  426. for(let item of items){
  427. itemsMapping[item.ID] = true;
  428. }
  429. for(let item of items){
  430. if(item.ParentID != -1 && !itemsMapping[item.ParentID]){
  431. delBulk.push({
  432. deleteOne: {
  433. filter: {ID: item.ID}
  434. }
  435. });
  436. }
  437. }
  438. if(delBulk.length > 0){
  439. console.log(`delBulk.length`);
  440. console.log(delBulk.length);
  441. await billsGuideItemsModel.bulkWrite(delBulk);
  442. }
  443. /* //查找同层节点含有相同NextSiblingID的节点
  444. let rst = [];
  445. let billsGroup = {};
  446. for(let item of items){
  447. if(!billsGroup[item.billsID]){
  448. billsGroup[item.billsID] = [item];
  449. }
  450. else {
  451. billsGroup[item.billsID].push(item);
  452. }
  453. }
  454. for(let bGroup in billsGroup){
  455. let group = billsGroup[bGroup];
  456. let parentGroup = {};
  457. for(let gItem of group){
  458. if(!parentGroup[gItem.ParentID]){
  459. parentGroup[gItem.ParentID] = [gItem]
  460. }
  461. else {
  462. parentGroup[gItem.ParentID].push(gItem);
  463. }
  464. }
  465. for(let pGroup in parentGroup){
  466. let pGroupData = parentGroup[pGroup];
  467. let nextGroup = {};
  468. for(let nItem of pGroupData){
  469. let sameNext = _.filter(pGroupData, {NextSiblingID: nItem.NextSiblingID});
  470. if(sameNext.length > 1){
  471. console.log(`sameNext`);
  472. console.log(sameNext);
  473. if(!nextGroup[nItem.ParentID + nItem.NextSiblingID]){
  474. rst.push({NextSiblingID: nItem.NextSiblingID, ParentID: nItem.ParentID});
  475. nextGroup[nItem.ParentID + nItem.NextSiblingID] = 1;
  476. }
  477. }
  478. }
  479. }
  480. }*/
  481. return delBulk.length;
  482. }