hsby.go 87 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635
  1. package models
  2. import (
  3. "encoding/hex"
  4. "fmt"
  5. "math"
  6. "mtp2_if/db"
  7. "mtp2_if/utils"
  8. "strconv"
  9. "strings"
  10. "time"
  11. )
  12. // Hsbygoodsex 商品扩展表
  13. type Hsbygoodsex struct {
  14. Goodsid int32 `json:"goodsid" xorm:"'GOODSID'" binding:"required"` // 商 品ID
  15. Hotindex int32 `json:"hotindex" xorm:"'HOTINDEX'"` // 景点热度
  16. Descprovinceid int64 `json:"descprovinceid" xorm:"'DESCPROVINCEID'"` // 目的地(省)
  17. Desccityid int64 `json:"desccityid" xorm:"'DESCCITYID'"` // 目的地(市)
  18. Vendorid int32 `json:"vendorid" xorm:"'VENDORID'"` // 供应商ID
  19. Goodsdesc string `json:"goodsdesc" xorm:"'GOODSDESC'"` // 商品详情
  20. Createtime time.Time `json:"createtime" xorm:"'CREATETIME'"` // 创建时间
  21. Creatorid int64 `json:"creatorid" xorm:"'CREATORID'"` // 创建人
  22. Modifierid int64 `json:"modifierid" xorm:"'MODIFIERID'"` // 修改人
  23. Modifytime time.Time `json:"modifytime" xorm:"'MODIFYTIME'"` // 修改时间
  24. Memberratio float64 `json:"memberratio" xorm:"'MEMBERRATIO'"` // 会员货款比例 [71]
  25. Videourls string `json:"videourls" xorm:"'VIDEOURLS'"` // 介绍视频[多张用逗号分隔]
  26. Picurls string `json:"picurls" xorm:"'PICURLS'"` // 介绍图片[多张用逗号分隔]
  27. Goodsprice float64 `json:"goodsprice" xorm:"'GOODSPRICE'"` // 商品价格
  28. Categoryid int32 `json:"categoryid" xorm:"'CATEGORYID'"` // 类别ID(WRCATEGORY)
  29. }
  30. // TableName is HSBY_GOODSEX
  31. func (Hsbygoodsex) TableName() string {
  32. return "HSBY_GOODSEX"
  33. }
  34. // Hsbysupplierinfo 供应商表
  35. type Hsbysupplierinfo struct {
  36. Vendorid int32 `json:"vendorid" xorm:"'VENDORID'" binding:"required"` // 供应商ID(SEQ_HSBY_SUPPLIERINFO)
  37. Vendorname string `json:"vendorname" xorm:"'VENDORNAME'"` // 名称描述客服电话名称
  38. Vendordesc string `json:"vendordesc" xorm:"'VENDORDESC'"` // 描述
  39. Vendorphone string `json:"vendorphone" xorm:"'VENDORPHONE'"` // 客服电话
  40. Vendorattr string `json:"vendorattr" xorm:"'VENDORATTR'"` // 附件(多张,逗号分隔)
  41. Createtime time.Time `json:"createtime" xorm:"'CREATETIME'"` // 创建时间
  42. Creatorid int64 `json:"creatorid" xorm:"'CREATORID'"` // 创建人
  43. Modifierid int64 `json:"modifierid" xorm:"'MODIFIERID'"` // 修改人
  44. Modifytime time.Time `json:"modifytime" xorm:"'MODIFYTIME'"` // 修改时间
  45. }
  46. // TableName is HSBY_SUPPLIERINFO
  47. func (Hsbysupplierinfo) TableName() string {
  48. return "HSBY_SUPPLIERINFO"
  49. }
  50. // Tradepayorder 交易待付款表
  51. type Tradepayorder struct {
  52. Tradeid int64 `json:"tradeid" xorm:"'TRADEID'" binding:"required"` // 成交单号(101+Unix秒时间戳(10位)+2位(MarketServiceID)+xxxx)
  53. Tradedate string `json:"tradedate" xorm:"'TRADEDATE'"` // 交易日(yyyyMMdd)
  54. Marketid int32 `json:"marketid" xorm:"'MARKETID'"` // 市场ID
  55. Goodsid int32 `json:"goodsid" xorm:"'GOODSID'"` // 商品ID
  56. Buyorderid int64 `json:"buyorderid" xorm:"'BUYORDERID'"` // 买方委托单号
  57. Buyaccountid int64 `json:"buyaccountid" xorm:"'BUYACCOUNTID'"` // 买方账号ID[报价币种]
  58. Sellorderid int64 `json:"sellorderid" xorm:"'SELLORDERID'"` // 卖方委托单号
  59. Sellaccountid int64 `json:"sellaccountid" xorm:"'SELLACCOUNTID'"` // 卖方账号ID[报价币种]
  60. Tradeamount float64 `json:"tradeamount" xorm:"'TRADEAMOUNT'"` // 成交金额
  61. Tradecharge float64 `json:"tradecharge" xorm:"'TRADECHARGE'"` // 成交手续费(买方)
  62. Payflag int32 `json:"payflag" xorm:"'PAYFLAG'"` // 付款标识 - 1:未支付 2:已支付 3:已过期 4:已撤销 5:结算过期 6:预售终止
  63. Createtime time.Time `json:"createtime" xorm:"'CREATETIME'"` // 创建时间
  64. Paytime time.Time `json:"paytime" xorm:"'PAYTIME'"` // 付款时间
  65. Paylimitedtime time.Time `json:"paylimitedtime" xorm:"'PAYLIMITEDTIME'"` // 支付期限
  66. Offamount float64 `json:"offamount" xorm:"'OFFAMOUNT'"` // 优惠金额
  67. Tradeprice float64 `json:"tradeprice" xorm:"'TRADEPRICE'"` // 成交价格
  68. Tradeqty int64 `json:"tradeqty" xorm:"'TRADEQTY'"` // 成交数量
  69. Payamount float64 `json:"payamount" xorm:"'PAYAMOUNT'"` // 支付金额 = TradeAmount + TradeCharge - OffAmount
  70. }
  71. // TableName is TRADE_PAYORDER
  72. func (Tradepayorder) TableName() string {
  73. return "TRADE_PAYORDER"
  74. }
  75. // Coupontype 优惠券类型表
  76. type Coupontype struct {
  77. Coupontypeid int32 `json:"coupontypeid" xorm:"'COUPONTYPEID'" binding:"required"` // 优惠券类型ID - SEQ_COUPONTYPE
  78. Areauserid int64 `json:"areauserid" xorm:"'AREAUSERID'"` // 所属机构
  79. Couponname string `json:"couponname" xorm:"'COUPONNAME'"` // 优惠券名称
  80. Couponcategroy int32 `json:"couponcategroy" xorm:"'COUPONCATEGROY'"` // 种类 - 1:现金券 2:折扣券 3:折扣券(单张)
  81. Conditionvalue float64 `json:"conditionvalue" xorm:"'CONDITIONVALUE'"` // 条件阈值(可为0)
  82. Couponvalue float64 `json:"couponvalue" xorm:"'COUPONVALUE'"` // 面值[1:现金券 - 抵扣值 2:折扣券-折扣值]
  83. Limitedflag int32 `json:"limitedflag" xorm:"'LIMITEDFLAG'"` // 是否指定商品 - 0:不限 1:限制
  84. Limitedgoodsids string `json:"limitedgoodsids" xorm:"'LIMITEDGOODSIDS'"` // 指定商品IDs[逗号分隔,前后加逗号]
  85. Isgeneral int32 `json:"isgeneral" xorm:"'ISGENERAL'"` // 是否通用券 - 0:否 1:是
  86. Userscope string `json:"userscope" xorm:"'USERSCOPE'"` // 卖家范围(用户ID,逗号分隔,前后加逗号) [IsGeneral =0时使用]
  87. Qty int64 `json:"qty" xorm:"'QTY'"` // 数量
  88. Curqty int64 `json:"curqty" xorm:"'CURQTY'"` // 剩余数量
  89. Createtime time.Time `json:"createtime" xorm:"'CREATETIME'"` // 创建时间
  90. Creatorid int64 `json:"creatorid" xorm:"'CREATORID'"` // 创建人
  91. Modifierid int64 `json:"modifierid" xorm:"'MODIFIERID'"` // 修改人
  92. Modifytime time.Time `json:"modifytime" xorm:"'MODIFYTIME'"` // 修改时间
  93. Usedqty int64 `json:"usedqty" xorm:"'USEDQTY'"` // 使用量
  94. Isvalid int32 `json:"isvalid" xorm:"'ISVALID'"` // 是否有效 - 0:无效 1:有效
  95. Expiredqty int64 `json:"expiredqty" xorm:"'EXPIREDQTY'"` // 失效量
  96. }
  97. // TableName is COUPONTYPE
  98. func (Coupontype) TableName() string {
  99. return "COUPONTYPE"
  100. }
  101. // Couponposition 优惠券头寸表 - 导历史(期末为0的清除)
  102. type Couponposition struct {
  103. Accountid int64 `json:"accountid" xorm:"'ACCOUNTID'" binding:"required"` // 资金账户ID
  104. Coupontypeid int32 `json:"coupontypeid" xorm:"'COUPONTYPEID'" binding:"required"` // 优惠券类型ID
  105. Userid int64 `json:"userid" xorm:"'USERID'"` // 用户ID
  106. Oriqty int64 `json:"oriqty" xorm:"'ORIQTY'"` // 期初数量
  107. Orifreezeqty int64 `json:"orifreezeqty" xorm:"'ORIFREEZEQTY'"` // 期初冻结数量
  108. Curqty int64 `json:"curqty" xorm:"'CURQTY'"` // 期末数量
  109. Curfreezeqty int64 `json:"curfreezeqty" xorm:"'CURFREEZEQTY'"` // 期末冻结数量
  110. Todayincrease int64 `json:"todayincrease" xorm:"'TODAYINCREASE'"` // 今日增加
  111. Todaydecrease int64 `json:"todaydecrease" xorm:"'TODAYDECREASE'"` // 今日减少
  112. }
  113. // TableName is COUPONPOSITION
  114. func (Couponposition) TableName() string {
  115. return "COUPONPOSITION"
  116. }
  117. // HsbyTopGoods 热卖商品(二级市场挂牌点选)
  118. type HsbyTopGoods struct {
  119. Goodsid int64 `json:"goodsid" xorm:"'GOODSID'" binding:"required"` // 商品ID(自增ID SEQ_GOODS)
  120. Goodscode string `json:"goodscode" xorm:"'GOODSCODE'" binding:"required"` // 商品代码(内部)
  121. Goodsname string `json:"goodsname" xorm:"'GOODSNAME'" binding:"required"` // 商品名称
  122. Decimalplace int64 `json:"decimalplace" xorm:"'DECIMALPLACE'"` // 报价小数位
  123. Marketid int64 `json:"marketid" xorm:"'MARKETID'" binding:"required"` // 所属市场ID
  124. Quoteminunit int64 `json:"quoteminunit" xorm:"'QUOTEMINUNIT'"` // 行情最小变动单位 [整数,报价小数位一起使用]
  125. Agreeunit float64 `json:"agreeunit" xorm:"'AGREEUNIT'"` // 合约单位
  126. Hotindex int32 `json:"hotindex" xorm:"'HOTINDEX'"` // 景点热度
  127. Videourls string `json:"videourls" xorm:"'VIDEOURLS'"` // 介绍视频[多张用逗号分隔]
  128. Picurls string `json:"picurls" xorm:"'PICURLS'"` // 介绍图片[多张用逗号分隔]
  129. Currency string `json:"currency" xorm:"'CURRENCY'"` // 货币
  130. Currencysign string `json:"currencysign" xorm:"'CURRENCYSIGN'"` // 货币符号
  131. Trademode int32 `json:"trademode" xorm:"'TRADEMODE'" binding:"required"` // 交易模式 - 10:做市 13:竞价 15:通道交易 16:挂牌点选 17:仓单贸易 18:期权 19:竞拍-降价式 20:竞拍-竞价式 21:竞拍-大宗式 22:受托竞价
  132. Last float64 `json:"last" xorm:"-"` // 现价
  133. }
  134. // GetHsbyTopGoodses 获取热门商品列表
  135. // 参数 marketIDs string 所属市场ID列表, 格式 1,2,3
  136. // 参数 descProvinceID int 目标省ID
  137. // 参数 descCityID int 目标城市ID
  138. // 返回 []TopGoods 热门商品列表
  139. // 返回 error error
  140. func GetHsbyTopGoodses(marketIDs string, descProvinceID, descCityID int) ([]HsbyTopGoods, error) {
  141. // 热门商品为二级市场(挂牌点选)下的商品信息
  142. engine := db.GetEngine()
  143. topGoodses := make([]HsbyTopGoods, 0)
  144. // 获取挂牌商品信息,以及扩展表信息
  145. // 与商城不一样的是,挂牌点选以商品表作为主表,同一商品可有多个委托单
  146. session := engine.Table("GOODS").
  147. Select(`GOODS.*,
  148. HSBY_GOODSEX.HOTINDEX, HSBY_GOODSEX.VIDEOURLS, HSBY_GOODSEX.PICURLS,
  149. ENUMDICITEM.ENUMDICNAME CURRENCY, ENUMDICITEM.PARAM2 CURRENCYSIGN,
  150. MARKET.TRADEMODE`).
  151. Join("LEFT", "HSBY_GOODSEX", "HSBY_GOODSEX.GOODSID = GOODS.GOODSID").
  152. Join("LEFT", "ENUMDICITEM", "ENUMDICITEM.ENUMITEMNAME = GOODS.CURRENCYID and ENUMDICITEM.ENUMDICCODE = 'currency'").
  153. Join("LEFT", "MARKET", "MARKET.MARKETID = GOODS.MARKETID").
  154. Where(fmt.Sprintf("GOODS.GOODSSTATUS = 3 and GOODS.MARKETID in (%s)", marketIDs)) // 二级市场只获取 3:上市 状态的商品
  155. if descProvinceID > 0 {
  156. session = session.And("HSBY_GOODSEX.DESCPROVINCEID = ?", descProvinceID)
  157. }
  158. if descCityID > 0 {
  159. session = session.And("HSBY_GOODSEX.DESCCITYID = ?", descCityID)
  160. }
  161. if err := session.Find(&topGoodses); err != nil {
  162. return nil, err
  163. }
  164. if len(topGoodses) == 0 {
  165. // 无数据
  166. return topGoodses, nil
  167. }
  168. // 获取商品ID列表
  169. goodsCodes := ""
  170. for _, v := range topGoodses {
  171. if len(goodsCodes) == 0 {
  172. goodsCodes = v.Goodscode
  173. } else {
  174. goodsCodes += "," + v.Goodscode
  175. }
  176. }
  177. // 获取商品现价
  178. quoteDays, err := GetQuoteDays(goodsCodes)
  179. if err != nil {
  180. return nil, err
  181. }
  182. for i, g := range topGoodses {
  183. topGoods := &topGoodses[i]
  184. // FIXME: - 这里应该使用 Duck Typing,后期再处理
  185. // 计算最小变动单位
  186. // lotSize := float64(topGoods.Quoteminunit) * math.Pow(0.1, float64(topGoods.Decimalplace))
  187. // topGoods.LotSize, _ = strconv.ParseFloat(utils.FormatFloat(lotSize, int(topGoods.Decimalplace)), 64)
  188. for _, q := range quoteDays {
  189. if g.Goodscode == q.Goodscode {
  190. if q.Last != 0 {
  191. topGoods.Last = utils.IntToFloat64(int(q.Last), int(g.Decimalplace))
  192. }
  193. // 没有现价则尝试用昨结
  194. if topGoods.Last == 0 && q.Presettle != 0 {
  195. topGoods.Last = utils.IntToFloat64(int(q.Presettle), int(g.Decimalplace))
  196. }
  197. continue
  198. }
  199. }
  200. }
  201. return topGoodses, nil
  202. }
  203. // HsbyListingGoodsDetail 二级市场(挂牌点选)商品信息详情
  204. type HsbyListingGoodsDetail struct {
  205. Goodsid int64 `json:"goodsid" xorm:"'GOODSID'" binding:"required"` // 商品ID(自增ID SEQ_GOODS)
  206. Goodscode string `json:"goodscode" xorm:"'GOODSCODE'" binding:"required"` // 商品代码(内部)
  207. Goodsname string `json:"goodsname" xorm:"'GOODSNAME'" binding:"required"` // 商品名称
  208. Decimalplace int64 `json:"decimalplace" xorm:"'DECIMALPLACE'"` // 报价小数位
  209. Marketid int64 `json:"marketid" xorm:"'MARKETID'" binding:"required"` // 所属市场ID
  210. Quoteminunit int64 `json:"quoteminunit" xorm:"'QUOTEMINUNIT'"` // 行情最小变动单位 [整数,报价小数位一起使用]
  211. Agreeunit float64 `json:"agreeunit" xorm:"'AGREEUNIT'"` // 合约单位
  212. Hotindex int32 `json:"hotindex" xorm:"'HOTINDEX'"` // 景点热度
  213. Videourls string `json:"videourls" xorm:"'VIDEOURLS'"` // 介绍视频[多张用逗号分隔]
  214. Picurls string `json:"picurls" xorm:"'PICURLS'"` // 介绍图片[多张用逗号分隔]
  215. Descprovinceid int64 `json:"descprovinceid" xorm:"'DESCPROVINCEID'"` // 目的地(省)ID
  216. Desccityid int64 `json:"desccityid" xorm:"'DESCCITYID'"` // 目的地(市)ID
  217. Goodsdesc string `json:"goodsdesc" xorm:"'GOODSDESC'"` // 商品详情
  218. Currency string `json:"currency" xorm:"'CURRENCY'"` // 货币
  219. Currencysign string `json:"currencysign" xorm:"'CURRENCYSIGN'"` // 货币符号
  220. Vendorname string `json:"vendorname" xorm:"'VENDORNAME'"` // 供应商名称
  221. Vendorphone string `json:"vendorphone" xorm:"'VENDORPHONE'"` // 供应商客服电话
  222. Vendorattr string `json:"vendorattr" xorm:"'VENDORATTR'"` // 供应商附件(多张,逗号分隔)
  223. Trademode int32 `json:"trademode" xorm:"'TRADEMODE'" binding:"required"` // 交易模式 - 10:做市 13:竞价 15:通道交易 16:挂牌点选 17:仓单贸易 18:期权 19:竞拍-降价式 20:竞拍-竞价式 21:竞拍-大宗式 22:受托竞价
  224. Last float64 `json:"last" xorm:"-"` // 现价
  225. Limitup float64 `json:"limitup" xorm:"-"` // 涨停价
  226. Limitdown float64 `json:"limitdown" xorm:"-"` // 跌停价
  227. StepValue float64 `json:"stepvalue" xorm:"-"` // 价格最小变动单位
  228. LotSize int `json:"lotsize" xorm:"-"` // 手数最小变动单位
  229. }
  230. // GetHsbyListingGoodsDetail 获取二级市场(挂牌点选)商品信息详情
  231. // 参数 goodsID int 目标商品ID
  232. // 返回 *HsbyListingGoodsDetail 二级市场(挂牌点选)商品信息详情,查询无结果返回nil
  233. // 返回 error error
  234. func GetHsbyListingGoodsDetail(goodsID int) (*HsbyListingGoodsDetail, error) {
  235. engine := db.GetEngine()
  236. details := make([]HsbyListingGoodsDetail, 0)
  237. // 获取挂牌商品信息,以及扩展表信息
  238. // FIXME: - 这里使用Get方法,会造成SQL语句的嵌套出错,后期再研究
  239. session := engine.Table("GOODS").
  240. Select(`GOODS.GOODSID, GOODS.GOODSCODE, GOODS.GOODSNAME, GOODS.DECIMALPLACE, GOODS.MARKETID, GOODS.QUOTEMINUNIT, GOODS.AGREEUNIT,
  241. HSBY_GOODSEX.HOTINDEX, HSBY_GOODSEX.VIDEOURLS, HSBY_GOODSEX.PICURLS, HSBY_GOODSEX.DESCPROVINCEID, HSBY_GOODSEX.Desccityid, HSBY_GOODSEX.Goodsdesc,
  242. ENUMDICITEM.ENUMDICNAME CURRENCY, ENUMDICITEM.PARAM2 CURRENCYSIGN,
  243. HSBY_SUPPLIERINFO.VENDORNAME, HSBY_SUPPLIERINFO.VENDORPHONE, HSBY_SUPPLIERINFO.VENDORATTR,
  244. MARKET.TRADEMODE`).
  245. Join("LEFT", "HSBY_GOODSEX", "HSBY_GOODSEX.GOODSID = GOODS.GOODSID").
  246. Join("LEFT", "ENUMDICITEM", "ENUMDICITEM.ENUMITEMNAME = GOODS.CURRENCYID and ENUMDICITEM.ENUMDICCODE = 'currency'").
  247. Join("LEFT", "HSBY_SUPPLIERINFO", "HSBY_SUPPLIERINFO.VENDORID = HSBY_GOODSEX.VENDORID").
  248. Join("LEFT", "MARKET", "MARKET.MARKETID = GOODS.MARKETID").
  249. Where("GOODS.GOODSID = ?", goodsID)
  250. if err := session.Find(&details); err != nil {
  251. return nil, err
  252. }
  253. // 无目标商品
  254. if len(details) == 0 {
  255. return nil, nil
  256. }
  257. hsbyListingGoodsDetail := details[0]
  258. // 获取商品现价和涨跌停价
  259. quoteDays, err := GetQuoteDays(hsbyListingGoodsDetail.Goodscode)
  260. if err != nil {
  261. return nil, err
  262. }
  263. if len(quoteDays) > 0 {
  264. if quoteDays[0].Last != 0 {
  265. hsbyListingGoodsDetail.Last = utils.IntToFloat64(int(quoteDays[0].Last), int(hsbyListingGoodsDetail.Decimalplace))
  266. }
  267. // 没有现价则尝试用昨结
  268. if hsbyListingGoodsDetail.Last == 0 && quoteDays[0].Presettle != 0 {
  269. hsbyListingGoodsDetail.Last = utils.IntToFloat64(int(quoteDays[0].Presettle), int(hsbyListingGoodsDetail.Decimalplace))
  270. }
  271. if quoteDays[0].Limitup != 0 {
  272. hsbyListingGoodsDetail.Limitup = utils.IntToFloat64(int(quoteDays[0].Limitup), int(hsbyListingGoodsDetail.Decimalplace))
  273. }
  274. if quoteDays[0].Limitdown != 0 {
  275. hsbyListingGoodsDetail.Limitdown = utils.IntToFloat64(int(quoteDays[0].Limitdown), int(hsbyListingGoodsDetail.Decimalplace))
  276. }
  277. }
  278. // 计算价格最小变动单位
  279. stepValue := float64(hsbyListingGoodsDetail.Quoteminunit) * math.Pow(0.1, float64(hsbyListingGoodsDetail.Decimalplace))
  280. hsbyListingGoodsDetail.StepValue, _ = strconv.ParseFloat(utils.FormatFloat(stepValue, int(hsbyListingGoodsDetail.Decimalplace)), 64)
  281. // 计算手数最小变动单位
  282. hsbyListingGoodsDetail.LotSize = 1
  283. return &hsbyListingGoodsDetail, nil
  284. }
  285. // HsbyGoodsOrderDetail 二级市场挂牌商品当前可摘委托单信息
  286. type HsbyGoodsOrderDetail struct {
  287. Orderid string `json:"orderid" xorm:"'ORDERID'" binding:"required"` // 委托单号(100+Unix秒时间戳(10位)+2位(MarketServiceID)+xxxx)
  288. Buyorsell int32 `json:"buyorsell" xorm:"'BUYORSELL'" binding:"required"` // 买卖 - 0:买 1:卖
  289. Ordertime time.Time `json:"ordertime" xorm:"'ORDERTIME'" binding:"required"` // 委托时间
  290. Orderprice float64 `json:"orderprice" xorm:"'ORDERPRICE'"` // 委托价格
  291. Enableqty int64 `json:"enableqty" xorm:"ENABLEQTY"` // 可用数量
  292. Customername string `json:"customername" xorm:"'CUSTOMERNAME'"` // 客户名称(企业名称),已脱敏
  293. Currencysign string `json:"currencysign" xorm:"-"` // 货币符号
  294. Goodunit string `json:"goodunit" xorm:"-"` // 报价单位
  295. }
  296. // GetHsbyGoodsOrderDetails 获取二级市场(挂牌点选)商品对应的挂牌委托单信息
  297. // 输入 goodsID int 商品ID
  298. // 输入 buyOrSell int 挂牌委托单方向(对手单方向),0:买 1:卖
  299. // 输入 price float64 参考价格。买方向委托单则价格小于等于(站在摘牌人的角度);卖方向委托单则价格大于等于
  300. // 输出 []HsbyGoodsOrderDetail 商品对应的挂牌委托单信息
  301. // 输出 error error
  302. func GetHsbyGoodsOrderDetails(goodsID, buyOrSell int, price float64, accountIDs string) ([]HsbyGoodsOrderDetail, error) {
  303. engine := db.GetEngine()
  304. // 获取与目标商品相关的挂牌委托单信息(ListingSelectType = 1 or 3; OrderStatus =3 or 7)
  305. hsbyGoodsOrderDetails := make([]HsbyGoodsOrderDetail, 0)
  306. session := engine.Table("TRADE_ORDERDETAIL").
  307. 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,
  308. substr(USERINFO.CUSTOMERNAME,0,1)||'****' as CUSTOMERNAME`).
  309. Join("LEFT", "TAACCOUNT", "TAACCOUNT.ACCOUNTID = TRADE_ORDERDETAIL.ACCOUNTID").
  310. Join("LEFT", "USERINFO", "USERINFO.USERID = TAACCOUNT.RELATEDUSERID").
  311. Where("(TRADE_ORDERDETAIL.LISTINGSELECTTYPE = 1 or TRADE_ORDERDETAIL.LISTINGSELECTTYPE = 3) and (TRADE_ORDERDETAIL.ORDERSTATUS = 3 or TRADE_ORDERDETAIL.ORDERSTATUS = 7)").
  312. And("TRADE_ORDERDETAIL.GOODSID = ?", goodsID).
  313. And("TRADE_ORDERDETAIL.BUYORSELL = ?", buyOrSell)
  314. if len(accountIDs) > 0 {
  315. session = session.And(fmt.Sprintf("TRADE_ORDERDETAIL.ACCOUNTID not in (%s)", accountIDs))
  316. }
  317. if price > 0 {
  318. if buyOrSell == 0 {
  319. // 对手单买方向委托单则价格大于等于(站在摘牌人的角度,摘牌方面是卖,我的闲置下单)
  320. session = session.And("TRADE_ORDERDETAIL.ORDERPRICE >= ?", price)
  321. } else {
  322. // 对手单卖方向委托单则价格小于等于(站在摘牌人的角度,摘牌方面是买,热门商品下单)
  323. session = session.And("TRADE_ORDERDETAIL.ORDERPRICE <= ?", price)
  324. }
  325. }
  326. if err := session.Find(&hsbyGoodsOrderDetails); err != nil {
  327. return nil, err
  328. }
  329. // 获取商品货币符号和报价单位
  330. goods, err := GetGoods(goodsID)
  331. if err != nil {
  332. return nil, err
  333. }
  334. currencyEnums, err := GetEnumDicItem("currency", int(goods.Currencyid))
  335. if err != nil {
  336. return nil, err
  337. }
  338. currencysign := ""
  339. if len(currencyEnums) > 0 {
  340. currencysign = currencyEnums[0].Param2
  341. }
  342. goodsUnitEnums, err := GetEnumDicItem("goodsunit", int(goods.Goodunitid))
  343. if err != nil {
  344. return nil, err
  345. }
  346. goodsUnit := ""
  347. if len(goodsUnitEnums) > 0 {
  348. goodsUnit = goodsUnitEnums[0].Enumdicname
  349. }
  350. for i := range hsbyGoodsOrderDetails {
  351. orderDetail := &hsbyGoodsOrderDetails[i]
  352. // 设置货币符号和报价单位
  353. orderDetail.Currencysign = currencysign
  354. orderDetail.Goodunit = goodsUnit
  355. }
  356. return hsbyGoodsOrderDetails, nil
  357. }
  358. // HybsMyBuyOrderDetail 我的订单(一二级市场的买方向委托单)
  359. type HybsMyBuyOrderDetail struct {
  360. Orderid string `json:"orderid" xorm:"'ORDERID'" binding:"required"` // 委托单号(100+Unix秒时间戳(10位)+2位(MarketServiceID)+xxxx)
  361. Marketid int32 `json:"marketid" xorm:"'MARKETID'" binding:"required"` // 市场ID
  362. Goodsid int32 `json:"goodsid" xorm:"'GOODSID'" binding:"required"` // 商品ID
  363. Accountid int64 `json:"accountid" xorm:"'ACCOUNTID'" binding:"required"` // 账户ID[报价币种]
  364. Buyorsell int32 `json:"buyorsell" xorm:"'BUYORSELL'" binding:"required"` // 买卖 - 0:买 1:卖
  365. Orderprice float64 `json:"orderprice" xorm:"'ORDERPRICE'"` // 委托价格
  366. Orderqty int64 `json:"orderqty" xorm:"'ORDERQTY'" binding:"required"` // 委托数量
  367. Tradeqty int64 `json:"tradeqty" xorm:"'TRADEQTY'"` // 成交数量
  368. Cancelqty int64 `json:"cancelqty" xorm:"'CANCELQTY'"` // 撤单数量
  369. Ordertime time.Time `json:"ordertime" xorm:"'ORDERTIME'" binding:"required"` // 委托时间
  370. Listingselecttype int32 `json:"listingselecttype" xorm:"'LISTINGSELECTTYPE'"` // 挂牌点选类型 - 1:挂牌 2:摘牌 3:先摘后挂
  371. Orderstatus int32 `json:"orderstatus" xorm:"'ORDERSTATUS'"` // 委托状态 - 1: 委托请求 2:待冻结 3:委托成功 4: 委托失败 5:配对成功 6: 已撤销 7:部分成交 8:已成交 9:部成部撤 10:成交失败 11:已拒绝 12:经过摘牌(先摘后挂专用-先摘后挂已摘过) 13:冻结成功(通道交易专用) 14:通道已撤 15:通道部成部撤 16:成交失败违约(荷兰式竞拍专用)
  372. Goodscode string `json:"goodscode" xorm:"'GOODSCODE'" binding:"required"` // 商品代码(内部)
  373. Goodsname string `json:"goodsname" xorm:"'GOODSNAME'" binding:"required"` // 商品名称
  374. Decimalplace int64 `json:"decimalplace" xorm:"'DECIMALPLACE'"` // 报价小数位
  375. Agreeunit float64 `json:"agreeunit" xorm:"'AGREEUNIT'"` // 合约单位
  376. Picurls1 string `json:"picurls1" xorm:"'PICURLS1'"` // 预售商品介绍图片[多张用逗号分隔]
  377. Picurls2 string `json:"picurls2" xorm:"'PICURLS2'"` // 挂牌商品介绍图片[多张用逗号分隔]
  378. Trademode uint32 `json:"trademode" xorm:"'TRADEMODE'" binding:"required"` // 交易模式 - 10:做市 13:竞价 15:通道交易 16:挂牌点选 17:仓单贸易 18:期权 19:竞拍-降价式 20:竞拍-竞价式 21:竞拍-大宗式 22:受托竞价
  379. Currencysign string `json:"currencysign" xorm:"'CURRENCYSIGN'"` // 货币符号
  380. Vendorname1 string `json:"vendorname1" xorm:"'VENDORNAME1'"` // 预售商品供应商名称
  381. Vendorname2 string `json:"vendorname2" xorm:"'VENDORNAME2'"` // 挂牌商品供应商名称
  382. MyBuyStatus int `json:"mybuystatus" xorm:"-"` // "我的订单"显示状态- 1:抢购中 2:求购中 3:已完成 4:已撤消 5:委托失败
  383. Orderamount float64 `json:"orderamount" xorm:"-"` // 委托金额
  384. }
  385. // GetHsbyBuyMyOrderDetails 获取“我的订单”数据(包括一二级市场)
  386. // 输入 accountIDs string 资金账户列表
  387. // 输入 myBuyStatus int "我的订单"状态, 1:抢购中 2:求购中 3:已完成
  388. // 输出 []HybsMyBuyOrderDetail “我的订单”数据
  389. // 输出 error error
  390. func GetHsbyBuyMyOrderDetails(accountIDs string, myBuyStatus int) ([]HybsMyBuyOrderDetail, error) {
  391. // 获取市场信息
  392. markets, err := GetMarkets()
  393. if err != nil {
  394. return nil, err
  395. }
  396. engine := db.GetEngine()
  397. // 委托状态
  398. orderStatus := "0" // 单据状态,为0的时候查询全部
  399. marketIDs := "" // 我的订单包括一二级市场的单据
  400. switch myBuyStatus {
  401. case 1: // 抢购中 (一级市场)
  402. // 获取市场ID
  403. for _, v := range markets {
  404. if v.Trademode == 71 { // 预售挂牌
  405. if len(marketIDs) == 0 {
  406. marketIDs = strconv.Itoa(int(v.Marketid))
  407. } else {
  408. marketIDs += "," + strconv.Itoa(int(v.Marketid))
  409. }
  410. }
  411. }
  412. orderStatus = "3,7"
  413. case 2: // 求购中(二级市场)
  414. // 获取市场ID
  415. for _, v := range markets {
  416. if v.Trademode == 16 { // 挂牌点选
  417. if len(marketIDs) == 0 {
  418. marketIDs = strconv.Itoa(int(v.Marketid))
  419. } else {
  420. marketIDs += "," + strconv.Itoa(int(v.Marketid))
  421. }
  422. }
  423. }
  424. orderStatus = "3,7"
  425. case 3: // 已完成
  426. orderStatus = "8,9"
  427. }
  428. // 默认取 TradeMode = 16 or 71 的市场
  429. if len(marketIDs) == 0 {
  430. for _, v := range markets {
  431. if v.Trademode == 16 || v.Trademode == 71 {
  432. if len(marketIDs) == 0 {
  433. marketIDs = strconv.Itoa(int(v.Marketid))
  434. } else {
  435. marketIDs += "," + strconv.Itoa(int(v.Marketid))
  436. }
  437. }
  438. }
  439. }
  440. hybsMyBuyOrderDetails := make([]HybsMyBuyOrderDetail, 0)
  441. // "我的订单"都是买委托
  442. session := engine.Table("TRADE_ORDERDETAIL").
  443. Select(`to_char(TRADE_ORDERDETAIL.ORDERID) ORDERID, TRADE_ORDERDETAIL.*,
  444. GOODS.GOODSCODE, GOODS.GOODSNAME, GOODS.DECIMALPLACE, GOODS.AGREEUNIT,
  445. HG1.PICURLS PICURLS1,
  446. HG2.PICURLS PICURLS2,
  447. MARKET.TRADEMODE,
  448. ENUMDICITEM.PARAM2 CURRENCYSIGN,
  449. HS1.VENDORNAME VENDORNAME1,
  450. HS2.VENDORNAME VENDORNAME2`).
  451. Join("LEFT", "GOODS", "GOODS.GOODSID = TRADE_ORDERDETAIL.GOODSID").
  452. Join("LEFT", "CPTRADE_PRESALEGOODSEX", "CPTRADE_PRESALEGOODSEX.GOODSID = TRADE_ORDERDETAIL.GOODSID").
  453. Join("LEFT", "HSBY_GOODSEX HG1", "HG1.GOODSID = CPTRADE_PRESALEGOODSEX.RELATEDGOODSID").
  454. Join("LEFT", "HSBY_GOODSEX HG2", "HG2.GOODSID = GOODS.GOODSID").
  455. Join("LEFT", "ENUMDICITEM", "GOODS.CURRENCYID = ENUMDICITEM.ENUMITEMNAME and ENUMDICITEM.ENUMDICCODE = 'currency'").
  456. Join("LEFT", "MARKET", "MARKET.MARKETID = TRADE_ORDERDETAIL.MARKETID").
  457. Join("LEFT", "HSBY_SUPPLIERINFO HS1", "HS1.VENDORID = HG1.VENDORID").
  458. Join("LEFT", "HSBY_SUPPLIERINFO HS2", "HS2.VENDORID = HG2.VENDORID").
  459. Where(fmt.Sprintf("TRADE_ORDERDETAIL.BUYORSELL = 0 and TRADE_ORDERDETAIL.ACCOUNTID in (%s)", accountIDs)).
  460. And(fmt.Sprintf("TRADE_ORDERDETAIL.MARKETID in (%s)", marketIDs))
  461. // 是否过滤状态(抢购中、求购中、已完成)
  462. if orderStatus != "0" {
  463. session = session.And(fmt.Sprintf("TRADE_ORDERDETAIL.ORDERSTATUS in (%s)", orderStatus))
  464. }
  465. if err := session.Find(&hybsMyBuyOrderDetails); err != nil {
  466. return nil, err
  467. }
  468. // 设置“我的订单”显示状态
  469. for i := range hybsMyBuyOrderDetails {
  470. detail := &hybsMyBuyOrderDetails[i]
  471. // 委托金额 = 委托价格 * 委托数量 * 合约单位
  472. detail.Orderamount = detail.Orderprice * float64(detail.Orderqty) * float64(detail.Agreeunit)
  473. if detail.Trademode == 71 && (detail.Orderstatus == 3 || detail.Orderstatus == 7) {
  474. // 抢购中
  475. detail.MyBuyStatus = 1
  476. } else if detail.Trademode == 16 && (detail.Orderstatus == 3 || detail.Orderstatus == 7) {
  477. // 求购中
  478. detail.MyBuyStatus = 2
  479. } else if detail.Orderstatus == 8 || detail.Orderstatus == 9 {
  480. // 已完成
  481. detail.MyBuyStatus = 3
  482. } else if detail.Orderstatus == 6 || detail.Orderstatus == 22 {
  483. // 已撤销
  484. detail.MyBuyStatus = 4
  485. } else if detail.Orderstatus == 4 {
  486. // 委托失败
  487. detail.MyBuyStatus = 5
  488. }
  489. }
  490. return hybsMyBuyOrderDetails, nil
  491. }
  492. // HsbyMyGoods 我的商品(买方向持仓)
  493. type HsbyMyGoods struct {
  494. Accountid int64 `json:"accountid" xorm:"'ACCOUNTID'" binding:"required"` // 账号Id
  495. Goodsid int32 `json:"goodsid" xorm:"'GOODSID'" binding:"required"` // 商品Id
  496. Buycurpositionqty int64 `json:"buycurpositionqty" xorm:"'BUYCURPOSITIONQTY'"` // 买当前持仓总数量
  497. Buycurholderamount float64 `json:"buycurholderamount" xorm:"'BUYCURHOLDERAMOUNT'"` // 买当前持仓总金额[商品币种]
  498. EnableQty int64 `json:"enableqty" xorm:"'ENABLEQTY'"` // 可用数量
  499. Goodscode string `json:"goodscode" xorm:"'GOODSCODE'" binding:"required"` // 商品代码(内部)
  500. Goodsname string `json:"goodsname" xorm:"'GOODSNAME'" binding:"required"` // 商品名称
  501. Decimalplace int64 `json:"decimalplace" xorm:"'DECIMALPLACE'"` // 报价小数位
  502. Agreeunit float64 `json:"agreeunit" xorm:"'AGREEUNIT'"` // 合约单位
  503. Goodsstatus int64 `json:"goodsstatus" xorm:"'GOODSSTATUS'"` // 商品状态- 1:待审核 2:未上市 3:上市 4:已注销 5:审核拒绝 6:退市 7:待退市
  504. Picurls string `json:"picurls" xorm:"'PICURLS'"` // 介绍图片[多张用逗号分隔]
  505. Currencysign string `json:"currencysign" xorm:"'CURRENCYSIGN'"` // 货币符号
  506. Buyaverageprice float64 `json:"buyaverageprice" xorm:"-"` // 持仓均价
  507. }
  508. // GetHsbyMyGoods 获取“我的商品”信息
  509. // 输入 accountIDs string 资金账户列表,格式 1,2,3
  510. // 输出 []HsbyMyGoods “我的商品”信息
  511. // 输入 error error
  512. func GetHsbyMyGoods(accountIDs string) ([]HsbyMyGoods, error) {
  513. engine := db.GetEngine()
  514. hsbyMyGoodses := make([]HsbyMyGoods, 0)
  515. // 此定制版本,只查询出当前手数(期末)大于0的数据 (TRADEPOSITION.BUYCURPOSITIONQTY > 0)
  516. if err := engine.Table("TRADEPOSITION").
  517. Select(`TRADEPOSITION.*, (TRADEPOSITION.BUYCURPOSITIONQTY - TRADEPOSITION.BUYFROZENQTY - TRADEPOSITION.BUYOTHERFROZENQTY) ENABLEQTY,
  518. GOODS.GOODSCODE, GOODS.GOODSNAME, GOODS.DECIMALPLACE, GOODS.AGREEUNIT, GOODS.GOODSSTATUS,
  519. HSBY_GOODSEX.PICURLS,
  520. ENUMDICITEM.PARAM2 CURRENCYSIGN`).
  521. Join("LEFT", "GOODS", "GOODS.GOODSID = TRADEPOSITION.GOODSID").
  522. Join("INNER", "HSBY_GOODSEX", "HSBY_GOODSEX.GOODSID = GOODS.GOODSID").
  523. Join("LEFT", "ENUMDICITEM", "GOODS.CURRENCYID = ENUMDICITEM.ENUMITEMNAME and ENUMDICITEM.ENUMDICCODE = 'currency'").
  524. Where(fmt.Sprintf("TRADEPOSITION.BUYCURPOSITIONQTY > 0 and TRADEPOSITION.ACCOUNTID in (%s)", accountIDs)).Find(&hsbyMyGoodses); err != nil {
  525. return nil, err
  526. }
  527. for i := range hsbyMyGoodses {
  528. myGoods := &hsbyMyGoodses[i]
  529. // 持仓均价
  530. averagePrice := myGoods.Buycurholderamount / float64(myGoods.Buycurpositionqty) / myGoods.Agreeunit
  531. myGoods.Buyaverageprice, _ = strconv.ParseFloat(utils.FormatFloat(averagePrice, int(myGoods.Decimalplace)), 64)
  532. }
  533. return hsbyMyGoodses, nil
  534. }
  535. // HsbyPreGoods 新品上市商品(一级市场产能预售)
  536. type HsbyPreGoods struct {
  537. Goodsid int64 `json:"goodsid" xorm:"'GOODSID'" binding:"required"` // 商品ID(自增ID SEQ_GOODS)
  538. Goodscode string `json:"goodscode" xorm:"'GOODSCODE'" binding:"required"` // 商品代码(内部)
  539. Goodsname string `json:"goodsname" xorm:"'GOODSNAME'" binding:"required"` // 商品名称
  540. Decimalplace int64 `json:"decimalplace" xorm:"'DECIMALPLACE'"` // 报价小数位
  541. Marketid int64 `json:"marketid" xorm:"'MARKETID'" binding:"required"` // 所属市场ID
  542. Goodsstatus int64 `json:"goodsstatus" xorm:"'GOODSSTATUS'"` // 商品状态- 2:未上市 3:上市
  543. Quoteminunit int64 `json:"quoteminunit" xorm:"'QUOTEMINUNIT'"` // 行情最小变动单位 [整数,报价小数位一起使用]
  544. Agreeunit float64 `json:"agreeunit" xorm:"'AGREEUNIT'"` // 合约单位
  545. Listingdate time.Time `json:"listingdate" xorm:"'LISTINGDATE'"` // 交易开始日期
  546. Lasttradedate time.Time `json:"lasttradedate" xorm:"'LASTTRADEDATE'"` // 最后交易日期(状态:待退市)
  547. Relatedgoodsid int64 `json:"relatedgoodsid" xorm:"'RELATEDGOODSID'"` // 关联交易合约ID
  548. Presaleqty int64 `json:"presaleqty" xorm:"'PRESALEQTY'"` // 预售数量
  549. Refprice float64 `json:"refprice" xorm:"'REFPRICE'"` // 参考价格[一口价]
  550. Presaledqty int64 `json:"presaledqty" xorm:"'PRESALEDQTY'"` // 已预售量(预售结束时更新)
  551. EnableQty int64 `json:"enableqty" xorm:"ENABLEQTY"` // 剩余数量
  552. Currency string `json:"currency" xorm:"'CURRENCY'"` // 货币
  553. Currencysign string `json:"currencysign" xorm:"'CURRENCYSIGN'"` // 货币符号
  554. Videourls string `json:"videourls" xorm:"'VIDEOURLS'"` // 介绍视频[多张用逗号分隔]
  555. Picurls string `json:"picurls" xorm:"'PICURLS'"` // 介绍图片[多张用逗号分隔]
  556. Trademode int32 `json:"trademode" xorm:"'TRADEMODE'" binding:"required"` // 交易模式 - 10:做市 13:竞价 15:通道交易 16:挂牌点选 17:仓单贸易 18:期权 19:竞拍-降价式 20:竞拍-竞价式 21:竞拍-大宗式 22:受托竞价
  557. }
  558. // GetHsbyPreGoodses 获取“新品上市”商品列表(一级市场产能预售)
  559. // 输入 marketIDs string 市场ID列表,格式 1,2,3
  560. // 输入 descProvinceID int 省ID
  561. // 输入 descCityID int 市ID
  562. // 输出 []HsbyPreGoods “新品上市”商品列表
  563. // 输出 error error
  564. func GetHsbyPreGoodses(marketIDs string, descProvinceID, descCityID int) ([]HsbyPreGoods, error) {
  565. engine := db.GetEngine()
  566. // 注意:一级产能预售市场使用的商品实际上是和二级市场有关系的,所以要使用 CPTRADE_PRESALEGOODSEX.RELATEDGOODSID 进行关联
  567. // 注意:CPTRADE_PRESALEGOODSEX 表的开始与结束时间是给大宗用的
  568. preGoodses := make([]HsbyPreGoods, 0)
  569. session := engine.Table("GOODS").
  570. Select(`GOODS.*,
  571. CPTRADE_PRESALEGOODSEX.RELATEDGOODSID, CPTRADE_PRESALEGOODSEX.PRESALEQTY, CPTRADE_PRESALEGOODSEX.REFPRICE,
  572. CPTRADE_PRESALEGOODSEX.PRESALEDQTY, (CPTRADE_PRESALEGOODSEX.PRESALEQTY - CPTRADE_PRESALEGOODSEX.PRESALEDQTY) ENABLEQTY,
  573. ENUMDICITEM.ENUMDICNAME CURRENCY, ENUMDICITEM.PARAM2 CURRENCYSIGN,
  574. HSBY_GOODSEX.VIDEOURLS, HSBY_GOODSEX.PICURLS,
  575. MARKET.TRADEMODE`).
  576. Join("INNER", "CPTRADE_PRESALEGOODSEX", "CPTRADE_PRESALEGOODSEX.GOODSID = GOODS.GOODSID").
  577. Join("LEFT", "ENUMDICITEM", "ENUMDICITEM.ENUMITEMNAME = GOODS.CURRENCYID and ENUMDICITEM.ENUMDICCODE = 'currency'").
  578. Join("LEFT", "HSBY_GOODSEX", "HSBY_GOODSEX.GOODSID = CPTRADE_PRESALEGOODSEX.RELATEDGOODSID").
  579. Join("LEFT", "MARKET", "MARKET.MARKETID = GOODS.MARKETID").
  580. Where(fmt.Sprintf("GOODS.GOODSSTATUS in (2,3) and GOODS.MARKETID in (%s)", marketIDs)) // 一级市场获取 2:未上市 3:上市 状态的商品
  581. if descProvinceID > 0 {
  582. session = session.And("HSBY_GOODSEX.DESCPROVINCEID = ?", descProvinceID)
  583. }
  584. if descCityID > 0 {
  585. session = session.And("HSBY_GOODSEX.DESCCITYID = ?", descCityID)
  586. }
  587. if err := session.Find(&preGoodses); err != nil {
  588. return nil, err
  589. }
  590. // for i := range preGoodses {
  591. // preGoods := &preGoodses[i]
  592. // // FIXME: - 这里应该使用 Duck Typing,后期再处理
  593. // // 计算最小变动单位
  594. // lotSize := float64(preGoods.Quoteminunit) * math.Pow(0.1, float64(preGoods.Decimalplace))
  595. // preGoods.LotSize, _ = strconv.ParseFloat(utils.FormatFloat(lotSize, int(preGoods.Decimalplace)), 64)
  596. // }
  597. return preGoodses, nil
  598. }
  599. // HsbyPreGoodsDetail 一级市场(产能预售)商品信息详情
  600. type HsbyPreGoodsDetail struct {
  601. Goodsid int64 `json:"goodsid" xorm:"'GOODSID'" binding:"required"` // 商品ID(自增ID SEQ_GOODS)
  602. Goodscode string `json:"goodscode" xorm:"'GOODSCODE'" binding:"required"` // 商品代码(内部)
  603. Goodsname string `json:"goodsname" xorm:"'GOODSNAME'" binding:"required"` // 商品名称
  604. Decimalplace int64 `json:"decimalplace" xorm:"'DECIMALPLACE'"` // 报价小数位
  605. Marketid int64 `json:"marketid" xorm:"'MARKETID'" binding:"required"` // 所属市场ID
  606. Goodsstatus int64 `json:"goodsstatus" xorm:"'GOODSSTATUS'"` // 商品状态- 2:未上市 3:上市
  607. Quoteminunit int64 `json:"quoteminunit" xorm:"'QUOTEMINUNIT'"` // 行情最小变动单位 [整数,报价小数位一起使用]
  608. Agreeunit float64 `json:"agreeunit" xorm:"'AGREEUNIT'"` // 合约单位
  609. Listingdate time.Time `json:"listingdate" xorm:"'LISTINGDATE'"` // 交易开始日期
  610. Lasttradedate time.Time `json:"lasttradedate" xorm:"'LASTTRADEDATE'"` // 最后交易日期(状态:待退市)
  611. Relatedgoodsid int64 `json:"relatedgoodsid" xorm:"'RELATEDGOODSID'"` // 关联交易合约ID
  612. Presaleqty int64 `json:"presaleqty" xorm:"'PRESALEQTY'"` // 预售数量
  613. Refprice float64 `json:"refprice" xorm:"'REFPRICE'"` // 参考价格[一口价]
  614. Presaledqty int64 `json:"presaledqty" xorm:"'PRESALEDQTY'"` // 已预售量(预售结束时更新)
  615. EnableQty int64 `json:"enableqty" xorm:"ENABLEQTY"` // 剩余数量
  616. Videourls string `json:"videourls" xorm:"'VIDEOURLS'"` // 介绍视频[多张用逗号分隔]
  617. Picurls string `json:"picurls" xorm:"'PICURLS'"` // 介绍图片[多张用逗号分隔]
  618. Descprovinceid int64 `json:"descprovinceid" xorm:"'DESCPROVINCEID'"` // 目的地(省)ID
  619. Desccityid int64 `json:"desccityid" xorm:"'DESCCITYID'"` // 目的地(市)ID
  620. Goodsdesc string `json:"goodsdesc" xorm:"'GOODSDESC'"` // 商品详情
  621. Currency string `json:"currency" xorm:"'CURRENCY'"` // 货币
  622. Currencysign string `json:"currencysign" xorm:"'CURRENCYSIGN'"` // 货币符号
  623. Goodunit string `json:"goodunit" xorm:"GOODUNIT"` // 报价单位
  624. Vendorname string `json:"vendorname" xorm:"'VENDORNAME'"` // 供应商名称
  625. Vendorphone string `json:"vendorphone" xorm:"'VENDORPHONE'"` // 供应商客服电话
  626. Vendorattr string `json:"vendorattr" xorm:"'VENDORATTR'"` // 供应商附件(多张,逗号分隔)
  627. Trademode int32 `json:"trademode" xorm:"'TRADEMODE'" binding:"required"` // 交易模式 - 10:做市 13:竞价 15:通道交易 16:挂牌点选 17:仓单贸易 18:期权 19:竞拍-降价式 20:竞拍-竞价式 21:竞拍-大宗式 22:受托竞价
  628. Customername string `json:"customername" xorm:"'CUSTOMERNAME'"` // 发行单位
  629. StepValue float64 `json:"stepvalue" xorm:"-"` // 价格最小变动单位
  630. LotSize int `json:"lotsize" xorm:"-"` // 手数最小变动单位
  631. Buymaxqty int `json:"buymaxqty" xorm:"'-'"` // 购买上限 [71] - 0为不限
  632. }
  633. // GetHsbyPreGoodsDetail 获取一级市场(产能预售)商品信息详情
  634. // 参数 goodsID int 目标商品ID
  635. // 返回 *HsbyListingGoodsDetail 二级市场(挂牌点选)商品信息详情,查询无结果返回nil
  636. // 返回 error error
  637. func GetHsbyPreGoodsDetail(goodsID, accountID int) (*HsbyPreGoodsDetail, error) {
  638. engine := db.GetEngine()
  639. details := make([]HsbyPreGoodsDetail, 0)
  640. // 获取挂牌商品信息,以及扩展表信息
  641. // FIXME: - 这里使用Get方法,会造成SQL语句的嵌套出错,后期再研究
  642. session := engine.Table("GOODS").
  643. Select(`GOODS.*,
  644. CPTRADE_PRESALEGOODSEX.RELATEDGOODSID, CPTRADE_PRESALEGOODSEX.PRESALEQTY, CPTRADE_PRESALEGOODSEX.REFPRICE,
  645. CPTRADE_PRESALEGOODSEX.PRESALEDQTY, (CPTRADE_PRESALEGOODSEX.PRESALEQTY - CPTRADE_PRESALEGOODSEX.PRESALEDQTY) ENABLEQTY,
  646. HSBY_GOODSEX.VIDEOURLS, HSBY_GOODSEX.PICURLS, HSBY_GOODSEX.DESCPROVINCEID, HSBY_GOODSEX.Desccityid, HSBY_GOODSEX.Goodsdesc,
  647. ENUMDICITEM.ENUMDICNAME CURRENCY, ENUMDICITEM.PARAM2 CURRENCYSIGN,
  648. E1.ENUMDICNAME GOODUNIT,
  649. HSBY_SUPPLIERINFO.VENDORNAME, HSBY_SUPPLIERINFO.VENDORPHONE, HSBY_SUPPLIERINFO.VENDORATTR,
  650. MARKET.TRADEMODE,
  651. USERINFO.CUSTOMERNAME`).
  652. Join("INNER", "CPTRADE_PRESALEGOODSEX", "CPTRADE_PRESALEGOODSEX.GOODSID = GOODS.GOODSID").
  653. Join("LEFT", "HSBY_GOODSEX", "HSBY_GOODSEX.GOODSID = CPTRADE_PRESALEGOODSEX.RELATEDGOODSID").
  654. Join("LEFT", "ENUMDICITEM", "ENUMDICITEM.ENUMITEMNAME = GOODS.CURRENCYID and ENUMDICITEM.ENUMDICCODE = 'currency'").
  655. Join("LEFT", "ENUMDICITEM E1", "E1.ENUMITEMNAME = GOODS.GOODUNITID and E1.ENUMDICCODE = 'goodsunit'").
  656. Join("LEFT", "HSBY_SUPPLIERINFO", "HSBY_SUPPLIERINFO.VENDORID = HSBY_GOODSEX.VENDORID").
  657. Join("LEFT", "MARKET", "MARKET.MARKETID = GOODS.MARKETID").
  658. Join("LEFT", "USERINFO", "USERINFO.USERID = CPTRADE_PRESALEGOODSEX.USERID").
  659. Where("GOODS.GOODSID = ?", goodsID)
  660. if err := session.Find(&details); err != nil {
  661. return nil, err
  662. }
  663. // 无目标商品
  664. if len(details) == 0 {
  665. return nil, nil
  666. }
  667. preGoodsDetail := details[0]
  668. // FIXME: - 这里应该使用 Duck Typing,后期再处理
  669. // 计算价格最小变动单位
  670. stepValue := float64(preGoodsDetail.Quoteminunit) * math.Pow(0.1, float64(preGoodsDetail.Decimalplace))
  671. preGoodsDetail.StepValue, _ = strconv.ParseFloat(utils.FormatFloat(stepValue, int(preGoodsDetail.Decimalplace)), 64)
  672. // 计算手数最小变动单位,一级市场固定为1
  673. preGoodsDetail.LotSize = 1
  674. // 获取购买上限
  675. if accountID != 0 {
  676. buyMaxQty, err := GetCPTradeBuyLimit(goodsID, accountID)
  677. if err != nil {
  678. return nil, err
  679. }
  680. preGoodsDetail.Buymaxqty = buyMaxQty
  681. }
  682. return &preGoodsDetail, nil
  683. }
  684. // HsbySellMyDetail "我的闲置"单据信息(发布中 - 二级市场卖挂,3:委托成功、7:部分成交; 已完成 - 二级市场成交单,包括历史数据)
  685. type HsbySellMyDetail struct {
  686. Orderid string `json:"orderid" xorm:"'ORDERID'" binding:"required"` // 单号(发布中 - 委托单号;已完成 - 成交单号)
  687. Marketid int32 `json:"marketid" xorm:"'MARKETID'" binding:"required"` // 市场ID
  688. Goodsid int32 `json:"goodsid" xorm:"'GOODSID'" binding:"required"` // 商品ID
  689. Accountid int64 `json:"accountid" xorm:"'ACCOUNTID'" binding:"required"` // 账户ID[报价币种]
  690. Buyorsell int32 `json:"buyorsell" xorm:"'BUYORSELL'" binding:"required"` // 买卖 - 0:买 1:卖
  691. Price float64 `json:"price" xorm:"'PRICE'"` // 价格
  692. Time time.Time `json:"time" xorm:"'TIME'" binding:"required"` // 时间
  693. Qty int64 `json:"qty" xorm:"'QTY'"` // 数量
  694. Goodscode string `json:"goodscode" xorm:"'GOODSCODE'" binding:"required"` // 商品代码(内部)
  695. Goodsname string `json:"goodsname" xorm:"'GOODSNAME'" binding:"required"` // 商品名称
  696. Decimalplace int64 `json:"decimalplace" xorm:"'DECIMALPLACE'"` // 报价小数位
  697. Agreeunit float64 `json:"agreeunit" xorm:"'AGREEUNIT'"` // 合约单位
  698. Picurls string `json:"picurls" xorm:"'PICURLS'"` // 介绍图片[多张用逗号分隔]
  699. Trademode uint32 `json:"trademode" xorm:"'TRADEMODE'" binding:"required"` // 交易模式 - 10:做市 13:竞价 15:通道交易 16:挂牌点选 17:仓单贸易 18:期权 19:竞拍-降价式 20:竞拍-竞价式 21:竞拍-大宗式 22:受托竞价
  700. Currencysign string `json:"currencysign" xorm:"'CURRENCYSIGN'"` // 货币符号
  701. Vendorname string `json:"vendorname" xorm:"'VENDORNAME'"` // 供应商名称
  702. OrderType int32 `json:"ordertype" xorm:"'ORDERTYPE'"` // 单据类型:0 - 发布中, 1 - 已完成
  703. }
  704. // GetHsbySellMyOrderDetails 获取"我的闲置 - 发布中"单据信息
  705. // 输入 accountIDs string 资金账户列表
  706. // 输出 []HsbySellMyDetail "我的闲置 - 发布中"单据信息
  707. // 输出 error error
  708. func GetHsbySellMyOrderDetails(accountIDs string) ([]HsbySellMyDetail, error) {
  709. // 获取市场信息
  710. markets, err := GetMarkets()
  711. if err != nil {
  712. return nil, err
  713. }
  714. engine := db.GetEngine()
  715. marketIDs := "" // 我的闲置-发布中: 二级市场卖挂牌
  716. // 默认取 TradeMode = 16
  717. for _, v := range markets {
  718. if v.Trademode == 16 {
  719. if len(marketIDs) == 0 {
  720. marketIDs = strconv.Itoa(int(v.Marketid))
  721. } else {
  722. marketIDs += "," + strconv.Itoa(int(v.Marketid))
  723. }
  724. }
  725. }
  726. orderDetails := make([]HsbySellMyDetail, 0)
  727. // “我的闲置 - 发布中”都是卖挂委托
  728. session := engine.Table("TRADE_ORDERDETAIL").
  729. Select(`to_char(TRADE_ORDERDETAIL.ORDERID) ORDERID,
  730. TRADE_ORDERDETAIL.MARKETID, TRADE_ORDERDETAIL.GOODSID, TRADE_ORDERDETAIL.ACCOUNTID, TRADE_ORDERDETAIL.BUYORSELL,
  731. TRADE_ORDERDETAIL.ORDERPRICE PRICE, TRADE_ORDERDETAIL.ORDERTIME TIME,
  732. (TRADE_ORDERDETAIL.ORDERQTY - TRADE_ORDERDETAIL.TRADEQTY - TRADE_ORDERDETAIL.CANCELQTY) QTY,
  733. GOODS.GOODSCODE, GOODS.GOODSNAME, GOODS.DECIMALPLACE, GOODS.AGREEUNIT,
  734. HSBY_GOODSEX.PICURLS,
  735. MARKET.TRADEMODE,
  736. ENUMDICITEM.PARAM2 CURRENCYSIGN,
  737. HSBY_SUPPLIERINFO.VENDORNAME,
  738. 0 ORDERTYPE`).
  739. Join("LEFT", "GOODS", "GOODS.GOODSID = TRADE_ORDERDETAIL.GOODSID").
  740. Join("LEFT", "HSBY_GOODSEX", "HSBY_GOODSEX.GOODSID = GOODS.GOODSID").
  741. Join("LEFT", "ENUMDICITEM", "GOODS.CURRENCYID = ENUMDICITEM.ENUMITEMNAME and ENUMDICITEM.ENUMDICCODE = 'currency'").
  742. Join("LEFT", "MARKET", "MARKET.MARKETID = TRADE_ORDERDETAIL.MARKETID").
  743. Join("LEFT", "HSBY_SUPPLIERINFO", "HSBY_SUPPLIERINFO.VENDORID = HSBY_GOODSEX.VENDORID").
  744. Where(fmt.Sprintf(`TRADE_ORDERDETAIL.BUYORSELL = 1 and TRADE_ORDERDETAIL.LISTINGSELECTTYPE = 1 and (TRADE_ORDERDETAIL.ORDERSTATUS = 3 or TRADE_ORDERDETAIL.ORDERSTATUS = 7)
  745. and TRADE_ORDERDETAIL.ACCOUNTID in (%s)`, accountIDs)).
  746. And(fmt.Sprintf("TRADE_ORDERDETAIL.MARKETID in (%s)", marketIDs))
  747. if err := session.Find(&orderDetails); err != nil {
  748. return nil, err
  749. }
  750. return orderDetails, nil
  751. }
  752. // GetHsbySellMyTradeDetails 获取"我的闲置 - 已完成"单据信息
  753. // 输入 accountIDs string 资金账户列表
  754. // 输出 []HsbySellMyDetail "我的闲置 - 已完成"单据信息
  755. // 输出 error error
  756. func GetHsbySellMyTradeDetails(accountIDs string) ([]HsbySellMyDetail, error) {
  757. // 获取市场信息
  758. markets, err := GetMarkets()
  759. if err != nil {
  760. return nil, err
  761. }
  762. engine := db.GetEngine()
  763. marketIDs := "" // 我的闲置-发布中: 二级市场卖方向成交单(包括历史成交单)
  764. // 默认取 TradeMode = 16
  765. for _, v := range markets {
  766. if v.Trademode == 16 {
  767. if len(marketIDs) == 0 {
  768. marketIDs = strconv.Itoa(int(v.Marketid))
  769. } else {
  770. marketIDs += "," + strconv.Itoa(int(v.Marketid))
  771. }
  772. }
  773. }
  774. orders := make([]HsbySellMyDetail, 0)
  775. // 当前成交单
  776. curOrders := make([]HsbySellMyDetail, 0)
  777. if err := engine.Table("TRADE_TRADEDETAIL").
  778. Select(`to_char(TRADE_TRADEDETAIL.TRADEID) ORDERID,
  779. TRADE_TRADEDETAIL.MARKETID, TRADE_TRADEDETAIL.GOODSID, TRADE_TRADEDETAIL.ACCOUNTID, TRADE_TRADEDETAIL.BUYORSELL,
  780. TRADE_TRADEDETAIL.TRADEPRICE PRICE, TRADE_TRADEDETAIL.TRADEQTY QTY, TRADE_TRADEDETAIL.TRADETIME TIME, TRADE_TRADEDETAIL.TRADEAMOUNT AMOUNT,
  781. GOODS.GOODSCODE, GOODS.GOODSNAME, GOODS.DECIMALPLACE, GOODS.AGREEUNIT,
  782. HSBY_GOODSEX.PICURLS,
  783. MARKET.TRADEMODE,
  784. ENUMDICITEM.PARAM2 CURRENCYSIGN,
  785. HSBY_SUPPLIERINFO.VENDORNAME,
  786. 1 ORDERTYPE`).
  787. Join("LEFT", "GOODS", "GOODS.GOODSID = TRADE_TRADEDETAIL.GOODSID").
  788. Join("LEFT", "HSBY_GOODSEX", "HSBY_GOODSEX.GOODSID = GOODS.GOODSID").
  789. Join("LEFT", "ENUMDICITEM", "GOODS.CURRENCYID = ENUMDICITEM.ENUMITEMNAME and ENUMDICITEM.ENUMDICCODE = 'currency'").
  790. Join("LEFT", "MARKET", "MARKET.MARKETID = TRADE_TRADEDETAIL.MARKETID").
  791. Join("LEFT", "HSBY_SUPPLIERINFO", "HSBY_SUPPLIERINFO.VENDORID = HSBY_GOODSEX.VENDORID").
  792. Where(fmt.Sprintf(`TRADE_TRADEDETAIL.BUYORSELL = 1 and TRADE_TRADEDETAIL.TRADETYPE in (1,24)
  793. and TRADE_TRADEDETAIL.ACCOUNTID in (%s)`, accountIDs)).
  794. And(fmt.Sprintf("TRADE_TRADEDETAIL.MARKETID in (%s)", marketIDs)).Find(&curOrders); err != nil {
  795. return nil, err
  796. }
  797. if len(curOrders) > 0 {
  798. orders = append(orders, curOrders...)
  799. }
  800. // 历史成交单
  801. hisOrders := make([]HsbySellMyDetail, 0)
  802. if err := engine.Table("HIS_TRADE_TRADEDETAIL").
  803. Select(`to_char(HIS_TRADE_TRADEDETAIL.TRADEID) ORDERID,
  804. HIS_TRADE_TRADEDETAIL.MARKETID, HIS_TRADE_TRADEDETAIL.GOODSID, HIS_TRADE_TRADEDETAIL.ACCOUNTID, HIS_TRADE_TRADEDETAIL.BUYORSELL,
  805. HIS_TRADE_TRADEDETAIL.TRADEPRICE PRICE, HIS_TRADE_TRADEDETAIL.TRADEQTY QTY, HIS_TRADE_TRADEDETAIL.TRADETIME TIME, HIS_TRADE_TRADEDETAIL.TRADEAMOUNT AMOUNT,
  806. GOODS.GOODSCODE, GOODS.GOODSNAME, GOODS.DECIMALPLACE, GOODS.AGREEUNIT,
  807. HSBY_GOODSEX.PICURLS,
  808. MARKET.TRADEMODE,
  809. ENUMDICITEM.PARAM2 CURRENCYSIGN,
  810. HSBY_SUPPLIERINFO.VENDORNAME,
  811. 1 ORDERTYPE`).
  812. Join("LEFT", "GOODS", "GOODS.GOODSID = HIS_TRADE_TRADEDETAIL.GOODSID").
  813. Join("LEFT", "HSBY_GOODSEX", "HSBY_GOODSEX.GOODSID = GOODS.GOODSID").
  814. Join("LEFT", "ENUMDICITEM", "GOODS.CURRENCYID = ENUMDICITEM.ENUMITEMNAME and ENUMDICITEM.ENUMDICCODE = 'currency'").
  815. Join("LEFT", "MARKET", "MARKET.MARKETID = HIS_TRADE_TRADEDETAIL.MARKETID").
  816. Join("LEFT", "HSBY_SUPPLIERINFO", "HSBY_SUPPLIERINFO.VENDORID = HSBY_GOODSEX.VENDORID").
  817. Where(fmt.Sprintf(`HIS_TRADE_TRADEDETAIL.BUYORSELL = 1 and HIS_TRADE_TRADEDETAIL.TRADETYPE in (1,24) and HIS_TRADE_TRADEDETAIL.ISVALIDDATA = 1
  818. and HIS_TRADE_TRADEDETAIL.ACCOUNTID in (%s)`, accountIDs)).
  819. And(fmt.Sprintf("HIS_TRADE_TRADEDETAIL.MARKETID in (%s)", marketIDs)).Find(&hisOrders); err != nil {
  820. return nil, err
  821. }
  822. if len(hisOrders) > 0 {
  823. orders = append(orders, hisOrders...)
  824. }
  825. return orders, nil
  826. }
  827. // HsbyMyPackage 我的包裹信息
  828. type HsbyMyPackage struct {
  829. Takeorderid string `json:"takeorderid" xorm:"'TAKEORDERID'" binding:"required"` // 提货单号(905+Unix秒时间戳(10位)+xxxxxx)
  830. Accountid int64 `json:"accountid" xorm:"'ACCOUNTID'"` // 账户ID
  831. Goodsid int64 `json:"goodsid" xorm:"'GOODSID'"` // 商品ID
  832. Userid int64 `json:"userid" xorm:"'USERID'"` // 用户ID
  833. Qty float64 `json:"qty" xorm:"'QTY'"` // 提货数量
  834. Reqtime time.Time `json:"reqtime" xorm:"'REQTIME'"` // 更新时间
  835. Recivername string `json:"recivername" xorm:"'RECIVERNAME'"` // 提货人姓名
  836. Cardtypeid int32 `json:"cardtypeid" xorm:"'CARDTYPEID'"` // 提货人证件类型
  837. Cardnum string `json:"cardnum" xorm:"'CARDNUM'"` // 提货人证件号码
  838. Phonenum string `json:"phonenum" xorm:"'PHONENUM'"` // 提货人联系方式
  839. Takemode int32 `json:"takemode" xorm:"'TAKEMODE'"` // 提货方式 - 2:自提 3:配送
  840. Address string `json:"address" xorm:"'ADDRESS'"` // 提货人详细地址
  841. Takeremark string `json:"takeremark" xorm:"'TAKEREMARK'"` // 提货备注
  842. Takeorderstatus int32 `json:"takeorderstatus" xorm:"'TAKEORDERSTATUS'"` // 提货状态 - 1:待发货 2:已发货 3:已收货
  843. Auditer int32 `json:"auditer" xorm:"'AUDITER'"` // 审核人
  844. Audittime time.Time `json:"audittime" xorm:"'AUDITTIME'"` // 审核时间
  845. Checkremark string `json:"checkremark" xorm:"'CHECKREMARK'"` // 审核备注
  846. Tradedate string `json:"tradedate" xorm:"'TRADEDATE'"` // 交易日(yyyyMMdd)
  847. Marketid int32 `json:"marketid" xorm:"'MARKETID'"` // 市场ID
  848. Handlestatus int32 `json:"handlestatus" xorm:"'HANDLESTATUS'"` // 处理状态
  849. Amount float64 `json:"amount" xorm:"'AMOUNT'"` // 提货金额
  850. Averageprice float64 `json:"averageprice" xorm:"'AVERAGEPRICE'"` // 均价
  851. Goodscode string `json:"goodscode" xorm:"'GOODSCODE'" binding:"required"` // 商品代码(内部)
  852. Goodsname string `json:"goodsname" xorm:"'GOODSNAME'" binding:"required"` // 商品名称
  853. Decimalplace int64 `json:"decimalplace" xorm:"'DECIMALPLACE'"` // 报价小数位
  854. Agreeunit float64 `json:"agreeunit" xorm:"'AGREEUNIT'"` // 合约单位
  855. Picurls string `json:"picurls" xorm:"'PICURLS'"` // 介绍图片[多张用逗号分隔]
  856. Currencysign string `json:"currencysign" xorm:"'CURRENCYSIGN'"` // 货币符号
  857. Vendorname string `json:"vendorname" xorm:"'VENDORNAME'"` // 供应商名称
  858. }
  859. // GetHsbyMyPackages 获取我的包裹信息
  860. // 输入 accountIDs string 资金账户列表
  861. // 输入 takeOrderStatus int 提货状态 - 1:待发货 2:已发货 3:已收货
  862. // 输出 []HsbyMyPackage 我的包裹信息
  863. // 输出 error error
  864. func GetHsbyMyPackages(accountIDs string, takeOrderStatus int) ([]HsbyMyPackage, error) {
  865. engine := db.GetEngine()
  866. myPackages := make([]HsbyMyPackage, 0)
  867. session := engine.Table("TRADE_GOODSPICKUP").
  868. Select(`to_char(TRADE_GOODSPICKUP.TAKEORDERID) TAKEORDERID, (DIVISION.PATHNAME || TRADE_GOODSPICKUP.ADDRESS) ADDRESS, TRADE_GOODSPICKUP.*,
  869. (TRADE_GOODSPICKUP.AMOUNT/TRADE_GOODSPICKUP.QTY/GOODS.AGREEUNIT) AVERAGEPRICE,
  870. GOODS.GOODSCODE, GOODS.GOODSNAME, GOODS.DECIMALPLACE, GOODS.AGREEUNIT,
  871. HSBY_GOODSEX.PICURLS,
  872. ENUMDICITEM.PARAM2 CURRENCYSIGN,
  873. HSBY_SUPPLIERINFO.VENDORNAME`).
  874. Join("LEFT", "GOODS", "GOODS.GOODSID = TRADE_GOODSPICKUP.GOODSID").
  875. Join("LEFT", "HSBY_GOODSEX", "HSBY_GOODSEX.GOODSID = GOODS.GOODSID").
  876. Join("LEFT", "ENUMDICITEM", "GOODS.CURRENCYID = ENUMDICITEM.ENUMITEMNAME and ENUMDICITEM.ENUMDICCODE = 'currency'").
  877. Join("LEFT", "HSBY_SUPPLIERINFO", "HSBY_SUPPLIERINFO.VENDORID = HSBY_GOODSEX.VENDORID").
  878. Join("LEFT", "DIVISION", "DIVISION.AUTOID = TRADE_GOODSPICKUP.DISTRICTID").
  879. Where(fmt.Sprintf("TRADE_GOODSPICKUP.ACCOUNTID in (%s)", accountIDs))
  880. if takeOrderStatus > 0 {
  881. session = session.And("TAKEORDERSTATUS = ?", takeOrderStatus)
  882. }
  883. if err := session.Find(&myPackages); err != nil {
  884. return nil, err
  885. }
  886. // FIXME:- 目前暂时在服务端对手机号码进行解密
  887. key, _ := hex.DecodeString(utils.AESSecretKey)
  888. for i := range myPackages {
  889. myPackage := &myPackages[i]
  890. if len(myPackage.Phonenum) > 0 {
  891. // 解密手机号码
  892. if phonenum, err := hex.DecodeString(myPackage.Phonenum); err == nil { // hex -> []byte
  893. if mobile, err := utils.AESDecrypt(phonenum, key); err == nil {
  894. myPackage.Phonenum = string(mobile)
  895. }
  896. }
  897. }
  898. }
  899. return myPackages, nil
  900. }
  901. // SetHsbyMyPackagesStatus 设置我的包裹提货状态
  902. func SetHsbyMyPackagesStatus(takeOrderID string, accountID, status int) error {
  903. engine := db.GetEngine()
  904. sql := fmt.Sprintf("update TRADE_GOODSPICKUP set TAKEORDERSTATUS = %d where TAKEORDERID = %s and ACCOUNTID = %d", status, takeOrderID, accountID)
  905. _, err := engine.Exec(sql)
  906. return err
  907. }
  908. // GetHsbyProvincesAndCities 获取省市信息数组
  909. // 参数 provinceID int 省ID,选填
  910. // 返回 []Division 枚举信息数组
  911. // 返回 error error
  912. func GetHsbyProvincesAndCities(provinceID int) ([]Division, error) {
  913. engine := db.GetEngine()
  914. divisions := make([]Division, 0)
  915. session := engine.Where("DIVISIONLEVEL = 'province' or DIVISIONLEVEL = 'city'").
  916. And(`AUTOID in (
  917. (select DESCPROVINCEID AUTOID from HSBY_GOODSEX inner join GOODS on GOODS.GOODSID = HSBY_GOODSEX.GOODSID and GOODS.GOODSSTATUS in (2,3))
  918. union all
  919. (select DESCCITYID AUTOID from HSBY_GOODSEX inner join GOODS on GOODS.GOODSID = HSBY_GOODSEX.GOODSID and GOODS.GOODSSTATUS in (2,3))
  920. )`)
  921. if provinceID > 0 {
  922. session = session.And("AUTOID = ?", provinceID)
  923. }
  924. if err := session.Find(&divisions); err != nil {
  925. return nil, err
  926. }
  927. return divisions, nil
  928. }
  929. // HsbyBuyMyTradeDetail "我的订单 - 已完成"单据信息(一二级市场成交单,包括历史数据)
  930. type HsbyBuyMyTradeDetail struct {
  931. Orderid string `json:"orderid" xorm:"'ORDERID'" binding:"required"` // 单号(成交单号)
  932. Marketid int32 `json:"marketid" xorm:"'MARKETID'" binding:"required"` // 市场ID
  933. Goodsid int32 `json:"goodsid" xorm:"'GOODSID'" binding:"required"` // 商品ID
  934. Accountid int64 `json:"accountid" xorm:"'ACCOUNTID'" binding:"required"` // 账户ID[报价币种]
  935. Buyorsell int32 `json:"buyorsell" xorm:"'BUYORSELL'" binding:"required"` // 买卖 - 0:买 1:卖
  936. Price float64 `json:"price" xorm:"'PRICE'"` // 价格
  937. Qty int64 `json:"qty" xorm:"'QTY'" binding:"required"` // 数量
  938. Time time.Time `json:"time" xorm:"'TIME'" binding:"required"` // 时间
  939. Amount float64 `json:"amount" xorm:"'AMOUNT'"` // 金额 = 价格 * 数量 * 合约单位
  940. Goodscode string `json:"goodscode" xorm:"'GOODSCODE'" binding:"required"` // 商品代码(内部)
  941. Goodsname string `json:"goodsname" xorm:"'GOODSNAME'" binding:"required"` // 商品名称
  942. Decimalplace int64 `json:"decimalplace" xorm:"'DECIMALPLACE'"` // 报价小数位
  943. Agreeunit float64 `json:"agreeunit" xorm:"'AGREEUNIT'"` // 合约单位
  944. Picurls1 string `json:"picurls1" xorm:"'PICURLS1'"` // 预售商品介绍图片[多张用逗号分隔]
  945. Picurls2 string `json:"picurls2" xorm:"'PICURLS2'"` // 挂牌商品介绍图片[多张用逗号分隔]
  946. Trademode uint32 `json:"trademode" xorm:"'TRADEMODE'" binding:"required"` // 交易模式 - 10:做市 13:竞价 15:通道交易 16:挂牌点选 17:仓单贸易 18:期权 19:竞拍-降价式 20:竞拍-竞价式 21:竞拍-大宗式 22:受托竞价
  947. Currencysign string `json:"currencysign" xorm:"'CURRENCYSIGN'"` // 货币符号
  948. Vendorname1 string `json:"vendorname1" xorm:"'VENDORNAME1'"` // 预售商品供应商名称
  949. Vendorname2 string `json:"vendorname2" xorm:"'VENDORNAME2'"` // 挂牌商品供应商名称
  950. }
  951. // GetHsbyBuyMyTradeDetails 获取"我的订单 - 已完成"单据信息(一二级市场成交单,包括历史数据)
  952. // 输入 accountIDs string 资金账户列表
  953. // 输出 []HsbyBuyMyTradeDetail "我的订单 - 已完成"单据信息
  954. // 输出 error error
  955. func GetHsbyBuyMyTradeDetails(accountIDs string) ([]HsbyBuyMyTradeDetail, error) {
  956. // 获取市场信息
  957. markets, err := GetMarkets()
  958. if err != nil {
  959. return nil, err
  960. }
  961. engine := db.GetEngine()
  962. marketIDs := "" // 我的订单包括一二级市场卖方向成交单(包括历史成交单)
  963. // 默认取 TradeMode = 16
  964. for _, v := range markets {
  965. if v.Trademode == 16 || v.Trademode == 71 {
  966. if len(marketIDs) == 0 {
  967. marketIDs = strconv.Itoa(int(v.Marketid))
  968. } else {
  969. marketIDs += "," + strconv.Itoa(int(v.Marketid))
  970. }
  971. }
  972. }
  973. orders := make([]HsbyBuyMyTradeDetail, 0)
  974. // 当前成交单
  975. curOrders := make([]HsbyBuyMyTradeDetail, 0)
  976. if err := engine.Table("TRADE_TRADEDETAIL").
  977. Select(`to_char(TRADE_TRADEDETAIL.TRADEID) ORDERID,
  978. TRADE_TRADEDETAIL.MARKETID, TRADE_TRADEDETAIL.GOODSID, TRADE_TRADEDETAIL.ACCOUNTID, TRADE_TRADEDETAIL.BUYORSELL,
  979. TRADE_TRADEDETAIL.TRADEPRICE PRICE, TRADE_TRADEDETAIL.TRADEQTY QTY, TRADE_TRADEDETAIL.TRADETIME TIME, TRADE_TRADEDETAIL.TRADEAMOUNT AMOUNT,
  980. GOODS.GOODSCODE, GOODS.GOODSNAME, GOODS.DECIMALPLACE, GOODS.AGREEUNIT,
  981. HG1.PICURLS PICURLS1, HG2.PICURLS PICURLS2,
  982. MARKET.TRADEMODE,
  983. ENUMDICITEM.PARAM2 CURRENCYSIGN,
  984. HS1.VENDORNAME VENDORNAME1, HS2.VENDORNAME VENDORNAME2`).
  985. Join("LEFT", "GOODS", "GOODS.GOODSID = TRADE_TRADEDETAIL.GOODSID").
  986. Join("LEFT", "CPTRADE_PRESALEGOODSEX", "CPTRADE_PRESALEGOODSEX.GOODSID = TRADE_TRADEDETAIL.GOODSID").
  987. Join("LEFT", "HSBY_GOODSEX HG1", "HG1.GOODSID = CPTRADE_PRESALEGOODSEX.RELATEDGOODSID").
  988. Join("LEFT", "HSBY_GOODSEX HG2", "HG2.GOODSID = GOODS.GOODSID").
  989. Join("LEFT", "ENUMDICITEM", "GOODS.CURRENCYID = ENUMDICITEM.ENUMITEMNAME and ENUMDICITEM.ENUMDICCODE = 'currency'").
  990. Join("LEFT", "MARKET", "MARKET.MARKETID = TRADE_TRADEDETAIL.MARKETID").
  991. Join("LEFT", "HSBY_SUPPLIERINFO HS1", "HS1.VENDORID = HG1.VENDORID").
  992. Join("LEFT", "HSBY_SUPPLIERINFO HS2", "HS2.VENDORID = HG2.VENDORID").
  993. Where(fmt.Sprintf(`TRADE_TRADEDETAIL.BUYORSELL = 0 and TRADE_TRADEDETAIL.TRADETYPE in (1,24)
  994. and TRADE_TRADEDETAIL.ACCOUNTID in (%s)`, accountIDs)).
  995. And(fmt.Sprintf("TRADE_TRADEDETAIL.MARKETID in (%s)", marketIDs)).Find(&curOrders); err != nil {
  996. return nil, err
  997. }
  998. if len(curOrders) > 0 {
  999. orders = append(orders, curOrders...)
  1000. }
  1001. // 历史成交单
  1002. hisOrders := make([]HsbyBuyMyTradeDetail, 0)
  1003. if err := engine.Table("HIS_TRADE_TRADEDETAIL").
  1004. Select(`to_char(HIS_TRADE_TRADEDETAIL.TRADEID) ORDERID,
  1005. HIS_TRADE_TRADEDETAIL.MARKETID, HIS_TRADE_TRADEDETAIL.GOODSID, HIS_TRADE_TRADEDETAIL.ACCOUNTID, HIS_TRADE_TRADEDETAIL.BUYORSELL,
  1006. HIS_TRADE_TRADEDETAIL.TRADEPRICE PRICE, HIS_TRADE_TRADEDETAIL.TRADEQTY QTY, HIS_TRADE_TRADEDETAIL.TRADETIME TIME, HIS_TRADE_TRADEDETAIL.TRADEAMOUNT AMOUNT,
  1007. GOODS.GOODSCODE, GOODS.GOODSNAME, GOODS.DECIMALPLACE, GOODS.AGREEUNIT,
  1008. HG1.PICURLS PICURLS1, HG2.PICURLS PICURLS2,
  1009. MARKET.TRADEMODE,
  1010. ENUMDICITEM.PARAM2 CURRENCYSIGN,
  1011. HS1.VENDORNAME VENDORNAME1, HS2.VENDORNAME VENDORNAME2`).
  1012. Join("LEFT", "GOODS", "GOODS.GOODSID = HIS_TRADE_TRADEDETAIL.GOODSID").
  1013. Join("LEFT", "CPTRADE_PRESALEGOODSEX", "CPTRADE_PRESALEGOODSEX.GOODSID = HIS_TRADE_TRADEDETAIL.GOODSID").
  1014. Join("LEFT", "HSBY_GOODSEX HG1", "HG1.GOODSID = CPTRADE_PRESALEGOODSEX.RELATEDGOODSID").
  1015. Join("LEFT", "HSBY_GOODSEX HG2", "HG2.GOODSID = GOODS.GOODSID").
  1016. Join("LEFT", "ENUMDICITEM", "GOODS.CURRENCYID = ENUMDICITEM.ENUMITEMNAME and ENUMDICITEM.ENUMDICCODE = 'currency'").
  1017. Join("LEFT", "MARKET", "MARKET.MARKETID = HIS_TRADE_TRADEDETAIL.MARKETID").
  1018. Join("LEFT", "HSBY_SUPPLIERINFO HS1", "HS1.VENDORID = HG1.VENDORID").
  1019. Join("LEFT", "HSBY_SUPPLIERINFO HS2", "HS2.VENDORID = HG2.VENDORID").
  1020. Where(fmt.Sprintf(`HIS_TRADE_TRADEDETAIL.BUYORSELL = 0 and HIS_TRADE_TRADEDETAIL.TRADETYPE in (1,24) and HIS_TRADE_TRADEDETAIL.ISVALIDDATA = 1
  1021. and HIS_TRADE_TRADEDETAIL.ACCOUNTID in (%s)`, accountIDs)).
  1022. And(fmt.Sprintf("HIS_TRADE_TRADEDETAIL.MARKETID in (%s)", marketIDs)).Find(&hisOrders); err != nil {
  1023. return nil, err
  1024. }
  1025. if len(hisOrders) > 0 {
  1026. orders = append(orders, hisOrders...)
  1027. }
  1028. return orders, nil
  1029. }
  1030. // HsbyBuyMyPayOrder "我的订单 - 待付款"单据信息(一二级市场待付款信息)
  1031. type HsbyBuyMyPayOrder struct {
  1032. Tradeid string `json:"tradeid" xorm:"'TRADEIDSTR'" binding:"required"` // 成交单号(101+Unix秒时间戳(10位)+2位(MarketServiceID)+xxxx)
  1033. Tradedate string `json:"tradedate" xorm:"'TRADEDATE'"` // 交易日(yyyyMMdd)
  1034. Marketid int32 `json:"marketid" xorm:"'MARKETID'"` // 市场ID
  1035. Goodsid int32 `json:"goodsid" xorm:"'GOODSID'"` // 商品ID
  1036. Buyorderid string `json:"buyorderid" xorm:"'BUYORDERIDSTR'"` // 买方委托单号
  1037. Buyaccountid int64 `json:"buyaccountid" xorm:"'BUYACCOUNTID'"` // 买方账号ID[报价币种]
  1038. Sellorderid string `json:"sellorderid" xorm:"'SELLORDERIDSTR'"` // 卖方委托单号
  1039. Sellaccountid int64 `json:"sellaccountid" xorm:"'SELLACCOUNTID'"` // 卖方账号ID[报价币种]
  1040. Tradeamount float64 `json:"tradeamount" xorm:"'TRADEAMOUNT'"` // 成交金额
  1041. Tradecharge float64 `json:"tradecharge" xorm:"'TRADECHARGE'"` // 成交手续费(买方)
  1042. Payflag int32 `json:"payflag" xorm:"'PAYFLAG'"` // 付款标识 - 1:未支付 2:已支付 3:已过期 4:已撤销 5:结算过期 6:预售终止
  1043. Createtime time.Time `json:"createtime" xorm:"'CREATETIME'"` // 创建时间
  1044. Paytime time.Time `json:"paytime" xorm:"'PAYTIME'"` // 付款时间
  1045. Paylimitedtime time.Time `json:"paylimitedtime" xorm:"'PAYLIMITEDTIME'"` // 支付期限
  1046. Offamount float64 `json:"offamount" xorm:"'OFFAMOUNT'"` // 优惠金额
  1047. Tradeprice float64 `json:"tradeprice" xorm:"'TRADEPRICE'"` // 成交价格
  1048. Tradeqty int64 `json:"tradeqty" xorm:"'TRADEQTY'"` // 成交数量
  1049. Payamount float64 `json:"payamount" xorm:"'PAYAMOUNT'"` // 支付金额 = TradeAmount + TradeCharge - OffAmount
  1050. Goodscode string `json:"goodscode" xorm:"'GOODSCODE'" binding:"required"` // 商品代码(内部)
  1051. Goodsname string `json:"goodsname" xorm:"'GOODSNAME'" binding:"required"` // 商品名称
  1052. Decimalplace int64 `json:"decimalplace" xorm:"'DECIMALPLACE'"` // 报价小数位
  1053. Agreeunit float64 `json:"agreeunit" xorm:"'AGREEUNIT'"` // 合约单位
  1054. Picurls1 string `json:"picurls1" xorm:"'PICURLS1'"` // 预售商品介绍图片[多张用逗号分隔]
  1055. Picurls2 string `json:"picurls2" xorm:"'PICURLS2'"` // 挂牌商品介绍图片[多张用逗号分隔]
  1056. Trademode uint32 `json:"trademode" xorm:"'TRADEMODE'" binding:"required"` // 交易模式 - 10:做市 13:竞价 15:通道交易 16:挂牌点选 17:仓单贸易 18:期权 19:竞拍-降价式 20:竞拍-竞价式 21:竞拍-大宗式 22:受托竞价
  1057. Currencysign string `json:"currencysign" xorm:"'CURRENCYSIGN'"` // 货币符号
  1058. Vendorname1 string `json:"vendorname1" xorm:"'VENDORNAME1'"` // 预售商品供应商名称
  1059. Vendorname2 string `json:"vendorname2" xorm:"'VENDORNAME2'"` // 挂牌商品供应商名称
  1060. }
  1061. // GetHsbyBuyMyPayOrders 获取待付款信息
  1062. func GetHsbyBuyMyPayOrders(accountIDs string, buyOrderID, sellOrderID int) ([]HsbyBuyMyPayOrder, error) {
  1063. // 获取市场信息
  1064. markets, err := GetMarkets()
  1065. if err != nil {
  1066. return nil, err
  1067. }
  1068. engine := db.GetEngine()
  1069. marketIDs := "" // 我的订单包括一二级市场卖方向成交单
  1070. // 默认取 TradeMode = 16
  1071. for _, v := range markets {
  1072. if v.Trademode == 16 || v.Trademode == 71 {
  1073. if len(marketIDs) == 0 {
  1074. marketIDs = strconv.Itoa(int(v.Marketid))
  1075. } else {
  1076. marketIDs += "," + strconv.Itoa(int(v.Marketid))
  1077. }
  1078. }
  1079. }
  1080. orders := make([]HsbyBuyMyPayOrder, 0)
  1081. session := engine.Table("TRADE_PAYORDER TP").
  1082. Select(`to_char(TP.TRADEID) TRADEIDSTR, to_char(TP.BUYORDERID) BUYORDERIDSTR, to_char(TP.SELLORDERID) SELLORDERIDSTR, TP.*,
  1083. GOODS.GOODSCODE, GOODS.GOODSNAME, GOODS.DECIMALPLACE, GOODS.AGREEUNIT,
  1084. HG1.PICURLS PICURLS1, HG2.PICURLS PICURLS2,
  1085. MARKET.TRADEMODE,
  1086. ENUMDICITEM.PARAM2 CURRENCYSIGN,
  1087. HS1.VENDORNAME VENDORNAME1, HS2.VENDORNAME VENDORNAME2`).
  1088. Join("LEFT", "GOODS", "GOODS.GOODSID = TP.GOODSID").
  1089. Join("LEFT", "CPTRADE_PRESALEGOODSEX CP", "CP.GOODSID = TP.GOODSID").
  1090. Join("LEFT", "HSBY_GOODSEX HG1", "HG1.GOODSID = CP.RELATEDGOODSID").
  1091. Join("LEFT", "HSBY_GOODSEX HG2", "HG2.GOODSID = GOODS.GOODSID").
  1092. Join("LEFT", "ENUMDICITEM", "GOODS.CURRENCYID = ENUMDICITEM.ENUMITEMNAME and ENUMDICITEM.ENUMDICCODE = 'currency'").
  1093. Join("LEFT", "MARKET", "MARKET.MARKETID = TP.MARKETID").
  1094. Join("LEFT", "HSBY_SUPPLIERINFO HS1", "HS1.VENDORID = HG1.VENDORID").
  1095. Join("LEFT", "HSBY_SUPPLIERINFO HS2", "HS2.VENDORID = HG2.VENDORID").
  1096. Where(fmt.Sprintf(`TP.PAYFLAG = 1 and TP.BUYACCOUNTID in (%s)`, accountIDs)).
  1097. And(fmt.Sprintf("TP.MARKETID in (%s)", marketIDs))
  1098. if buyOrderID != 0 {
  1099. session = session.And("TP.BUYORDERID = ?", buyOrderID)
  1100. }
  1101. if sellOrderID != 0 {
  1102. session = session.And("TP.SELLORDERID = ?", sellOrderID)
  1103. }
  1104. if err := session.Find(&orders); err != nil {
  1105. return nil, err
  1106. }
  1107. return orders, nil
  1108. }
  1109. // HsbyMarketInfo 海商报业相关市场信息
  1110. type HsbyMarketInfo struct {
  1111. Marketid int32 `json:"marketid" xorm:"'MARKETID'" binding:"required"` // 市场ID正常5位,前三位固定:两位表示交易模式, 一位表示交易属性(1:收益权,2:所有权) 其它特殊市场:0-系统 1-交割服务 2-账户服务3-履约服务 4-仓单服务 5-积分服务 6-银行服务
  1112. Marketname string `json:"marketname" xorm:"'MARKETNAME'"` // 市场名称
  1113. Trademode int32 `json:"trademode" xorm:"'TRADEMODE'" binding:"required"` // 交易模式 - 10:做市 13:竞价 15:通道交易 16:挂牌点选 17:仓单贸易 18:期权 19:竞拍-降价式 20:竞拍-竞价式 21:竞拍-大宗式 22:受托竞价
  1114. Marketstatus int32 `json:"marketstatus" xorm:"'MARKETSTATUS'" binding:"required"` // 生效状态(ValidStatus枚举): 1:待生效 2:正常 3:注销
  1115. }
  1116. // GetHsbyMarketInfos 获取海商报业相关市场信息
  1117. func GetHsbyMarketInfos() ([]HsbyMarketInfo, error) {
  1118. engine := db.GetEngine()
  1119. markets := make([]HsbyMarketInfo, 0)
  1120. if err := engine.Table("MARKET").Where("TRADEMODE in (71,16,70)").Find(&markets); err != nil {
  1121. return nil, err
  1122. }
  1123. return markets, nil
  1124. }
  1125. // HsbySellCollectionOrder 我的闲置中收款信息
  1126. type HsbySellCollectionOrder struct {
  1127. Tradeid string `json:"tradeid" xorm:"'TRADEIDSTR'" binding:"required"` // 成交单号(101+Unix秒时间戳(10位)+2位(MarketServiceID)+xxxx)
  1128. Tradedate string `json:"tradedate" xorm:"'TRADEDATE'"` // 交易日(yyyyMMdd)
  1129. Marketid int32 `json:"marketid" xorm:"'MARKETID'"` // 市场ID
  1130. Goodsid int32 `json:"goodsid" xorm:"'GOODSID'"` // 商品ID
  1131. Buyorderid string `json:"buyorderid" xorm:"'BUYORDERIDSTR'"` // 买方委托单号
  1132. Buyaccountid int64 `json:"buyaccountid" xorm:"'BUYACCOUNTID'"` // 买方账号ID[报价币种]
  1133. Sellorderid string `json:"sellorderid" xorm:"'SELLORDERIDSTR'"` // 卖方委托单号
  1134. Sellaccountid int64 `json:"sellaccountid" xorm:"'SELLACCOUNTID'"` // 卖方账号ID[报价币种]
  1135. Tradeamount float64 `json:"tradeamount" xorm:"'TRADEAMOUNT'"` // 成交金额
  1136. Tradecharge float64 `json:"tradecharge" xorm:"'TRADECHARGE'"` // 成交手续费(买方)
  1137. Payflag int32 `json:"payflag" xorm:"'PAYFLAG'"` // 付款标识 - 1:未支付 2:已支付 3:已过期 4:已撤销 5:结算过期 6:预售终止
  1138. Createtime time.Time `json:"createtime" xorm:"'CREATETIME'"` // 创建时间
  1139. Paytime time.Time `json:"paytime" xorm:"'PAYTIME'"` // 付款时间
  1140. Paylimitedtime time.Time `json:"paylimitedtime" xorm:"'PAYLIMITEDTIME'"` // 支付期限
  1141. Offamount float64 `json:"offamount" xorm:"'OFFAMOUNT'"` // 优惠金额
  1142. Tradeprice float64 `json:"tradeprice" xorm:"'TRADEPRICE'"` // 成交价格
  1143. Tradeqty int64 `json:"tradeqty" xorm:"'TRADEQTY'"` // 成交数量
  1144. Payamount float64 `json:"payamount" xorm:"'PAYAMOUNT'"` // 支付金额 = TradeAmount + TradeCharge - OffAmount
  1145. Goodscode string `json:"goodscode" xorm:"'GOODSCODE'" binding:"required"` // 商品代码(内部)
  1146. Goodsname string `json:"goodsname" xorm:"'GOODSNAME'" binding:"required"` // 商品名称
  1147. Decimalplace int64 `json:"decimalplace" xorm:"'DECIMALPLACE'"` // 报价小数位
  1148. Agreeunit float64 `json:"agreeunit" xorm:"'AGREEUNIT'"` // 合约单位
  1149. Picurls string `json:"picurls" xorm:"'PICURLS'"` // 商品介绍图片[多张用逗号分隔]
  1150. Trademode uint32 `json:"trademode" xorm:"'TRADEMODE'" binding:"required"` // 交易模式 - 10:做市 13:竞价 15:通道交易 16:挂牌点选 17:仓单贸易 18:期权 19:竞拍-降价式 20:竞拍-竞价式 21:竞拍-大宗式 22:受托竞价
  1151. Currencysign string `json:"currencysign" xorm:"'CURRENCYSIGN'"` // 货币符号
  1152. }
  1153. // GetHsbySellCollectionOrders 获取我的闲置中待收款信息
  1154. func GetHsbySellCollectionOrders(accountIDs string) ([]HsbySellCollectionOrder, error) {
  1155. // 获取市场信息
  1156. markets, err := GetMarkets()
  1157. if err != nil {
  1158. return nil, err
  1159. }
  1160. engine := db.GetEngine()
  1161. marketIDs := ""
  1162. for _, v := range markets {
  1163. if v.Trademode == 71 || v.Trademode == 70 {
  1164. if len(marketIDs) == 0 {
  1165. marketIDs = strconv.Itoa(int(v.Marketid))
  1166. } else {
  1167. marketIDs += "," + strconv.Itoa(int(v.Marketid))
  1168. }
  1169. }
  1170. }
  1171. orders := make([]HsbySellCollectionOrder, 0)
  1172. session := engine.Table("TRADE_PAYORDER TP").
  1173. Select(`to_char(TP.TRADEID) TRADEIDSTR, to_char(TP.BUYORDERID) BUYORDERIDSTR, to_char(TP.SELLORDERID) SELLORDERIDSTR, TP.*,
  1174. G.GOODSCODE, G.GOODSNAME, G.DECIMALPLACE, G.AGREEUNIT,
  1175. HG.PICURLS,
  1176. M.TRADEMODE,
  1177. E.PARAM2 CURRENCYSIGN`).
  1178. Join("LEFT", "GOODS G", "G.GOODSID = TP.GOODSID").
  1179. Join("LEFT", "HSBY_GOODSEX HG", "HG.GOODSID = G.GOODSID").
  1180. Join("LEFT", "ENUMDICITEM E", "G.CURRENCYID = E.ENUMITEMNAME and E.ENUMDICCODE = 'currency'").
  1181. Join("LEFT", "MARKET M", "M.MARKETID = TP.MARKETID").
  1182. Join("LEFT", "HSBY_SUPPLIERINFO HS", "HS.VENDORID = HG.VENDORID").
  1183. Where(fmt.Sprintf(`TP.PAYFLAG = 1 and TP.SELLACCOUNTID in (%s)`, accountIDs)).
  1184. And(fmt.Sprintf("TP.MARKETID in (%s)", marketIDs))
  1185. if err := session.Find(&orders); err != nil {
  1186. return nil, err
  1187. }
  1188. return orders, nil
  1189. }
  1190. // HsbyMarketGoods 特卖商城商品(三级市场商场)
  1191. type HsbyMarketGoods struct {
  1192. Orderid string `json:"orderid" xorm:"'ORDERIDSTR'" binding:"required"` // 委托单号(100+Unix秒时间戳(10位)+2位(MarketServiceID)+xxxx)
  1193. Marketid int32 `json:"marketid" xorm:"'MARKETID'" binding:"required"` // 市场ID
  1194. Goodsid int32 `json:"goodsid" xorm:"'GOODSID'" binding:"required"` // 商品ID
  1195. Accountid int64 `json:"accountid" xorm:"'ACCOUNTID'" binding:"required"` // 账户ID[报价币种]
  1196. Buyorsell int32 `json:"buyorsell" xorm:"'BUYORSELL'" binding:"required"` // 买卖 - 0:买 1:卖
  1197. Orderprice float64 `json:"orderprice" xorm:"'ORDERPRICE'"` // 委托价格
  1198. Orderstatus int32 `json:"orderstatus" xorm:"'ORDERSTATUS'"` // 委托状态 - 1: 委托请求 2:待冻结 3:委托成功 4: 委托失败 5:配对成功 6: 已撤销 7:部分成交 8:已成交 9:部成部撤 10:成交失败 11:已拒绝 12:经过摘牌(先摘后挂专用-先摘后挂已摘过) 13:冻结成功(通道交易专用) 14:通道已撤 15:通道部成部撤 16:成交失败违约(荷兰式竞拍专用)
  1199. Goodscode string `json:"goodscode" xorm:"'GOODSCODE'" binding:"required"` // 商品代码(内部)
  1200. Goodsname string `json:"goodsname" xorm:"'GOODSNAME'" binding:"required"` // 商品名称
  1201. Decimalplace int64 `json:"decimalplace" xorm:"'DECIMALPLACE'"` // 报价小数位
  1202. Quoteminunit int64 `json:"quoteminunit" xorm:"'QUOTEMINUNIT'"` // 行情最小变动单位 [整数,报价小数位一起使用]
  1203. Agreeunit float64 `json:"agreeunit" xorm:"'AGREEUNIT'"` // 合约单位
  1204. Hotindex int32 `json:"hotindex" xorm:"'HOTINDEX'"` // 景点热度
  1205. Videourls string `json:"videourls" xorm:"'VIDEOURLS'"` // 介绍视频[多张用逗号分隔]
  1206. Picurls string `json:"picurls" xorm:"'PICURLS'"` // 介绍图片[多张用逗号分隔]
  1207. Categoryid int32 `json:"categoryid" xorm:"'CATEGORYID'"` // 类别ID(WRCATEGORY)
  1208. Currency string `json:"currency" xorm:"'CURRENCY'"` // 货币
  1209. Currencysign string `json:"currencysign" xorm:"'CURRENCYSIGN'"` // 货币符号
  1210. Trademode int32 `json:"trademode" xorm:"'TRADEMODE'" binding:"required"` // 交易模式 - 10:做市 13:竞价 15:通道交易 16:挂牌点选 17:仓单贸易 18:期权 19:竞拍-降价式 20:竞拍-竞价式 21:竞拍-大宗式 22:受托竞价
  1211. Customername string `json:"customername" xorm:"'CUSTOMERNAME'"` // 卖家名称
  1212. HasCoupon bool `json:"hascoupon" xorm:"-"` // 是否可用优惠卷
  1213. SellUserID int `json:"sellUserID" xorm:"-"` // 卖方UserID
  1214. }
  1215. // GetHsbyMarketGoodses 获取商城商品列表(三级商城)
  1216. func GetHsbyMarketGoodses(marketIDs string, accountID, categoryID, goodsID int) ([]HsbyMarketGoods, error) {
  1217. engine := db.GetEngine()
  1218. orders := make([]HsbyMarketGoods, 0)
  1219. // 与挂牌点选不一样的是商城是以委托单为主表;已经卖完的委托单不显示
  1220. session := engine.Table("TRADE_ORDERDETAIL T").
  1221. Select(`to_char(T.ORDERID) ORDERIDSTR, T.*,
  1222. G.GOODSCODE, G.GOODSNAME, G.DECIMALPLACE, G.QUOTEMINUNIT, G.AGREEUNIT,
  1223. GX.HOTINDEX, GX.PICURLS, GX.CATEGORYID, GX.GOODSDESC,
  1224. ENUMDICITEM.ENUMDICNAME CURRENCY, ENUMDICITEM.PARAM2 CURRENCYSIGN,
  1225. M.TRADEMODE`).
  1226. Join("LEFT", "GOODS G", "G.GOODSID = T.GOODSID").
  1227. Join("LEFT", "HSBY_GOODSEX GX", "GX.GOODSID = T.GOODSID").
  1228. Join("LEFT", "ENUMDICITEM E", "E.ENUMITEMNAME = G.CURRENCYID and E.ENUMDICCODE = 'currency'").
  1229. Join("LEFT", "MARKET M", "M.MARKETID = T.MARKETID").
  1230. Where(fmt.Sprintf("T.MARKETID in (%s)", marketIDs)).
  1231. And("T.ORDERSTATUS in (3,7) and T.BUYORSELL = 1 and (T.ORDERQTY - T.TRADEQTY - T.CANCELQTY) > 0")
  1232. if categoryID != 0 {
  1233. session = session.And("GX.CATEGORYID = ?", categoryID)
  1234. }
  1235. if goodsID != 0 {
  1236. session = session.And("G.GOODSID = ?", goodsID)
  1237. }
  1238. if err := session.Find(&orders); err != nil {
  1239. return nil, err
  1240. }
  1241. // 获取当前账户的优惠卷
  1242. type myCouponPosition struct {
  1243. Accountid int64 `json:"accountid" xorm:"'ACCOUNTID'" binding:"required"` // 资金账户ID
  1244. Coupontypeid int32 `json:"coupontypeid" xorm:"'COUPONTYPEID'" binding:"required"` // 优惠券类型ID
  1245. Curqty int64 `json:"curqty" xorm:"'CURQTY'"` // 期末数量
  1246. Curfreezeqty int64 `json:"curfreezeqty" xorm:"'CURFREEZEQTY'"` // 期末冻结数量
  1247. Limitedflag int32 `json:"limitedflag" xorm:"'LIMITEDFLAG'"` // 是否指定商品 - 0:不限 1:限制
  1248. Limitedgoodsids string `json:"limitedgoodsids" xorm:"'LIMITEDGOODSIDS'"` // 指定商品IDs[逗号分隔,前后加逗号]
  1249. Isgeneral int32 `json:"isgeneral" xorm:"'ISGENERAL'"` // 是否通用券 - 0:否 1:是
  1250. Userscope string `json:"userscope" xorm:"'USERSCOPE'"` // 卖家范围(用户ID,逗号分隔,前后加逗号) [IsGeneral =0时使用]
  1251. }
  1252. myCouponPositions := make([]myCouponPosition, 0)
  1253. if accountID != 0 {
  1254. if err := engine.Table("COUPONPOSITION C").
  1255. Select(`C.ACCOUNTID, C.COUPONTYPEID, C.CURQTY, C.CURFREEZEQTY,
  1256. CT.LIMITEDFLAG, CT.LIMITEDGOODSIDS, CT.ISGENERAL, CT.USERSCOPE`).
  1257. Join("INNER", "COUPONTYPE CT", "CT.COUPONTYPEID = C.COUPONTYPEID").
  1258. Where("C.ACCOUNTID = ?", accountID).Find(&myCouponPositions); err != nil {
  1259. return nil, err
  1260. }
  1261. }
  1262. // 是否有优惠卷可用
  1263. if len(orders) > 0 && len(myCouponPositions) > 0 {
  1264. for _, coupon := range myCouponPositions {
  1265. if coupon.Limitedflag == 0 && coupon.Isgeneral == 1 {
  1266. // 有不指定商品的通用卷则所有委托单都可以用卷
  1267. for i := range orders {
  1268. order := &orders[i]
  1269. order.HasCoupon = true
  1270. }
  1271. break
  1272. } else {
  1273. for i := range orders {
  1274. order := &orders[i]
  1275. // 获取卖家UserID
  1276. userInfo, err := GetUserInfoByAccountID(int(order.Accountid))
  1277. if err != nil {
  1278. continue
  1279. }
  1280. order.SellUserID = int(userInfo.Userid)
  1281. order.Customername = userInfo.Customername
  1282. tmpGoodsID := fmt.Sprintf(",%d,", order.Goodsid)
  1283. tmpUserID := fmt.Sprintf(",%d,", userInfo.Userid)
  1284. // 优惠卷是否可用于商品
  1285. isRightGoods := false
  1286. if coupon.Limitedflag == 0 {
  1287. isRightGoods = true
  1288. } else {
  1289. if strings.Contains(coupon.Limitedgoodsids, tmpGoodsID) {
  1290. isRightGoods = true
  1291. }
  1292. }
  1293. // 优惠卷是否可用于卖家
  1294. isRightAccount := false
  1295. if coupon.Isgeneral == 1 {
  1296. isRightAccount = true
  1297. } else {
  1298. if strings.Contains(coupon.Userscope, tmpUserID) {
  1299. isRightAccount = true
  1300. }
  1301. }
  1302. if isRightGoods && isRightAccount {
  1303. order.HasCoupon = true
  1304. }
  1305. }
  1306. }
  1307. }
  1308. }
  1309. return orders, nil
  1310. }
  1311. // HsbyMarketGoodsDetail 特卖商城商品(三级市场商场)
  1312. type HsbyMarketGoodsDetail struct {
  1313. Orderid string `json:"orderid" xorm:"'ORDERIDSTR'" binding:"required"` // 委托单号(100+Unix秒时间戳(10位)+2位(MarketServiceID)+xxxx)
  1314. Marketid int32 `json:"marketid" xorm:"'MARKETID'" binding:"required"` // 市场ID
  1315. Goodsid int32 `json:"goodsid" xorm:"'GOODSID'" binding:"required"` // 商品ID
  1316. Accountid int64 `json:"accountid" xorm:"'ACCOUNTID'" binding:"required"` // 账户ID[报价币种]
  1317. Buyorsell int32 `json:"buyorsell" xorm:"'BUYORSELL'" binding:"required"` // 买卖 - 0:买 1:卖
  1318. Orderprice float64 `json:"orderprice" xorm:"'ORDERPRICE'"` // 委托价格
  1319. Orderqty int64 `json:"orderqty" xorm:"'ORDERQTY'" binding:"required"` // 委托数量
  1320. Tradeqty int64 `json:"tradeqty" xorm:"'TRADEQTY'"` // 成交数量
  1321. Cancelqty int64 `json:"cancelqty" xorm:"'CANCELQTY'"` // 撤单数量
  1322. Orderstatus int32 `json:"orderstatus" xorm:"'ORDERSTATUS'"` // 委托状态 - 1: 委托请求 2:待冻结 3:委托成功 4: 委托失败 5:配对成功 6: 已撤销 7:部分成交 8:已成交 9:部成部撤 10:成交失败 11:已拒绝 12:经过摘牌(先摘后挂专用-先摘后挂已摘过) 13:冻结成功(通道交易专用) 14:通道已撤 15:通道部成部撤 16:成交失败违约(荷兰式竞拍专用)
  1323. Goodscode string `json:"goodscode" xorm:"'GOODSCODE'" binding:"required"` // 商品代码(内部)
  1324. Goodsname string `json:"goodsname" xorm:"'GOODSNAME'" binding:"required"` // 商品名称
  1325. Decimalplace int64 `json:"decimalplace" xorm:"'DECIMALPLACE'"` // 报价小数位
  1326. Quoteminunit int64 `json:"quoteminunit" xorm:"'QUOTEMINUNIT'"` // 行情最小变动单位 [整数,报价小数位一起使用]
  1327. Agreeunit float64 `json:"agreeunit" xorm:"'AGREEUNIT'"` // 合约单位
  1328. Hotindex int32 `json:"hotindex" xorm:"'HOTINDEX'"` // 景点热度
  1329. Videourls string `json:"videourls" xorm:"'VIDEOURLS'"` // 介绍视频[多张用逗号分隔]
  1330. Picurls string `json:"picurls" xorm:"'PICURLS'"` // 介绍图片[多张用逗号分隔]
  1331. Categoryid int32 `json:"categoryid" xorm:"'CATEGORYID'"` // 类别ID(WRCATEGORY)
  1332. Goodsdesc string `json:"goodsdesc" xorm:"'GOODSDESC'"` // 商品详情
  1333. Currency string `json:"currency" xorm:"'CURRENCY'"` // 货币
  1334. Currencysign string `json:"currencysign" xorm:"'CURRENCYSIGN'"` // 货币符号
  1335. Trademode int32 `json:"trademode" xorm:"'TRADEMODE'" binding:"required"` // 交易模式 - 10:做市 13:竞价 15:通道交易 16:挂牌点选 17:仓单贸易 18:期权 19:竞拍-降价式 20:竞拍-竞价式 21:竞拍-大宗式 22:受托竞价
  1336. Vendorname string `json:"vendorname" xorm:"'VENDORNAME'"` // 供应商名称
  1337. Customername string `json:"customername" xorm:"'CUSTOMERNAME'"` // 卖家名称
  1338. SellUserID int `json:"sellUserID" xorm:"SELLUSERID"` // 卖方UserID
  1339. }
  1340. // GetHsbyMarketGoodsDetail 获取商城商品详情(三级商城)
  1341. func GetHsbyMarketGoodsDetail(orderID int) (*HsbyMarketGoodsDetail, error) {
  1342. engine := db.GetEngine()
  1343. orders := make([]HsbyMarketGoodsDetail, 0)
  1344. // 与挂牌点选不一样的是商城是以委托单为主表
  1345. session := engine.Table("TRADE_ORDERDETAIL T").
  1346. Select(`to_char(T.ORDERID) ORDERIDSTR, T.*,
  1347. G.GOODSCODE, G.GOODSNAME, G.DECIMALPLACE, G.QUOTEMINUNIT, G.AGREEUNIT,
  1348. GX.HOTINDEX, GX.VIDEOURLS, GX.PICURLS, GX.CATEGORYID, GX.GOODSDESC,
  1349. ENUMDICITEM.ENUMDICNAME CURRENCY, ENUMDICITEM.PARAM2 CURRENCYSIGN,
  1350. M.TRADEMODE,
  1351. H.VENDORNAME,
  1352. U.USERID SELLUSERID, U.CUSTOMERNAME`).
  1353. Join("LEFT", "GOODS G", "G.GOODSID = T.GOODSID").
  1354. Join("LEFT", "HSBY_GOODSEX GX", "GX.GOODSID = T.GOODSID").
  1355. Join("LEFT", "ENUMDICITEM E", "E.ENUMITEMNAME = G.CURRENCYID and E.ENUMDICCODE = 'currency'").
  1356. Join("LEFT", "MARKET M", "M.MARKETID = T.MARKETID").
  1357. Join("LEFT", "HSBY_SUPPLIERINFO H", "H.VENDORID = GX.VENDORID").
  1358. Join("LEFT", "TAACCOUNT TA", "TA.ACCOUNTID = T.ACCOUNTID").
  1359. Join("LEFT", "USERINFO U", "U.USERID = TA.RELATEDUSERID").
  1360. Where("T.ORDERID = ?", orderID)
  1361. if err := session.Find(&orders); err != nil {
  1362. return nil, err
  1363. }
  1364. // 无目标商品
  1365. if len(orders) == 0 {
  1366. return nil, nil
  1367. }
  1368. hsbyMarketGoodsDetail := orders[0]
  1369. return &hsbyMarketGoodsDetail, nil
  1370. }
  1371. // MyCoupon 我的优惠卷
  1372. type MyCoupon struct {
  1373. Accountid int64 `json:"accountid" xorm:"'ACCOUNTID'" binding:"required"` // 资金账户ID
  1374. Coupontypeid int32 `json:"coupontypeid" xorm:"'COUPONTYPEID'" binding:"required"` // 优惠券类型ID
  1375. Userid int64 `json:"userid" xorm:"'USERID'"` // 用户ID
  1376. Oriqty int64 `json:"oriqty" xorm:"'ORIQTY'"` // 期初数量
  1377. Orifreezeqty int64 `json:"orifreezeqty" xorm:"'ORIFREEZEQTY'"` // 期初冻结数量
  1378. Curqty int64 `json:"curqty" xorm:"'CURQTY'"` // 期末数量
  1379. Curfreezeqty int64 `json:"curfreezeqty" xorm:"'CURFREEZEQTY'"` // 期末冻结数量
  1380. Todayincrease int64 `json:"todayincrease" xorm:"'TODAYINCREASE'"` // 今日增加
  1381. Todaydecrease int64 `json:"todaydecrease" xorm:"'TODAYDECREASE'"` // 今日减少
  1382. Areauserid int64 `json:"areauserid" xorm:"'AREAUSERID'"` // 所属机构
  1383. Couponname string `json:"couponname" xorm:"'COUPONNAME'"` // 优惠券名称
  1384. Couponcategroy int32 `json:"couponcategroy" xorm:"'COUPONCATEGROY'"` // 种类 - 1:现金券 2:折扣券 3:折扣券(单张)
  1385. Conditionvalue float64 `json:"conditionvalue" xorm:"'CONDITIONVALUE'"` // 条件阈值(可为0)
  1386. Couponvalue float64 `json:"couponvalue" xorm:"'COUPONVALUE'"` // 面值[1:现金券 - 抵扣值 2:折扣券-折扣值]
  1387. Limitedflag int32 `json:"limitedflag" xorm:"'LIMITEDFLAG'"` // 是否指定商品 - 0:不限 1:限制
  1388. Limitedgoodsids string `json:"limitedgoodsids" xorm:"'LIMITEDGOODSIDS'"` // 指定商品IDs[逗号分隔,前后加逗号]
  1389. Isgeneral int32 `json:"isgeneral" xorm:"'ISGENERAL'"` // 是否通用券 - 0:否 1:是
  1390. Userscope string `json:"userscope" xorm:"'USERSCOPE'"` // 卖家范围(用户ID,逗号分隔,前后加逗号) [IsGeneral =0时使用]
  1391. IsUnusable bool `json:"isunusable" xorm:"-"` // 是否不可用
  1392. ReasonType int `json:"reasontype"` // 不可用原因 - 0:未确认 1:不可用于此商品 2:不可用于此卖家
  1393. }
  1394. // GetMyCoupons 获取我的优惠卷信息
  1395. func GetMyCoupons(accountIDs string, goodsID, sellUserID int) ([]MyCoupon, error) {
  1396. engine := db.GetEngine()
  1397. myCoupons := make([]MyCoupon, 0)
  1398. session := engine.Table("COUPONPOSITION CP").
  1399. Join("INNER", "COUPONTYPE CT", "CT.COUPONTYPEID = CP.COUPONTYPEID").
  1400. Where(fmt.Sprintf("CP.ACCOUNTID in (%s)", accountIDs))
  1401. if err := session.Find(&myCoupons); err != nil {
  1402. return nil, err
  1403. }
  1404. // 如果传入了商品ID和卖方UserID,则判断优惠卷是否可用
  1405. if goodsID != 0 && sellUserID != 0 && len(myCoupons) > 0 {
  1406. for i := range myCoupons {
  1407. myCoupon := &myCoupons[i]
  1408. // 优惠卷是否可用于商品
  1409. isRightGoods := false
  1410. if myCoupon.Limitedflag == 0 {
  1411. isRightGoods = true
  1412. } else {
  1413. if strings.Contains(myCoupon.Limitedgoodsids, fmt.Sprintf(",%d,", goodsID)) {
  1414. isRightGoods = true
  1415. }
  1416. }
  1417. // 优惠卷是否可用于卖家
  1418. isRightAccount := false
  1419. if myCoupon.Isgeneral == 1 {
  1420. isRightAccount = true
  1421. } else {
  1422. if strings.Contains(myCoupon.Userscope, fmt.Sprintf(",%d,", sellUserID)) {
  1423. isRightAccount = true
  1424. }
  1425. }
  1426. if !isRightGoods {
  1427. myCoupon.IsUnusable = true
  1428. myCoupon.ReasonType = 1
  1429. }
  1430. if !isRightAccount {
  1431. myCoupon.IsUnusable = true
  1432. myCoupon.ReasonType = 2
  1433. }
  1434. }
  1435. }
  1436. return myCoupons, nil
  1437. }