purge.go 7.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218
  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. "fmt"
  19. "strings"
  20. "yunion.io/x/pkg/errors"
  21. "yunion.io/x/sqlchemy"
  22. "yunion.io/x/onecloud/pkg/cloudcommon/db"
  23. "yunion.io/x/onecloud/pkg/cloudcommon/db/lockman"
  24. )
  25. type purgePair struct {
  26. manager db.IModelManager
  27. key string
  28. q *sqlchemy.SQuery
  29. }
  30. func (self *purgePair) queryIds() ([]string, error) {
  31. ids := []string{}
  32. sq := self.q.SubQuery()
  33. q := sq.Query(sq.Field(self.key)).Distinct()
  34. rows, err := q.Rows()
  35. if err != nil {
  36. if errors.Cause(err) == sql.ErrNoRows {
  37. return ids, nil
  38. }
  39. return ids, errors.Wrap(err, "Query")
  40. }
  41. defer rows.Close()
  42. for rows.Next() {
  43. var id string
  44. err := rows.Scan(&id)
  45. if err != nil {
  46. return ids, errors.Wrap(err, "rows.Scan")
  47. }
  48. ids = append(ids, id)
  49. }
  50. return ids, nil
  51. }
  52. func (self *purgePair) purgeAll(ctx context.Context) error {
  53. purgeIds, err := self.queryIds()
  54. if err != nil {
  55. return errors.Wrapf(err, "Query ids")
  56. }
  57. if len(purgeIds) == 0 {
  58. return nil
  59. }
  60. var purge = func(ids []string) error {
  61. vars := []interface{}{}
  62. placeholders := make([]string, len(ids))
  63. for i := range placeholders {
  64. placeholders[i] = "?"
  65. vars = append(vars, ids[i])
  66. }
  67. placeholder := strings.Join(placeholders, ",")
  68. sql := fmt.Sprintf(
  69. "delete from %s where %s in (%s)",
  70. self.manager.TableSpec().Name(), self.key, placeholder,
  71. )
  72. lockman.LockRawObject(ctx, self.manager.Keyword(), "purge")
  73. defer lockman.ReleaseRawObject(ctx, self.manager.Keyword(), "purge")
  74. _, err = sqlchemy.GetDB().Exec(
  75. sql, vars...,
  76. )
  77. if err != nil {
  78. return errors.Wrapf(err, strings.ReplaceAll(sql, "?", "%s"), vars...)
  79. }
  80. return nil
  81. }
  82. var splitByLen = func(data []string, splitLen int) [][]string {
  83. var result [][]string
  84. for i := 0; i < len(data); i += splitLen {
  85. end := i + splitLen
  86. if end > len(data) {
  87. end = len(data)
  88. }
  89. result = append(result, data[i:end])
  90. }
  91. return result
  92. }
  93. idsArr := splitByLen(purgeIds, 100)
  94. for i := range idsArr {
  95. err = purge(idsArr[i])
  96. if err != nil {
  97. return err
  98. }
  99. }
  100. return nil
  101. }
  102. func (self *SCloudaccount) purge(ctx context.Context) error {
  103. cloudproviders := CloudproviderManager.Query("id").Equals("cloudaccount_id", self.Id)
  104. cloudusers := ClouduserManager.Query("id").Equals("cloudaccount_id", self.Id)
  105. ups := ClouduserPolicyManager.Query("row_id").In("clouduser_id", cloudusers.SubQuery())
  106. ugs := CloudgroupUserManager.Query("row_id").In("clouduser_id", cloudusers.SubQuery())
  107. cloudgroups := CloudgroupManager.Query("id").Equals("cloudaccount_id", self.Id)
  108. gps := CloudgroupPolicyManager.Query("row_id").In("cloudgroup_id", cloudgroups.SubQuery())
  109. gus := CloudgroupUserManager.Query("row_id").In("cloudgroup_id", cloudgroups.SubQuery())
  110. samlproviders := SAMLProviderManager.Query("id").Equals("cloudaccount_id", self.Id)
  111. roles := CloudroleManager.Query("id").Equals("cloudaccount_id", self.Id)
  112. samlusers := SamluserManager.Query("id").In("cloudgroup_id", cloudgroups.SubQuery())
  113. policies := CloudpolicyManager.Query("id").Equals("cloudaccount_id", self.Id)
  114. pairs := []purgePair{
  115. {manager: CloudpolicyManager, key: "id", q: policies},
  116. {manager: SamluserManager, key: "id", q: samlusers},
  117. {manager: CloudroleManager, key: "id", q: roles},
  118. {manager: SAMLProviderManager, key: "id", q: samlproviders},
  119. {manager: CloudgroupPolicyManager, key: "row_id", q: gps},
  120. {manager: CloudgroupUserManager, key: "row_id", q: gus},
  121. {manager: CloudgroupManager, key: "id", q: cloudgroups},
  122. {manager: ClouduserPolicyManager, key: "row_id", q: ups},
  123. {manager: CloudgroupUserManager, key: "row_id", q: ugs},
  124. {manager: ClouduserManager, key: "id", q: cloudusers},
  125. {manager: CloudproviderManager, key: "id", q: cloudproviders},
  126. }
  127. for i := range pairs {
  128. err := pairs[i].purgeAll(ctx)
  129. if err != nil {
  130. return err
  131. }
  132. }
  133. return self.SStandaloneResourceBase.Delete(ctx, nil)
  134. }
  135. func (self *SClouduser) purge(ctx context.Context) error {
  136. ups := ClouduserPolicyManager.Query("row_id").Equals("clouduser_id", self.Id)
  137. gps := CloudgroupUserManager.Query("row_id").Equals("clouduser_id", self.Id)
  138. pairs := []purgePair{
  139. {manager: ClouduserPolicyManager, key: "row_id", q: ups},
  140. {manager: CloudgroupUserManager, key: "row_id", q: gps},
  141. }
  142. for i := range pairs {
  143. err := pairs[i].purgeAll(ctx)
  144. if err != nil {
  145. return err
  146. }
  147. }
  148. return self.SStatusDomainLevelResourceBase.Delete(ctx, nil)
  149. }
  150. func (self *SCloudprovider) purge(ctx context.Context) error {
  151. cloudusers := ClouduserManager.Query("id").Equals("manager_id", self.Id)
  152. ups := ClouduserPolicyManager.Query("row_id").In("clouduser_id", cloudusers.SubQuery())
  153. ugs := CloudgroupUserManager.Query("row_id").In("clouduser_id", cloudusers.SubQuery())
  154. cloudgroups := CloudgroupManager.Query("id").Equals("manager_id", self.Id)
  155. gps := CloudgroupPolicyManager.Query("row_id").In("cloudgroup_id", cloudgroups.SubQuery())
  156. gus := CloudgroupUserManager.Query("row_id").In("cloudgroup_id", cloudgroups.SubQuery())
  157. samlproviders := SAMLProviderManager.Query("id").Equals("manager_id", self.Id)
  158. samlusers := SamluserManager.Query("id").In("cloudgroup_id", cloudgroups.SubQuery())
  159. policies := CloudpolicyManager.Query("id").Equals("manager_id", self.Id)
  160. roles := CloudroleManager.Query("id").Equals("manager_id", self.Id)
  161. pairs := []purgePair{
  162. {manager: CloudpolicyManager, key: "id", q: policies},
  163. {manager: SamluserManager, key: "id", q: samlusers},
  164. {manager: CloudroleManager, key: "id", q: roles},
  165. {manager: SAMLProviderManager, key: "id", q: samlproviders},
  166. {manager: CloudgroupPolicyManager, key: "row_id", q: gps},
  167. {manager: CloudgroupUserManager, key: "row_id", q: gus},
  168. {manager: CloudgroupManager, key: "id", q: cloudgroups},
  169. {manager: ClouduserPolicyManager, key: "row_id", q: ups},
  170. {manager: CloudgroupUserManager, key: "row_id", q: ugs},
  171. {manager: ClouduserManager, key: "id", q: cloudusers},
  172. }
  173. for i := range pairs {
  174. err := pairs[i].purgeAll(ctx)
  175. if err != nil {
  176. return err
  177. }
  178. }
  179. return self.SStandaloneAnonResourceBase.Delete(ctx, nil)
  180. }
  181. func (self *SCloudgroup) purge(ctx context.Context) error {
  182. ups := CloudgroupPolicyManager.Query("row_id").Equals("cloudgroup_id", self.Id)
  183. gps := CloudgroupUserManager.Query("row_id").Equals("cloudgroup_id", self.Id)
  184. saml := SamluserManager.Query("id").Equals("cloudgroup_id", self.Id)
  185. pairs := []purgePair{
  186. {manager: SamluserManager, key: "id", q: saml},
  187. {manager: ClouduserPolicyManager, key: "row_id", q: ups},
  188. {manager: CloudgroupUserManager, key: "row_id", q: gps},
  189. }
  190. for i := range pairs {
  191. err := pairs[i].purgeAll(ctx)
  192. if err != nil {
  193. return err
  194. }
  195. }
  196. return self.SStatusInfrasResourceBase.Delete(ctx, nil)
  197. }