thjNtf.go 23 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616
  1. package service
  2. import (
  3. "encoding/hex"
  4. "fmt"
  5. "mtp20_assisted/global"
  6. "mtp20_assisted/model"
  7. "mtp20_assisted/res/pb"
  8. "mtp20_assisted/utils"
  9. "os"
  10. "strconv"
  11. "strings"
  12. "time"
  13. "github.com/gofrs/uuid"
  14. "github.com/nguyenthenguyen/docx"
  15. "go.uber.org/zap"
  16. "google.golang.org/protobuf/proto"
  17. )
  18. // THJNtf 铁合通知处理模型
  19. type THJNtf struct{}
  20. func (t *THJNtf) Process(msg *[]byte) {
  21. // 分解总线包信息
  22. funcode := utils.BytesToUint32((*msg)[0:4])
  23. sessionId := utils.BytesToUint32((*msg)[4:8])
  24. bytes := (*msg)[8:]
  25. global.M2A_LOG.Info("[S->C]", zap.Any("funcode", funcode), zap.Any("sessionId", sessionId), zap.Any("count", len(bytes)))
  26. switch funcode {
  27. case uint32(global.THJPurchaseTradeNtf):
  28. // 等待两秒
  29. <-time.After(2 * time.Second)
  30. onTHJPurchaseTradeNtf(&bytes)
  31. case uint32(global.PurchaseTransferNtf):
  32. <-time.After(2 * time.Second)
  33. onTHJPurchaseTransferNtf(&bytes)
  34. case uint32(global.WRTradeDealedNtf):
  35. <-time.After(2 * time.Second)
  36. onWRTradeDealedNtf(&bytes)
  37. }
  38. }
  39. // onTHJPurchaseTradeNtf 铁合金成交通知
  40. func onTHJPurchaseTradeNtf(bytes *[]byte) {
  41. // Read from docx file
  42. r, err := docx.ReadDocxFile("./static/产能预售合同.docx")
  43. if err != nil {
  44. global.M2A_LOG.Error("[铁合金成交通知] 读取合同文件失败", zap.Error(err))
  45. return
  46. }
  47. // 获取网上开户地址(用于客户端下载文件)
  48. openconfig := model.Wskhopenaccountconfig{CONFIGID: 6}
  49. if has, err := openconfig.Get(); err != nil || !has {
  50. global.M2A_LOG.Error("[铁合金成交通知] 获取网上开户地址失败", zap.Error(err))
  51. return
  52. }
  53. var p pb.THJPurchaseTradeNtf
  54. if err := proto.Unmarshal(*bytes, &p); err != nil {
  55. global.M2A_LOG.Error("[铁合金成交通知] 总线回复数据反序列化失败", zap.Error(err))
  56. return
  57. }
  58. // 铁合金采购成交扩展信息
  59. thjpurchasetradedetail := model.Thjpurchasetradedetail{WRTRADEDETAILID: int64(p.GetWRTradeDetailID())}
  60. if has, err := thjpurchasetradedetail.Get(); err != nil || !has {
  61. global.M2A_LOG.Error("[铁合金成交通知] 获取铁合金采购成交扩展信息失败", zap.Error(err))
  62. return
  63. }
  64. // 交割方式
  65. thjdeliverymode := "否"
  66. if thjpurchasetradedetail.THJDELIVERYMODE == 1 {
  67. thjdeliverymode = "是"
  68. }
  69. // 现货商品信息
  70. wrstandard := model.Wrstandard{WRSTANDARDID: int64(thjpurchasetradedetail.WRSTANDARDID)}
  71. if has, err := wrstandard.Get(); err != nil || !has {
  72. global.M2A_LOG.Error("[铁合金成交通知] 获取现货商品信息失败", zap.Error(err))
  73. return
  74. }
  75. // 仓单预售信息
  76. wrpresaleinfo := model.Wrpresaleinfo{PRESALEAPPLYID: thjpurchasetradedetail.PRESALEAPPLYID}
  77. if has, err := wrpresaleinfo.Get(); err != nil || !has {
  78. global.M2A_LOG.Error("[铁合金成交通知] 获取仓单预售信息失败", zap.Error(err))
  79. return
  80. }
  81. // 甲方信息
  82. userinfo := model.Userinfo{USERID: thjpurchasetradedetail.BUYUSERID}
  83. if has, err := userinfo.Get(); err != nil || !has {
  84. global.M2A_LOG.Error("[铁合金成交通知] 获取甲方信息失败", zap.Error(err))
  85. return
  86. }
  87. division := model.Division{AUTOID: int64(userinfo.DISTRICTID)}
  88. totalAddress, err := division.GetTotalAddressByDistrictID(userinfo.ADDRESS)
  89. if err != nil {
  90. global.M2A_LOG.Error("[铁合金成交通知] 获取甲方地址信息失败", zap.Error(err))
  91. return
  92. }
  93. // 证件号码解密
  94. key, _ := hex.DecodeString(utils.AESSecretKey)
  95. if len(userinfo.CARDNUM) > 0 {
  96. if cardnum, err := hex.DecodeString(userinfo.CARDNUM); err == nil { // hex -> []byte
  97. if c, err := utils.AESDecrypt(cardnum, key); err == nil {
  98. userinfo.CARDNUM = string(c)
  99. }
  100. }
  101. }
  102. // 手机号码解密
  103. if len(userinfo.MOBILE) > 0 {
  104. if phonenum, err := hex.DecodeString(userinfo.MOBILE); err == nil { // hex -> []byte
  105. if mobile, err := utils.AESDecrypt(phonenum, key); err == nil {
  106. userinfo.MOBILE = string(mobile)
  107. }
  108. }
  109. }
  110. // 甲方签约信息
  111. bankaccountsign := model.Bankaccountsign{}
  112. if has, err := bankaccountsign.GetByUserID(uint64(thjpurchasetradedetail.BUYUSERID)); err != nil || !has {
  113. global.M2A_LOG.Error("[铁合金成交通知] 获取甲方签约信息失败", zap.Error(err))
  114. return
  115. }
  116. // 账号解密
  117. if len(bankaccountsign.BANKACCOUNTNO) > 0 {
  118. if tt, err := hex.DecodeString(bankaccountsign.BANKACCOUNTNO); err == nil { // hex -> []byte
  119. if dd, err := utils.AESDecrypt(tt, key); err == nil {
  120. bankaccountsign.BANKACCOUNTNO = string(dd)
  121. }
  122. }
  123. }
  124. // 银行信息
  125. bankinfo := model.Bankbankinfo{BANKID: bankaccountsign.BANKID}
  126. if has, err := bankinfo.Get(); err != nil || !has {
  127. global.M2A_LOG.Error("[铁合金成交通知] 获取银行信息失败", zap.Error(err))
  128. return
  129. }
  130. // 生产厂家信息
  131. selluserinfo := model.Userinfo{USERID: thjpurchasetradedetail.SELLUSERID}
  132. if has, err := selluserinfo.Get(); err != nil || !has {
  133. global.M2A_LOG.Error("[铁合金成交通知] 获取生产厂家信息失败", zap.Error(err))
  134. return
  135. }
  136. docx1 := r.Editable()
  137. // global.M2A_LOG.Info(docx1.GetContent())
  138. docx1.Replace("${WRTRADEDETAILID}", strconv.Itoa(int(thjpurchasetradedetail.WRTRADEDETAILID)), -1)
  139. docx1.Replace("${TRADETIME}", thjpurchasetradedetail.TRADETIME.Format("2006年01月02日"), -1)
  140. docx1.Replace("${DESADDRESS}", totalAddress, -1)
  141. // 判断是公司还是个人
  142. if userinfo.USERINFOTYPE == 1 {
  143. // 个人
  144. docx1.Replace("COMPANY", "", -1)
  145. docx1.Replace("${LEGALPERSONNAME}", userinfo.CUSTOMERNAME, -1)
  146. } else {
  147. // 公司
  148. docx1.Replace("COMPANY", userinfo.CUSTOMERNAME, -1)
  149. docx1.Replace("${LEGALPERSONNAME}", userinfo.LEGALPERSONNAME, -1)
  150. }
  151. docx1.Replace("CARDNUM", userinfo.CARDNUM, -1)
  152. docx1.Replace("${MOBILE}", userinfo.MOBILE, -1)
  153. docx1.Replace("${BANK}", bankinfo.BANKNAME, -1)
  154. docx1.Replace("${BANKACCOUNTNO}", bankaccountsign.BANKACCOUNTNO, -1)
  155. docx1.Replace("${WRSTANDARDNAME}", wrstandard.WRSTANDARDNAME, -1)
  156. docx1.Replace("${SELLUSER}", selluserinfo.CUSTOMERNAME, -1)
  157. docx1.Replace("${LASTAMOUNT}", strconv.FormatFloat(thjpurchasetradedetail.LASTAMOUNT, 'f', -1, 64), -1)
  158. docx1.Replace("${DEPOSITRATE}", strconv.FormatFloat(thjpurchasetradedetail.DEPOSITRATE*100, 'f', -1, 64)+"%", -1)
  159. docx1.Replace("${TRADEQTY}", strconv.Itoa(int(thjpurchasetradedetail.TRADEQTY)), -1)
  160. docx1.Replace("${TRADEPRICE}", strconv.FormatFloat(thjpurchasetradedetail.TRADEPRICE, 'f', -1, 64), -1)
  161. docx1.Replace("${ENDDATEMONTH}", strconv.Itoa(int(wrpresaleinfo.ENDDATE.Local().Month())), -1)
  162. docx1.Replace("${THJDELIVERYMODE}", thjdeliverymode, -1)
  163. docx1.Replace("${ENDDATE}", wrpresaleinfo.ENDDATE.Format("200601"), -1)
  164. docx1.Replace("${STORAGEFEE}", strconv.FormatFloat(wrstandard.STORAGEFEE, 'f', -1, 64), -1)
  165. // 暂存docx文件
  166. if exist, _ := utils.PathExists("./.tmp"); !exist {
  167. os.Mkdir("./.tmp", os.ModePerm)
  168. }
  169. uid, _ := uuid.NewV4()
  170. docxFilename := fmt.Sprintf("%v_%v.docx", strconv.Itoa(int(thjpurchasetradedetail.WRTRADEDETAILID)), uid.String())
  171. pdfFilename := strings.Replace(docxFilename, "docx", "pdf", -1)
  172. docx1.WriteToFile("./.tmp/" + docxFilename)
  173. r.Close()
  174. // 导出pdf到目标目录
  175. folderPath := "Purchase_Contract/" + time.Now().Format("20060102")
  176. savePath := openconfig.CONFIGVALUE + "/" + folderPath
  177. if exist, _ := utils.PathExists(savePath); !exist {
  178. os.MkdirAll(savePath, os.ModePerm)
  179. }
  180. success := utils.ConvertToPDF("./.tmp/"+docxFilename, savePath)
  181. if !success {
  182. // 转换失败
  183. global.M2A_LOG.Error("[铁合金成交通知] docx转换pdf失败", zap.Error(err))
  184. return
  185. }
  186. // 更新数据库
  187. // pdfFilename = docxFilename // 临时代码
  188. thjpurchasetradedetail.CONTRACTADDRBUY = fmt.Sprintf("./uploadFile/%v/%v", folderPath, pdfFilename)
  189. if err = thjpurchasetradedetail.UpdateContractAddrBuy(); err != nil {
  190. global.M2A_LOG.Error("[铁合金成交通知] 更新数据库失败", zap.Error(err))
  191. return
  192. }
  193. global.M2A_LOG.Info("[铁合金成交通知] 合同生成成功", zap.Any("file", savePath+pdfFilename))
  194. }
  195. // onTHJPurchaseTransferNtf 铁合金协议转让通知
  196. func onTHJPurchaseTransferNtf(bytes *[]byte) {
  197. // Read from docx file
  198. r, err := docx.ReadDocxFile("./static/协议转让合同.docx")
  199. if err != nil {
  200. global.M2A_LOG.Error("[铁合金协议转让通知] 读取合同文件失败", zap.Error(err))
  201. return
  202. }
  203. // 获取网上开户地址(用于客户端下载文件)
  204. openconfig := model.Wskhopenaccountconfig{CONFIGID: 6}
  205. if has, err := openconfig.Get(); err != nil || !has {
  206. global.M2A_LOG.Error("[铁合金协议转让通知] 获取网上开户地址失败", zap.Error(err))
  207. return
  208. }
  209. var p pb.PurchaseTransferNtf
  210. if err := proto.Unmarshal(*bytes, &p); err != nil {
  211. global.M2A_LOG.Error("[铁合金协议转让通知] 总线回复数据反序列化失败", zap.Error(err))
  212. return
  213. }
  214. // 铁合金采购协议表
  215. thjpurchasetransfer := model.Thjpurchasetransfer{TRANSFERID: int64(p.GetTransferID())}
  216. if has, err := thjpurchasetransfer.Get(); err != nil || !has {
  217. global.M2A_LOG.Error("[铁合金协议转让通知] 获取铁合金采购协议表信息失败", zap.Error(err))
  218. return
  219. }
  220. // 铁合金采购成交扩展信息
  221. thjpurchasetradedetail := model.Thjpurchasetradedetail{WRTRADEDETAILID: thjpurchasetransfer.WRTRADEDETAILID}
  222. if has, err := thjpurchasetradedetail.Get(); err != nil || !has {
  223. global.M2A_LOG.Error("[铁合金协议转让通知] 获取铁合金采购成交扩展信息失败", zap.Error(err))
  224. return
  225. }
  226. // 现货商品信息
  227. wrstandard := model.Wrstandard{WRSTANDARDID: int64(thjpurchasetradedetail.WRSTANDARDID)}
  228. if has, err := wrstandard.Get(); err != nil || !has {
  229. global.M2A_LOG.Error("[铁合金协议转让通知] 获取现货商品信息失败", zap.Error(err))
  230. return
  231. }
  232. // ******************* 甲方信息(卖方)*******************
  233. a_userinfo := model.Userinfo{USERID: thjpurchasetransfer.SELLUSERID}
  234. if has, err := a_userinfo.Get(); err != nil || !has {
  235. global.M2A_LOG.Error("[铁合金协议转让通知] 获取甲方信息失败", zap.Error(err))
  236. return
  237. }
  238. a_division := model.Division{AUTOID: int64(a_userinfo.DISTRICTID)}
  239. a_totalAddress, err := a_division.GetTotalAddressByDistrictID(a_userinfo.ADDRESS)
  240. if err != nil {
  241. global.M2A_LOG.Error("[铁合金协议转让通知] 获取甲方地址信息失败", zap.Error(err))
  242. return
  243. }
  244. // 证件号码解密
  245. key, _ := hex.DecodeString(utils.AESSecretKey)
  246. if len(a_userinfo.CARDNUM) > 0 {
  247. if cardnum, err := hex.DecodeString(a_userinfo.CARDNUM); err == nil { // hex -> []byte
  248. if c, err := utils.AESDecrypt(cardnum, key); err == nil {
  249. a_userinfo.CARDNUM = string(c)
  250. }
  251. }
  252. }
  253. // 手机号码解密
  254. if len(a_userinfo.MOBILE) > 0 {
  255. if phonenum, err := hex.DecodeString(a_userinfo.MOBILE); err == nil { // hex -> []byte
  256. if mobile, err := utils.AESDecrypt(phonenum, key); err == nil {
  257. a_userinfo.MOBILE = string(mobile)
  258. }
  259. }
  260. }
  261. // 签约信息
  262. a_bankaccountsign := model.Bankaccountsign{}
  263. if has, err := a_bankaccountsign.GetByUserID(uint64(thjpurchasetransfer.SELLUSERID)); err != nil || !has {
  264. global.M2A_LOG.Error("[铁合金协议转让通知] 获取甲方签约信息失败", zap.Error(err))
  265. return
  266. }
  267. // 账号解密
  268. if len(a_bankaccountsign.BANKACCOUNTNO) > 0 {
  269. if tt, err := hex.DecodeString(a_bankaccountsign.BANKACCOUNTNO); err == nil { // hex -> []byte
  270. if dd, err := utils.AESDecrypt(tt, key); err == nil {
  271. a_bankaccountsign.BANKACCOUNTNO = string(dd)
  272. }
  273. }
  274. }
  275. // 银行信息
  276. a_bankinfo := model.Bankbankinfo{BANKID: a_bankaccountsign.BANKID}
  277. if has, err := a_bankinfo.Get(); err != nil || !has {
  278. global.M2A_LOG.Error("[铁合金协议转让通知] 获取银行信息失败", zap.Error(err))
  279. return
  280. }
  281. // ******************* 乙方信息(买方)*******************
  282. b_userinfo := model.Userinfo{USERID: thjpurchasetransfer.BUYUSERID}
  283. if has, err := b_userinfo.Get(); err != nil || !has {
  284. global.M2A_LOG.Error("[铁合金协议转让通知] 获取乙方信息失败", zap.Error(err))
  285. return
  286. }
  287. b_division := model.Division{AUTOID: int64(b_userinfo.DISTRICTID)}
  288. b_totalAddress, err := b_division.GetTotalAddressByDistrictID(b_userinfo.ADDRESS)
  289. if err != nil {
  290. global.M2A_LOG.Error("[铁合金协议转让通知] 获取乙方地址信息失败", zap.Error(err))
  291. return
  292. }
  293. // 证件号码解密
  294. if len(b_userinfo.CARDNUM) > 0 {
  295. if cardnum, err := hex.DecodeString(b_userinfo.CARDNUM); err == nil { // hex -> []byte
  296. if c, err := utils.AESDecrypt(cardnum, key); err == nil {
  297. b_userinfo.CARDNUM = string(c)
  298. }
  299. }
  300. }
  301. // 手机号码解密
  302. if len(b_userinfo.MOBILE) > 0 {
  303. if phonenum, err := hex.DecodeString(b_userinfo.MOBILE); err == nil { // hex -> []byte
  304. if mobile, err := utils.AESDecrypt(phonenum, key); err == nil {
  305. b_userinfo.MOBILE = string(mobile)
  306. }
  307. }
  308. }
  309. // 签约信息
  310. b_bankaccountsign := model.Bankaccountsign{}
  311. if has, err := b_bankaccountsign.GetByUserID(uint64(thjpurchasetransfer.BUYUSERID)); err != nil || !has {
  312. global.M2A_LOG.Error("[铁合金协议转让通知] 获取乙方签约信息失败", zap.Error(err))
  313. return
  314. }
  315. // 账号解密
  316. if len(b_bankaccountsign.BANKACCOUNTNO) > 0 {
  317. if tt, err := hex.DecodeString(b_bankaccountsign.BANKACCOUNTNO); err == nil { // hex -> []byte
  318. if dd, err := utils.AESDecrypt(tt, key); err == nil {
  319. b_bankaccountsign.BANKACCOUNTNO = string(dd)
  320. }
  321. }
  322. }
  323. // 银行信息
  324. b_bankinfo := model.Bankbankinfo{BANKID: b_bankaccountsign.BANKID}
  325. if has, err := b_bankinfo.Get(); err != nil || !has {
  326. global.M2A_LOG.Error("[铁合金协议转让通知] 获取乙方银行信息失败", zap.Error(err))
  327. return
  328. }
  329. docx1 := r.Editable()
  330. docx1.Replace("TRANSFERID", strconv.Itoa(int(thjpurchasetransfer.TRANSFERID)), -1)
  331. if transfertradedate, e := time.Parse("20060102", thjpurchasetransfer.TRANSFERTRADEDATE); e != nil {
  332. docx1.Replace("TRANSFERTRADEDATE", transfertradedate.Format("2006年01月02日"), -1)
  333. }
  334. docx1.Replace("WRSTANDARDNAME", wrstandard.WRSTANDARDNAME, -1)
  335. docx1.Replace("WRTRADEDETAILID", strconv.Itoa(int(thjpurchasetradedetail.WRTRADEDETAILID)), -1)
  336. docx1.Replace("TRANSFERQTY", strconv.Itoa(int(thjpurchasetransfer.TRANSFERQTY)), -1)
  337. docx1.Replace("TRADEAMOUNT", strconv.FormatFloat(thjpurchasetradedetail.TRADEAMOUNT, 'f', -1, 64), -1)
  338. docx1.Replace("TRANSFERAMOUNT", strconv.FormatFloat(thjpurchasetransfer.TRANSFERAMOUNT, 'f', -1, 64), -1)
  339. docx1.Replace("TRANSFERPRICE", strconv.FormatFloat(thjpurchasetransfer.TRANSFERPRICE, 'f', -1, 64), -1)
  340. // ********************* 甲方 *********************
  341. // 判断甲方是公司还是个人
  342. if a_userinfo.USERINFOTYPE == 1 {
  343. // 个人
  344. docx1.Replace("PARTYA_COMPANY", "", -1)
  345. docx1.Replace("PARTYA_LEGALPERSONNAME", a_userinfo.CUSTOMERNAME, -1)
  346. } else {
  347. // 公司
  348. docx1.Replace("PARTYA_COMPANY", a_userinfo.CUSTOMERNAME, -1)
  349. docx1.Replace("PARTYA_LEGALPERSONNAME", a_userinfo.LEGALPERSONNAME, -1)
  350. }
  351. docx1.Replace("PARTYA_DESADDRESS", a_totalAddress, -1)
  352. docx1.Replace("PARTYA_CARDNUM", a_userinfo.CARDNUM, -1)
  353. docx1.Replace("PARTYA_MOBILE", a_userinfo.MOBILE, -1)
  354. docx1.Replace("PARTYA_BANK_BANKACCOUNTNO",
  355. fmt.Sprintf("%v%v", a_bankinfo.BANKNAME, a_bankaccountsign.BANKACCOUNTNO), -1)
  356. // ********************* 乙方 *********************
  357. // 判断乙方是公司还是个人
  358. if b_userinfo.USERINFOTYPE == 1 {
  359. // 个人
  360. docx1.Replace("PARTYB_COMPANY", "", -1)
  361. docx1.Replace("PARTYB_LEGALPERSONNAME", b_userinfo.CUSTOMERNAME, -1)
  362. } else {
  363. // 公司
  364. docx1.Replace("PARTYB_COMPANY", b_userinfo.CUSTOMERNAME, -1)
  365. docx1.Replace("PARTYB_LEGALPERSONNAME", b_userinfo.LEGALPERSONNAME, -1)
  366. }
  367. docx1.Replace("PARTYB_DESADDRESS", b_totalAddress, -1)
  368. docx1.Replace("PARTYB_CARDNUM", b_userinfo.CARDNUM, -1)
  369. docx1.Replace("PARTYB_MOBILE", b_userinfo.MOBILE, -1)
  370. docx1.Replace("PARTYB_BANK_BANKACCOUNTNO",
  371. fmt.Sprintf("%v%v", b_bankinfo.BANKNAME, b_bankaccountsign.BANKACCOUNTNO), -1)
  372. // 暂存docx文件
  373. if exist, _ := utils.PathExists("./.tmp"); !exist {
  374. os.Mkdir("./.tmp", os.ModePerm)
  375. }
  376. uid, _ := uuid.NewV4()
  377. docxFilename := fmt.Sprintf("%v_%v.docx", strconv.Itoa(int(thjpurchasetransfer.TRANSFERID)), uid.String())
  378. pdfFilename := strings.Replace(docxFilename, "docx", "pdf", -1)
  379. docx1.WriteToFile("./.tmp/" + docxFilename)
  380. r.Close()
  381. // 导出pdf到目标目录
  382. folderPath := "Agreement_Assignment/" + time.Now().Format("20060102")
  383. savePath := openconfig.CONFIGVALUE + "/" + folderPath
  384. if exist, _ := utils.PathExists(savePath); !exist {
  385. os.MkdirAll(savePath, os.ModePerm)
  386. }
  387. success := utils.ConvertToPDF("./.tmp/"+docxFilename, savePath)
  388. if !success {
  389. // 转换失败
  390. global.M2A_LOG.Error("[铁合金协议转让通知] docx转换pdf失败", zap.Error(err))
  391. return
  392. }
  393. // 更新数据库
  394. thjpurchasetransfer.CONTRACTADDR = fmt.Sprintf("./uploadFile/%v/%v", folderPath, pdfFilename)
  395. if err = thjpurchasetransfer.UpdateContractAddr(); err != nil {
  396. global.M2A_LOG.Error("[铁合金协议转让通知] 更新数据库失败", zap.Error(err))
  397. return
  398. }
  399. global.M2A_LOG.Info("[铁合金协议转让通知] 合同生成成功", zap.Any("file", savePath+pdfFilename))
  400. }
  401. // onWRTradeDealedNtf 仓单贸易成交通知
  402. func onWRTradeDealedNtf(bytes *[]byte) {
  403. var p pb.WRTradeDealedNtf
  404. if err := proto.Unmarshal(*bytes, &p); err != nil {
  405. global.M2A_LOG.Error("[仓单贸易成交通知] 总线回复数据反序列化失败", zap.Error(err))
  406. return
  407. }
  408. if p.GetHeader().GetMarketID() != 65201 {
  409. return
  410. }
  411. // 铁合金采购协议表
  412. wrtradetradequote := model.Wrtradetradequote{WRTRADEDETAILID: int64(p.GetTradeID())}
  413. if has, err := wrtradetradequote.Get(); err != nil || !has {
  414. global.M2A_LOG.Error("[仓单贸易成交通知] 获取仓单贸易成交行情表信息失败", zap.Error(err))
  415. return
  416. }
  417. buildWRTradeDealedContract(wrtradetradequote, 0)
  418. buildWRTradeDealedContract(wrtradetradequote, 1)
  419. }
  420. /*
  421. buildWRTradeDealedContract 生成现货采购合同
  422. buyOrSell 0-买方合同 1-卖方合同
  423. */
  424. func buildWRTradeDealedContract(wrtradetradequote model.Wrtradetradequote, buyOrSell int) {
  425. tempDocx := "./static/现货贸易合同(买家模板).docx"
  426. if buyOrSell == 1 {
  427. tempDocx = "./static/现货贸易合同(卖家模板).docx"
  428. }
  429. r, err := docx.ReadDocxFile(tempDocx)
  430. if err != nil {
  431. global.M2A_LOG.Error("[仓单贸易成交通知] 读取合同文件失败", zap.Error(err))
  432. return
  433. }
  434. // 获取网上开户地址(用于客户端下载文件)
  435. openconfig := model.Wskhopenaccountconfig{CONFIGID: 6}
  436. if has, err := openconfig.Get(); err != nil || !has {
  437. global.M2A_LOG.Error("[仓单贸易成交通知] 获取网上开户地址失败", zap.Error(err))
  438. return
  439. }
  440. // 现货商品信息
  441. wrstandard := model.Wrstandard{WRSTANDARDID: int64(wrtradetradequote.WRSTANDARDID)}
  442. if has, err := wrstandard.Get(); err != nil || !has {
  443. global.M2A_LOG.Error("[仓单贸易成交通知] 获取现货商品信息失败", zap.Error(err))
  444. return
  445. }
  446. // 甲方或乙方信息
  447. userId := wrtradetradequote.BUYUSERID
  448. if buyOrSell == 1 {
  449. userId = wrtradetradequote.SELLUSERID
  450. }
  451. userinfo := model.Userinfo{USERID: userId}
  452. if has, err := userinfo.Get(); err != nil || !has {
  453. global.M2A_LOG.Error("[仓单贸易成交通知] 获取用户信息失败", zap.Error(err))
  454. return
  455. }
  456. division := model.Division{AUTOID: int64(userinfo.DISTRICTID)}
  457. totalAddress, err := division.GetTotalAddressByDistrictID(userinfo.ADDRESS)
  458. if err != nil {
  459. global.M2A_LOG.Error("[仓单贸易成交通知] 获取用户地址信息失败", zap.Error(err))
  460. return
  461. }
  462. // 证件号码解密
  463. key, _ := hex.DecodeString(utils.AESSecretKey)
  464. if len(userinfo.CARDNUM) > 0 {
  465. if cardnum, err := hex.DecodeString(userinfo.CARDNUM); err == nil { // hex -> []byte
  466. if c, err := utils.AESDecrypt(cardnum, key); err == nil {
  467. userinfo.CARDNUM = string(c)
  468. }
  469. }
  470. }
  471. // 手机号码解密
  472. if len(userinfo.MOBILE) > 0 {
  473. if phonenum, err := hex.DecodeString(userinfo.MOBILE); err == nil { // hex -> []byte
  474. if mobile, err := utils.AESDecrypt(phonenum, key); err == nil {
  475. userinfo.MOBILE = string(mobile)
  476. }
  477. }
  478. }
  479. // 签约信息
  480. bankaccountsign := model.Bankaccountsign{}
  481. if has, err := bankaccountsign.GetByUserID(uint64(userId)); err != nil || !has {
  482. global.M2A_LOG.Error("[仓单贸易成交通知] 获取用户签约信息失败", zap.Error(err))
  483. return
  484. }
  485. // 账号解密
  486. if len(bankaccountsign.BANKACCOUNTNO) > 0 {
  487. if tt, err := hex.DecodeString(bankaccountsign.BANKACCOUNTNO); err == nil { // hex -> []byte
  488. if dd, err := utils.AESDecrypt(tt, key); err == nil {
  489. bankaccountsign.BANKACCOUNTNO = string(dd)
  490. }
  491. }
  492. }
  493. // 银行信息
  494. bankinfo := model.Bankbankinfo{BANKID: bankaccountsign.BANKID}
  495. if has, err := bankinfo.Get(); err != nil || !has {
  496. global.M2A_LOG.Error("[仓单贸易成交通知] 获取银行信息失败", zap.Error(err))
  497. return
  498. }
  499. // 电子仓单号
  500. type temp struct {
  501. LADINGBILLID string `xorm:"LADINGBILLID"`
  502. }
  503. _temp := temp{}
  504. sql := fmt.Sprintf(`
  505. select
  506. to_char(wrd.LADINGBILLID) LADINGBILLID
  507. from wrtrade_tradedetail td
  508. left join WRChangeDetail wrd on td.wrid = wrd.wrid and wrd.changetype = 1
  509. where td.WRTRADEDETAILID = %v and td.wrtradetype = 2
  510. `, wrtradetradequote.WRTRADEDETAILID)
  511. global.M2A_DB.SQL(sql).Get(&_temp)
  512. docx1 := r.Editable()
  513. docx1.Replace("${WRTRADEDETAILID}", strconv.Itoa(int(wrtradetradequote.WRTRADEDETAILID)), -1)
  514. if transfertradedate, e := time.Parse("20060102", wrtradetradequote.TRADEDATE); e != nil {
  515. docx1.Replace("TRADEDATE", transfertradedate.Format("2006年01月02日"), -1)
  516. }
  517. // 判断是公司还是个人
  518. if userinfo.USERINFOTYPE == 1 {
  519. // 个人
  520. docx1.Replace("COMPANY", "", -1)
  521. docx1.Replace("LEGALPERSONNAME", userinfo.CUSTOMERNAME, -1)
  522. } else {
  523. // 公司
  524. docx1.Replace("COMPANY", userinfo.CUSTOMERNAME, -1)
  525. docx1.Replace("LEGALPERSONNAME", userinfo.LEGALPERSONNAME, -1)
  526. }
  527. docx1.Replace("DESADDRESS", totalAddress, -1)
  528. docx1.Replace("CARDNUM", userinfo.CARDNUM, -1)
  529. docx1.Replace("MOBILE", userinfo.MOBILE, -1)
  530. docx1.Replace("BANK", bankinfo.BANKNAME, -1)
  531. docx1.Replace("BANKACCOUNTNO", bankaccountsign.BANKACCOUNTNO, -1)
  532. docx1.Replace("WRSTANDARDNAME", wrstandard.WRSTANDARDNAME, -1)
  533. docx1.Replace("TRADEQTY", strconv.Itoa(int(wrtradetradequote.TRADEQTY)), -1)
  534. docx1.Replace("TRADEPRICE", strconv.FormatFloat(wrtradetradequote.TRADEPRICE, 'f', -1, 64), -1)
  535. docx1.Replace("TRADEAMOUNT", strconv.FormatFloat(wrtradetradequote.TRADEAMOUNT, 'f', -1, 64), -1)
  536. docx1.Replace("STORAGEFEE", strconv.FormatFloat(wrstandard.STORAGEFEE, 'f', -1, 64), -1)
  537. docx1.Replace("LADINGBILLID", _temp.LADINGBILLID, -1)
  538. // 暂存docx文件
  539. if exist, _ := utils.PathExists("./.tmp"); !exist {
  540. os.Mkdir("./.tmp", os.ModePerm)
  541. }
  542. uid, _ := uuid.NewV4()
  543. docxFilename := fmt.Sprintf("%v_%v_%v.docx", buyOrSell, strconv.Itoa(int(wrtradetradequote.WRTRADEDETAILID)), uid.String())
  544. pdfFilename := strings.Replace(docxFilename, "docx", "pdf", -1)
  545. docx1.WriteToFile("./.tmp/" + docxFilename)
  546. r.Close()
  547. // 导出pdf到目标目录
  548. folderPath := "Spot_Contract/" + time.Now().Format("20060102")
  549. savePath := openconfig.CONFIGVALUE + "/" + folderPath
  550. if exist, _ := utils.PathExists(savePath); !exist {
  551. os.MkdirAll(savePath, os.ModePerm)
  552. }
  553. success := utils.ConvertToPDF("./.tmp/"+docxFilename, savePath)
  554. if !success {
  555. // 转换失败
  556. global.M2A_LOG.Error("[仓单贸易成交通知] docx转换pdf失败", zap.Error(err))
  557. return
  558. }
  559. // 更新数据库
  560. if buyOrSell == 0 {
  561. wrtradetradequote.CONTRACTADDRBUY = fmt.Sprintf("./uploadFile/%v/%v", folderPath, pdfFilename)
  562. } else {
  563. wrtradetradequote.CONTRACTADDRSELL = fmt.Sprintf("./uploadFile/%v/%v", folderPath, pdfFilename)
  564. }
  565. if err = wrtradetradequote.UpdateContractAddr(buyOrSell); err != nil {
  566. global.M2A_LOG.Error("[仓单贸易成交通知] 更新数据库失败", zap.Error(err))
  567. return
  568. }
  569. global.M2A_LOG.Info("[仓单贸易成交通知] 合同生成成功", zap.Any("file", savePath+pdfFilename))
  570. }