update.go 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456
  1. package main
  2. import (
  3. "github.com/go-xweb/log"
  4. "go.mongodb.org/mongo-driver/bson/primitive"
  5. "math"
  6. qu "qfw/util"
  7. "reflect"
  8. "sort"
  9. "strings"
  10. )
  11. func (p *ProjectTask) modifyUpdate(pInfoId string, index int, info *Info, tmp map[string]interface{}, tmpPro map[string]interface{}, modifyMap map[string]interface{}) {
  12. backupPro(tmpPro)
  13. infoList := []interface{}(tmpPro["list"].(primitive.A))
  14. infoMap := infoList[index].(map[string]interface{})
  15. infoList[index] = updateValue(infoMap, modifyMap)
  16. set := map[string]interface{}{
  17. "$set": tmpPro,
  18. }
  19. _ = MongoTool.UpdateById(ProjectColl, pInfoId, set)
  20. }
  21. func (p *ProjectTask) mergeAndModify(pInfoId string, index int, info *Info, tmp map[string]interface{}, tmpPro map[string]interface{}, modifyMap map[string]interface{}) {
  22. p.AllIdsMapLock.Lock()
  23. currentPro := p.AllIdsMap[pInfoId]
  24. p.AllIdsMapLock.Unlock()
  25. merge, pId, i, comStr, mergePro := p.updateMerge(index, info, pInfoId, tmp, tmpPro)
  26. //招标信息在list中第1个
  27. if index == 0 {
  28. log.Println("第1条合并", "------1------")
  29. //快照
  30. backupPro(tmpPro)
  31. p.AllIdsMapLock.Lock()
  32. delete(p.AllIdsMap, pInfoId)
  33. p.AllIdsMapLock.Unlock()
  34. count := MongoTool.Delete(ProjectColl, pInfoId)
  35. log.Println(count)
  36. for _, infoId := range currentPro.P.Ids{
  37. if infoId == info.Id {
  38. continue
  39. }
  40. tmp1 := MongoTool.FindById(ExtractColl, infoId)
  41. info1 := ParseInfo(tmp1)
  42. p.CommonMerge(tmp1, info1)
  43. }
  44. }else {
  45. log.Println("第n条合并", "------n------")
  46. //快照 招标信息所在的项目
  47. backupPro(tmpPro)
  48. if merge {
  49. //跟原项目list第0条信息作对比,可以合并
  50. if pId == pInfoId {
  51. //合并到当前项目
  52. mergeProject(p, currentPro.P, info, tmpPro, index)
  53. p.AllIdsMapLock.Lock()
  54. p.AllIdsMap[currentPro.P.Id.Hex()].P = currentPro.P
  55. p.AllIdsMapLock.Unlock()
  56. infoList := []interface{}(tmpPro["list"].(primitive.A))
  57. infoMap := infoList[index].(map[string]interface{})
  58. infoList[index] = updateValue(infoMap, modifyMap)
  59. set := map[string]interface{}{
  60. "$set": tmpPro,
  61. }
  62. MongoTool.UpdateById(ProjectColl, pInfoId, set)
  63. }else {
  64. //合并到的其它项目里边
  65. //更新其它的项目
  66. pro := MongoTool.FindById(ProjectColl, mergePro.Id.Hex())
  67. backupPro(pro, )
  68. choose, ex := p.CompareStatus(mergePro, info)
  69. if !choose {
  70. p.UpdateProject(tmp, info, mergePro, i, comStr, ex)
  71. }else {
  72. id, p1 := p.NewProject(tmp, info)
  73. p.AllIdsMapLock.Lock()
  74. p.AllIdsMap[id] = &ID{Id: id, P: p1}
  75. p.AllIdsMapLock.Unlock()
  76. }
  77. //内部合并
  78. p.innerMerge(currentPro.P, info, tmp, tmpPro)
  79. }
  80. }else {
  81. //合并到新建的项目里边,原有项目内部合并
  82. p.innerMerge(currentPro.P, info, tmp, tmpPro)
  83. }
  84. }
  85. }
  86. func (p *ProjectTask) updateMerge(index int, info *Info, pInfoId string, tmp map[string]interface{}, tmpPro map[string]interface{}) (bool, string, int, string, *ProjectInfo) {
  87. bpn, bpc, bptc, bpb, pids, _, IDArr := p.getCompareIds(info.ProjectName, info.ProjectCode, info.PTC, info.Buyer)
  88. defer p.wg.Done()
  89. for _, m := range pids {
  90. defer m.Lock.Unlock()
  91. }
  92. for _, id := range IDArr {
  93. defer id.Lock.Unlock()
  94. }
  95. bFindProject := false
  96. findPid := ""
  97. //获取完id,进行计算
  98. //定义两组
  99. comRes1 := []*ProjectInfo{} //优先级最高的对比结果数组
  100. comRes2 := []*ProjectInfo{} //优化级其次
  101. comRes3 := []*ProjectInfo{}
  102. for _, v := range IDArr {
  103. if index == 0 && pInfoId == v.P.Id.Hex() {
  104. //招标条目信息在原项目中是第0条时,忽略
  105. continue
  106. }
  107. comStr := ""
  108. compareProject := v.P
  109. compareProject.score = 0
  110. diffTime := int64(math.Abs(float64(info.Publishtime - compareProject.LastTime)))
  111. if diffTime <= p.validTime {
  112. //代理机构完全不相同,直接新建项目
  113. if CheckContain(compareProject.Agency, info.Agency) == 3 {
  114. continue
  115. }
  116. //地区(省、市、区)不同,直接新建项目
  117. if ComparePlace(compareProject, info) {
  118. continue
  119. }
  120. info.PNBH = 0
  121. info.PCBH = 0
  122. info.PTCBH = 0
  123. compareStr, score := comparePNC(info, compareProject)
  124. resVal, pjVal := Select(compareStr, info, compareProject)
  125. if resVal > 0 {
  126. compareBuyer, compareCity, compareTime, compareAgency, compareBudget, compareBidmount, score2 := p.compareBCTABB(info, compareProject, diffTime, score)
  127. //项目名称、项目编号、标题项目编号、采购单位、省、市、发布时间、代理机构
  128. comStr = compareStr + compareBuyer + compareCity + compareTime + compareAgency + compareBudget + compareBidmount
  129. compareProject.comStr = comStr
  130. compareProject.pjVal = pjVal
  131. compareProject.resVal = resVal
  132. eqV := compareResult(resVal, pjVal, score2, comStr, compareBuyer, compareCity, compareTime, compareAgency, compareBudget, compareBidmount)
  133. if eqV == 1 {
  134. comRes1 = append(comRes1, compareProject)
  135. } else if eqV == 2 {
  136. comRes2 = append(comRes2, compareProject)
  137. } else if eqV == 3 {
  138. comRes3 = append(comRes3, compareProject)
  139. }
  140. }
  141. }
  142. }
  143. //--------------------------------对比完成-----------------------
  144. k := -1
  145. comStr := ""
  146. mergePro := &ProjectInfo{}
  147. for kv, resN := range [][]*ProjectInfo{comRes1, comRes2, comRes3} {
  148. if len(resN) > 0 {
  149. if len(resN) > 1 {
  150. sort.Slice(resN, func(i, j int) bool {
  151. return resN[i].score > resN[j].score
  152. })
  153. }
  154. ex := 0
  155. resArr := []*ProjectInfo{}
  156. for _, res := range resN{
  157. choose, e := p.CompareStatus(resN[0], info)
  158. if !choose {
  159. ex = e
  160. resArr = append(resArr, res)
  161. }
  162. }
  163. if len(resArr) > 0 {
  164. bFindProject = true
  165. findPid = resArr[0].Id.Hex()
  166. for k2, bv := range []int{bpn, bpc, bptc, bpb} {
  167. if bv > -1 {
  168. pids[bv].Arr = append(pids[bv].Arr, findPid)
  169. if k2 == 0 {
  170. if resArr[0].ProjectName == "" {
  171. resArr[0].ProjectName = info.ProjectName
  172. } else {
  173. if resArr[0].MPN == nil {
  174. resArr[0].MPN = []string{info.ProjectName}
  175. } else {
  176. resArr[0].MPN = append(resArr[0].MPN, info.ProjectName)
  177. }
  178. }
  179. } else if k2 < 3 {
  180. if resArr[0].ProjectCode == "" {
  181. resArr[0].ProjectCode = qu.If(k2 == 1, info.ProjectCode, info.PTC).(string)
  182. } else {
  183. if resArr[0].MPC == nil {
  184. resArr[0].MPC = []string{qu.If(k2 == 1, info.ProjectCode, info.PTC).(string)}
  185. } else {
  186. resArr[0].MPC = append(resArr[0].MPC, qu.If(k2 == 1, info.ProjectCode, info.PTC).(string))
  187. }
  188. }
  189. } else {
  190. if resArr[0].Buyer == "" {
  191. resArr[0].Buyer = info.Buyer
  192. }
  193. }
  194. }
  195. }
  196. if index != 0 {
  197. //招标条目信息在原项目中是第非0条时,返回对比结果
  198. k = kv+1
  199. comStr = resArr[0].comStr
  200. mergePro = resArr[0]
  201. bFindProject = false
  202. findPid = ""
  203. }else {
  204. mergePro = resArr[0]
  205. //如果有合并,合并之前备份(快照)
  206. backupPro(tmpPro)
  207. p.UpdateProject(tmp, info, resArr[0], kv+1, resArr[0].comStr, ex)
  208. }
  209. }else {
  210. bFindProject = false
  211. findPid = ""
  212. }
  213. break
  214. }
  215. }
  216. if !bFindProject {
  217. id, p1 := p.NewProject(tmp, info)
  218. p.AllIdsMapLock.Lock()
  219. p.AllIdsMap[id] = &ID{Id: id, P: p1}
  220. p.AllIdsMapLock.Unlock()
  221. for _, m := range pids {
  222. m.Arr = append(m.Arr, id)
  223. }
  224. }
  225. return bFindProject, findPid, k, comStr, mergePro
  226. }
  227. //内部合并
  228. func (p *ProjectTask) innerMerge(pInfo *ProjectInfo, info *Info, tmp map[string]interface{}, tmpPro map[string]interface{}) {
  229. infoList := []interface{}(tmpPro["list"].(primitive.A))
  230. for index, info1 := range infoList{
  231. info2 := info1.(map[string]interface{})
  232. if info2["infoid"] == info.Id {
  233. deleteSlice1(infoList, info1)
  234. continue
  235. }
  236. mergeProject(p, pInfo, info, tmpPro, index)
  237. }
  238. set := map[string]interface{}{
  239. "$set": tmpPro,
  240. }
  241. MongoTool.UpdateById(ProjectColl, pInfo.Id.Hex(), set)
  242. }
  243. //合并字段到project
  244. func mergeProject(p *ProjectTask, pInfo *ProjectInfo, thisinfo *Info, set map[string]interface{}, index int) map[string]interface{} {
  245. //1--firsttime
  246. if thisinfo.Publishtime < pInfo.FirstTime && thisinfo.Publishtime > 0 {
  247. pInfo.FirstTime = thisinfo.Publishtime
  248. set["firsttime"] = thisinfo.Publishtime
  249. if thisinfo.TopType == "招标" {
  250. set["zbtime"] = thisinfo.Publishtime
  251. }
  252. }
  253. //2--lasttime
  254. if thisinfo.Publishtime > pInfo.LastTime {
  255. pInfo.LastTime = thisinfo.Publishtime
  256. set["lasttime"] = thisinfo.Publishtime
  257. p.mapBidLock.Lock()
  258. bt := bidtype[thisinfo.SubType]
  259. p.mapBidLock.Unlock()
  260. if bt != "" {
  261. set["bidtype"] = bt
  262. }
  263. if thisinfo.SubType != "" {
  264. set["bidstatus"] = thisinfo.SubType
  265. if thisinfo.SubType != "预告" {
  266. set["jgtime"] = thisinfo.Publishtime
  267. }
  268. }else if thisinfo.Infoformat == 2 {
  269. set["bidstatus"] = "拟建"
  270. }else if thisinfo.SubType == "招标" {
  271. set["bidstatus"] = thisinfo.TopType
  272. }else {
  273. set["bidstatus"] = thisinfo.SubType
  274. }
  275. }
  276. //3\4\5--省、市、县
  277. if thisinfo.Area != "全国" {
  278. //xt := true
  279. if pInfo.Area == "全国" {
  280. pInfo.Area = thisinfo.Area
  281. set["area"] = thisinfo.Area
  282. } else if pInfo.Area != thisinfo.Area {
  283. //xt = false
  284. }
  285. if pInfo.City == "" && thisinfo.City != "" {
  286. pInfo.City = thisinfo.City
  287. set["city"] = thisinfo.City
  288. } else if pInfo.City != thisinfo.City {
  289. //xt = false
  290. }
  291. if thisinfo.District != "" && pInfo.District == "" {
  292. pInfo.District = thisinfo.District
  293. set["district"] = thisinfo.District
  294. }
  295. }
  296. //6--项目名称
  297. if (thisinfo.ProjectName != "" && pInfo.ProjectName == "") || (len([]rune(pInfo.ProjectName)) < 6 && thisinfo.LenPN > 6) {
  298. pInfo.ProjectName = thisinfo.ProjectName
  299. set["projectname"] = thisinfo.ProjectName
  300. }
  301. //7--项目编号
  302. if (pInfo.ProjectCode == "" && thisinfo.ProjectCode != "") || (len([]rune(pInfo.ProjectCode)) < 6 && len([]rune(thisinfo.ProjectCode)) > 6) {
  303. pInfo.ProjectCode = thisinfo.ProjectCode
  304. set["projectcode"] = thisinfo.ProjectCode
  305. }
  306. //7--采购单位
  307. if (pInfo.Buyer == "" && thisinfo.Buyer != "") || (len([]rune(pInfo.Buyer)) < 5 && len([]rune(thisinfo.Buyer)) > 5) {
  308. pInfo.Buyer = thisinfo.Buyer
  309. set["buyer"] = thisinfo.Buyer
  310. pInfo.Buyerclass = thisinfo.Buyerclass
  311. set["buyerclass"] = thisinfo.Buyerclass
  312. }
  313. if pInfo.Buyer == "" {
  314. set["buyerclass"] = ""
  315. }
  316. //8--代理机构
  317. if (pInfo.Agency == "" && thisinfo.Agency != "") || (len([]rune(pInfo.Agency)) < 5 && len([]rune(thisinfo.Agency)) > 5) {
  318. pInfo.Agency = thisinfo.Agency
  319. set["agency"] = thisinfo.Agency
  320. }
  321. //9--采购单位联系人
  322. if thisinfo.Buyerperson != "" && strings.Index(pInfo.Buyerperson, thisinfo.Buyerperson) < 0 {
  323. pInfo.Buyerperson = thisinfo.Buyerperson
  324. set["buyerperson"] = pInfo.Buyerperson
  325. }
  326. //10--采购单位電話
  327. if thisinfo.Buyertel != "" && strings.Index(pInfo.Buyertel, thisinfo.Buyertel) < 0 {
  328. pInfo.Buyertel = thisinfo.Buyertel
  329. set["buyertel"] = pInfo.Buyertel
  330. }
  331. if thisinfo.Bidopentime > pInfo.Bidopentime {
  332. pInfo.Bidopentime = thisinfo.Bidopentime
  333. set["bidopentime"] = pInfo.Bidopentime
  334. }
  335. //废标、流标 处理时间
  336. if thisinfo.SubType == "流标" || thisinfo.SubType == "废标" {
  337. pInfo.FirstTime = thisinfo.Publishtime
  338. pInfo.Bidopentime = int64(0)
  339. pInfo.LastTime = thisinfo.Publishtime
  340. set["firsttime"] = thisinfo.Publishtime
  341. set["zbtime"] = int64(0)
  342. set["publishtime"] = thisinfo.Publishtime
  343. set["bidopentime"] = int64(0)
  344. }
  345. if len(thisinfo.Topscopeclass) > 0 {
  346. sort.Strings(pInfo.Topscopeclass)
  347. for _, k := range thisinfo.Topscopeclass {
  348. if BinarySearch(pInfo.Topscopeclass, k) == -1 {
  349. pInfo.Topscopeclass = append(pInfo.Topscopeclass, k)
  350. sort.Strings(pInfo.Topscopeclass)
  351. }
  352. }
  353. set["topscopeclass"] = pInfo.Topscopeclass
  354. }
  355. if len(thisinfo.Subscopeclass) > 0 {
  356. sort.Strings(pInfo.Subscopeclass)
  357. for _, k := range thisinfo.Subscopeclass {
  358. if BinarySearch(pInfo.Subscopeclass, k) == -1 {
  359. pInfo.Subscopeclass = append(pInfo.Subscopeclass, k)
  360. sort.Strings(pInfo.Subscopeclass)
  361. }
  362. }
  363. set["subscopeclass"] = pInfo.Subscopeclass
  364. set["s_subscopeclass"] = strings.Join(pInfo.Subscopeclass, ",")
  365. }
  366. //winner
  367. if len(thisinfo.Winners) > 0 {
  368. sort.Strings(pInfo.Winners)
  369. for _, k := range thisinfo.Winners {
  370. if BinarySearch(pInfo.Winners, k) == -1 {
  371. pInfo.Winners = append(pInfo.Winners, k)
  372. sort.Strings(pInfo.Winners)
  373. }
  374. }
  375. //set["winners"] = pInfo.Winners
  376. set["s_winner"] = strings.Join(pInfo.Winners, ",")
  377. }
  378. if thisinfo.HasPackage {
  379. pkg := PackageFormat(thisinfo, pInfo)
  380. set["multipackage"] = 1
  381. pInfo.Package = pkg
  382. }else {
  383. set["multipackage"] = 0
  384. }
  385. //处理多包后,计算预算金额、中标金额
  386. CountAmount(pInfo, thisinfo)
  387. set["budget"] = pInfo.Budget
  388. set["bidamount"] = pInfo.Bidamount
  389. set["mpn"] = pInfo.MPN
  390. set["mpc"] = pInfo.MPC
  391. set["pici"] = p.pici
  392. return set
  393. }
  394. func deleteSlice1(arr []interface{}, v interface{}) []interface{} {
  395. for k, v1 := range arr {
  396. if reflect.DeepEqual(v1, v) {
  397. return append(arr[:k], arr[k+1:]...)
  398. }
  399. }
  400. return arr
  401. }
  402. func updateValue(info map[string]interface{}, tmp map[string]interface{}) map[string]interface{} {
  403. for k := range info{
  404. for k1 := range tmp {
  405. if k == k1 && info[k] != tmp[k1] {
  406. info[k] = tmp[k]
  407. break
  408. }
  409. }
  410. }
  411. return info
  412. }
  413. //备份(快照)
  414. func backupPro(tmp map[string]interface{}) {
  415. tmp1 := make(map[string]interface{})
  416. for k, v := range tmp{
  417. tmp1[k] = v
  418. }
  419. if Sysconfig["backupFlag"].(bool) {
  420. tmp1["sourceprojectid"] = tmp1["_id"]
  421. delete(tmp1, "_id")
  422. MongoTool.Save(BackupColl, tmp1)
  423. }
  424. }