thjNtf.go 23 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618
  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 == 2 {
  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.TAKESTARTDATE.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 + "/uploadFile/" + 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. // global.M2A_LOG.Info(docx1.GetContent())
  331. docx1.Replace("TRANSFERID", strconv.Itoa(int(thjpurchasetransfer.TRANSFERID)), -1)
  332. if transfertradedate, e := time.Parse("20060102", thjpurchasetransfer.TRANSFERTRADEDATE); e == nil {
  333. docx1.Replace("TRANSFERTRADEDATE", transfertradedate.Format("2006年01月02日"), -1)
  334. }
  335. docx1.Replace("WRSTANDARDNAME", wrstandard.WRSTANDARDNAME, -1)
  336. docx1.Replace("WRTRADEDETAILID", strconv.Itoa(int(thjpurchasetradedetail.WRTRADEDETAILID)), -1)
  337. docx1.Replace("TRANSFERQTY", strconv.Itoa(int(thjpurchasetransfer.TRANSFERQTY)), -1)
  338. docx1.Replace("TRADEAMOUNT", strconv.FormatFloat(thjpurchasetradedetail.TRADEAMOUNT, 'f', -1, 64), -1)
  339. docx1.Replace("TRANSFERAMOUNT", strconv.FormatFloat(thjpurchasetransfer.TRANSFERAMOUNT, 'f', -1, 64), -1)
  340. docx1.Replace("TRANSFERPRICE", strconv.FormatFloat(thjpurchasetransfer.TRANSFERPRICE, 'f', -1, 64), -1)
  341. // ********************* 甲方 *********************
  342. // 判断甲方是公司还是个人
  343. if a_userinfo.USERINFOTYPE == 1 {
  344. // 个人
  345. docx1.Replace("PARTYA_COMPANY", "", -1)
  346. docx1.Replace("PARTYA_LEGALPERSONNAME", a_userinfo.CUSTOMERNAME, -1)
  347. } else {
  348. // 公司
  349. docx1.Replace("PARTYA_COMPANY", a_userinfo.CUSTOMERNAME, -1)
  350. docx1.Replace("PARTYA_LEGALPERSONNAME", a_userinfo.LEGALPERSONNAME, -1)
  351. }
  352. docx1.Replace("PARTYA_DESADDRESS", a_totalAddress, -1)
  353. docx1.Replace("PARTYA_CARDNUM", a_userinfo.CARDNUM, -1)
  354. docx1.Replace("PARTYA_MOBILE", a_userinfo.MOBILE, -1)
  355. docx1.Replace("AAAAAAA",
  356. fmt.Sprintf("%v%v", a_bankinfo.BANKNAME, a_bankaccountsign.BANKACCOUNTNO), -1)
  357. // ********************* 乙方 *********************
  358. // 判断乙方是公司还是个人
  359. if b_userinfo.USERINFOTYPE == 1 {
  360. // 个人
  361. docx1.Replace("COMPANY", "", -1)
  362. docx1.Replace("LEGALPERSONNAME", b_userinfo.CUSTOMERNAME, -1)
  363. } else {
  364. // 公司
  365. docx1.Replace("COMPANY", b_userinfo.CUSTOMERNAME, -1)
  366. docx1.Replace("LEGALPERSONNAME", b_userinfo.LEGALPERSONNAME, -1)
  367. }
  368. docx1.Replace("DESADDRESS", b_totalAddress, -1)
  369. docx1.Replace("CARDNUM", b_userinfo.CARDNUM, -1)
  370. docx1.Replace("MOBILE", b_userinfo.MOBILE, -1)
  371. docx1.Replace("BBBBBBB",
  372. fmt.Sprintf("%v%v", b_bankinfo.BANKNAME, b_bankaccountsign.BANKACCOUNTNO), -1)
  373. // 暂存docx文件
  374. if exist, _ := utils.PathExists("./.tmp"); !exist {
  375. os.Mkdir("./.tmp", os.ModePerm)
  376. }
  377. uid, _ := uuid.NewV4()
  378. docxFilename := fmt.Sprintf("%v_%v.docx", strconv.Itoa(int(thjpurchasetransfer.TRANSFERID)), uid.String())
  379. pdfFilename := strings.Replace(docxFilename, "docx", "pdf", -1)
  380. docx1.WriteToFile("./.tmp/" + docxFilename)
  381. r.Close()
  382. // 导出pdf到目标目录
  383. folderPath := "Agreement_Assignment/" + time.Now().Format("20060102")
  384. savePath := openconfig.CONFIGVALUE + "/uploadFile/" + folderPath
  385. if exist, _ := utils.PathExists(savePath); !exist {
  386. os.MkdirAll(savePath, os.ModePerm)
  387. }
  388. success := utils.ConvertToPDF("./.tmp/"+docxFilename, savePath)
  389. if !success {
  390. // 转换失败
  391. global.M2A_LOG.Error("[铁合金协议转让通知] docx转换pdf失败", zap.Error(err))
  392. return
  393. }
  394. // 更新数据库
  395. thjpurchasetransfer.CONTRACTADDR = fmt.Sprintf("./uploadFile/%v/%v", folderPath, pdfFilename)
  396. if err = thjpurchasetransfer.UpdateContractAddr(); err != nil {
  397. global.M2A_LOG.Error("[铁合金协议转让通知] 更新数据库失败", zap.Error(err))
  398. return
  399. }
  400. global.M2A_LOG.Info("[铁合金协议转让通知] 合同生成成功", zap.Any("file", savePath+pdfFilename))
  401. }
  402. // onWRTradeDealedNtf 仓单贸易成交通知
  403. func onWRTradeDealedNtf(bytes *[]byte) {
  404. var p pb.WRTradeDealedNtf
  405. if err := proto.Unmarshal(*bytes, &p); err != nil {
  406. global.M2A_LOG.Error("[仓单贸易成交通知] 总线回复数据反序列化失败", zap.Error(err))
  407. return
  408. }
  409. if p.GetHeader().GetMarketID() != 65201 {
  410. return
  411. }
  412. // 铁合金采购协议表
  413. wrtradetradequote := model.Wrtradetradequote{WRTRADEDETAILID: int64(p.GetTradeID())}
  414. if has, err := wrtradetradequote.Get(); err != nil || !has {
  415. global.M2A_LOG.Error("[仓单贸易成交通知] 获取仓单贸易成交行情表信息失败", zap.Error(err))
  416. return
  417. }
  418. buildWRTradeDealedContract(wrtradetradequote, 0)
  419. buildWRTradeDealedContract(wrtradetradequote, 1)
  420. }
  421. /*
  422. buildWRTradeDealedContract 生成现货采购合同
  423. buyOrSell 0-买方合同 1-卖方合同
  424. */
  425. func buildWRTradeDealedContract(wrtradetradequote model.Wrtradetradequote, buyOrSell int) {
  426. tempDocx := "./static/现货贸易合同(买家模板).docx"
  427. if buyOrSell == 1 {
  428. tempDocx = "./static/现货贸易合同(卖家模板).docx"
  429. }
  430. r, err := docx.ReadDocxFile(tempDocx)
  431. if err != nil {
  432. global.M2A_LOG.Error("[仓单贸易成交通知] 读取合同文件失败", zap.Error(err))
  433. return
  434. }
  435. // 获取网上开户地址(用于客户端下载文件)
  436. openconfig := model.Wskhopenaccountconfig{CONFIGID: 6}
  437. if has, err := openconfig.Get(); err != nil || !has {
  438. global.M2A_LOG.Error("[仓单贸易成交通知] 获取网上开户地址失败", zap.Error(err))
  439. return
  440. }
  441. // 现货商品信息
  442. wrstandard := model.Wrstandard{WRSTANDARDID: int64(wrtradetradequote.WRSTANDARDID)}
  443. if has, err := wrstandard.Get(); err != nil || !has {
  444. global.M2A_LOG.Error("[仓单贸易成交通知] 获取现货商品信息失败", zap.Error(err))
  445. return
  446. }
  447. // 甲方或乙方信息
  448. userId := wrtradetradequote.BUYUSERID
  449. if buyOrSell == 1 {
  450. userId = wrtradetradequote.SELLUSERID
  451. }
  452. userinfo := model.Userinfo{USERID: userId}
  453. if has, err := userinfo.Get(); err != nil || !has {
  454. global.M2A_LOG.Error("[仓单贸易成交通知] 获取用户信息失败", zap.Error(err))
  455. return
  456. }
  457. division := model.Division{AUTOID: int64(userinfo.DISTRICTID)}
  458. totalAddress, err := division.GetTotalAddressByDistrictID(userinfo.ADDRESS)
  459. if err != nil {
  460. global.M2A_LOG.Error("[仓单贸易成交通知] 获取用户地址信息失败", zap.Error(err))
  461. return
  462. }
  463. // 证件号码解密
  464. key, _ := hex.DecodeString(utils.AESSecretKey)
  465. if len(userinfo.CARDNUM) > 0 {
  466. if cardnum, err := hex.DecodeString(userinfo.CARDNUM); err == nil { // hex -> []byte
  467. if c, err := utils.AESDecrypt(cardnum, key); err == nil {
  468. userinfo.CARDNUM = string(c)
  469. }
  470. }
  471. }
  472. // 手机号码解密
  473. if len(userinfo.MOBILE) > 0 {
  474. if phonenum, err := hex.DecodeString(userinfo.MOBILE); err == nil { // hex -> []byte
  475. if mobile, err := utils.AESDecrypt(phonenum, key); err == nil {
  476. userinfo.MOBILE = string(mobile)
  477. }
  478. }
  479. }
  480. // 签约信息
  481. bankaccountsign := model.Bankaccountsign{}
  482. if has, err := bankaccountsign.GetByUserID(uint64(userId)); err != nil || !has {
  483. global.M2A_LOG.Error("[仓单贸易成交通知] 获取用户签约信息失败", zap.Error(err))
  484. return
  485. }
  486. // 账号解密
  487. if len(bankaccountsign.BANKACCOUNTNO) > 0 {
  488. if tt, err := hex.DecodeString(bankaccountsign.BANKACCOUNTNO); err == nil { // hex -> []byte
  489. if dd, err := utils.AESDecrypt(tt, key); err == nil {
  490. bankaccountsign.BANKACCOUNTNO = string(dd)
  491. }
  492. }
  493. }
  494. // 银行信息
  495. bankinfo := model.Bankbankinfo{BANKID: bankaccountsign.BANKID}
  496. if has, err := bankinfo.Get(); err != nil || !has {
  497. global.M2A_LOG.Error("[仓单贸易成交通知] 获取银行信息失败", zap.Error(err))
  498. return
  499. }
  500. // 电子仓单号
  501. type temp struct {
  502. LADINGBILLID string `xorm:"LADINGBILLID"`
  503. }
  504. _temp := temp{}
  505. sql := fmt.Sprintf(`
  506. select
  507. to_char(wrd.LADINGBILLID) LADINGBILLID
  508. from wrtrade_tradedetail td
  509. left join WRChangeDetail wrd on td.wrid = wrd.wrid and wrd.changetype = 1
  510. where td.WRTRADEDETAILID = %v and td.wrtradetype = 1
  511. `, wrtradetradequote.WRTRADEDETAILID)
  512. global.M2A_DB.SQL(sql).Get(&_temp)
  513. docx1 := r.Editable()
  514. // global.M2A_LOG.Info(docx1.GetContent())
  515. docx1.Replace("WRTRADEDETAILID", strconv.Itoa(int(wrtradetradequote.WRTRADEDETAILID)), -1)
  516. if transfertradedate, e := time.Parse("20060102", wrtradetradequote.TRADEDATE); e == nil {
  517. docx1.Replace("TRADEDATE", transfertradedate.Format("2006年01月02日"), -1)
  518. }
  519. // 判断是公司还是个人
  520. if userinfo.USERINFOTYPE == 1 {
  521. // 个人
  522. docx1.Replace("COMPANY", "", -1)
  523. docx1.Replace("LEGALPERSONNAME", userinfo.CUSTOMERNAME, -1)
  524. } else {
  525. // 公司
  526. docx1.Replace("COMPANY", userinfo.CUSTOMERNAME, -1)
  527. docx1.Replace("LEGALPERSONNAME", userinfo.LEGALPERSONNAME, -1)
  528. }
  529. docx1.Replace("DESADDRESS", totalAddress, -1)
  530. docx1.Replace("CARDNUM", userinfo.CARDNUM, -1)
  531. docx1.Replace("MOBILE", userinfo.MOBILE, -1)
  532. docx1.Replace("BANKACCOUNTNO", bankaccountsign.BANKACCOUNTNO, -1)
  533. docx1.Replace("BANK", bankinfo.BANKNAME, -1)
  534. docx1.Replace("WRSTANDARDNAME", wrstandard.WRSTANDARDNAME, -1)
  535. docx1.Replace("TRADEQTY", strconv.Itoa(int(wrtradetradequote.TRADEQTY)), -1)
  536. docx1.Replace("TRADEPRICE", strconv.FormatFloat(wrtradetradequote.TRADEPRICE, 'f', -1, 64), -1)
  537. docx1.Replace("TRADEAMOUNT", strconv.FormatFloat(wrtradetradequote.TRADEAMOUNT, 'f', -1, 64), -1)
  538. docx1.Replace("STORAGEFEE", strconv.FormatFloat(wrstandard.STORAGEFEE, 'f', -1, 64), -1)
  539. docx1.Replace("LADINGBILLID", _temp.LADINGBILLID, -1)
  540. // 暂存docx文件
  541. if exist, _ := utils.PathExists("./.tmp"); !exist {
  542. os.Mkdir("./.tmp", os.ModePerm)
  543. }
  544. uid, _ := uuid.NewV4()
  545. docxFilename := fmt.Sprintf("%v_%v_%v.docx", buyOrSell, strconv.Itoa(int(wrtradetradequote.WRTRADEDETAILID)), uid.String())
  546. pdfFilename := strings.Replace(docxFilename, "docx", "pdf", -1)
  547. docx1.WriteToFile("./.tmp/" + docxFilename)
  548. r.Close()
  549. // 导出pdf到目标目录
  550. folderPath := "Spot_Contract/" + time.Now().Format("20060102")
  551. savePath := openconfig.CONFIGVALUE + "/uploadFile/" + folderPath
  552. if exist, _ := utils.PathExists(savePath); !exist {
  553. os.MkdirAll(savePath, os.ModePerm)
  554. }
  555. success := utils.ConvertToPDF("./.tmp/"+docxFilename, savePath)
  556. if !success {
  557. // 转换失败
  558. global.M2A_LOG.Error("[仓单贸易成交通知] docx转换pdf失败", zap.Error(err))
  559. return
  560. }
  561. // 更新数据库
  562. if buyOrSell == 0 {
  563. wrtradetradequote.CONTRACTADDRBUY = fmt.Sprintf("./uploadFile/%v/%v", folderPath, pdfFilename)
  564. } else {
  565. wrtradetradequote.CONTRACTADDRSELL = fmt.Sprintf("./uploadFile/%v/%v", folderPath, pdfFilename)
  566. }
  567. if err = wrtradetradequote.UpdateContractAddr(buyOrSell); err != nil {
  568. global.M2A_LOG.Error("[仓单贸易成交通知] 更新数据库失败", zap.Error(err))
  569. return
  570. }
  571. global.M2A_LOG.Info("[仓单贸易成交通知] 合同生成成功", zap.Any("file", savePath+pdfFilename))
  572. }