query_rule.go 9.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313
  1. package query_rule
  2. import (
  3. "encoding/json"
  4. "git.sxidc.com/go-framework/baize/convenient/domain/query_rule/definition"
  5. "git.sxidc.com/go-framework/baize/convenient/domain/query_rule/rule"
  6. "git.sxidc.com/go-framework/baize/framework/binding"
  7. "git.sxidc.com/go-framework/baize/framework/core/api"
  8. "git.sxidc.com/go-framework/baize/framework/core/api/request"
  9. "git.sxidc.com/go-framework/baize/framework/core/api/response"
  10. "git.sxidc.com/go-framework/baize/framework/core/application"
  11. "git.sxidc.com/go-framework/baize/framework/core/domain"
  12. "git.sxidc.com/go-framework/baize/framework/core/domain/entity"
  13. "git.sxidc.com/go-framework/baize/framework/core/infrastructure"
  14. "git.sxidc.com/go-framework/baize/framework/core/infrastructure/database"
  15. "git.sxidc.com/go-framework/baize/framework/core/infrastructure/database/clause"
  16. "git.sxidc.com/go-framework/baize/framework/core/infrastructure/database/sql"
  17. "git.sxidc.com/go-tools/utils/strutils"
  18. "github.com/pkg/errors"
  19. )
  20. // Simple Bind参数
  21. type Simple struct {
  22. // schema
  23. Schema string
  24. // 关闭记录查询规则
  25. DisableRecordRule bool
  26. }
  27. func (simple *Simple) bind(binder *binding.Binder) {
  28. (&definition.Simple{Schema: simple.Schema}).Bind(binder)
  29. if !simple.DisableRecordRule {
  30. (&rule.Simple{Schema: simple.Schema}).Bind(binder)
  31. }
  32. }
  33. func Bind(app *application.App, simple *Simple) {
  34. binder := binding.NewBinder(app.Api().ChooseRouter(api.RouterPrefix, ""), app.Infrastructure())
  35. simple.bind(binder)
  36. }
  37. type AdvanceQueryParams interface {
  38. GetRuleConditions() (string, []any)
  39. GetPageNo() int
  40. GetPageSize() int
  41. }
  42. type BaseAdvanceQueryParams struct {
  43. RuleClause string `json:"ruleClause" assign:"-"`
  44. RuleClauseArgs []any `json:"ruleClauseArgs" assign:"-"`
  45. PageNo int `json:"pageNo" assign:"-"`
  46. PageSize int `json:"pageSize" assign:"-"`
  47. }
  48. func (queryParams *BaseAdvanceQueryParams) GetRuleConditions() (string, []any) {
  49. return queryParams.RuleClause, queryParams.RuleClauseArgs
  50. }
  51. func (queryParams *BaseAdvanceQueryParams) GetPageNo() int {
  52. return queryParams.PageNo
  53. }
  54. func (queryParams *BaseAdvanceQueryParams) GetPageSize() int {
  55. return queryParams.PageSize
  56. }
  57. type FormAdditionalSelectClausesFunc func(queryParams AdvanceQueryParams) ([]string, error)
  58. type FormAdditionalFromFunc func(queryParams AdvanceQueryParams) ([]clause.Clause, error)
  59. type FormAdditionalConditionsFunc func(queryParams AdvanceQueryParams) (clause.Clause, error)
  60. type FormOtherClausesFunc func(queryParams AdvanceQueryParams) ([]clause.Clause, error)
  61. type AddUseQueryRuleQueryRouteParams struct {
  62. DBSchema string
  63. QueryParams AdvanceQueryParams
  64. FormAdditionalSelectClausesFunc FormAdditionalSelectClausesFunc
  65. FormAdditionalFromFunc FormAdditionalFromFunc
  66. FormAdditionalConditionsFunc FormAdditionalConditionsFunc
  67. FormOtherClausesFunc FormOtherClausesFunc
  68. Object domain.Object
  69. QueryMiddlewares []binding.Middleware
  70. }
  71. func (params *AddUseQueryRuleQueryRouteParams) check() error {
  72. if params.DBSchema == "" {
  73. return errors.New("没有传递DBSchema")
  74. }
  75. if params.Object == nil {
  76. return errors.New("没有传递Object")
  77. }
  78. return nil
  79. }
  80. func AddUseQueryRuleQueryRoute[O any](binder *binding.Binder, addParams *AddUseQueryRuleQueryRouteParams) error {
  81. err := addParams.check()
  82. if err != nil {
  83. return err
  84. }
  85. domainPath := domain.RelativeDomainPath(addParams.Object)
  86. var queryParams AdvanceQueryParams
  87. queryParams = new(BaseAdvanceQueryParams)
  88. if addParams.QueryParams != nil {
  89. queryParams = addParams.QueryParams
  90. }
  91. binding.PostBind(binder, &binding.SimpleBindItem[response.InfosData[O]]{
  92. Path: domainPath + "/advancedQuery",
  93. SendResponseFunc: response.SendInfosResponse[O],
  94. RequestParams: queryParams,
  95. Objects: []domain.Object{addParams.Object},
  96. ServiceFunc: func(c *api.Context, params request.Params, objects []domain.Object, i *infrastructure.Infrastructure) (response.InfosData[O], error) {
  97. errResponse := response.InfosData[O]{
  98. Infos: make([]O, 0),
  99. }
  100. if params == nil {
  101. return errResponse, errors.New("请求参数为空")
  102. }
  103. object := objects[0]
  104. if object == nil {
  105. return errResponse, errors.New("领域实体为空")
  106. }
  107. advanceQueryParams, ok := params.(AdvanceQueryParams)
  108. if !ok {
  109. return errResponse, errors.New("请求参数不是Query接口")
  110. }
  111. e, ok := objects[0].(entity.Entity)
  112. if !ok {
  113. return errResponse, errors.New("需要传递领域对象应该为实体")
  114. }
  115. var selectClause *clause.Select
  116. var countClause *clause.Select
  117. var fromClause *clause.From
  118. var additionalConditions clause.Clause
  119. selectClauses := make([]string, 0)
  120. fromClauses := []clause.Clause{clause.TableName(domain.TableName(addParams.DBSchema, e))}
  121. selectOtherClauses := make([]clause.Clause, 0)
  122. countOtherClauses := make([]clause.Clause, 0)
  123. if addParams.FormAdditionalSelectClausesFunc != nil {
  124. additionalSelectClauses, err := addParams.FormAdditionalSelectClausesFunc(advanceQueryParams)
  125. if err != nil {
  126. return errResponse, err
  127. }
  128. selectClauses = additionalSelectClauses
  129. }
  130. if addParams.FormAdditionalFromFunc != nil {
  131. additionalFromClauses, err := addParams.FormAdditionalFromFunc(advanceQueryParams)
  132. if err != nil {
  133. return errResponse, err
  134. }
  135. fromClauses = append(fromClauses, additionalFromClauses...)
  136. }
  137. fromClause = clause.NewFrom(fromClauses)
  138. limitClause := clause.NewLimit(advanceQueryParams.GetPageNo(), advanceQueryParams.GetPageSize())
  139. if addParams.FormAdditionalConditionsFunc != nil {
  140. conditions, err := addParams.FormAdditionalConditionsFunc(advanceQueryParams)
  141. if err != nil {
  142. return errResponse, err
  143. }
  144. additionalConditions = conditions
  145. }
  146. ruleClause, ruleClauseArgs := advanceQueryParams.GetRuleConditions()
  147. if strutils.IsStringEmpty(ruleClause) {
  148. if additionalConditions != nil {
  149. selectOtherClauses = append(selectOtherClauses, clause.NewWhere(additionalConditions), limitClause)
  150. countOtherClauses = append(countOtherClauses, clause.NewWhere(additionalConditions))
  151. } else {
  152. selectOtherClauses = append(selectOtherClauses, limitClause)
  153. }
  154. } else {
  155. var conditionClause clause.Clause
  156. conditionClause = clause.NewConditions().AddCondition(ruleClause, ruleClauseArgs...).And()
  157. if additionalConditions != nil {
  158. conditionClause = clause.NewConditionJoin(conditionClause, additionalConditions).And()
  159. }
  160. selectOtherClauses = append(selectOtherClauses, clause.NewWhere(conditionClause), limitClause)
  161. countOtherClauses = append(countOtherClauses, clause.NewWhere(conditionClause))
  162. }
  163. if addParams.FormOtherClausesFunc != nil {
  164. otherClauses, err := addParams.FormOtherClausesFunc(advanceQueryParams)
  165. if err != nil {
  166. return errResponse, err
  167. }
  168. selectOtherClauses = append(selectOtherClauses, otherClauses...)
  169. countOtherClauses = append(countOtherClauses, otherClauses...)
  170. }
  171. selectClause = clause.NewSelect(selectClauses, fromClause, selectOtherClauses...)
  172. countClause = clause.NewSelect([]string{"COUNT(*) AS total"}, fromClause, countOtherClauses...)
  173. selectClauseStr, err := selectClause.Clause()
  174. if err != nil {
  175. return errResponse, err
  176. }
  177. results, err := database.ExecuteRawSql(i.DBExecutor(), selectClauseStr, selectClause.Args()...)
  178. if err != nil {
  179. return errResponse, err
  180. }
  181. countClauseStr, err := countClause.Clause()
  182. if err != nil {
  183. return errResponse, err
  184. }
  185. countResults, err := database.ExecuteRawSql(i.DBExecutor(), countClauseStr, countClause.Args()...)
  186. if err != nil {
  187. return errResponse, err
  188. }
  189. infos := make([]O, 0)
  190. err = sql.ParseSqlResult(results, &infos)
  191. if err != nil {
  192. return errResponse, err
  193. }
  194. return response.InfosData[O]{
  195. Infos: infos,
  196. TotalCount: countResults[0].ColumnValueInt64("total"),
  197. PageNo: advanceQueryParams.GetPageNo(),
  198. }, nil
  199. },
  200. }, addParams.QueryMiddlewares...)
  201. return nil
  202. }
  203. type QueryRuleParseJsonBody struct {
  204. Rule string `json:"rule" assign:"-"`
  205. }
  206. type AddUseQueryRuleParseRouteParams struct {
  207. Object domain.Object
  208. }
  209. func (params *AddUseQueryRuleParseRouteParams) check() error {
  210. if params.Object == nil {
  211. return errors.New("没有传递Object")
  212. }
  213. return nil
  214. }
  215. func AddUseQueryRuleParseRoute(binder *binding.Binder, addParams *AddUseQueryRuleParseRouteParams) error {
  216. err := addParams.check()
  217. if err != nil {
  218. return err
  219. }
  220. domainPath := domain.RelativeDomainPath(addParams.Object)
  221. binding.PostBind(binder, &binding.SimpleBindItem[map[string]any]{
  222. Path: domainPath + "/queryRule/parse",
  223. SendResponseFunc: response.SendMapResponse,
  224. RequestParams: &QueryRuleParseJsonBody{},
  225. ServiceFunc: func(c *api.Context, params request.Params, objects []domain.Object, i *infrastructure.Infrastructure) (map[string]any, error) {
  226. errResponse := map[string]any{
  227. "ruleClause": "",
  228. "ruleClauseArgs": make([]any, 0),
  229. }
  230. jsonBody, err := request.ToConcrete[*QueryRuleParseJsonBody](params)
  231. if err != nil {
  232. return errResponse, err
  233. }
  234. r := new(rule.Rule)
  235. err = json.Unmarshal([]byte(jsonBody.Rule), r)
  236. if err != nil {
  237. return errResponse, err
  238. }
  239. ruleConditionClause, err := rule.FormConditionClauseByRule(addParams.Object.DomainCamelName(), *r, nil)
  240. if err != nil {
  241. return errResponse, err
  242. }
  243. ruleConditionClauseStr, err := ruleConditionClause.Clause()
  244. if err != nil {
  245. return errResponse, err
  246. }
  247. return map[string]any{
  248. "ruleClause": ruleConditionClauseStr,
  249. "ruleClauseArgs": ruleConditionClause.Args(),
  250. }, nil
  251. },
  252. })
  253. return nil
  254. }