hsby.go 63 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206
  1. package models
  2. import (
  3. "encoding/hex"
  4. "fmt"
  5. "math"
  6. "mtp2_if/db"
  7. "mtp2_if/utils"
  8. "strconv"
  9. "time"
  10. )
  11. // Hsbygoodsex 商品扩展表
  12. type Hsbygoodsex struct {
  13. Goodsid int32 `json:"goodsid" xorm:"'GOODSID'" binding:"required"` // 商 品ID
  14. Hotindex int32 `json:"hotindex" xorm:"'HOTINDEX'"` // 景点热度
  15. Descprovinceid int64 `json:"descprovinceid" xorm:"'DESCPROVINCEID'"` // 目的地(省)
  16. Desccityid int64 `json:"desccityid" xorm:"'DESCCITYID'"` // 目的地(市)
  17. Vendorid int32 `json:"vendorid" xorm:"'VENDORID'"` // 供应商ID
  18. Goodsdesc string `json:"goodsdesc" xorm:"'GOODSDESC'"` // 商品详情
  19. Createtime time.Time `json:"createtime" xorm:"'CREATETIME'"` // 创建时间
  20. Creatorid int64 `json:"creatorid" xorm:"'CREATORID'"` // 创建人
  21. Modifierid int64 `json:"modifierid" xorm:"'MODIFIERID'"` // 修改人
  22. Modifytime time.Time `json:"modifytime" xorm:"'MODIFYTIME'"` // 修改时间
  23. Memberratio float64 `json:"memberratio" xorm:"'MEMBERRATIO'"` // 会员货款比例 [71]
  24. Videourls string `json:"videourls" xorm:"'VIDEOURLS'"` // 介绍视频[多张用逗号分隔]
  25. Picurls string `json:"picurls" xorm:"'PICURLS'"` // 介绍图片[多张用逗号分隔]
  26. }
  27. // TableName is HSBY_GOODSEX
  28. func (Hsbygoodsex) TableName() string {
  29. return "HSBY_GOODSEX"
  30. }
  31. // Hsbysupplierinfo 供应商表
  32. type Hsbysupplierinfo struct {
  33. Vendorid int32 `json:"vendorid" xorm:"'VENDORID'" binding:"required"` // 供应商ID(SEQ_HSBY_SUPPLIERINFO)
  34. Vendorname string `json:"vendorname" xorm:"'VENDORNAME'"` // 名称描述客服电话名称
  35. Vendordesc string `json:"vendordesc" xorm:"'VENDORDESC'"` // 描述
  36. Vendorphone string `json:"vendorphone" xorm:"'VENDORPHONE'"` // 客服电话
  37. Vendorattr string `json:"vendorattr" xorm:"'VENDORATTR'"` // 附件(多张,逗号分隔)
  38. Createtime time.Time `json:"createtime" xorm:"'CREATETIME'"` // 创建时间
  39. Creatorid int64 `json:"creatorid" xorm:"'CREATORID'"` // 创建人
  40. Modifierid int64 `json:"modifierid" xorm:"'MODIFIERID'"` // 修改人
  41. Modifytime time.Time `json:"modifytime" xorm:"'MODIFYTIME'"` // 修改时间
  42. }
  43. // TableName is HSBY_SUPPLIERINFO
  44. func (Hsbysupplierinfo) TableName() string {
  45. return "HSBY_SUPPLIERINFO"
  46. }
  47. // Tradepayorder 交易待付款表
  48. type Tradepayorder struct {
  49. Tradeid string `json:"tradeid" xorm:"'TRADEID'" binding:"required"` // 成交单号(101+Unix秒时间戳(10位)+2位(MarketServiceID)+xxxx)
  50. Tradedate string `json:"tradedate" xorm:"'TRADEDATE'"` // 交易日(yyyyMMdd)
  51. Marketid int32 `json:"marketid" xorm:"'MARKETID'"` // 市场ID
  52. Goodsid int32 `json:"goodsid" xorm:"'GOODSID'"` // 商品ID
  53. Buyorderid string `json:"buyorderid" xorm:"'BUYORDERID'"` // 买方委托单号
  54. Buyaccountid int64 `json:"buyaccountid" xorm:"'BUYACCOUNTID'"` // 买方账号ID[报价币种]
  55. Sellorderid string `json:"sellorderid" xorm:"'SELLORDERID'"` // 卖方委托单号
  56. Sellaccountid int64 `json:"sellaccountid" xorm:"'SELLACCOUNTID'"` // 卖方账号ID[报价币种]
  57. Tradeamount float64 `json:"tradeamount" xorm:"'TRADEAMOUNT'"` // 成交金额
  58. Tradecharge float64 `json:"tradecharge" xorm:"'TRADECHARGE'"` // 成交手续费(买方)
  59. Payflag int32 `json:"payflag" xorm:"'PAYFLAG'"` // 付款标识 - 1:未支付 2:已支付 3:已过期 4:已撤销 5:结算过期
  60. Createtime time.Time `json:"createtime" xorm:"'CREATETIME'"` // 创建时间
  61. Paytime time.Time `json:"paytime" xorm:"'PAYTIME'"` // 付款时间
  62. Paylimitedtime time.Time `json:"paylimitedtime" xorm:"'PAYLIMITEDTIME'"` // 支付期限
  63. Offamount float64 `json:"offamount" xorm:"'OFFAMOUNT'"` // 优惠金额
  64. Tradeprice float64 `json:"tradeprice" xorm:"'TRADEPRICE'"` // 成交价格
  65. Tradeqty int64 `json:"tradeqty" xorm:"'TRADEQTY'"` // 成交数量
  66. }
  67. // TableName is TRADE_PAYORDER
  68. func (Tradepayorder) TableName() string {
  69. return "TRADE_PAYORDER"
  70. }
  71. // HsbyTopGoods 热卖商品(二级市场挂牌点选)
  72. type HsbyTopGoods struct {
  73. Goodsid int64 `json:"goodsid" xorm:"'GOODSID'" binding:"required"` // 商品ID(自增ID SEQ_GOODS)
  74. Goodscode string `json:"goodscode" xorm:"'GOODSCODE'" binding:"required"` // 商品代码(内部)
  75. Goodsname string `json:"goodsname" xorm:"'GOODSNAME'" binding:"required"` // 商品名称
  76. Decimalplace int64 `json:"decimalplace" xorm:"'DECIMALPLACE'"` // 报价小数位
  77. Marketid int64 `json:"marketid" xorm:"'MARKETID'" binding:"required"` // 所属市场ID
  78. Quoteminunit int64 `json:"quoteminunit" xorm:"'QUOTEMINUNIT'"` // 行情最小变动单位 [整数,报价小数位一起使用]
  79. Agreeunit float64 `json:"agreeunit" xorm:"'AGREEUNIT'"` // 合约单位
  80. Hotindex int32 `json:"hotindex" xorm:"'HOTINDEX'"` // 景点热度
  81. Videourls string `json:"videourls" xorm:"'VIDEOURLS'"` // 介绍视频[多张用逗号分隔]
  82. Picurls string `json:"picurls" xorm:"'PICURLS'"` // 介绍图片[多张用逗号分隔]
  83. Currency string `json:"currency" xorm:"'CURRENCY'"` // 货币
  84. Currencysign string `json:"currencysign" xorm:"'CURRENCYSIGN'"` // 货币符号
  85. Trademode int32 `json:"trademode" xorm:"'TRADEMODE'" binding:"required"` // 交易模式 - 10:做市 13:竞价 15:通道交易 16:挂牌点选 17:仓单贸易 18:期权 19:竞拍-降价式 20:竞拍-竞价式 21:竞拍-大宗式 22:受托竞价
  86. Last float64 `json:"last" xorm:"-"` // 现价
  87. }
  88. // GetHsbyTopGoodses 获取热门商品列表
  89. // 参数 marketIDs string 所属市场ID列表, 格式 1,2,3
  90. // 参数 descProvinceID int 目标省ID
  91. // 参数 descCityID int 目标城市ID
  92. // 返回 []TopGoods 热门商品列表
  93. // 返回 error error
  94. func GetHsbyTopGoodses(marketIDs string, descProvinceID, descCityID int) ([]HsbyTopGoods, error) {
  95. // 热门商品为二级市场(挂牌点选)下的商品信息
  96. engine := db.GetEngine()
  97. topGoodses := make([]HsbyTopGoods, 0)
  98. // 获取挂牌商品信息,以及扩展表信息
  99. session := engine.Table("GOODS").
  100. Select(`GOODS.*,
  101. HSBY_GOODSEX.HOTINDEX, HSBY_GOODSEX.VIDEOURLS, HSBY_GOODSEX.PICURLS,
  102. ENUMDICITEM.ENUMDICNAME CURRENCY, ENUMDICITEM.PARAM2 CURRENCYSIGN,
  103. MARKET.TRADEMODE`).
  104. Join("LEFT", "HSBY_GOODSEX", "HSBY_GOODSEX.GOODSID = GOODS.GOODSID").
  105. Join("LEFT", "ENUMDICITEM", "ENUMDICITEM.ENUMITEMNAME = GOODS.CURRENCYID and ENUMDICITEM.ENUMDICCODE = 'currency'").
  106. Join("LEFT", "MARKET", "MARKET.MARKETID = GOODS.MARKETID").
  107. Where(fmt.Sprintf("GOODS.GOODSSTATUS = 3 and GOODS.MARKETID in (%s)", marketIDs)) // 二级市场只获取 3:上市 状态的商品
  108. if descProvinceID > 0 {
  109. session = session.And("HSBY_GOODSEX.DESCPROVINCEID = ?", descProvinceID)
  110. }
  111. if descCityID > 0 {
  112. session = session.And("HSBY_GOODSEX.DESCCITYID = ?", descCityID)
  113. }
  114. if err := session.Find(&topGoodses); err != nil {
  115. return nil, err
  116. }
  117. if len(topGoodses) == 0 {
  118. // 无数据
  119. return topGoodses, nil
  120. }
  121. // 获取商品ID列表
  122. goodsCodes := ""
  123. for _, v := range topGoodses {
  124. if len(goodsCodes) == 0 {
  125. goodsCodes = v.Goodscode
  126. } else {
  127. goodsCodes += "," + v.Goodscode
  128. }
  129. }
  130. // 获取商品现价
  131. quoteDays, err := GetQuoteDays(goodsCodes)
  132. if err != nil {
  133. return nil, err
  134. }
  135. for i, g := range topGoodses {
  136. topGoods := &topGoodses[i]
  137. // FIXME: - 这里应该使用 Duck Typing,后期再处理
  138. // 计算最小变动单位
  139. // lotSize := float64(topGoods.Quoteminunit) * math.Pow(0.1, float64(topGoods.Decimalplace))
  140. // topGoods.LotSize, _ = strconv.ParseFloat(utils.FormatFloat(lotSize, int(topGoods.Decimalplace)), 64)
  141. for _, q := range quoteDays {
  142. if g.Goodscode == q.Goodscode {
  143. if q.Last != 0 {
  144. topGoods.Last = utils.IntToFloat64(int(q.Last), int(g.Decimalplace))
  145. }
  146. // 没有现价则尝试用昨结
  147. if topGoods.Last == 0 && q.Presettle != 0 {
  148. topGoods.Last = utils.IntToFloat64(int(q.Presettle), int(g.Decimalplace))
  149. }
  150. continue
  151. }
  152. }
  153. }
  154. return topGoodses, nil
  155. }
  156. // HsbyListingGoodsDetail 二级市场(挂牌点选)商品信息详情
  157. type HsbyListingGoodsDetail struct {
  158. Goodsid int64 `json:"goodsid" xorm:"'GOODSID'" binding:"required"` // 商品ID(自增ID SEQ_GOODS)
  159. Goodscode string `json:"goodscode" xorm:"'GOODSCODE'" binding:"required"` // 商品代码(内部)
  160. Goodsname string `json:"goodsname" xorm:"'GOODSNAME'" binding:"required"` // 商品名称
  161. Decimalplace int64 `json:"decimalplace" xorm:"'DECIMALPLACE'"` // 报价小数位
  162. Marketid int64 `json:"marketid" xorm:"'MARKETID'" binding:"required"` // 所属市场ID
  163. Quoteminunit int64 `json:"quoteminunit" xorm:"'QUOTEMINUNIT'"` // 行情最小变动单位 [整数,报价小数位一起使用]
  164. Agreeunit float64 `json:"agreeunit" xorm:"'AGREEUNIT'"` // 合约单位
  165. Hotindex int32 `json:"hotindex" xorm:"'HOTINDEX'"` // 景点热度
  166. Videourls string `json:"videourls" xorm:"'VIDEOURLS'"` // 介绍视频[多张用逗号分隔]
  167. Picurls string `json:"picurls" xorm:"'PICURLS'"` // 介绍图片[多张用逗号分隔]
  168. Descprovinceid int64 `json:"descprovinceid" xorm:"'DESCPROVINCEID'"` // 目的地(省)ID
  169. Desccityid int64 `json:"desccityid" xorm:"'DESCCITYID'"` // 目的地(市)ID
  170. Goodsdesc string `json:"goodsdesc" xorm:"'GOODSDESC'"` // 商品详情
  171. Currency string `json:"currency" xorm:"'CURRENCY'"` // 货币
  172. Currencysign string `json:"currencysign" xorm:"'CURRENCYSIGN'"` // 货币符号
  173. Vendorname string `json:"vendorname" xorm:"'VENDORNAME'"` // 供应商名称
  174. Vendorphone string `json:"vendorphone" xorm:"'VENDORPHONE'"` // 供应商客服电话
  175. Vendorattr string `json:"vendorattr" xorm:"'VENDORATTR'"` // 供应商附件(多张,逗号分隔)
  176. Trademode int32 `json:"trademode" xorm:"'TRADEMODE'" binding:"required"` // 交易模式 - 10:做市 13:竞价 15:通道交易 16:挂牌点选 17:仓单贸易 18:期权 19:竞拍-降价式 20:竞拍-竞价式 21:竞拍-大宗式 22:受托竞价
  177. Last float64 `json:"last" xorm:"-"` // 现价
  178. Limitup float64 `json:"limitup" xorm:"-"` // 涨停价
  179. Limitdown float64 `json:"limitdown" xorm:"-"` // 跌停价
  180. StepValue float64 `json:"stepvalue" xorm:"-"` // 价格最小变动单位
  181. LotSize int `json:"lotsize" xorm:"-"` // 手数最小变动单位
  182. }
  183. // GetHsbyListingGoodsDetail 获取二级市场(挂牌点选)商品信息详情
  184. // 参数 goodsID int 目标商品ID
  185. // 返回 *HsbyListingGoodsDetail 二级市场(挂牌点选)商品信息详情,查询无结果返回nil
  186. // 返回 error error
  187. func GetHsbyListingGoodsDetail(goodsID int) (*HsbyListingGoodsDetail, error) {
  188. engine := db.GetEngine()
  189. details := make([]HsbyListingGoodsDetail, 0)
  190. // 获取挂牌商品信息,以及扩展表信息
  191. // FIXME: - 这里使用Get方法,会造成SQL语句的嵌套出错,后期再研究
  192. session := engine.Table("GOODS").
  193. Select(`GOODS.GOODSID, GOODS.GOODSCODE, GOODS.GOODSNAME, GOODS.DECIMALPLACE, GOODS.MARKETID, GOODS.QUOTEMINUNIT, GOODS.AGREEUNIT,
  194. HSBY_GOODSEX.HOTINDEX, HSBY_GOODSEX.VIDEOURLS, HSBY_GOODSEX.PICURLS, HSBY_GOODSEX.DESCPROVINCEID, HSBY_GOODSEX.Desccityid, HSBY_GOODSEX.Goodsdesc,
  195. ENUMDICITEM.ENUMDICNAME CURRENCY, ENUMDICITEM.PARAM2 CURRENCYSIGN,
  196. HSBY_SUPPLIERINFO.VENDORNAME, HSBY_SUPPLIERINFO.VENDORPHONE, HSBY_SUPPLIERINFO.VENDORATTR,
  197. MARKET.TRADEMODE`).
  198. Join("LEFT", "HSBY_GOODSEX", "HSBY_GOODSEX.GOODSID = GOODS.GOODSID").
  199. Join("LEFT", "ENUMDICITEM", "ENUMDICITEM.ENUMITEMNAME = GOODS.CURRENCYID and ENUMDICITEM.ENUMDICCODE = 'currency'").
  200. Join("LEFT", "HSBY_SUPPLIERINFO", "HSBY_SUPPLIERINFO.VENDORID = HSBY_GOODSEX.VENDORID").
  201. Join("LEFT", "MARKET", "MARKET.MARKETID = GOODS.MARKETID").
  202. Where("GOODS.GOODSID = ?", goodsID)
  203. if err := session.Find(&details); err != nil {
  204. return nil, err
  205. }
  206. // 无目标商品
  207. if len(details) == 0 {
  208. return nil, nil
  209. }
  210. hsbyListingGoodsDetail := details[0]
  211. // 获取商品现价和涨跌停价
  212. quoteDays, err := GetQuoteDays(hsbyListingGoodsDetail.Goodscode)
  213. if err != nil {
  214. return nil, err
  215. }
  216. if len(quoteDays) > 0 {
  217. if quoteDays[0].Last != 0 {
  218. hsbyListingGoodsDetail.Last = utils.IntToFloat64(int(quoteDays[0].Last), int(hsbyListingGoodsDetail.Decimalplace))
  219. }
  220. // 没有现价则尝试用昨结
  221. if hsbyListingGoodsDetail.Last == 0 && quoteDays[0].Presettle != 0 {
  222. hsbyListingGoodsDetail.Last = utils.IntToFloat64(int(quoteDays[0].Presettle), int(hsbyListingGoodsDetail.Decimalplace))
  223. }
  224. if quoteDays[0].Limitup != 0 {
  225. hsbyListingGoodsDetail.Limitup = utils.IntToFloat64(int(quoteDays[0].Limitup), int(hsbyListingGoodsDetail.Decimalplace))
  226. }
  227. if quoteDays[0].Limitdown != 0 {
  228. hsbyListingGoodsDetail.Limitdown = utils.IntToFloat64(int(quoteDays[0].Limitdown), int(hsbyListingGoodsDetail.Decimalplace))
  229. }
  230. }
  231. // 计算价格最小变动单位
  232. stepValue := float64(hsbyListingGoodsDetail.Quoteminunit) * math.Pow(0.1, float64(hsbyListingGoodsDetail.Decimalplace))
  233. hsbyListingGoodsDetail.StepValue, _ = strconv.ParseFloat(utils.FormatFloat(stepValue, int(hsbyListingGoodsDetail.Decimalplace)), 64)
  234. // 计算手数最小变动单位
  235. hsbyListingGoodsDetail.LotSize = 1
  236. return &hsbyListingGoodsDetail, nil
  237. }
  238. // HsbyGoodsOrderDetail 二级市场挂牌商品当前可摘委托单信息
  239. type HsbyGoodsOrderDetail struct {
  240. Orderid string `json:"orderid" xorm:"'ORDERID'" binding:"required"` // 委托单号(100+Unix秒时间戳(10位)+2位(MarketServiceID)+xxxx)
  241. Buyorsell int32 `json:"buyorsell" xorm:"'BUYORSELL'" binding:"required"` // 买卖 - 0:买 1:卖
  242. Ordertime time.Time `json:"ordertime" xorm:"'ORDERTIME'" binding:"required"` // 委托时间
  243. Orderprice float64 `json:"orderprice" xorm:"'ORDERPRICE'"` // 委托价格
  244. Enableqty int64 `json:"enableqty" xorm:"ENABLEQTY"` // 可用数量
  245. Customername string `json:"customername" xorm:"'CUSTOMERNAME'"` // 客户名称(企业名称),已脱敏
  246. Currencysign string `json:"currencysign" xorm:"-"` // 货币符号
  247. Goodunit string `json:"goodunit" xorm:"-"` // 报价单位
  248. }
  249. // GetHsbyGoodsOrderDetails 获取二级市场(挂牌点选)商品对应的挂牌委托单信息
  250. // 输入 goodsID int 商品ID
  251. // 输入 buyOrSell int 挂牌委托单方向(对手单方向),0:买 1:卖
  252. // 输入 price float64 参考价格。买方向委托单则价格小于等于(站在摘牌人的角度);卖方向委托单则价格大于等于
  253. // 输出 []HsbyGoodsOrderDetail 商品对应的挂牌委托单信息
  254. // 输出 error error
  255. func GetHsbyGoodsOrderDetails(goodsID, buyOrSell int, price float64, accountIDs string) ([]HsbyGoodsOrderDetail, error) {
  256. engine := db.GetEngine()
  257. // 获取与目标商品相关的挂牌委托单信息(ListingSelectType = 1 or 3; OrderStatus =3 or 7)
  258. hsbyGoodsOrderDetails := make([]HsbyGoodsOrderDetail, 0)
  259. session := engine.Table("TRADE_ORDERDETAIL").
  260. Select(`to_char(TRADE_ORDERDETAIL.ORDERID) ORDERID, TRADE_ORDERDETAIL.BUYORSELL, TRADE_ORDERDETAIL.ORDERTIME, TRADE_ORDERDETAIL.ORDERPRICE, (TRADE_ORDERDETAIL.ORDERQTY - TRADE_ORDERDETAIL.TRADEQTY - TRADE_ORDERDETAIL.CANCELQTY) ENABLEQTY,
  261. substr(USERINFO.CUSTOMERNAME,0,1)||'****' as CUSTOMERNAME`).
  262. Join("LEFT", "TAACCOUNT", "TAACCOUNT.ACCOUNTID = TRADE_ORDERDETAIL.ACCOUNTID").
  263. Join("LEFT", "USERINFO", "USERINFO.USERID = TAACCOUNT.RELATEDUSERID").
  264. Where("(TRADE_ORDERDETAIL.LISTINGSELECTTYPE = 1 or TRADE_ORDERDETAIL.LISTINGSELECTTYPE = 3) and (TRADE_ORDERDETAIL.ORDERSTATUS = 3 or TRADE_ORDERDETAIL.ORDERSTATUS = 7)").
  265. And("TRADE_ORDERDETAIL.GOODSID = ?", goodsID).
  266. And("TRADE_ORDERDETAIL.BUYORSELL = ?", buyOrSell).
  267. And(fmt.Sprintf("TRADE_ORDERDETAIL.ACCOUNTID not in (%s)", accountIDs))
  268. if price > 0 {
  269. if buyOrSell == 0 {
  270. // 对手单买方向委托单则价格大于等于(站在摘牌人的角度,摘牌方面是卖,我的闲置下单)
  271. session = session.And("TRADE_ORDERDETAIL.ORDERPRICE >= ?", price)
  272. } else {
  273. // 对手单卖方向委托单则价格小于等于(站在摘牌人的角度,摘牌方面是买,热门商品下单)
  274. session = session.And("TRADE_ORDERDETAIL.ORDERPRICE <= ?", price)
  275. }
  276. }
  277. if err := session.Find(&hsbyGoodsOrderDetails); err != nil {
  278. return nil, err
  279. }
  280. // 获取商品货币符号和报价单位
  281. goods, err := GetGoods(goodsID)
  282. if err != nil {
  283. return nil, err
  284. }
  285. currencyEnums, err := GetEnumDicItem("currency", int(goods.Currencyid))
  286. if err != nil {
  287. return nil, err
  288. }
  289. currencysign := ""
  290. if len(currencyEnums) > 0 {
  291. currencysign = currencyEnums[0].Param2
  292. }
  293. goodsUnitEnums, err := GetEnumDicItem("goodsunit", int(goods.Goodunitid))
  294. if err != nil {
  295. return nil, err
  296. }
  297. goodsUnit := ""
  298. if len(goodsUnitEnums) > 0 {
  299. goodsUnit = goodsUnitEnums[0].Enumdicname
  300. }
  301. for i := range hsbyGoodsOrderDetails {
  302. orderDetail := &hsbyGoodsOrderDetails[i]
  303. // 设置货币符号和报价单位
  304. orderDetail.Currencysign = currencysign
  305. orderDetail.Goodunit = goodsUnit
  306. }
  307. return hsbyGoodsOrderDetails, nil
  308. }
  309. // HybsMyBuyOrderDetail 我的订单(一二级市场的买方向委托单)
  310. type HybsMyBuyOrderDetail struct {
  311. Orderid string `json:"orderid" xorm:"'ORDERID'" binding:"required"` // 委托单号(100+Unix秒时间戳(10位)+2位(MarketServiceID)+xxxx)
  312. Marketid int32 `json:"marketid" xorm:"'MARKETID'" binding:"required"` // 市场ID
  313. Goodsid int32 `json:"goodsid" xorm:"'GOODSID'" binding:"required"` // 商品ID
  314. Accountid int64 `json:"accountid" xorm:"'ACCOUNTID'" binding:"required"` // 账户ID[报价币种]
  315. Buyorsell int32 `json:"buyorsell" xorm:"'BUYORSELL'" binding:"required"` // 买卖 - 0:买 1:卖
  316. Orderprice float64 `json:"orderprice" xorm:"'ORDERPRICE'"` // 委托价格
  317. Orderqty int64 `json:"orderqty" xorm:"'ORDERQTY'" binding:"required"` // 委托数量
  318. Tradeqty int64 `json:"tradeqty" xorm:"'TRADEQTY'"` // 成交数量
  319. Cancelqty int64 `json:"cancelqty" xorm:"'CANCELQTY'"` // 撤单数量
  320. Ordertime time.Time `json:"ordertime" xorm:"'ORDERTIME'" binding:"required"` // 委托时间
  321. Listingselecttype int32 `json:"listingselecttype" xorm:"'LISTINGSELECTTYPE'"` // 挂牌点选类型 - 1:挂牌 2:摘牌 3:先摘后挂
  322. Orderstatus int32 `json:"orderstatus" xorm:"'ORDERSTATUS'"` // 委托状态 - 1: 委托请求 2:待冻结 3:委托成功 4: 委托失败 5:配对成功 6: 已撤销 7:部分成交 8:已成交 9:部成部撤 10:成交失败 11:已拒绝 12:经过摘牌(先摘后挂专用-先摘后挂已摘过) 13:冻结成功(通道交易专用) 14:通道已撤 15:通道部成部撤 16:成交失败违约(荷兰式竞拍专用)
  323. Goodscode string `json:"goodscode" xorm:"'GOODSCODE'" binding:"required"` // 商品代码(内部)
  324. Goodsname string `json:"goodsname" xorm:"'GOODSNAME'" binding:"required"` // 商品名称
  325. Decimalplace int64 `json:"decimalplace" xorm:"'DECIMALPLACE'"` // 报价小数位
  326. Agreeunit float64 `json:"agreeunit" xorm:"'AGREEUNIT'"` // 合约单位
  327. Picurls1 string `json:"picurls1" xorm:"'PICURLS1'"` // 预售商品介绍图片[多张用逗号分隔]
  328. Picurls2 string `json:"picurls2" xorm:"'PICURLS2'"` // 挂牌商品介绍图片[多张用逗号分隔]
  329. Trademode uint32 `json:"trademode" xorm:"'TRADEMODE'" binding:"required"` // 交易模式 - 10:做市 13:竞价 15:通道交易 16:挂牌点选 17:仓单贸易 18:期权 19:竞拍-降价式 20:竞拍-竞价式 21:竞拍-大宗式 22:受托竞价
  330. Currencysign string `json:"currencysign" xorm:"'CURRENCYSIGN'"` // 货币符号
  331. Vendorname1 string `json:"vendorname1" xorm:"'VENDORNAME1'"` // 预售商品供应商名称
  332. Vendorname2 string `json:"vendorname2" xorm:"'VENDORNAME2'"` // 挂牌商品供应商名称
  333. MyBuyStatus int `json:"mybuystatus" xorm:"-"` // "我的订单"显示状态- 1:抢购中 2:求购中 3:已完成 4:已撤消 5:委托失败
  334. Orderamount float64 `json:"orderamount" xorm:"-"` // 委托金额
  335. }
  336. // GetHsbyBuyMyOrderDetails 获取“我的订单”数据(包括一二级市场)
  337. // 输入 accountIDs string 资金账户列表
  338. // 输入 myBuyStatus int "我的订单"状态, 1:抢购中 2:求购中 3:已完成
  339. // 输出 []HybsMyBuyOrderDetail “我的订单”数据
  340. // 输出 error error
  341. func GetHsbyBuyMyOrderDetails(accountIDs string, myBuyStatus int) ([]HybsMyBuyOrderDetail, error) {
  342. // 获取市场信息
  343. markets, err := GetMarkets()
  344. if err != nil {
  345. return nil, err
  346. }
  347. engine := db.GetEngine()
  348. // 委托状态
  349. orderStatus := "0" // 单据状态,为0的时候查询全部
  350. marketIDs := "" // 我的订单包括一二级市场的单据
  351. switch myBuyStatus {
  352. case 1: // 抢购中 (一级市场)
  353. // 获取市场ID
  354. for _, v := range markets {
  355. if v.Trademode == 71 { // 预售挂牌
  356. if len(marketIDs) == 0 {
  357. marketIDs = strconv.Itoa(int(v.Marketid))
  358. } else {
  359. marketIDs += "," + strconv.Itoa(int(v.Marketid))
  360. }
  361. }
  362. }
  363. orderStatus = "3,7"
  364. case 2: // 求购中(二级市场)
  365. // 获取市场ID
  366. for _, v := range markets {
  367. if v.Trademode == 16 { // 挂牌点选
  368. if len(marketIDs) == 0 {
  369. marketIDs = strconv.Itoa(int(v.Marketid))
  370. } else {
  371. marketIDs += "," + strconv.Itoa(int(v.Marketid))
  372. }
  373. }
  374. }
  375. orderStatus = "3,7"
  376. case 3: // 已完成
  377. orderStatus = "8,9"
  378. }
  379. // 默认取 TradeMode = 16 or 71 的市场
  380. if len(marketIDs) == 0 {
  381. for _, v := range markets {
  382. if v.Trademode == 16 || v.Trademode == 71 {
  383. if len(marketIDs) == 0 {
  384. marketIDs = strconv.Itoa(int(v.Marketid))
  385. } else {
  386. marketIDs += "," + strconv.Itoa(int(v.Marketid))
  387. }
  388. }
  389. }
  390. }
  391. hybsMyBuyOrderDetails := make([]HybsMyBuyOrderDetail, 0)
  392. // "我的订单"都是买委托
  393. session := engine.Table("TRADE_ORDERDETAIL").
  394. Select(`to_char(TRADE_ORDERDETAIL.ORDERID) ORDERID, TRADE_ORDERDETAIL.*,
  395. GOODS.GOODSCODE, GOODS.GOODSNAME, GOODS.DECIMALPLACE, GOODS.AGREEUNIT,
  396. HG1.PICURLS PICURLS1,
  397. HG2.PICURLS PICURLS2,
  398. MARKET.TRADEMODE,
  399. ENUMDICITEM.PARAM2 CURRENCYSIGN,
  400. HS1.VENDORNAME VENDORNAME1,
  401. HS2.VENDORNAME VENDORNAME2`).
  402. Join("LEFT", "GOODS", "GOODS.GOODSID = TRADE_ORDERDETAIL.GOODSID").
  403. Join("LEFT", "CPTRADE_PRESALEGOODSEX", "CPTRADE_PRESALEGOODSEX.GOODSID = TRADE_ORDERDETAIL.GOODSID").
  404. Join("LEFT", "HSBY_GOODSEX HG1", "HG1.GOODSID = CPTRADE_PRESALEGOODSEX.RELATEDGOODSID").
  405. Join("LEFT", "HSBY_GOODSEX HG2", "HG2.GOODSID = GOODS.GOODSID").
  406. Join("LEFT", "ENUMDICITEM", "GOODS.CURRENCYID = ENUMDICITEM.ENUMITEMNAME and ENUMDICITEM.ENUMDICCODE = 'currency'").
  407. Join("LEFT", "MARKET", "MARKET.MARKETID = TRADE_ORDERDETAIL.MARKETID").
  408. Join("LEFT", "HSBY_SUPPLIERINFO HS1", "HS1.VENDORID = HG1.VENDORID").
  409. Join("LEFT", "HSBY_SUPPLIERINFO HS2", "HS2.VENDORID = HG2.VENDORID").
  410. Where(fmt.Sprintf("TRADE_ORDERDETAIL.BUYORSELL = 0 and TRADE_ORDERDETAIL.ACCOUNTID in (%s)", accountIDs)).
  411. And(fmt.Sprintf("TRADE_ORDERDETAIL.MARKETID in (%s)", marketIDs))
  412. // 是否过滤状态(抢购中、求购中、已完成)
  413. if orderStatus != "0" {
  414. session = session.And(fmt.Sprintf("TRADE_ORDERDETAIL.ORDERSTATUS in (%s)", orderStatus))
  415. }
  416. if err := session.Find(&hybsMyBuyOrderDetails); err != nil {
  417. return nil, err
  418. }
  419. // 设置“我的订单”显示状态
  420. for i := range hybsMyBuyOrderDetails {
  421. detail := &hybsMyBuyOrderDetails[i]
  422. // 委托金额 = 委托价格 * 委托数量 * 合约单位
  423. detail.Orderamount = detail.Orderprice * float64(detail.Orderqty) * float64(detail.Agreeunit)
  424. if detail.Trademode == 71 && (detail.Orderstatus == 3 || detail.Orderstatus == 7) {
  425. // 抢购中
  426. detail.MyBuyStatus = 1
  427. } else if detail.Trademode == 16 && (detail.Orderstatus == 3 || detail.Orderstatus == 7) {
  428. // 求购中
  429. detail.MyBuyStatus = 2
  430. } else if detail.Orderstatus == 8 || detail.Orderstatus == 9 {
  431. // 已完成
  432. detail.MyBuyStatus = 3
  433. } else if detail.Orderstatus == 6 {
  434. // 已撤销
  435. detail.MyBuyStatus = 4
  436. } else if detail.Orderstatus == 4 {
  437. // 委托失败
  438. detail.MyBuyStatus = 5
  439. }
  440. }
  441. return hybsMyBuyOrderDetails, nil
  442. }
  443. // HsbyMyGoods 我的商品(买方向持仓)
  444. type HsbyMyGoods struct {
  445. Accountid int64 `json:"accountid" xorm:"'ACCOUNTID'" binding:"required"` // 账号Id
  446. Goodsid int32 `json:"goodsid" xorm:"'GOODSID'" binding:"required"` // 商品Id
  447. Buycurpositionqty int64 `json:"buycurpositionqty" xorm:"'BUYCURPOSITIONQTY'"` // 买当前持仓总数量
  448. Buycurholderamount float64 `json:"buycurholderamount" xorm:"'BUYCURHOLDERAMOUNT'"` // 买当前持仓总金额[商品币种]
  449. EnableQty int64 `json:"enableqty" xorm:"'ENABLEQTY'"` // 可用数量
  450. Goodscode string `json:"goodscode" xorm:"'GOODSCODE'" binding:"required"` // 商品代码(内部)
  451. Goodsname string `json:"goodsname" xorm:"'GOODSNAME'" binding:"required"` // 商品名称
  452. Decimalplace int64 `json:"decimalplace" xorm:"'DECIMALPLACE'"` // 报价小数位
  453. Agreeunit float64 `json:"agreeunit" xorm:"'AGREEUNIT'"` // 合约单位
  454. Goodsstatus int64 `json:"goodsstatus" xorm:"'GOODSSTATUS'"` // 商品状态- 1:待审核 2:未上市 3:上市 4:已注销 5:审核拒绝 6:退市 7:待退市
  455. Picurls string `json:"picurls" xorm:"'PICURLS'"` // 介绍图片[多张用逗号分隔]
  456. Currencysign string `json:"currencysign" xorm:"'CURRENCYSIGN'"` // 货币符号
  457. Buyaverageprice float64 `json:"buyaverageprice" xorm:"-"` // 持仓均价
  458. }
  459. // GetHsbyMyGoods 获取“我的商品”信息
  460. // 输入 accountIDs string 资金账户列表,格式 1,2,3
  461. // 输出 []HsbyMyGoods “我的商品”信息
  462. // 输入 error error
  463. func GetHsbyMyGoods(accountIDs string) ([]HsbyMyGoods, error) {
  464. engine := db.GetEngine()
  465. hsbyMyGoodses := make([]HsbyMyGoods, 0)
  466. // 此定制版本,只查询出当前手数(期末)大于0的数据 (TRADEPOSITION.BUYCURPOSITIONQTY > 0)
  467. if err := engine.Table("TRADEPOSITION").
  468. Select(`TRADEPOSITION.*, (TRADEPOSITION.BUYCURPOSITIONQTY - TRADEPOSITION.BUYFROZENQTY - TRADEPOSITION.BUYOTHERFROZENQTY) ENABLEQTY,
  469. GOODS.GOODSCODE, GOODS.GOODSNAME, GOODS.DECIMALPLACE, GOODS.AGREEUNIT, GOODS.GOODSSTATUS,
  470. HSBY_GOODSEX.PICURLS,
  471. ENUMDICITEM.PARAM2 CURRENCYSIGN`).
  472. Join("LEFT", "GOODS", "GOODS.GOODSID = TRADEPOSITION.GOODSID").
  473. Join("INNER", "HSBY_GOODSEX", "HSBY_GOODSEX.GOODSID = GOODS.GOODSID").
  474. Join("LEFT", "ENUMDICITEM", "GOODS.CURRENCYID = ENUMDICITEM.ENUMITEMNAME and ENUMDICITEM.ENUMDICCODE = 'currency'").
  475. Where(fmt.Sprintf("TRADEPOSITION.BUYCURPOSITIONQTY > 0 and TRADEPOSITION.ACCOUNTID in (%s)", accountIDs)).Find(&hsbyMyGoodses); err != nil {
  476. return nil, err
  477. }
  478. for i := range hsbyMyGoodses {
  479. myGoods := &hsbyMyGoodses[i]
  480. // 持仓均价
  481. averagePrice := myGoods.Buycurholderamount / float64(myGoods.Buycurpositionqty) / myGoods.Agreeunit
  482. myGoods.Buyaverageprice, _ = strconv.ParseFloat(utils.FormatFloat(averagePrice, int(myGoods.Decimalplace)), 64)
  483. }
  484. return hsbyMyGoodses, nil
  485. }
  486. // HsbyPreGoods 新品上市商品(一级市场产能预售)
  487. type HsbyPreGoods struct {
  488. Goodsid int64 `json:"goodsid" xorm:"'GOODSID'" binding:"required"` // 商品ID(自增ID SEQ_GOODS)
  489. Goodscode string `json:"goodscode" xorm:"'GOODSCODE'" binding:"required"` // 商品代码(内部)
  490. Goodsname string `json:"goodsname" xorm:"'GOODSNAME'" binding:"required"` // 商品名称
  491. Decimalplace int64 `json:"decimalplace" xorm:"'DECIMALPLACE'"` // 报价小数位
  492. Marketid int64 `json:"marketid" xorm:"'MARKETID'" binding:"required"` // 所属市场ID
  493. Goodsstatus int64 `json:"goodsstatus" xorm:"'GOODSSTATUS'"` // 商品状态- 2:未上市 3:上市
  494. Quoteminunit int64 `json:"quoteminunit" xorm:"'QUOTEMINUNIT'"` // 行情最小变动单位 [整数,报价小数位一起使用]
  495. Agreeunit float64 `json:"agreeunit" xorm:"'AGREEUNIT'"` // 合约单位
  496. Listingdate time.Time `json:"listingdate" xorm:"'LISTINGDATE'"` // 交易开始日期
  497. Lasttradedate time.Time `json:"lasttradedate" xorm:"'LASTTRADEDATE'"` // 最后交易日期(状态:待退市)
  498. Relatedgoodsid int64 `json:"relatedgoodsid" xorm:"'RELATEDGOODSID'"` // 关联交易合约ID
  499. Presaleqty int64 `json:"presaleqty" xorm:"'PRESALEQTY'"` // 预售数量
  500. Refprice float64 `json:"refprice" xorm:"'REFPRICE'"` // 参考价格[一口价]
  501. Presaledqty int64 `json:"presaledqty" xorm:"'PRESALEDQTY'"` // 已预售量(预售结束时更新)
  502. EnableQty int64 `json:"enableqty" xorm:"ENABLEQTY"` // 剩余数量
  503. Currency string `json:"currency" xorm:"'CURRENCY'"` // 货币
  504. Currencysign string `json:"currencysign" xorm:"'CURRENCYSIGN'"` // 货币符号
  505. Videourls string `json:"videourls" xorm:"'VIDEOURLS'"` // 介绍视频[多张用逗号分隔]
  506. Picurls string `json:"picurls" xorm:"'PICURLS'"` // 介绍图片[多张用逗号分隔]
  507. Trademode int32 `json:"trademode" xorm:"'TRADEMODE'" binding:"required"` // 交易模式 - 10:做市 13:竞价 15:通道交易 16:挂牌点选 17:仓单贸易 18:期权 19:竞拍-降价式 20:竞拍-竞价式 21:竞拍-大宗式 22:受托竞价
  508. }
  509. // GetHsbyPreGoodses 获取“新品上市”商品列表(一级市场产能预售)
  510. // 输入 marketIDs string 市场ID列表,格式 1,2,3
  511. // 输入 descProvinceID int 省ID
  512. // 输入 descCityID int 市ID
  513. // 输出 []HsbyPreGoods “新品上市”商品列表
  514. // 输出 error error
  515. func GetHsbyPreGoodses(marketIDs string, descProvinceID, descCityID int) ([]HsbyPreGoods, error) {
  516. engine := db.GetEngine()
  517. // 注意:一级产能预售市场使用的商品实际上是和二级市场有关系的,所以要使用 CPTRADE_PRESALEGOODSEX.RELATEDGOODSID 进行关联
  518. // 注意:CPTRADE_PRESALEGOODSEX 表的开始与结束时间是给大宗用的
  519. preGoodses := make([]HsbyPreGoods, 0)
  520. session := engine.Table("GOODS").
  521. Select(`GOODS.*,
  522. CPTRADE_PRESALEGOODSEX.RELATEDGOODSID, CPTRADE_PRESALEGOODSEX.PRESALEQTY, CPTRADE_PRESALEGOODSEX.REFPRICE,
  523. CPTRADE_PRESALEGOODSEX.PRESALEDQTY, (CPTRADE_PRESALEGOODSEX.PRESALEQTY - CPTRADE_PRESALEGOODSEX.PRESALEDQTY) ENABLEQTY,
  524. ENUMDICITEM.ENUMDICNAME CURRENCY, ENUMDICITEM.PARAM2 CURRENCYSIGN,
  525. HSBY_GOODSEX.VIDEOURLS, HSBY_GOODSEX.PICURLS,
  526. MARKET.TRADEMODE`).
  527. Join("INNER", "CPTRADE_PRESALEGOODSEX", "CPTRADE_PRESALEGOODSEX.GOODSID = GOODS.GOODSID").
  528. Join("LEFT", "ENUMDICITEM", "ENUMDICITEM.ENUMITEMNAME = GOODS.CURRENCYID and ENUMDICITEM.ENUMDICCODE = 'currency'").
  529. Join("LEFT", "HSBY_GOODSEX", "HSBY_GOODSEX.GOODSID = CPTRADE_PRESALEGOODSEX.RELATEDGOODSID").
  530. Join("LEFT", "MARKET", "MARKET.MARKETID = GOODS.MARKETID").
  531. Where(fmt.Sprintf("GOODS.GOODSSTATUS in (2,3) and GOODS.MARKETID in (%s)", marketIDs)) // 一级市场获取 2:未上市 3:上市 状态的商品
  532. if descProvinceID > 0 {
  533. session = session.And("HSBY_GOODSEX.DESCPROVINCEID = ?", descProvinceID)
  534. }
  535. if descCityID > 0 {
  536. session = session.And("HSBY_GOODSEX.DESCCITYID = ?", descCityID)
  537. }
  538. if err := session.Find(&preGoodses); err != nil {
  539. return nil, err
  540. }
  541. // for i := range preGoodses {
  542. // preGoods := &preGoodses[i]
  543. // // FIXME: - 这里应该使用 Duck Typing,后期再处理
  544. // // 计算最小变动单位
  545. // lotSize := float64(preGoods.Quoteminunit) * math.Pow(0.1, float64(preGoods.Decimalplace))
  546. // preGoods.LotSize, _ = strconv.ParseFloat(utils.FormatFloat(lotSize, int(preGoods.Decimalplace)), 64)
  547. // }
  548. return preGoodses, nil
  549. }
  550. // HsbyPreGoodsDetail 一级市场(产能预售)商品信息详情
  551. type HsbyPreGoodsDetail struct {
  552. Goodsid int64 `json:"goodsid" xorm:"'GOODSID'" binding:"required"` // 商品ID(自增ID SEQ_GOODS)
  553. Goodscode string `json:"goodscode" xorm:"'GOODSCODE'" binding:"required"` // 商品代码(内部)
  554. Goodsname string `json:"goodsname" xorm:"'GOODSNAME'" binding:"required"` // 商品名称
  555. Decimalplace int64 `json:"decimalplace" xorm:"'DECIMALPLACE'"` // 报价小数位
  556. Marketid int64 `json:"marketid" xorm:"'MARKETID'" binding:"required"` // 所属市场ID
  557. Goodsstatus int64 `json:"goodsstatus" xorm:"'GOODSSTATUS'"` // 商品状态- 2:未上市 3:上市
  558. Quoteminunit int64 `json:"quoteminunit" xorm:"'QUOTEMINUNIT'"` // 行情最小变动单位 [整数,报价小数位一起使用]
  559. Agreeunit float64 `json:"agreeunit" xorm:"'AGREEUNIT'"` // 合约单位
  560. Listingdate time.Time `json:"listingdate" xorm:"'LISTINGDATE'"` // 交易开始日期
  561. Lasttradedate time.Time `json:"lasttradedate" xorm:"'LASTTRADEDATE'"` // 最后交易日期(状态:待退市)
  562. Relatedgoodsid int64 `json:"relatedgoodsid" xorm:"'RELATEDGOODSID'"` // 关联交易合约ID
  563. Presaleqty int64 `json:"presaleqty" xorm:"'PRESALEQTY'"` // 预售数量
  564. Refprice float64 `json:"refprice" xorm:"'REFPRICE'"` // 参考价格[一口价]
  565. Presaledqty int64 `json:"presaledqty" xorm:"'PRESALEDQTY'"` // 已预售量(预售结束时更新)
  566. EnableQty int64 `json:"enableqty" xorm:"ENABLEQTY"` // 剩余数量
  567. Videourls string `json:"videourls" xorm:"'VIDEOURLS'"` // 介绍视频[多张用逗号分隔]
  568. Picurls string `json:"picurls" xorm:"'PICURLS'"` // 介绍图片[多张用逗号分隔]
  569. Descprovinceid int64 `json:"descprovinceid" xorm:"'DESCPROVINCEID'"` // 目的地(省)ID
  570. Desccityid int64 `json:"desccityid" xorm:"'DESCCITYID'"` // 目的地(市)ID
  571. Goodsdesc string `json:"goodsdesc" xorm:"'GOODSDESC'"` // 商品详情
  572. Currency string `json:"currency" xorm:"'CURRENCY'"` // 货币
  573. Currencysign string `json:"currencysign" xorm:"'CURRENCYSIGN'"` // 货币符号
  574. Goodunit string `json:"goodunit" xorm:"GOODUNIT"` // 报价单位
  575. Vendorname string `json:"vendorname" xorm:"'VENDORNAME'"` // 供应商名称
  576. Vendorphone string `json:"vendorphone" xorm:"'VENDORPHONE'"` // 供应商客服电话
  577. Vendorattr string `json:"vendorattr" xorm:"'VENDORATTR'"` // 供应商附件(多张,逗号分隔)
  578. Trademode int32 `json:"trademode" xorm:"'TRADEMODE'" binding:"required"` // 交易模式 - 10:做市 13:竞价 15:通道交易 16:挂牌点选 17:仓单贸易 18:期权 19:竞拍-降价式 20:竞拍-竞价式 21:竞拍-大宗式 22:受托竞价
  579. Customername string `json:"customername" xorm:"'CUSTOMERNAME'"` // 发行单位
  580. StepValue float64 `json:"stepvalue" xorm:"-"` // 价格最小变动单位
  581. LotSize int `json:"lotsize" xorm:"-"` // 手数最小变动单位
  582. Buymaxqty int `json:"buymaxqty" xorm:"'-'"` // 购买上限 [71] - 0为不限
  583. }
  584. // GetHsbyPreGoodsDetail 获取一级市场(产能预售)商品信息详情
  585. // 参数 goodsID int 目标商品ID
  586. // 返回 *HsbyListingGoodsDetail 二级市场(挂牌点选)商品信息详情,查询无结果返回nil
  587. // 返回 error error
  588. func GetHsbyPreGoodsDetail(goodsID, accountID int) (*HsbyPreGoodsDetail, error) {
  589. engine := db.GetEngine()
  590. details := make([]HsbyPreGoodsDetail, 0)
  591. // 获取挂牌商品信息,以及扩展表信息
  592. // FIXME: - 这里使用Get方法,会造成SQL语句的嵌套出错,后期再研究
  593. session := engine.Table("GOODS").
  594. Select(`GOODS.*,
  595. CPTRADE_PRESALEGOODSEX.RELATEDGOODSID, CPTRADE_PRESALEGOODSEX.PRESALEQTY, CPTRADE_PRESALEGOODSEX.REFPRICE,
  596. CPTRADE_PRESALEGOODSEX.PRESALEDQTY, (CPTRADE_PRESALEGOODSEX.PRESALEQTY - CPTRADE_PRESALEGOODSEX.PRESALEDQTY) ENABLEQTY,
  597. HSBY_GOODSEX.VIDEOURLS, HSBY_GOODSEX.PICURLS, HSBY_GOODSEX.DESCPROVINCEID, HSBY_GOODSEX.Desccityid, HSBY_GOODSEX.Goodsdesc,
  598. ENUMDICITEM.ENUMDICNAME CURRENCY, ENUMDICITEM.PARAM2 CURRENCYSIGN,
  599. E1.ENUMDICNAME GOODUNIT,
  600. HSBY_SUPPLIERINFO.VENDORNAME, HSBY_SUPPLIERINFO.VENDORPHONE, HSBY_SUPPLIERINFO.VENDORATTR,
  601. MARKET.TRADEMODE,
  602. USERINFO.CUSTOMERNAME`).
  603. Join("INNER", "CPTRADE_PRESALEGOODSEX", "CPTRADE_PRESALEGOODSEX.GOODSID = GOODS.GOODSID").
  604. Join("LEFT", "HSBY_GOODSEX", "HSBY_GOODSEX.GOODSID = CPTRADE_PRESALEGOODSEX.RELATEDGOODSID").
  605. Join("LEFT", "ENUMDICITEM", "ENUMDICITEM.ENUMITEMNAME = GOODS.CURRENCYID and ENUMDICITEM.ENUMDICCODE = 'currency'").
  606. Join("LEFT", "ENUMDICITEM E1", "E1.ENUMITEMNAME = GOODS.GOODUNITID and E1.ENUMDICCODE = 'goodsunit'").
  607. Join("LEFT", "HSBY_SUPPLIERINFO", "HSBY_SUPPLIERINFO.VENDORID = HSBY_GOODSEX.VENDORID").
  608. Join("LEFT", "MARKET", "MARKET.MARKETID = GOODS.MARKETID").
  609. Join("LEFT", "USERINFO", "USERINFO.USERID = CPTRADE_PRESALEGOODSEX.USERID").
  610. Where("GOODS.GOODSID = ?", goodsID)
  611. if err := session.Find(&details); err != nil {
  612. return nil, err
  613. }
  614. // 无目标商品
  615. if len(details) == 0 {
  616. return nil, nil
  617. }
  618. preGoodsDetail := details[0]
  619. // FIXME: - 这里应该使用 Duck Typing,后期再处理
  620. // 计算价格最小变动单位
  621. stepValue := float64(preGoodsDetail.Quoteminunit) * math.Pow(0.1, float64(preGoodsDetail.Decimalplace))
  622. preGoodsDetail.StepValue, _ = strconv.ParseFloat(utils.FormatFloat(stepValue, int(preGoodsDetail.Decimalplace)), 64)
  623. // 计算手数最小变动单位,一级市场固定为1
  624. preGoodsDetail.LotSize = 1
  625. // 获取购买上限
  626. buyMaxQty, err := GetCPTradeBuyLimit(goodsID, accountID)
  627. if err != nil {
  628. return nil, err
  629. }
  630. preGoodsDetail.Buymaxqty = buyMaxQty
  631. return &preGoodsDetail, nil
  632. }
  633. // HsbySellMyDetail "我的闲置"单据信息(已发布 - 二级市场卖挂,3:委托成功、7:部分成交; 已完成 - 二级市场成交单,包括历史数据)
  634. type HsbySellMyDetail struct {
  635. Orderid string `json:"orderid" xorm:"'ORDERID'" binding:"required"` // 单号(已发布 - 委托单号;已完成 - 成交单号)
  636. Marketid int32 `json:"marketid" xorm:"'MARKETID'" binding:"required"` // 市场ID
  637. Goodsid int32 `json:"goodsid" xorm:"'GOODSID'" binding:"required"` // 商品ID
  638. Accountid int64 `json:"accountid" xorm:"'ACCOUNTID'" binding:"required"` // 账户ID[报价币种]
  639. Buyorsell int32 `json:"buyorsell" xorm:"'BUYORSELL'" binding:"required"` // 买卖 - 0:买 1:卖
  640. Price float64 `json:"price" xorm:"'PRICE'"` // 价格
  641. Qty int64 `json:"qty" xorm:"'QTY'" binding:"required"` // 数量
  642. Time time.Time `json:"time" xorm:"'TIME'" binding:"required"` // 时间
  643. Amount float64 `json:"amount" xorm:"'AMOUNT'"` // 金额 = 价格 * 数量 * 合约单位
  644. Goodscode string `json:"goodscode" xorm:"'GOODSCODE'" binding:"required"` // 商品代码(内部)
  645. Goodsname string `json:"goodsname" xorm:"'GOODSNAME'" binding:"required"` // 商品名称
  646. Decimalplace int64 `json:"decimalplace" xorm:"'DECIMALPLACE'"` // 报价小数位
  647. Agreeunit float64 `json:"agreeunit" xorm:"'AGREEUNIT'"` // 合约单位
  648. Picurls string `json:"picurls" xorm:"'PICURLS'"` // 介绍图片[多张用逗号分隔]
  649. Trademode uint32 `json:"trademode" xorm:"'TRADEMODE'" binding:"required"` // 交易模式 - 10:做市 13:竞价 15:通道交易 16:挂牌点选 17:仓单贸易 18:期权 19:竞拍-降价式 20:竞拍-竞价式 21:竞拍-大宗式 22:受托竞价
  650. Currencysign string `json:"currencysign" xorm:"'CURRENCYSIGN'"` // 货币符号
  651. Vendorname string `json:"vendorname" xorm:"'VENDORNAME'"` // 供应商名称
  652. OrderType int32 `json:"ordertype" xorm:"'ORDERTYPE'"` // 单据类型:0 - 已发布, 1 - 已完成
  653. }
  654. // GetHsbySellMyOrderDetails 获取"我的闲置 - 已发布"单据信息
  655. // 输入 accountIDs string 资金账户列表
  656. // 输出 []HsbySellMyDetail "我的闲置 - 已发布"单据信息
  657. // 输出 error error
  658. func GetHsbySellMyOrderDetails(accountIDs string) ([]HsbySellMyDetail, error) {
  659. // 获取市场信息
  660. markets, err := GetMarkets()
  661. if err != nil {
  662. return nil, err
  663. }
  664. engine := db.GetEngine()
  665. marketIDs := "" // 我的闲置-已发布: 二级市场卖挂牌
  666. // 默认取 TradeMode = 16
  667. for _, v := range markets {
  668. if v.Trademode == 16 {
  669. if len(marketIDs) == 0 {
  670. marketIDs = strconv.Itoa(int(v.Marketid))
  671. } else {
  672. marketIDs += "," + strconv.Itoa(int(v.Marketid))
  673. }
  674. }
  675. }
  676. orderDetails := make([]HsbySellMyDetail, 0)
  677. // “我的闲置 - 已发布”都是卖挂委托
  678. session := engine.Table("TRADE_ORDERDETAIL").
  679. Select(`to_char(TRADE_ORDERDETAIL.ORDERID) ORDERID,
  680. TRADE_ORDERDETAIL.MARKETID, TRADE_ORDERDETAIL.GOODSID, TRADE_ORDERDETAIL.ACCOUNTID, TRADE_ORDERDETAIL.BUYORSELL,
  681. TRADE_ORDERDETAIL.ORDERPRICE PRICE, TRADE_ORDERDETAIL.ORDERQTY QTY, TRADE_ORDERDETAIL.ORDERTIME TIME,
  682. (TRADE_ORDERDETAIL.ORDERPRICE * TRADE_ORDERDETAIL.ORDERQTY * GOODS.AGREEUNIT) AMOUNT,
  683. GOODS.GOODSCODE, GOODS.GOODSNAME, GOODS.DECIMALPLACE, GOODS.AGREEUNIT,
  684. HSBY_GOODSEX.PICURLS,
  685. MARKET.TRADEMODE,
  686. ENUMDICITEM.PARAM2 CURRENCYSIGN,
  687. HSBY_SUPPLIERINFO.VENDORNAME,
  688. 0 ORDERTYPE`).
  689. Join("LEFT", "GOODS", "GOODS.GOODSID = TRADE_ORDERDETAIL.GOODSID").
  690. Join("LEFT", "HSBY_GOODSEX", "HSBY_GOODSEX.GOODSID = GOODS.GOODSID").
  691. Join("LEFT", "ENUMDICITEM", "GOODS.CURRENCYID = ENUMDICITEM.ENUMITEMNAME and ENUMDICITEM.ENUMDICCODE = 'currency'").
  692. Join("LEFT", "MARKET", "MARKET.MARKETID = TRADE_ORDERDETAIL.MARKETID").
  693. Join("LEFT", "HSBY_SUPPLIERINFO", "HSBY_SUPPLIERINFO.VENDORID = HSBY_GOODSEX.VENDORID").
  694. Where(fmt.Sprintf(`TRADE_ORDERDETAIL.BUYORSELL = 1 and TRADE_ORDERDETAIL.LISTINGSELECTTYPE = 1 and (TRADE_ORDERDETAIL.ORDERSTATUS = 3 or TRADE_ORDERDETAIL.ORDERSTATUS = 7)
  695. and TRADE_ORDERDETAIL.ACCOUNTID in (%s)`, accountIDs)).
  696. And(fmt.Sprintf("TRADE_ORDERDETAIL.MARKETID in (%s)", marketIDs))
  697. if err := session.Find(&orderDetails); err != nil {
  698. return nil, err
  699. }
  700. return orderDetails, nil
  701. }
  702. // GetHsbySellMyTradeDetails 获取"我的闲置 - 已完成"单据信息
  703. // 输入 accountIDs string 资金账户列表
  704. // 输出 []HsbySellMyDetail "我的闲置 - 已完成"单据信息
  705. // 输出 error error
  706. func GetHsbySellMyTradeDetails(accountIDs string) ([]HsbySellMyDetail, error) {
  707. // 获取市场信息
  708. markets, err := GetMarkets()
  709. if err != nil {
  710. return nil, err
  711. }
  712. engine := db.GetEngine()
  713. marketIDs := "" // 我的闲置-已发布: 二级市场卖方向成交单(包括历史成交单)
  714. // 默认取 TradeMode = 16
  715. for _, v := range markets {
  716. if v.Trademode == 16 {
  717. if len(marketIDs) == 0 {
  718. marketIDs = strconv.Itoa(int(v.Marketid))
  719. } else {
  720. marketIDs += "," + strconv.Itoa(int(v.Marketid))
  721. }
  722. }
  723. }
  724. orders := make([]HsbySellMyDetail, 0)
  725. // 当前成交单
  726. curOrders := make([]HsbySellMyDetail, 0)
  727. if err := engine.Table("TRADE_TRADEDETAIL").
  728. Select(`to_char(TRADE_TRADEDETAIL.TRADEID) ORDERID,
  729. TRADE_TRADEDETAIL.MARKETID, TRADE_TRADEDETAIL.GOODSID, TRADE_TRADEDETAIL.ACCOUNTID, TRADE_TRADEDETAIL.BUYORSELL,
  730. TRADE_TRADEDETAIL.TRADEPRICE PRICE, TRADE_TRADEDETAIL.TRADEQTY QTY, TRADE_TRADEDETAIL.TRADETIME TIME, TRADE_TRADEDETAIL.TRADEAMOUNT AMOUNT,
  731. GOODS.GOODSCODE, GOODS.GOODSNAME, GOODS.DECIMALPLACE, GOODS.AGREEUNIT,
  732. HSBY_GOODSEX.PICURLS,
  733. MARKET.TRADEMODE,
  734. ENUMDICITEM.PARAM2 CURRENCYSIGN,
  735. HSBY_SUPPLIERINFO.VENDORNAME,
  736. 1 ORDERTYPE`).
  737. Join("LEFT", "GOODS", "GOODS.GOODSID = TRADE_TRADEDETAIL.GOODSID").
  738. Join("LEFT", "HSBY_GOODSEX", "HSBY_GOODSEX.GOODSID = GOODS.GOODSID").
  739. Join("LEFT", "ENUMDICITEM", "GOODS.CURRENCYID = ENUMDICITEM.ENUMITEMNAME and ENUMDICITEM.ENUMDICCODE = 'currency'").
  740. Join("LEFT", "MARKET", "MARKET.MARKETID = TRADE_TRADEDETAIL.MARKETID").
  741. Join("LEFT", "HSBY_SUPPLIERINFO", "HSBY_SUPPLIERINFO.VENDORID = HSBY_GOODSEX.VENDORID").
  742. Where(fmt.Sprintf(`TRADE_TRADEDETAIL.BUYORSELL = 1
  743. and TRADE_TRADEDETAIL.ACCOUNTID in (%s)`, accountIDs)).
  744. And(fmt.Sprintf("TRADE_TRADEDETAIL.MARKETID in (%s)", marketIDs)).Find(&curOrders); err != nil {
  745. return nil, err
  746. }
  747. if len(curOrders) > 0 {
  748. orders = append(orders, curOrders...)
  749. }
  750. // 历史成交单
  751. hisOrders := make([]HsbySellMyDetail, 0)
  752. if err := engine.Table("HIS_TRADE_TRADEDETAIL").
  753. Select(`to_char(HIS_TRADE_TRADEDETAIL.TRADEID) ORDERID,
  754. HIS_TRADE_TRADEDETAIL.MARKETID, HIS_TRADE_TRADEDETAIL.GOODSID, HIS_TRADE_TRADEDETAIL.ACCOUNTID, HIS_TRADE_TRADEDETAIL.BUYORSELL,
  755. HIS_TRADE_TRADEDETAIL.TRADEPRICE PRICE, HIS_TRADE_TRADEDETAIL.TRADEQTY QTY, HIS_TRADE_TRADEDETAIL.TRADETIME TIME, HIS_TRADE_TRADEDETAIL.TRADEAMOUNT AMOUNT,
  756. GOODS.GOODSCODE, GOODS.GOODSNAME, GOODS.DECIMALPLACE, GOODS.AGREEUNIT,
  757. HSBY_GOODSEX.PICURLS,
  758. MARKET.TRADEMODE,
  759. ENUMDICITEM.PARAM2 CURRENCYSIGN,
  760. HSBY_SUPPLIERINFO.VENDORNAME,
  761. 1 ORDERTYPE`).
  762. Join("LEFT", "GOODS", "GOODS.GOODSID = HIS_TRADE_TRADEDETAIL.GOODSID").
  763. Join("LEFT", "HSBY_GOODSEX", "HSBY_GOODSEX.GOODSID = GOODS.GOODSID").
  764. Join("LEFT", "ENUMDICITEM", "GOODS.CURRENCYID = ENUMDICITEM.ENUMITEMNAME and ENUMDICITEM.ENUMDICCODE = 'currency'").
  765. Join("LEFT", "MARKET", "MARKET.MARKETID = HIS_TRADE_TRADEDETAIL.MARKETID").
  766. Join("LEFT", "HSBY_SUPPLIERINFO", "HSBY_SUPPLIERINFO.VENDORID = HSBY_GOODSEX.VENDORID").
  767. Where(fmt.Sprintf(`HIS_TRADE_TRADEDETAIL.BUYORSELL = 1 and HIS_TRADE_TRADEDETAIL.ISVALIDDATA = 1
  768. and HIS_TRADE_TRADEDETAIL.ACCOUNTID in (%s)`, accountIDs)).
  769. And(fmt.Sprintf("HIS_TRADE_TRADEDETAIL.MARKETID in (%s)", marketIDs)).Find(&hisOrders); err != nil {
  770. return nil, err
  771. }
  772. if len(hisOrders) > 0 {
  773. orders = append(orders, hisOrders...)
  774. }
  775. return orders, nil
  776. }
  777. // HsbyMyPackage 我的包裹信息
  778. type HsbyMyPackage struct {
  779. Takeorderid string `json:"takeorderid" xorm:"'TAKEORDERID'" binding:"required"` // 提货单号(905+Unix秒时间戳(10位)+xxxxxx)
  780. Accountid int64 `json:"accountid" xorm:"'ACCOUNTID'"` // 账户ID
  781. Goodsid int64 `json:"goodsid" xorm:"'GOODSID'"` // 商品ID
  782. Userid int64 `json:"userid" xorm:"'USERID'"` // 用户ID
  783. Qty float64 `json:"qty" xorm:"'QTY'"` // 提货数量
  784. Reqtime time.Time `json:"reqtime" xorm:"'REQTIME'"` // 更新时间
  785. Recivername string `json:"recivername" xorm:"'RECIVERNAME'"` // 提货人姓名
  786. Cardtypeid int32 `json:"cardtypeid" xorm:"'CARDTYPEID'"` // 提货人证件类型
  787. Cardnum string `json:"cardnum" xorm:"'CARDNUM'"` // 提货人证件号码
  788. Phonenum string `json:"phonenum" xorm:"'PHONENUM'"` // 提货人联系方式
  789. Takemode int32 `json:"takemode" xorm:"'TAKEMODE'"` // 提货方式 - 2:自提 3:配送
  790. Address string `json:"address" xorm:"'ADDRESS'"` // 提货人详细地址
  791. Takeremark string `json:"takeremark" xorm:"'TAKEREMARK'"` // 提货备注
  792. Takeorderstatus int32 `json:"takeorderstatus" xorm:"'TAKEORDERSTATUS'"` // 提货状态 - 1:待发货 2:已发货 3:已收货
  793. Auditer int32 `json:"auditer" xorm:"'AUDITER'"` // 审核人
  794. Audittime time.Time `json:"audittime" xorm:"'AUDITTIME'"` // 审核时间
  795. Checkremark string `json:"checkremark" xorm:"'CHECKREMARK'"` // 审核备注
  796. Tradedate string `json:"tradedate" xorm:"'TRADEDATE'"` // 交易日(yyyyMMdd)
  797. Marketid int32 `json:"marketid" xorm:"'MARKETID'"` // 市场ID
  798. Handlestatus int32 `json:"handlestatus" xorm:"'HANDLESTATUS'"` // 处理状态
  799. Amount float64 `json:"amount" xorm:"'AMOUNT'"` // 提货金额
  800. Averageprice float64 `json:"averageprice" xorm:"'AVERAGEPRICE'"` // 均价
  801. Goodscode string `json:"goodscode" xorm:"'GOODSCODE'" binding:"required"` // 商品代码(内部)
  802. Goodsname string `json:"goodsname" xorm:"'GOODSNAME'" binding:"required"` // 商品名称
  803. Decimalplace int64 `json:"decimalplace" xorm:"'DECIMALPLACE'"` // 报价小数位
  804. Agreeunit float64 `json:"agreeunit" xorm:"'AGREEUNIT'"` // 合约单位
  805. Picurls string `json:"picurls" xorm:"'PICURLS'"` // 介绍图片[多张用逗号分隔]
  806. Currencysign string `json:"currencysign" xorm:"'CURRENCYSIGN'"` // 货币符号
  807. Vendorname string `json:"vendorname" xorm:"'VENDORNAME'"` // 供应商名称
  808. }
  809. // GetHsbyMyPackages 获取我的包裹信息
  810. // 输入 accountIDs string 资金账户列表
  811. // 输入 takeOrderStatus int 提货状态 - 1:待发货 2:已发货 3:已收货
  812. // 输出 []HsbyMyPackage 我的包裹信息
  813. // 输出 error error
  814. func GetHsbyMyPackages(accountIDs string, takeOrderStatus int) ([]HsbyMyPackage, error) {
  815. engine := db.GetEngine()
  816. myPackages := make([]HsbyMyPackage, 0)
  817. session := engine.Table("TRADE_GOODSPICKUP").
  818. Select(`to_char(TRADE_GOODSPICKUP.TAKEORDERID) TAKEORDERID, (DIVISION.PATHNAME || TRADE_GOODSPICKUP.ADDRESS) ADDRESS, TRADE_GOODSPICKUP.*,
  819. (TRADE_GOODSPICKUP.AMOUNT/TRADE_GOODSPICKUP.QTY/GOODS.AGREEUNIT) AVERAGEPRICE,
  820. GOODS.GOODSCODE, GOODS.GOODSNAME, GOODS.DECIMALPLACE, GOODS.AGREEUNIT,
  821. HSBY_GOODSEX.PICURLS,
  822. ENUMDICITEM.PARAM2 CURRENCYSIGN,
  823. HSBY_SUPPLIERINFO.VENDORNAME`).
  824. Join("LEFT", "GOODS", "GOODS.GOODSID = TRADE_GOODSPICKUP.GOODSID").
  825. Join("LEFT", "HSBY_GOODSEX", "HSBY_GOODSEX.GOODSID = GOODS.GOODSID").
  826. Join("LEFT", "ENUMDICITEM", "GOODS.CURRENCYID = ENUMDICITEM.ENUMITEMNAME and ENUMDICITEM.ENUMDICCODE = 'currency'").
  827. Join("LEFT", "HSBY_SUPPLIERINFO", "HSBY_SUPPLIERINFO.VENDORID = HSBY_GOODSEX.VENDORID").
  828. Join("LEFT", "DIVISION", "DIVISION.AUTOID = TRADE_GOODSPICKUP.DISTRICTID").
  829. Where(fmt.Sprintf("TRADE_GOODSPICKUP.ACCOUNTID in (%s)", accountIDs))
  830. if takeOrderStatus > 0 {
  831. session = session.And("TAKEORDERSTATUS = ?", takeOrderStatus)
  832. }
  833. if err := session.Find(&myPackages); err != nil {
  834. return nil, err
  835. }
  836. // FIXME:- 目前暂时在服务端对手机号码进行解密
  837. key, _ := hex.DecodeString(utils.AESSecretKey)
  838. for i := range myPackages {
  839. myPackage := &myPackages[i]
  840. // 解密手机号码
  841. if phonenum, err := hex.DecodeString(myPackage.Phonenum); err == nil { // hex -> []byte
  842. if mobile, err := utils.AESDecrypt(phonenum, key); err == nil {
  843. myPackage.Phonenum = string(mobile)
  844. }
  845. }
  846. }
  847. return myPackages, nil
  848. }
  849. // SetHsbyMyPackagesStatus 设置我的包裹提货状态
  850. func SetHsbyMyPackagesStatus(takeOrderID string, accountID, status int) error {
  851. engine := db.GetEngine()
  852. sql := fmt.Sprintf("update TRADE_GOODSPICKUP set TAKEORDERSTATUS = %d where TAKEORDERID = %s and ACCOUNTID = %d", status, takeOrderID, accountID)
  853. _, err := engine.Exec(sql)
  854. return err
  855. }
  856. // GetHsbyProvincesAndCities 获取省市信息数组
  857. // 参数 provinceID int 省ID,选填
  858. // 返回 []Division 枚举信息数组
  859. // 返回 error error
  860. func GetHsbyProvincesAndCities(provinceID int) ([]Division, error) {
  861. engine := db.GetEngine()
  862. divisions := make([]Division, 0)
  863. session := engine.Where("DIVISIONLEVEL = 'province' or DIVISIONLEVEL = 'city'").
  864. And(`AUTOID in (
  865. (select DESCPROVINCEID AUTOID from HSBY_GOODSEX inner join GOODS on GOODS.GOODSID = HSBY_GOODSEX.GOODSID and GOODS.GOODSSTATUS in (2,3))
  866. union all
  867. (select DESCCITYID AUTOID from HSBY_GOODSEX inner join GOODS on GOODS.GOODSID = HSBY_GOODSEX.GOODSID and GOODS.GOODSSTATUS in (2,3))
  868. )`)
  869. if provinceID > 0 {
  870. session = session.And("AUTOID = ?", provinceID)
  871. }
  872. if err := session.Find(&divisions); err != nil {
  873. return nil, err
  874. }
  875. return divisions, nil
  876. }
  877. // HsbyBuyMyTradeDetail "我的订单 - 已完成"单据信息(一二级市场成交单,包括历史数据)
  878. type HsbyBuyMyTradeDetail struct {
  879. Orderid string `json:"orderid" xorm:"'ORDERID'" binding:"required"` // 单号(成交单号)
  880. Marketid int32 `json:"marketid" xorm:"'MARKETID'" binding:"required"` // 市场ID
  881. Goodsid int32 `json:"goodsid" xorm:"'GOODSID'" binding:"required"` // 商品ID
  882. Accountid int64 `json:"accountid" xorm:"'ACCOUNTID'" binding:"required"` // 账户ID[报价币种]
  883. Buyorsell int32 `json:"buyorsell" xorm:"'BUYORSELL'" binding:"required"` // 买卖 - 0:买 1:卖
  884. Price float64 `json:"price" xorm:"'PRICE'"` // 价格
  885. Qty int64 `json:"qty" xorm:"'QTY'" binding:"required"` // 数量
  886. Time time.Time `json:"time" xorm:"'TIME'" binding:"required"` // 时间
  887. Amount float64 `json:"amount" xorm:"'AMOUNT'"` // 金额 = 价格 * 数量 * 合约单位
  888. Goodscode string `json:"goodscode" xorm:"'GOODSCODE'" binding:"required"` // 商品代码(内部)
  889. Goodsname string `json:"goodsname" xorm:"'GOODSNAME'" binding:"required"` // 商品名称
  890. Decimalplace int64 `json:"decimalplace" xorm:"'DECIMALPLACE'"` // 报价小数位
  891. Agreeunit float64 `json:"agreeunit" xorm:"'AGREEUNIT'"` // 合约单位
  892. Picurls1 string `json:"picurls1" xorm:"'PICURLS1'"` // 预售商品介绍图片[多张用逗号分隔]
  893. Picurls2 string `json:"picurls2" xorm:"'PICURLS2'"` // 挂牌商品介绍图片[多张用逗号分隔]
  894. Trademode uint32 `json:"trademode" xorm:"'TRADEMODE'" binding:"required"` // 交易模式 - 10:做市 13:竞价 15:通道交易 16:挂牌点选 17:仓单贸易 18:期权 19:竞拍-降价式 20:竞拍-竞价式 21:竞拍-大宗式 22:受托竞价
  895. Currencysign string `json:"currencysign" xorm:"'CURRENCYSIGN'"` // 货币符号
  896. Vendorname1 string `json:"vendorname1" xorm:"'VENDORNAME1'"` // 预售商品供应商名称
  897. Vendorname2 string `json:"vendorname2" xorm:"'VENDORNAME2'"` // 挂牌商品供应商名称
  898. }
  899. // GetHsbyBuyMyTradeDetails 获取"我的订单 - 已完成"单据信息(一二级市场成交单,包括历史数据)
  900. // 输入 accountIDs string 资金账户列表
  901. // 输出 []HsbyBuyMyTradeDetail "我的订单 - 已完成"单据信息
  902. // 输出 error error
  903. func GetHsbyBuyMyTradeDetails(accountIDs string) ([]HsbyBuyMyTradeDetail, error) {
  904. // 获取市场信息
  905. markets, err := GetMarkets()
  906. if err != nil {
  907. return nil, err
  908. }
  909. engine := db.GetEngine()
  910. marketIDs := "" // 我的订单包括一二级市场卖方向成交单(包括历史成交单)
  911. // 默认取 TradeMode = 16
  912. for _, v := range markets {
  913. if v.Trademode == 16 || v.Trademode == 71 {
  914. if len(marketIDs) == 0 {
  915. marketIDs = strconv.Itoa(int(v.Marketid))
  916. } else {
  917. marketIDs += "," + strconv.Itoa(int(v.Marketid))
  918. }
  919. }
  920. }
  921. orders := make([]HsbyBuyMyTradeDetail, 0)
  922. // 当前成交单
  923. curOrders := make([]HsbyBuyMyTradeDetail, 0)
  924. if err := engine.Table("TRADE_TRADEDETAIL").
  925. Select(`to_char(TRADE_TRADEDETAIL.TRADEID) ORDERID,
  926. TRADE_TRADEDETAIL.MARKETID, TRADE_TRADEDETAIL.GOODSID, TRADE_TRADEDETAIL.ACCOUNTID, TRADE_TRADEDETAIL.BUYORSELL,
  927. TRADE_TRADEDETAIL.TRADEPRICE PRICE, TRADE_TRADEDETAIL.TRADEQTY QTY, TRADE_TRADEDETAIL.TRADETIME TIME, TRADE_TRADEDETAIL.TRADEAMOUNT AMOUNT,
  928. GOODS.GOODSCODE, GOODS.GOODSNAME, GOODS.DECIMALPLACE, GOODS.AGREEUNIT,
  929. HG1.PICURLS PICURLS1, HG2.PICURLS PICURLS2,
  930. MARKET.TRADEMODE,
  931. ENUMDICITEM.PARAM2 CURRENCYSIGN,
  932. HS1.VENDORNAME VENDORNAME1, HS2.VENDORNAME VENDORNAME2`).
  933. Join("LEFT", "GOODS", "GOODS.GOODSID = TRADE_TRADEDETAIL.GOODSID").
  934. Join("LEFT", "CPTRADE_PRESALEGOODSEX", "CPTRADE_PRESALEGOODSEX.GOODSID = TRADE_TRADEDETAIL.GOODSID").
  935. Join("LEFT", "HSBY_GOODSEX HG1", "HG1.GOODSID = CPTRADE_PRESALEGOODSEX.RELATEDGOODSID").
  936. Join("LEFT", "HSBY_GOODSEX HG2", "HG2.GOODSID = GOODS.GOODSID").
  937. Join("LEFT", "ENUMDICITEM", "GOODS.CURRENCYID = ENUMDICITEM.ENUMITEMNAME and ENUMDICITEM.ENUMDICCODE = 'currency'").
  938. Join("LEFT", "MARKET", "MARKET.MARKETID = TRADE_TRADEDETAIL.MARKETID").
  939. Join("LEFT", "HSBY_SUPPLIERINFO HS1", "HS1.VENDORID = HG1.VENDORID").
  940. Join("LEFT", "HSBY_SUPPLIERINFO HS2", "HS2.VENDORID = HG2.VENDORID").
  941. Where(fmt.Sprintf(`TRADE_TRADEDETAIL.BUYORSELL = 0
  942. and TRADE_TRADEDETAIL.ACCOUNTID in (%s)`, accountIDs)).
  943. And(fmt.Sprintf("TRADE_TRADEDETAIL.MARKETID in (%s)", marketIDs)).Find(&curOrders); err != nil {
  944. return nil, err
  945. }
  946. if len(curOrders) > 0 {
  947. orders = append(orders, curOrders...)
  948. }
  949. // 历史成交单
  950. hisOrders := make([]HsbyBuyMyTradeDetail, 0)
  951. if err := engine.Table("HIS_TRADE_TRADEDETAIL").
  952. Select(`to_char(HIS_TRADE_TRADEDETAIL.TRADEID) ORDERID,
  953. HIS_TRADE_TRADEDETAIL.MARKETID, HIS_TRADE_TRADEDETAIL.GOODSID, HIS_TRADE_TRADEDETAIL.ACCOUNTID, HIS_TRADE_TRADEDETAIL.BUYORSELL,
  954. HIS_TRADE_TRADEDETAIL.TRADEPRICE PRICE, HIS_TRADE_TRADEDETAIL.TRADEQTY QTY, HIS_TRADE_TRADEDETAIL.TRADETIME TIME, HIS_TRADE_TRADEDETAIL.TRADEAMOUNT AMOUNT,
  955. GOODS.GOODSCODE, GOODS.GOODSNAME, GOODS.DECIMALPLACE, GOODS.AGREEUNIT,
  956. HG1.PICURLS PICURLS1, HG2.PICURLS PICURLS2,
  957. MARKET.TRADEMODE,
  958. ENUMDICITEM.PARAM2 CURRENCYSIGN,
  959. HS1.VENDORNAME VENDORNAME1, HS2.VENDORNAME VENDORNAME2`).
  960. Join("LEFT", "GOODS", "GOODS.GOODSID = HIS_TRADE_TRADEDETAIL.GOODSID").
  961. Join("LEFT", "CPTRADE_PRESALEGOODSEX", "CPTRADE_PRESALEGOODSEX.GOODSID = HIS_TRADE_TRADEDETAIL.GOODSID").
  962. Join("LEFT", "HSBY_GOODSEX HG1", "HG1.GOODSID = CPTRADE_PRESALEGOODSEX.RELATEDGOODSID").
  963. Join("LEFT", "HSBY_GOODSEX HG2", "HG2.GOODSID = GOODS.GOODSID").
  964. Join("LEFT", "ENUMDICITEM", "GOODS.CURRENCYID = ENUMDICITEM.ENUMITEMNAME and ENUMDICITEM.ENUMDICCODE = 'currency'").
  965. Join("LEFT", "MARKET", "MARKET.MARKETID = HIS_TRADE_TRADEDETAIL.MARKETID").
  966. Join("LEFT", "HSBY_SUPPLIERINFO HS1", "HS1.VENDORID = HG1.VENDORID").
  967. Join("LEFT", "HSBY_SUPPLIERINFO HS2", "HS2.VENDORID = HG2.VENDORID").
  968. Where(fmt.Sprintf(`HIS_TRADE_TRADEDETAIL.BUYORSELL = 0 and HIS_TRADE_TRADEDETAIL.ISVALIDDATA = 1
  969. and HIS_TRADE_TRADEDETAIL.ACCOUNTID in (%s)`, accountIDs)).
  970. And(fmt.Sprintf("HIS_TRADE_TRADEDETAIL.MARKETID in (%s)", marketIDs)).Find(&hisOrders); err != nil {
  971. return nil, err
  972. }
  973. if len(hisOrders) > 0 {
  974. orders = append(orders, hisOrders...)
  975. }
  976. return orders, nil
  977. }
  978. // HsbyBuyMyPayOrder "我的订单 - 待付款"单据信息(一二级市场成交单和待付款信息)
  979. type HsbyBuyMyPayOrder struct {
  980. Orderid string `json:"orderid" xorm:"'ORDERID'" binding:"required"` // 单号(成交单号)
  981. Marketid int32 `json:"marketid" xorm:"'MARKETID'" binding:"required"` // 市场ID
  982. Goodsid int32 `json:"goodsid" xorm:"'GOODSID'" binding:"required"` // 商品ID
  983. Accountid int64 `json:"accountid" xorm:"'ACCOUNTID'" binding:"required"` // 账户ID[报价币种]
  984. Buyorsell int32 `json:"buyorsell" xorm:"'BUYORSELL'" binding:"required"` // 买卖 - 0:买 1:卖
  985. Goodscode string `json:"goodscode" xorm:"'GOODSCODE'" binding:"required"` // 商品代码(内部)
  986. Goodsname string `json:"goodsname" xorm:"'GOODSNAME'" binding:"required"` // 商品名称
  987. Decimalplace int64 `json:"decimalplace" xorm:"'DECIMALPLACE'"` // 报价小数位
  988. Agreeunit float64 `json:"agreeunit" xorm:"'AGREEUNIT'"` // 合约单位
  989. Picurls1 string `json:"picurls1" xorm:"'PICURLS1'"` // 预售商品介绍图片[多张用逗号分隔]
  990. Picurls2 string `json:"picurls2" xorm:"'PICURLS2'"` // 挂牌商品介绍图片[多张用逗号分隔]
  991. Trademode uint32 `json:"trademode" xorm:"'TRADEMODE'" binding:"required"` // 交易模式 - 10:做市 13:竞价 15:通道交易 16:挂牌点选 17:仓单贸易 18:期权 19:竞拍-降价式 20:竞拍-竞价式 21:竞拍-大宗式 22:受托竞价
  992. Currencysign string `json:"currencysign" xorm:"'CURRENCYSIGN'"` // 货币符号
  993. Vendorname1 string `json:"vendorname1" xorm:"'VENDORNAME1'"` // 预售商品供应商名称
  994. Vendorname2 string `json:"vendorname2" xorm:"'VENDORNAME2'"` // 挂牌商品供应商名称
  995. Tradeamount float64 `json:"tradeamount" xorm:"'TRADEAMOUNT'"` // 成交金额
  996. Tradecharge float64 `json:"tradecharge" xorm:"'TRADECHARGE'"` // 成交手续费(买方)
  997. Payflag int32 `json:"payflag" xorm:"'PAYFLAG'"` // 付款标识 - 1:未支付 2:已支付 3:已过期 4:已撤销 5:结算过期
  998. Createtime time.Time `json:"createtime" xorm:"'CREATETIME'"` // 创建时间
  999. Paylimitedtime time.Time `json:"paylimitedtime" xorm:"'PAYLIMITEDTIME'"` // 支付期限
  1000. Offamount float64 `json:"offamount" xorm:"'OFFAMOUNT'"` // 优惠金额
  1001. Tradeprice float64 `json:"tradeprice" xorm:"'TRADEPRICE'"` // 成交价格
  1002. Tradeqty int64 `json:"tradeqty" xorm:"'TRADEQTY'"` // 成交数量
  1003. Accountname string `json:"accountname" xorm:"'ACCOUNTNAME'"` // 购买方名称
  1004. }
  1005. // GetHsbyBuyMyPayOrders 获取待付款信息
  1006. func GetHsbyBuyMyPayOrders(accountIDs string, sellOrderID, sellAccountID int) ([]HsbyBuyMyPayOrder, error) {
  1007. // 获取市场信息
  1008. markets, err := GetMarkets()
  1009. if err != nil {
  1010. return nil, err
  1011. }
  1012. engine := db.GetEngine()
  1013. marketIDs := "" // 我的订单包括一二级市场卖方向成交单
  1014. // 默认取 TradeMode = 16
  1015. for _, v := range markets {
  1016. if v.Trademode == 16 || v.Trademode == 71 {
  1017. if len(marketIDs) == 0 {
  1018. marketIDs = strconv.Itoa(int(v.Marketid))
  1019. } else {
  1020. marketIDs += "," + strconv.Itoa(int(v.Marketid))
  1021. }
  1022. }
  1023. }
  1024. orders := make([]HsbyBuyMyPayOrder, 0)
  1025. session := engine.Table("TRADE_TRADEDETAIL").
  1026. Select(`to_char(TRADE_TRADEDETAIL.TRADEID) ORDERID,
  1027. TRADE_TRADEDETAIL.MARKETID, TRADE_TRADEDETAIL.GOODSID, TRADE_TRADEDETAIL.ACCOUNTID, TRADE_TRADEDETAIL.BUYORSELL,
  1028. GOODS.GOODSCODE, GOODS.GOODSNAME, GOODS.DECIMALPLACE, GOODS.AGREEUNIT,
  1029. HG1.PICURLS PICURLS1, HG2.PICURLS PICURLS2,
  1030. MARKET.TRADEMODE,
  1031. ENUMDICITEM.PARAM2 CURRENCYSIGN,
  1032. HS1.VENDORNAME VENDORNAME1, HS2.VENDORNAME VENDORNAME2,
  1033. TP.TRADEAMOUNT, TP.TRADECHARGE, TP.PAYFLAG, TP.CREATETIME, TP.PAYLIMITEDTIME, TP.OFFAMOUNT, TP.TRADEPRICE, TP.TRADEQTY,
  1034. U.ACCOUNTNAME`).
  1035. Join("LEFT", "GOODS", "GOODS.GOODSID = TRADE_TRADEDETAIL.GOODSID").
  1036. Join("LEFT", "CPTRADE_PRESALEGOODSEX", "CPTRADE_PRESALEGOODSEX.GOODSID = TRADE_TRADEDETAIL.GOODSID").
  1037. Join("LEFT", "HSBY_GOODSEX HG1", "HG1.GOODSID = CPTRADE_PRESALEGOODSEX.RELATEDGOODSID").
  1038. Join("LEFT", "HSBY_GOODSEX HG2", "HG2.GOODSID = GOODS.GOODSID").
  1039. Join("LEFT", "ENUMDICITEM", "GOODS.CURRENCYID = ENUMDICITEM.ENUMITEMNAME and ENUMDICITEM.ENUMDICCODE = 'currency'").
  1040. Join("LEFT", "MARKET", "MARKET.MARKETID = TRADE_TRADEDETAIL.MARKETID").
  1041. Join("LEFT", "HSBY_SUPPLIERINFO HS1", "HS1.VENDORID = HG1.VENDORID").
  1042. Join("LEFT", "HSBY_SUPPLIERINFO HS2", "HS2.VENDORID = HG2.VENDORID").
  1043. Join("INNER", "TRADE_PAYORDER TP", "TP.TRADEID = TRADE_TRADEDETAIL.TRADEID and TP.PAYFLAG = 1").
  1044. Join("LEFT", "TAACCOUNT TA", "TA.ACCOUNTID = TP.BUYACCOUNTID").
  1045. Join("LEFT", "USERACCOUNT U", "U.USERID = TA.RELATEDUSERID").
  1046. Where(`TRADE_TRADEDETAIL.BUYORSELL = 0`).
  1047. And(fmt.Sprintf("TRADE_TRADEDETAIL.MARKETID in (%s)", marketIDs))
  1048. if len(accountIDs) > 0 {
  1049. session = session.And(fmt.Sprintf("TRADE_TRADEDETAIL.ACCOUNTID in (%s)", accountIDs))
  1050. }
  1051. if sellOrderID != 0 {
  1052. session = session.And("TP.SELLORDERID = ?", sellOrderID)
  1053. }
  1054. if sellAccountID != 0 {
  1055. session = session.And("TP.SELLACCOUNTID = ?", sellAccountID)
  1056. }
  1057. if err := session.Find(&orders); err != nil {
  1058. return nil, err
  1059. }
  1060. return orders, nil
  1061. }