query_rule.go 6.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203
  1. package query_rule
  2. import (
  3. "git.sxidc.com/go-framework/baize/convenient/domain/query_rule/definition"
  4. "git.sxidc.com/go-framework/baize/convenient/domain/query_rule/rule"
  5. "git.sxidc.com/go-framework/baize/framework/binding"
  6. "git.sxidc.com/go-framework/baize/framework/core/api"
  7. "git.sxidc.com/go-framework/baize/framework/core/api/request"
  8. "git.sxidc.com/go-framework/baize/framework/core/api/response"
  9. "git.sxidc.com/go-framework/baize/framework/core/application"
  10. "git.sxidc.com/go-framework/baize/framework/core/domain"
  11. "git.sxidc.com/go-framework/baize/framework/core/domain/entity"
  12. "git.sxidc.com/go-framework/baize/framework/core/infrastructure"
  13. "git.sxidc.com/go-framework/baize/framework/core/infrastructure/database"
  14. "git.sxidc.com/go-framework/baize/framework/core/infrastructure/database/clause"
  15. "git.sxidc.com/go-framework/baize/framework/core/infrastructure/database/sql"
  16. "git.sxidc.com/go-tools/utils/strutils"
  17. "github.com/goccy/go-json"
  18. "github.com/pkg/errors"
  19. )
  20. // Simple Bind参数
  21. type Simple struct {
  22. // schema
  23. Schema string
  24. }
  25. func (simple *Simple) bind(binder *binding.Binder) {
  26. (&definition.Simple{Schema: simple.Schema}).Bind(binder)
  27. (&rule.Simple{Schema: simple.Schema}).Bind(binder)
  28. }
  29. func Bind(app *application.App, simple *Simple) {
  30. binder := binding.NewBinder(app.Api().ChooseRouter(api.RouterPrefix, ""), app.Infrastructure())
  31. simple.bind(binder)
  32. }
  33. type AdvanceQueryParams interface {
  34. GetRule() string
  35. GetPageNo() int
  36. GetPageSize() int
  37. }
  38. type BaseAdvanceQueryParams struct {
  39. Rule string `json:"rule" assign:"-"`
  40. PageNo int `json:"pageNo" assign:"-"`
  41. PageSize int `json:"pageSize" assign:"-"`
  42. }
  43. func (queryParams *BaseAdvanceQueryParams) GetRule() string {
  44. return queryParams.Rule
  45. }
  46. func (queryParams *BaseAdvanceQueryParams) GetPageNo() int {
  47. return queryParams.PageNo
  48. }
  49. func (queryParams *BaseAdvanceQueryParams) GetPageSize() int {
  50. return queryParams.PageSize
  51. }
  52. type FormAdditionalConditionsFunc func(queryParams AdvanceQueryParams) (clause.Clause, error)
  53. type FormOtherClausesFunc func(queryParams AdvanceQueryParams) ([]clause.Clause, error)
  54. type AddUseQueryRuleQueryRouteParams struct {
  55. DBSchema string
  56. QueryParams AdvanceQueryParams
  57. FormAdditionalConditionsFunc FormAdditionalConditionsFunc
  58. FormOtherClausesFunc FormOtherClausesFunc
  59. Object domain.Object
  60. QueryMiddlewares []binding.Middleware
  61. }
  62. func AddUseQueryRuleQueryRoute[O any](binder *binding.Binder, addParams *AddUseQueryRuleQueryRouteParams) {
  63. domainPath := domain.RelativeDomainPath(addParams.Object)
  64. var queryParams AdvanceQueryParams
  65. queryParams = new(BaseAdvanceQueryParams)
  66. if addParams.QueryParams != nil {
  67. queryParams = addParams.QueryParams
  68. }
  69. binding.PostBind(binder, &binding.SimpleBindItem[response.InfosData[O]]{
  70. Path: domainPath + "/advancedQuery",
  71. SendResponseFunc: response.SendInfosResponse[O],
  72. RequestParams: queryParams,
  73. Objects: []domain.Object{addParams.Object},
  74. ServiceFunc: func(c *api.Context, params request.Params, objects []domain.Object, i *infrastructure.Infrastructure) (response.InfosData[O], error) {
  75. errResponse := response.InfosData[O]{
  76. Infos: make([]O, 0),
  77. }
  78. if params == nil {
  79. return errResponse, errors.New("请求参数为空")
  80. }
  81. object := objects[0]
  82. if object == nil {
  83. return errResponse, errors.New("领域实体为空")
  84. }
  85. advanceQueryParams, ok := params.(AdvanceQueryParams)
  86. if !ok {
  87. return errResponse, errors.New("请求参数不是Query接口")
  88. }
  89. e, ok := objects[0].(entity.Entity)
  90. if !ok {
  91. return errResponse, errors.New("需要传递领域对象应该为实体")
  92. }
  93. var selectClause *clause.Select
  94. var countClause *clause.Select
  95. var additionalConditions clause.Clause
  96. selectOtherClauses := make([]clause.Clause, 0)
  97. countOtherClauses := make([]clause.Clause, 0)
  98. fromClause := clause.NewFrom([]clause.Clause{clause.TableName(domain.TableName(addParams.DBSchema, e))})
  99. limitClause := clause.NewLimit(advanceQueryParams.GetPageNo(), advanceQueryParams.GetPageSize())
  100. if addParams.FormAdditionalConditionsFunc != nil {
  101. conditions, err := addParams.FormAdditionalConditionsFunc(advanceQueryParams)
  102. if err != nil {
  103. return errResponse, err
  104. }
  105. additionalConditions = conditions
  106. }
  107. if strutils.IsStringEmpty(advanceQueryParams.GetRule()) {
  108. if additionalConditions != nil {
  109. selectOtherClauses = append(selectOtherClauses, clause.NewWhere(additionalConditions), limitClause)
  110. countOtherClauses = append(countOtherClauses, clause.NewWhere(additionalConditions))
  111. } else {
  112. selectOtherClauses = append(selectOtherClauses, limitClause)
  113. }
  114. } else {
  115. queryRule := new(rule.Rule)
  116. err := json.Unmarshal([]byte(advanceQueryParams.GetRule()), queryRule)
  117. if err != nil {
  118. return errResponse, err
  119. }
  120. conditionClause, err := rule.FormConditionClauseByRules(e.DomainCamelName(), *queryRule, nil)
  121. if err != nil {
  122. return errResponse, err
  123. }
  124. if additionalConditions != nil {
  125. conditionClause = clause.NewConditionJoin(conditionClause, additionalConditions).And()
  126. }
  127. selectOtherClauses = append(selectOtherClauses, clause.NewWhere(conditionClause), limitClause)
  128. countOtherClauses = append(countOtherClauses, clause.NewWhere(conditionClause))
  129. }
  130. if addParams.FormOtherClausesFunc != nil {
  131. otherClauses, err := addParams.FormOtherClausesFunc(advanceQueryParams)
  132. if err != nil {
  133. return errResponse, err
  134. }
  135. selectOtherClauses = append(selectOtherClauses, otherClauses...)
  136. countOtherClauses = append(countOtherClauses, otherClauses...)
  137. }
  138. selectClause = clause.NewSelect(nil, fromClause, selectOtherClauses...)
  139. countClause = clause.NewSelect([]string{"COUNT(*) AS total"}, fromClause, countOtherClauses...)
  140. selectClauseStr, err := selectClause.Clause()
  141. if err != nil {
  142. return errResponse, err
  143. }
  144. results, err := database.ExecuteRawSql(i.DBExecutor(), selectClauseStr, selectClause.Args()...)
  145. if err != nil {
  146. return errResponse, err
  147. }
  148. countClauseStr, err := countClause.Clause()
  149. if err != nil {
  150. return errResponse, err
  151. }
  152. countResults, err := database.ExecuteRawSql(i.DBExecutor(), countClauseStr, countClause.Args()...)
  153. if err != nil {
  154. return errResponse, err
  155. }
  156. infos := make([]O, 0)
  157. err = sql.ParseSqlResult(results, &infos)
  158. if err != nil {
  159. return errResponse, err
  160. }
  161. return response.InfosData[O]{
  162. Infos: infos,
  163. TotalCount: countResults[0].ColumnValueInt64("total"),
  164. PageNo: advanceQueryParams.GetPageNo(),
  165. }, nil
  166. },
  167. }, addParams.QueryMiddlewares...)
  168. }