table.go 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382
  1. package common
  2. import (
  3. "fmt"
  4. "mtp2_if/db"
  5. "mtp2_if/global/app"
  6. "mtp2_if/global/e"
  7. "mtp2_if/global/utils"
  8. "mtp2_if/logger"
  9. "mtp2_if/models"
  10. "net/http"
  11. "strconv"
  12. "strings"
  13. "github.com/gin-gonic/gin"
  14. )
  15. // QueryTraderMenuReq 查询交易端菜单请求参数
  16. type QueryTraderMenuReq struct {
  17. LoginID int `form:"loginid" binding:"required"`
  18. }
  19. // QueryTraderMenuRsp 交易端菜单
  20. type QueryTraderMenuRsp struct {
  21. QuoteMenu []QuotePrimaryMenu `json:"QuoteMenu"` // 报价牌分类菜单
  22. OperationMenu []OperationPrimaryMenu `json:"OperationMenu"` // 功能菜单
  23. }
  24. // QuotePrimaryMenu 报价牌一级分类菜单
  25. type QuotePrimaryMenu struct {
  26. Index int `json:"Index"` // 序号
  27. Key string `json:"Key"` // 键名
  28. Name string `json:"Name"` // 菜单名称
  29. SubTitleType int `json:"SubTitleType"` // 子菜单标题模式:0-市场名称;1-外部交易所名称
  30. TradeModes string `json:"TradeModes"` // 包含市场交易类型
  31. SubMenus []QuoteSecondaryMenu `json:"SubMenus"` // 子菜单
  32. }
  33. // QuoteSecondaryMenu 报价牌二级分类菜单
  34. type QuoteSecondaryMenu struct {
  35. Index int `json:"Index"` // 序号
  36. MarketID int `json:"MarketID"` // 市场ID
  37. TradeMode int `json:"TradeMode"` // 交易模式
  38. MenuTitle string `json:"MenuTitle" xorm:"'ExExchangeName'"` // 菜单标题(市场名称或外部交易所名称)
  39. GoodsGroupIDs []int `json:"GoodsGroupIDs"` // 商品组ID列表
  40. ExExchangeID int `json:"ExExchangeID" xorm:"'ExExchangeID'"` // 外部交易所ID
  41. ExExchangeCode string `json:"ExExchangeCode" xorm:"'ExExchangeCode'"` // 外部交易所代码
  42. }
  43. // OperationPrimaryMenu 一级功能菜单
  44. type OperationPrimaryMenu struct {
  45. Key string `json:"Key"` // 菜单KEY
  46. Label string `json:"Label"` // 菜单标题
  47. Children []OperationSecondaryMenu `json:"Children"` // 二级功能菜单
  48. }
  49. // OperationSecondaryMenu 二级功能菜单
  50. type OperationSecondaryMenu struct {
  51. Key string `json:"Key"` // 菜单KEY
  52. Label string `json:"Label"` // 菜单标题
  53. TabList []OperationTabMenu `json:"TabList"` // 三级功能菜单
  54. }
  55. // OperationTabMenu 三级功能菜单
  56. type OperationTabMenu struct {
  57. Key string `json:"Key"` // 菜单KEY
  58. Label string `json:"Label"` // 菜单标题
  59. }
  60. // QueryTraderMenu 查询交易端菜单
  61. // @Summary 查询交易端菜单
  62. // @Produce json
  63. // @Param loginid query int true "登录账号"
  64. // @Success 200 {object} QueryTraderMenuRsp
  65. // @Failure 500 {object} app.Response
  66. // @Router /Common/QueryTraderMenu [get]
  67. // @Tags 通用服务
  68. func QueryTraderMenu(c *gin.Context) {
  69. appG := app.Gin{C: c}
  70. // 获取请求参数
  71. var req QueryTraderMenuReq
  72. if err := appG.C.ShouldBindQuery(&req); err != nil {
  73. logger.GetLogger().Errorf("QueryTraderMenu failed: %s", err.Error())
  74. appG.Response(http.StatusBadRequest, e.INVALID_PARAMS, nil)
  75. return
  76. }
  77. var queryTraderMenuRsp QueryTraderMenuRsp
  78. // 获取行情报价牌分类菜单
  79. if datas, err := getQuoteMenu(req.LoginID); err == nil {
  80. queryTraderMenuRsp.QuoteMenu = datas
  81. } else {
  82. logger.GetLogger().Errorf("QueryTraderMenu failed: %s", err.Error())
  83. appG.Response(http.StatusBadRequest, e.ERROR_QUERY_QUOTEMENU_FAIL, nil)
  84. return
  85. }
  86. // 获取功能菜单
  87. if datas, err := getOperationMenu(); err == nil {
  88. queryTraderMenuRsp.OperationMenu = datas
  89. } else {
  90. logger.GetLogger().Errorf("QueryTraderMenu failed: %s", err.Error())
  91. appG.Response(http.StatusBadRequest, e.ERROR_QUERY_OPERATIONMENU_FAIL, nil)
  92. return
  93. }
  94. // 查询成功
  95. logger.GetLogger().Infof("QueryTraderMenu successed: %v", queryTraderMenuRsp)
  96. appG.Response(http.StatusOK, e.SUCCESS, queryTraderMenuRsp)
  97. }
  98. // getQuoteMenu 获取行情报价牌分类菜单
  99. func getQuoteMenu(loginID int) ([]QuotePrimaryMenu, error) {
  100. engine := db.GetEngine()
  101. rst := make([]QuotePrimaryMenu, 0)
  102. // 账户下有权限的市场ID列表
  103. var marketIDs []int
  104. // 获取账户类型 - 1:交易所 2:机构 3:会员子机构 4:经纪人 5:投资者 6:客户 (目前可能登录交易端的账号类型为 2 5)
  105. userAccount := new(models.Useraccount)
  106. has, err := engine.Join("INNER", "LOGINACCOUNT", "USERACCOUNT.UserID = LOGINACCOUNT.UserID").Where("LOGINACCOUNT.LoginID = ?", loginID).Get(userAccount)
  107. if err != nil || !has {
  108. return nil, err
  109. }
  110. if userAccount.Usertype == 5 {
  111. // 如果账户类型为5(投资者),则需要通过其所属经济会员来获取市场权限(表:AreaRoleMarket, 条件:对应市场状态为正常;角色类型:经济会员)
  112. if err := engine.Table("AREAROLEMARKET").
  113. Cols("AREAROLEMARKET.MARKETID").
  114. Join("INNER", "MARKET", "MARKET.MARKETID = AREAROLEMARKET.MARKETID").
  115. Where("MARKET.MARKETSTATUS = 2 and AREAROLEMARKET.ROLETYPE = 7 and AREAROLEMARKET.AREAUSERID = ?", userAccount.Memberuserid).Find(&marketIDs); err != nil {
  116. return nil, err
  117. }
  118. } else {
  119. // 非投资者账号直接通过资金账号获取市场权限(表:TAAccountMarket)
  120. var taAccounts []string // 账户下所有资金账户
  121. // 先要获取当前登录账户对应的资金账户;如果为外部子资金账号(TaAccount.TaAccountType = 1),则使用TaAccount.FromAccountID来获取市场权限
  122. type taAccount struct {
  123. AccountID int `xorm:"ACCOUNTID"`
  124. TaAccountType int `xorm:"TAACCOUNTTYPE"`
  125. FromAccountID int `xorm:"FROMACCOUNTID"`
  126. }
  127. datas := make([]taAccount, 0)
  128. if err := engine.Table("LOGINTAACCOUNT").
  129. Join("INNER", "TAACCOUNT", "LOGINTAACCOUNT.ACCOUNTID = TAACCOUNT.ACCOUNTID").
  130. Cols("TAACCOUNT.ACCOUNTID", "TAACCOUNT.TAACCOUNTTYPE", "TAACCOUNT.FROMACCOUNTID").
  131. Where("LOGINTAACCOUNT.LOGINID = ?", loginID).Find(&datas); err != nil {
  132. return nil, err
  133. }
  134. // 如果一条记录都没有(未配置自营会员资金账户)则直接通过资金账户表获取
  135. if len(datas) == 0 {
  136. // 这里要注意,TaAccount表与LoginAccount表关联时,要使用TaAccount.RelatedUserID
  137. if err := engine.Table("TAACCOUNT").
  138. Join("INNER", "LOGINACCOUNT", "TAACCOUNT.RELATEDUSERID = LOGINACCOUNT.USERID").
  139. Cols("TAACCOUNT.ACCOUNTID", "TAACCOUNT.TAACCOUNTTYPE", "TAACCOUNT.FROMACCOUNTID").
  140. Where("LOGINACCOUNT.LOGINID = ?", loginID).Find(&datas); err != nil {
  141. return nil, err
  142. }
  143. }
  144. for _, v := range datas {
  145. if v.TaAccountType == 1 {
  146. // 外部资金账户使用TaAccount.FromAccountID来获取市场权限
  147. taAccounts = append(taAccounts, strconv.Itoa(v.FromAccountID))
  148. } else {
  149. taAccounts = append(taAccounts, strconv.Itoa(v.AccountID))
  150. }
  151. }
  152. // 获取资金账户对应的市场权限(有权限的市场ID)
  153. if len(taAccounts) > 0 {
  154. taAccountStr := strings.Join(taAccounts, ",")
  155. if err := engine.Table("TAACCOUNTMARKET").
  156. Join("INNER", "MARKET", "MARKET.MARKETID = TAACCOUNTMARKET.MARKETID").
  157. Cols("TAACCOUNTMARKET.MARKETID").
  158. In("TAACCOUNTMARKET.ACCOUNTID", taAccountStr).
  159. And("MARKET.MARKETSTATUS = 2").Find(&marketIDs); err != nil {
  160. return nil, err
  161. }
  162. }
  163. }
  164. // ********************* 构建行情报价牌菜单 *********************
  165. // 获取一级菜单列表
  166. // datas := make([]models.Tablecolumnconfig, 0)
  167. // if err := engine.Where("TableKey = 'trader_master_menu' and IsShow = 1").Find(&datas); err != nil {
  168. // return nil, err
  169. // }
  170. datas := make([]models.Funcmenulist, 0)
  171. if err := engine.Join("INNER", "ROLEFUNCMENU", "FUNCMENULIST.RESOURCECODE = ROLEFUNCMENU.RESOURCECODE").
  172. Where("FUNCMENULIST.PARENTCODE = 'trader_master_menu'").
  173. Asc("SORT").
  174. Find(&datas); err != nil {
  175. return nil, err
  176. }
  177. // 构建一级菜单对象
  178. for i, v := range datas {
  179. quotePrimaryMenu := QuotePrimaryMenu{
  180. Index: i,
  181. Key: v.Resourcecode,
  182. Name: v.Resourcename,
  183. SubMenus: make([]QuoteSecondaryMenu, 0),
  184. }
  185. // 跳过自选
  186. if v.Resourcecode == "optional" {
  187. rst = append(rst, quotePrimaryMenu)
  188. continue
  189. }
  190. // URL:模式,0-使用市场名称,1-使用外部交易所名称;Remark:包含市场交易模式
  191. quotePrimaryMenu.SubTitleType, _ = strconv.Atoi(v.URL)
  192. quotePrimaryMenu.TradeModes = v.Remark
  193. // 如果传入的LoginID获取不到有权限的市场ID列表(有可能LoginID是错误的),则不构建二级子菜单
  194. if len(marketIDs) == 0 {
  195. rst = append(rst, quotePrimaryMenu)
  196. continue
  197. }
  198. // 构建二级子菜单对象
  199. marketIDsStr := utils.JoinItoString(marketIDs, ",")
  200. if quotePrimaryMenu.SubTitleType == 0 {
  201. // 获取目标交易模式的市场信息
  202. markets := make([]models.Market, 0)
  203. if err := engine.Where(fmt.Sprintf(`TradeMode in (%s) and MarketID in (%s)`, quotePrimaryMenu.TradeModes, marketIDsStr)).Find(&markets); err != nil {
  204. return nil, err
  205. }
  206. // 使用市场名称
  207. for mi, mv := range markets {
  208. quoteSecondaryMenu := QuoteSecondaryMenu{
  209. Index: mi,
  210. MarketID: int(mv.Marketid),
  211. TradeMode: int(mv.Trademode),
  212. MenuTitle: mv.Marketname,
  213. GoodsGroupIDs: make([]int, 0),
  214. }
  215. quotePrimaryMenu.SubMenus = append(quotePrimaryMenu.SubMenus, quoteSecondaryMenu)
  216. }
  217. } else {
  218. // 使用外部交易所名称
  219. quoteSecondaryMenus := make([]QuoteSecondaryMenu, 0)
  220. sql := fmt.Sprintf(`select distinct
  221. e.autoid ExExchangeID,
  222. e.ExExchangeName,
  223. e.ExExchangeCode
  224. from ExternalExchange e
  225. inner join goodsgroup g on g.exexchangeid = e.autoid
  226. inner join Market m on g.marketid = m.marketid
  227. where m.trademode in (%s) and m.marketid in (%s)`, quotePrimaryMenu.TradeModes, marketIDsStr)
  228. if err := engine.SQL(sql).Find(&quoteSecondaryMenus); err != nil {
  229. return nil, err
  230. }
  231. // 获取外部交易所对应的商品组信息
  232. for ei, ev := range quoteSecondaryMenus {
  233. q := &quoteSecondaryMenus[ei]
  234. q.Index = ei
  235. // 商品组列表
  236. goodsgroups := make([]models.Goodsgroup, 0)
  237. if err := engine.Where("Exexchangeid = ?", ev.ExExchangeID).Find(&goodsgroups); err != nil {
  238. return nil, err
  239. }
  240. marketID := 0
  241. var goodsGroupIDs []int
  242. for _, gv := range goodsgroups {
  243. marketID = int(gv.Marketid)
  244. goodsGroupIDs = append(goodsGroupIDs, int(gv.Goodsgroupid))
  245. }
  246. q.MarketID = marketID
  247. q.GoodsGroupIDs = goodsGroupIDs
  248. }
  249. quotePrimaryMenu.SubMenus = quoteSecondaryMenus
  250. }
  251. rst = append(rst, quotePrimaryMenu)
  252. }
  253. return rst, nil
  254. }
  255. // getOperationMenu 获取功能菜单
  256. func getOperationMenu() ([]OperationPrimaryMenu, error) {
  257. engine := db.GetEngine()
  258. rst := make([]OperationPrimaryMenu, 0)
  259. // 获取一级功能菜单
  260. opm := make([]models.Funcmenulist, 0)
  261. if err := engine.Join("INNER", "ROLEFUNCMENU", "FUNCMENULIST.RESOURCECODE = ROLEFUNCMENU.RESOURCECODE").
  262. Where("FUNCMENULIST.PARENTCODE = 'trader_operation_master_menu'").
  263. Asc("SORT").
  264. Find(&opm); err != nil {
  265. return nil, err
  266. }
  267. for _, pv := range opm {
  268. var operationPrimaryMenu = OperationPrimaryMenu{
  269. Key: pv.Resourcecode,
  270. Label: pv.Resourcename,
  271. Children: make([]OperationSecondaryMenu, 0),
  272. }
  273. // 获取二级功能菜单
  274. osm := make([]models.Funcmenulist, 0)
  275. if err := engine.Join("INNER", "ROLEFUNCMENU", "FUNCMENULIST.RESOURCECODE = ROLEFUNCMENU.RESOURCECODE").
  276. Where("FUNCMENULIST.PARENTCODE = ?", operationPrimaryMenu.Key).
  277. Asc("SORT").
  278. Find(&osm); err != nil {
  279. return nil, err
  280. }
  281. for _, sv := range osm {
  282. var operationSecondaryMenu = OperationSecondaryMenu{
  283. Key: sv.Resourcecode,
  284. Label: sv.Resourcename,
  285. TabList: make([]OperationTabMenu, 0),
  286. }
  287. // 获取三级功能菜单
  288. otm := make([]models.Funcmenulist, 0)
  289. if err := engine.Join("INNER", "ROLEFUNCMENU", "FUNCMENULIST.RESOURCECODE = ROLEFUNCMENU.RESOURCECODE").
  290. Where("FUNCMENULIST.PARENTCODE = ?", operationSecondaryMenu.Key).
  291. Asc("SORT").
  292. Find(&otm); err != nil {
  293. return nil, err
  294. }
  295. for _, tv := range otm {
  296. var operationTabMenu = OperationTabMenu{
  297. Key: tv.Resourcecode,
  298. Label: tv.Resourcename,
  299. }
  300. operationSecondaryMenu.TabList = append(operationSecondaryMenu.TabList, operationTabMenu)
  301. }
  302. operationPrimaryMenu.Children = append(operationPrimaryMenu.Children, operationSecondaryMenu)
  303. }
  304. rst = append(rst, operationPrimaryMenu)
  305. }
  306. return rst, nil
  307. }
  308. // QueryTableColumnConfigReq 查询交易端列表头信息请求参数
  309. type QueryTableColumnConfigReq struct {
  310. TableKey string `json:"TableKey"` // 表key
  311. }
  312. // QueryTableColumnConfig 查询交易端列表头信息
  313. // @Summary 查询交易端列表头信息
  314. // @Produce json
  315. // @Param TableKey query string false "表key"
  316. // @Success 200 {object} models.Tablecolumnconfig
  317. // @Failure 500 {object} app.Response
  318. // @Router /Common/QueryTableColumnConfig [get]
  319. // @Tags 通用服务
  320. func QueryTableColumnConfig(c *gin.Context) {
  321. appG := app.Gin{C: c}
  322. // 获取请求参数
  323. var req QueryTableColumnConfigReq
  324. if err := appG.C.ShouldBindQuery(&req); err != nil {
  325. logger.GetLogger().Errorf("QueryTableColumnConfig failed: %s", err.Error())
  326. appG.Response(http.StatusBadRequest, e.INVALID_PARAMS, nil)
  327. return
  328. }
  329. engine := db.GetEngine()
  330. tablecolumnconfigs := make([]models.Tablecolumnconfig, 0)
  331. // 这里的表名必须要大写 (Oracle的表名都是大写)
  332. s := engine.Join("INNER", "TABLEDEFINE", "TABLEDEFINE.TableKey = TABLECOLUMNCONFIG.TableKey")
  333. if len(req.TableKey) > 0 {
  334. s = s.Where("TABLEDEFINE.TableType = 2 and TABLEDEFINE.TableKey = ?", req.TableKey)
  335. } else {
  336. s = s.Where("TABLEDEFINE.TableType = 2")
  337. }
  338. if err := s.Find(&tablecolumnconfigs); err != nil {
  339. logger.GetLogger().Errorf("QueryTableColumnConfig failed: %s", err.Error())
  340. appG.Response(http.StatusBadRequest, e.ERROR_QUERY_FAIL, nil)
  341. return
  342. }
  343. // 查询成功
  344. logger.GetLogger().Infof("QueryTableColumnConfig successed: %v", tablecolumnconfigs)
  345. appG.Response(http.StatusOK, e.SUCCESS, tablecolumnconfigs)
  346. }