index.js 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464
  1. const mongoose = require('mongoose');
  2. const uuidV1 = require('uuid/v1');
  3. const { CRAWL_LOG_KEY, ProcessStatus } = require('../../../public/constants/price_info_constant');
  4. const priceInfoLibModel = mongoose.model('std_price_info_lib');
  5. const priceInfoClassModel = mongoose.model('std_price_info_class');
  6. const priceInfoItemModel = mongoose.model('std_price_info_items');
  7. const priceInfoAreaModel = mongoose.model('std_price_info_areas');
  8. const compilationModel = mongoose.model('compilation');
  9. const importLogsModel = mongoose.model('import_logs');
  10. async function getLibs(query) {
  11. return await priceInfoLibModel.find(query).lean();
  12. }
  13. async function createLib(name, period, compilationID) {
  14. // 将2020-01变成2020年01月
  15. const reg = /(\d{4})-(\d{2})/;
  16. const formattedPeriod = period.replace(reg, '$1年-$2月');
  17. const lib = {
  18. ID: uuidV1(),
  19. name,
  20. period: formattedPeriod,
  21. compilationID,
  22. createDate: Date.now(),
  23. };
  24. await priceInfoLibModel.create(lib);
  25. return lib;
  26. }
  27. async function updateLib(query, updateData) {
  28. await priceInfoLibModel.update(query, updateData);
  29. }
  30. async function deleteLib(libID) {
  31. await priceInfoClassModel.remove({ libID });
  32. await priceInfoItemModel.remove({ libID });
  33. await priceInfoLibModel.remove({ ID: libID });
  34. }
  35. async function processChecking(key) {
  36. const logData = key
  37. ? await importLogsModel.findOne({ key })
  38. : await importLogsModel.findOne({ key: CRAWL_LOG_KEY });
  39. if (!logData) {
  40. return { status: ProcessStatus.FINISH };
  41. }
  42. if (logData.status === ProcessStatus.FINISH || logData.status === ProcessStatus.ERROR) {
  43. await importLogsModel.remove({ key: logData.key });
  44. }
  45. return { status: logData.status, errorMsg: logData.errorMsg || '', key: logData.key };
  46. }
  47. // 爬取数据
  48. async function crawlDataByCompilation(compilationID, from, to) {
  49. if (!compilationID) {
  50. throw '无有效费用定额。';
  51. }
  52. const compilationData = await compilationModel.findOne({ _id: mongoose.Types.ObjectId(compilationID) }, 'overWriteUrl').lean();
  53. if (!compilationData || !compilationData.overWriteUrl) {
  54. throw '无有效费用定额。';
  55. }
  56. // 从overWriteUrl提取并组装爬虫文件
  57. const reg = /\/([^/]+)\.js/;
  58. const matched = compilationData.overWriteUrl.match(reg);
  59. const crawlURL = `${matched[1]}_price_crawler.js`;
  60. let crawlData;
  61. try {
  62. const crawler = require(`../../../web/over_write/crawler/${crawlURL}`);
  63. crawlData = crawler.crawlData;
  64. } catch (e) {
  65. throw '该费用定额无可用爬虫方法。'
  66. }
  67. //await crawlData(from, to);
  68. // 异步不等结果,结果由checking来获取
  69. crawlDataByMiddleware(crawlData, from, to, compilationID);
  70. }
  71. // 爬取数据中间件,主要处理checking初始化
  72. async function crawlDataByMiddleware(crawlFunc, from, to, compilationID) {
  73. const logUpdateData = { status: ProcessStatus.FINISH };
  74. try {
  75. const logData = {
  76. key: CRAWL_LOG_KEY,
  77. content: '正在爬取数据,请稍候……',
  78. status: ProcessStatus.START,
  79. create_time: Date.now()
  80. };
  81. await importLogsModel.create(logData);
  82. await crawlFunc(from, to, compilationID);
  83. } catch (err) {
  84. console.log(err);
  85. logUpdateData.errorMsg = String(err);
  86. logUpdateData.status = ProcessStatus.ERROR;
  87. } finally {
  88. await importLogsModel.update({ key: CRAWL_LOG_KEY }, logUpdateData);
  89. }
  90. }
  91. // 导入excel数据,格式如下
  92. // 格式1:
  93. //地区 分类 编码 名称 规格型号 单位 不含税价 含税价
  94. //江北区 黑色及有色金属 热轧光圆钢筋 φ6(6.5) 3566.37 4030
  95. //江北区 木、竹材料及其制品 柏木门套线 60×10 8.76 9.9
  96. // 格式2:
  97. //地区 分类 编码 名称 规格型号 不含税价 含税价
  98. //江北区 黑色及有色金属 热轧光圆钢筋 φ6(6.5) 3566.37 4030
  99. // 柏木门套线 60×10 8.76 9.9
  100. // 沥青混凝土 AC-13 982.3 1110
  101. //
  102. //北碚区 木、竹材料及其制品 热轧光圆钢筋 φ6(6.5) 3566.37 4030
  103. async function importExcelData(libID, sheetData) {
  104. const libs = await getLibs({ ID: libID });
  105. const compilationID = libs[0].compilationID;
  106. // 建立区映射表:名称-ID映射、ID-名称映射
  107. const areaList = await getAreas(compilationID);
  108. const areaMap = {};
  109. areaList.forEach(({ ID, name }) => {
  110. areaMap[name] = ID;
  111. areaMap[ID] = name;
  112. });
  113. // 建立分类映射表:地区名称@分类名称:ID映射
  114. /* const classMap = {};
  115. const classList = await getClassData(libID);
  116. classList.forEach(({ ID, areaID, name }) => {
  117. const areaName = areaMap[areaID] || '';
  118. classMap[`${areaName}@${name}`] = ID;
  119. }); */
  120. // 第一行获取行映射
  121. const colMap = {};
  122. for (let col = 0; col < sheetData[0].length; col++) {
  123. const cellText = sheetData[0][col];
  124. switch (cellText) {
  125. case '地区':
  126. colMap.area = col;
  127. break;
  128. case '分类':
  129. colMap.class = col;
  130. break;
  131. case '编码':
  132. colMap.code = col;
  133. break;
  134. case '名称':
  135. colMap.name = col;
  136. break;
  137. case '规格型号':
  138. colMap.specs = col;
  139. break;
  140. case '单位':
  141. colMap.unit = col;
  142. break;
  143. case '不含税价':
  144. colMap.noTaxPrice = col;
  145. break;
  146. case '含税价':
  147. colMap.taxPrice = col;
  148. break;
  149. }
  150. }
  151. // 提取数据
  152. const data = [];
  153. const classData = [];
  154. const areaClassDataMap = {};
  155. let curAreaName;
  156. let curClassName;
  157. let curClassID;
  158. for (let row = 1; row < sheetData.length; row++) {
  159. const areaName = sheetData[row][colMap.area] || '';
  160. const className = sheetData[row][colMap.class] || '';
  161. const code = sheetData[row][colMap.code] || '';
  162. const name = sheetData[row][colMap.name] || '';
  163. const specs = sheetData[row][colMap.specs] || '';
  164. const unit = sheetData[row][colMap.unit] || '';
  165. const noTaxPrice = sheetData[row][colMap.noTaxPrice] || '';
  166. const taxPrice = sheetData[row][colMap.taxPrice] || '';
  167. if (!className && !code && !name && !specs && !noTaxPrice && !taxPrice) { // 认为是空数据
  168. continue;
  169. }
  170. if (areaName && areaName !== curAreaName) {
  171. curAreaName = areaName;
  172. }
  173. const areaID = areaMap[curAreaName];
  174. if (!areaID) {
  175. continue;
  176. }
  177. if (className && className !== curClassName) {
  178. curClassName = className;
  179. const classItem = {
  180. libID,
  181. areaID,
  182. ID: uuidV1(),
  183. ParentID: '-1',
  184. NextSiblingID: '-1',
  185. name: curClassName
  186. };
  187. curClassID = classItem.ID;
  188. classData.push(classItem);
  189. (areaClassDataMap[areaID] || (areaClassDataMap[areaID] = [])).push(classItem);
  190. const preClassItem = areaClassDataMap[areaID][areaClassDataMap[areaID].length - 2];
  191. if (preClassItem) {
  192. preClassItem.NextSiblingID = classItem.ID;
  193. }
  194. }
  195. if (!curClassID) {
  196. continue;
  197. }
  198. data.push({
  199. ID: uuidV1(),
  200. compilationID,
  201. libID,
  202. areaID,
  203. classID: curClassID,
  204. period: libs[0].period,
  205. code,
  206. name,
  207. specs,
  208. unit,
  209. noTaxPrice,
  210. taxPrice
  211. });
  212. }
  213. if (classData.length) {
  214. await priceInfoClassModel.remove({ libID });
  215. await priceInfoClassModel.insertMany(classData);
  216. }
  217. if (data.length) {
  218. await priceInfoItemModel.remove({ libID });
  219. await priceInfoItemModel.insertMany(data);
  220. } else {
  221. throw 'excel没有有效数据。'
  222. }
  223. }
  224. /* async function importExcelData(libID, sheetData) {
  225. const libs = await getLibs({ ID: libID });
  226. const compilationID = libs[0].compilationID;
  227. // 建立区映射表:名称-ID映射、ID-名称映射
  228. const areaList = await getAreas(compilationID);
  229. const areaMap = {};
  230. areaList.forEach(({ ID, name }) => {
  231. areaMap[name] = ID;
  232. areaMap[ID] = name;
  233. });
  234. // 建立分类映射表:地区名称@分类名称:ID映射
  235. const classMap = {};
  236. const classList = await getClassData(libID);
  237. classList.forEach(({ ID, areaID, name }) => {
  238. const areaName = areaMap[areaID] || '';
  239. classMap[`${areaName}@${name}`] = ID;
  240. });
  241. // 第一行获取行映射
  242. const colMap = {};
  243. for (let col = 0; col < sheetData[0].length; col++) {
  244. const cellText = sheetData[0][col];
  245. switch (cellText) {
  246. case '地区':
  247. colMap.area = col;
  248. break;
  249. case '分类':
  250. colMap.class = col;
  251. break;
  252. case '编码':
  253. colMap.code = col;
  254. break;
  255. case '名称':
  256. colMap.name = col;
  257. break;
  258. case '规格型号':
  259. colMap.specs = col;
  260. break;
  261. case '单位':
  262. colMap.unit = col;
  263. break;
  264. case '不含税价':
  265. colMap.noTaxPrice = col;
  266. break;
  267. case '含税价':
  268. colMap.taxPrice = col;
  269. break;
  270. }
  271. }
  272. // 提取数据
  273. const data = [];
  274. let curAreaName;
  275. let curClassName;
  276. for (let row = 1; row < sheetData.length; row++) {
  277. const areaName = sheetData[row][colMap.area] || '';
  278. const className = sheetData[row][colMap.class] || '';
  279. const code = sheetData[row][colMap.code] || '';
  280. const name = sheetData[row][colMap.name] || '';
  281. const specs = sheetData[row][colMap.specs] || '';
  282. const unit = sheetData[row][colMap.unit] || '';
  283. const noTaxPrice = sheetData[row][colMap.noTaxPrice] || '';
  284. const taxPrice = sheetData[row][colMap.taxPrice] || '';
  285. if (!code && !name && !specs && !noTaxPrice && !taxPrice) { // 认为是空数据
  286. continue;
  287. }
  288. if (areaName && areaName !== curAreaName) {
  289. curAreaName = areaName;
  290. }
  291. if (className && className !== curClassName) {
  292. curClassName = className;
  293. }
  294. const areaID = areaMap[curAreaName];
  295. if (!areaID) {
  296. continue;
  297. }
  298. const classID = classMap[`${curAreaName}@${curClassName}`];
  299. if (!classID) {
  300. continue;
  301. }
  302. data.push({
  303. ID: uuidV1(),
  304. compilationID,
  305. libID,
  306. areaID,
  307. classID,
  308. period: libs[0].period,
  309. code,
  310. name,
  311. specs,
  312. unit,
  313. noTaxPrice,
  314. taxPrice
  315. });
  316. }
  317. if (data.length) {
  318. await priceInfoItemModel.remove({ libID });
  319. await priceInfoItemModel.insertMany(data);
  320. } else {
  321. throw 'excel没有有效数据。'
  322. }
  323. } */
  324. // 获取费用定额的地区数据
  325. async function getAreas(compilationID) {
  326. return await priceInfoAreaModel.find({ compilationID }, '-_id ID name serialNo').lean();
  327. }
  328. async function updateAres(updateData) {
  329. const bulks = [];
  330. updateData.forEach(({ ID, name }) => bulks.push({
  331. updateOne: {
  332. filter: { ID },
  333. update: { name }
  334. }
  335. }));
  336. if (bulks.length) {
  337. await priceInfoAreaModel.bulkWrite(bulks);
  338. }
  339. }
  340. async function insertAreas(insertData) {
  341. await priceInfoAreaModel.insertMany(insertData);
  342. }
  343. async function deleteAreas(deleteData) {
  344. await priceInfoClassModel.remove({ areaID: { $in: deleteData } });
  345. await priceInfoItemModel.remove({ areaID: { $in: deleteData } });
  346. await priceInfoAreaModel.remove({ ID: { $in: deleteData } });
  347. }
  348. async function getClassData(libID, areaID) {
  349. if (libID && areaID) {
  350. return await priceInfoClassModel.find({ libID, areaID }, '-_id').lean();
  351. }
  352. if (libID) {
  353. return await priceInfoClassModel.find({ libID }, '-_id').lean();
  354. }
  355. if (areaID) {
  356. return await priceInfoClassModel.find({ areaID }, '-_id').lean();
  357. }
  358. }
  359. async function getPriceData(classIDList) {
  360. return await priceInfoItemModel.find({ classID: { $in: classIDList } }, '-_id').lean();
  361. }
  362. const UpdateType = {
  363. UPDATE: 'update',
  364. DELETE: 'delete',
  365. CREATE: 'create',
  366. };
  367. async function editPriceData(postData) {
  368. const bulks = [];
  369. postData.forEach(data => {
  370. if (data.type === UpdateType.UPDATE) {
  371. bulks.push({
  372. updateOne: {
  373. filter: { ID: data.ID },
  374. update: { ...data.data }
  375. }
  376. });
  377. } else if (data.type === UpdateType.DELETE) {
  378. bulks.push({
  379. deleteOne: {
  380. filter: { ID: data.ID }
  381. }
  382. });
  383. } else {
  384. bulks.push({
  385. insertOne: {
  386. document: data.data
  387. }
  388. });
  389. }
  390. });
  391. if (bulks.length) {
  392. await priceInfoItemModel.bulkWrite(bulks);
  393. }
  394. }
  395. async function editClassData(updateData) {
  396. const bulks = [];
  397. const deleteIDList = [];
  398. updateData.forEach(({ type, filter, update, document }) => {
  399. if (type === UpdateType.UPDATE) {
  400. bulks.push({
  401. updateOne: {
  402. filter,
  403. update
  404. }
  405. });
  406. } else if (type === UpdateType.DELETE) {
  407. deleteIDList.push(filter.ID);
  408. bulks.push({
  409. deleteOne: {
  410. filter
  411. }
  412. });
  413. } else {
  414. bulks.push({
  415. insertOne: {
  416. document
  417. }
  418. });
  419. }
  420. });
  421. if (deleteIDList.length) {
  422. await priceInfoItemModel.remove({ classID: { $in: deleteIDList } });
  423. }
  424. if (bulks.length) {
  425. await priceInfoClassModel.bulkWrite(bulks);
  426. }
  427. }
  428. module.exports = {
  429. getLibs,
  430. createLib,
  431. updateLib,
  432. deleteLib,
  433. processChecking,
  434. crawlDataByCompilation,
  435. importExcelData,
  436. getAreas,
  437. updateAres,
  438. insertAreas,
  439. deleteAreas,
  440. getClassData,
  441. getPriceData,
  442. editPriceData,
  443. editClassData
  444. }