groupresource.go 5.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181
  1. // Copyright 2019 Yunion
  2. //
  3. // Licensed under the Apache License, Version 2.0 (the "License");
  4. // you may not use this file except in compliance with the License.
  5. // You may obtain a copy of the License at
  6. //
  7. // http://www.apache.org/licenses/LICENSE-2.0
  8. //
  9. // Unless required by applicable law or agreed to in writing, software
  10. // distributed under the License is distributed on an "AS IS" BASIS,
  11. // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  12. // See the License for the specific language governing permissions and
  13. // limitations under the License.
  14. package models
  15. import (
  16. "context"
  17. "database/sql"
  18. "yunion.io/x/jsonutils"
  19. "yunion.io/x/log"
  20. "yunion.io/x/pkg/errors"
  21. "yunion.io/x/pkg/util/reflectutils"
  22. "yunion.io/x/sqlchemy"
  23. api "yunion.io/x/onecloud/pkg/apis/compute"
  24. "yunion.io/x/onecloud/pkg/cloudcommon/db"
  25. "yunion.io/x/onecloud/pkg/httperrors"
  26. "yunion.io/x/onecloud/pkg/mcclient"
  27. "yunion.io/x/onecloud/pkg/util/stringutils2"
  28. )
  29. type SGroupResourceBase struct {
  30. // 实例组ID
  31. GroupId string `width:"36" charset:"ascii" nullable:"true" list:"user" create:"optional"`
  32. }
  33. type SGroupResourceBaseManager struct {
  34. }
  35. func ValidateGroupResourceInput(ctx context.Context, userCred mcclient.TokenCredential, input api.GroupResourceInput) (*SGroup, api.GroupResourceInput, error) {
  36. groupObj, err := GroupManager.FetchByIdOrName(ctx, userCred, input.GroupId)
  37. if err != nil {
  38. if errors.Cause(err) == sql.ErrNoRows {
  39. return nil, input, errors.Wrapf(httperrors.ErrResourceNotFound, "%s %s", GroupManager.Keyword(), input.GroupId)
  40. } else {
  41. return nil, input, errors.Wrap(err, "GroupManager.FetchByIdOrName")
  42. }
  43. }
  44. input.GroupId = groupObj.GetId()
  45. return groupObj.(*SGroup), input, nil
  46. }
  47. func (self *SGroupResourceBase) GetGroup() *SGroup {
  48. obj, _ := GroupManager.FetchById(self.GroupId)
  49. if obj != nil {
  50. return obj.(*SGroup)
  51. }
  52. return nil
  53. }
  54. func (manager *SGroupResourceBaseManager) FetchCustomizeColumns(
  55. ctx context.Context,
  56. userCred mcclient.TokenCredential,
  57. query jsonutils.JSONObject,
  58. objs []interface{},
  59. fields stringutils2.SSortedStrings,
  60. isList bool,
  61. ) []api.GroupResourceInfo {
  62. rows := make([]api.GroupResourceInfo, len(objs))
  63. groupIds := make([]string, len(objs))
  64. for i := range objs {
  65. var base *SGroupResourceBase
  66. err := reflectutils.FindAnonymouStructPointer(objs[i], &base)
  67. if err != nil {
  68. log.Errorf("Cannot find SGroupResourceBase in object %s", objs[i])
  69. continue
  70. }
  71. groupIds[i] = base.GroupId
  72. }
  73. groupNames, err := db.FetchIdNameMap2(GroupManager, groupIds)
  74. if err != nil {
  75. log.Errorf("FetchIdNameMap2 fail %s", err)
  76. return rows
  77. }
  78. for i := range rows {
  79. if name, ok := groupNames[groupIds[i]]; ok {
  80. rows[i].Group = name
  81. }
  82. }
  83. return rows
  84. }
  85. func (manager *SGroupResourceBaseManager) ListItemFilter(
  86. ctx context.Context,
  87. q *sqlchemy.SQuery,
  88. userCred mcclient.TokenCredential,
  89. query api.GroupFilterListInput,
  90. ) (*sqlchemy.SQuery, error) {
  91. if len(query.GroupId) > 0 {
  92. groupObj, _, err := ValidateGroupResourceInput(ctx, userCred, query.GroupResourceInput)
  93. if err != nil {
  94. return nil, errors.Wrap(err, "ValidateGroupResourceInput")
  95. }
  96. q = q.Equals("group_id", groupObj.GetId())
  97. }
  98. return q, nil
  99. }
  100. func (manager *SGroupResourceBaseManager) QueryDistinctExtraField(q *sqlchemy.SQuery, field string) (*sqlchemy.SQuery, error) {
  101. switch field {
  102. case "group":
  103. groupQuery := GroupManager.Query("name", "id").SubQuery()
  104. q = q.AppendField(groupQuery.Field("name", field)).Distinct()
  105. q = q.Join(groupQuery, sqlchemy.Equals(q.Field("group_id"), groupQuery.Field("id")))
  106. return q, nil
  107. }
  108. return q, httperrors.ErrNotFound
  109. }
  110. func (manager *SGroupResourceBaseManager) OrderByExtraFields(
  111. ctx context.Context,
  112. q *sqlchemy.SQuery,
  113. userCred mcclient.TokenCredential,
  114. query api.GroupFilterListInput,
  115. ) (*sqlchemy.SQuery, error) {
  116. if !db.NeedOrderQuery(manager.GetOrderByFields(query)) {
  117. return q, nil
  118. }
  119. orderQ := GroupManager.Query("id")
  120. orderSubQ := orderQ.SubQuery()
  121. orderQ, orders, fields := manager.GetOrderBySubQuery(orderQ, orderSubQ, orderQ.Field("id"), userCred, query, nil, nil)
  122. q = q.LeftJoin(orderSubQ, sqlchemy.Equals(q.Field("group_id"), orderSubQ.Field("id")))
  123. q = db.OrderByFields(q, orders, fields)
  124. return q, nil
  125. }
  126. func (manager *SGroupResourceBaseManager) GetOrderBySubQuery(
  127. q *sqlchemy.SQuery,
  128. subq *sqlchemy.SSubQuery,
  129. joinField sqlchemy.IQueryField,
  130. userCred mcclient.TokenCredential,
  131. query api.GroupFilterListInput,
  132. orders []string,
  133. fields []sqlchemy.IQueryField,
  134. ) (*sqlchemy.SQuery, []string, []sqlchemy.IQueryField) {
  135. if !db.NeedOrderQuery(manager.GetOrderByFields(query)) {
  136. return q, orders, fields
  137. }
  138. groupQ := GroupManager.Query().SubQuery()
  139. q = q.LeftJoin(groupQ, sqlchemy.Equals(joinField, groupQ.Field("id")))
  140. q = q.AppendField(groupQ.Field("name").Label("group"))
  141. orders = append(orders, query.OrderByGroup)
  142. fields = append(fields, subq.Field("group"))
  143. return q, orders, fields
  144. }
  145. func (manager *SGroupResourceBaseManager) GetOrderByFields(query api.GroupFilterListInput) []string {
  146. return []string{query.OrderByGroup}
  147. }
  148. func (manager *SGroupResourceBaseManager) ListItemExportKeys(ctx context.Context,
  149. q *sqlchemy.SQuery,
  150. userCred mcclient.TokenCredential,
  151. keys stringutils2.SSortedStrings,
  152. ) (*sqlchemy.SQuery, error) {
  153. if keys.ContainsAny(manager.GetExportKeys()...) {
  154. subq := GroupManager.Query("id", "name").SubQuery()
  155. q = q.LeftJoin(subq, sqlchemy.Equals(q.Field("group_id"), subq.Field("id")))
  156. if keys.Contains("group") {
  157. q = q.AppendField(subq.Field("name", "group"))
  158. }
  159. }
  160. return q, nil
  161. }
  162. func (manager *SGroupResourceBaseManager) GetExportKeys() []string {
  163. return []string{"group"}
  164. }