hsby.go 100 KB

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