db_joint_dispatcher.go 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333
  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 db
  15. import (
  16. "context"
  17. "database/sql"
  18. "fmt"
  19. "yunion.io/x/jsonutils"
  20. "yunion.io/x/log"
  21. "yunion.io/x/pkg/errors"
  22. "yunion.io/x/pkg/util/printutils"
  23. "yunion.io/x/onecloud/pkg/cloudcommon/db/lockman"
  24. "yunion.io/x/onecloud/pkg/cloudcommon/policy"
  25. "yunion.io/x/onecloud/pkg/httperrors"
  26. "yunion.io/x/onecloud/pkg/mcclient"
  27. "yunion.io/x/onecloud/pkg/util/logclient"
  28. )
  29. type DBJointModelDispatcher struct {
  30. DBModelDispatcher
  31. }
  32. func NewJointModelHandler(manager IJointModelManager) *DBJointModelDispatcher {
  33. // registerModelManager(manager)
  34. return &DBJointModelDispatcher{DBModelDispatcher: DBModelDispatcher{manager: manager}}
  35. }
  36. func (dispatcher *DBJointModelDispatcher) JointModelManager() IJointModelManager {
  37. return dispatcher.manager.(IJointModelManager)
  38. }
  39. func (dispatcher *DBJointModelDispatcher) MasterKeywordPlural() string {
  40. jointManager := dispatcher.JointModelManager()
  41. if jointManager == nil {
  42. log.Fatalf("nil jointModelManager")
  43. }
  44. return jointManager.GetMasterManager().KeywordPlural()
  45. }
  46. func (dispatcher *DBJointModelDispatcher) SlaveKeywordPlural() string {
  47. jointManager := dispatcher.JointModelManager()
  48. if jointManager == nil {
  49. log.Fatalf("nil jointModelManager")
  50. }
  51. return jointManager.GetSlaveManager().KeywordPlural()
  52. }
  53. func (dispatcher *DBJointModelDispatcher) ListMasterDescendent(ctx context.Context, idStr string, query jsonutils.JSONObject) (*printutils.ListResult, error) {
  54. //log.Debugf("ListMasterDescendent %s %s", dispatcher.JointModelManager().GetMasterManager().Keyword(), idStr)
  55. userCred := fetchUserCredential(ctx)
  56. var queryDict *jsonutils.JSONDict
  57. if query != nil {
  58. queryDict, _ = query.(*jsonutils.JSONDict)
  59. if queryDict == nil {
  60. return nil, fmt.Errorf("fail to convert query to dict")
  61. }
  62. }
  63. manager := dispatcher.manager.GetImmutableInstance(ctx, userCred, query).(IJointModelManager)
  64. ctx = manager.PrepareQueryContext(ctx, userCred, query)
  65. model, err := fetchItem(manager.GetMasterManager(), ctx, userCred, idStr, query)
  66. if err != nil {
  67. if errors.Cause(err) == sql.ErrNoRows {
  68. return nil, httperrors.NewResourceNotFoundError2(manager.GetMasterManager().Keyword(), idStr)
  69. } else {
  70. return nil, err
  71. }
  72. }
  73. queryDict.Add(jsonutils.NewString(model.GetId()), fmt.Sprintf("%s_id", manager.GetMasterManager().Keyword()))
  74. queryDict.Add(jsonutils.NewString(model.GetId()), manager.GetMasterFieldName())
  75. if len(manager.GetMasterManager().Alias()) > 0 {
  76. queryDict.Add(jsonutils.NewString(model.GetId()), fmt.Sprintf("%s_id", manager.GetMasterManager().Alias()))
  77. }
  78. return _listJoint(manager, ctx, userCred, model.(IStandaloneModel), queryDict)
  79. }
  80. func (dispatcher *DBJointModelDispatcher) ListSlaveDescendent(ctx context.Context, idStr string, query jsonutils.JSONObject) (*printutils.ListResult, error) {
  81. //log.Debugf("ListSlaveDescendent %s %s", dispatcher.JointModelManager().GetMasterManager().Keyword(), idStr)
  82. userCred := fetchUserCredential(ctx)
  83. var queryDict *jsonutils.JSONDict
  84. if query != nil {
  85. queryDict, _ = query.(*jsonutils.JSONDict)
  86. if queryDict == nil {
  87. return nil, fmt.Errorf("fail to convert query to dict")
  88. }
  89. }
  90. manager := dispatcher.manager.GetImmutableInstance(ctx, userCred, query).(IJointModelManager)
  91. ctx = manager.PrepareQueryContext(ctx, userCred, query)
  92. model, err := fetchItem(manager.GetSlaveManager(), ctx, userCred, idStr, query)
  93. if err != nil {
  94. if errors.Cause(err) == sql.ErrNoRows {
  95. return nil, httperrors.NewResourceNotFoundError2(manager.GetSlaveManager().Keyword(), idStr)
  96. } else {
  97. return nil, httperrors.NewGeneralError(err)
  98. }
  99. }
  100. queryDict.Add(jsonutils.NewString(model.GetId()), fmt.Sprintf("%s_id", manager.GetSlaveManager().Keyword()))
  101. queryDict.Add(jsonutils.NewString(model.GetId()), manager.GetSlaveFieldName())
  102. if len(manager.GetSlaveManager().Alias()) > 0 {
  103. queryDict.Add(jsonutils.NewString(model.GetId()), fmt.Sprintf("%s_id", manager.GetSlaveManager().Alias()))
  104. }
  105. return _listJoint(manager, ctx, userCred, model.(IStandaloneModel), queryDict)
  106. }
  107. func _listJoint(manager IModelManager, ctx context.Context, userCred mcclient.TokenCredential, ctxModel IStandaloneModel, queryDict jsonutils.JSONObject) (*printutils.ListResult, error) {
  108. items, err := ListItems(manager, ctx, userCred, queryDict, nil)
  109. if err != nil {
  110. log.Errorf("Fail to list items: %s", err)
  111. return nil, httperrors.NewGeneralError(err)
  112. }
  113. return items, nil
  114. }
  115. func fetchJointItem(manager IJointModelManager, ctx context.Context, userCred mcclient.TokenCredential, id1 string, id2 string, query jsonutils.JSONObject) (IStandaloneModel, IStandaloneModel, IJointModel, error) {
  116. master, err := fetchItem(manager.GetMasterManager(), ctx, userCred, id1, query)
  117. if err != nil {
  118. if errors.Cause(err) == sql.ErrNoRows {
  119. return nil, nil, nil, httperrors.NewResourceNotFoundError2(manager.GetMasterManager().Keyword(), id1)
  120. } else {
  121. return nil, nil, nil, httperrors.NewGeneralError(err)
  122. }
  123. }
  124. slave, err := fetchItem(manager.GetSlaveManager(), ctx, userCred, id2, query)
  125. if err != nil {
  126. if errors.Cause(err) == sql.ErrNoRows {
  127. return nil, nil, nil, httperrors.NewResourceNotFoundError2(manager.GetSlaveManager().Keyword(), id2)
  128. } else {
  129. return nil, nil, nil, httperrors.NewGeneralError(err)
  130. }
  131. }
  132. item, err := FetchJointByIds(manager, master.GetId(), slave.GetId(), query)
  133. if err != nil {
  134. return nil, nil, nil, err
  135. }
  136. return master.(IStandaloneModel), slave.(IStandaloneModel), item, nil
  137. }
  138. func (dispatcher *DBJointModelDispatcher) Get(ctx context.Context, id1 string, id2 string, query jsonutils.JSONObject) (jsonutils.JSONObject, error) {
  139. userCred := fetchUserCredential(ctx)
  140. manager := dispatcher.manager.GetImmutableInstance(ctx, userCred, query).(IJointModelManager)
  141. ctx = manager.PrepareQueryContext(ctx, userCred, query)
  142. _, _, item, err := fetchJointItem(manager, ctx, userCred, id1, id2, query)
  143. if err == sql.ErrNoRows {
  144. return nil, httperrors.NewResourceNotFoundError2(manager.Keyword(), id1+"-"+id2)
  145. } else if err != nil {
  146. return nil, httperrors.NewGeneralError(err)
  147. }
  148. err = isJointObjectRbacAllowed(ctx, item, userCred, policy.PolicyActionGet)
  149. if err != nil {
  150. return nil, err
  151. }
  152. return getItemDetails(manager, item, ctx, userCred, query)
  153. }
  154. func attachItems(
  155. dispatcher *DBJointModelDispatcher,
  156. master IStandaloneModel,
  157. slave IStandaloneModel,
  158. ctx context.Context,
  159. userCred mcclient.TokenCredential,
  160. query jsonutils.JSONObject,
  161. data jsonutils.JSONObject,
  162. ) (jsonutils.JSONObject, error) {
  163. err := isObjectRbacAllowed(ctx, master, userCred, policy.PolicyActionPerform, "attach")
  164. if err != nil {
  165. return nil, err
  166. }
  167. err = isObjectRbacAllowed(ctx, slave, userCred, policy.PolicyActionPerform, "attach")
  168. if err != nil {
  169. return nil, err
  170. }
  171. // ownerProjId, err := fetchOwnerId(ctx, dispatcher.JointModelManager(), userCred, data)
  172. dataDict, ok := data.(*jsonutils.JSONDict)
  173. if !ok {
  174. return nil, fmt.Errorf("body not a json dict")
  175. }
  176. dataDict.Add(jsonutils.NewString(master.GetId()), fmt.Sprintf("%s_id", dispatcher.JointModelManager().GetMasterManager().Keyword()))
  177. if len(dispatcher.JointModelManager().GetMasterManager().Alias()) > 0 {
  178. dataDict.Add(jsonutils.NewString(master.GetId()), fmt.Sprintf("%s_id", dispatcher.JointModelManager().GetMasterManager().Alias()))
  179. }
  180. dataDict.Add(jsonutils.NewString(slave.GetId()), fmt.Sprintf("%s_id", dispatcher.JointModelManager().GetSlaveManager().Keyword()))
  181. if len(dispatcher.JointModelManager().GetSlaveManager().Alias()) > 0 {
  182. dataDict.Add(jsonutils.NewString(slave.GetId()), fmt.Sprintf("%s_id", dispatcher.JointModelManager().GetSlaveManager().Alias()))
  183. }
  184. item, err := doCreateItem(dispatcher.JointModelManager(), ctx, userCred, nil, query, data)
  185. if err != nil {
  186. return nil, httperrors.NewGeneralError(err)
  187. }
  188. item.PostCreate(ctx, userCred, nil, query, data)
  189. if err := dispatcher.JointModelManager().GetExtraHook().AfterPostCreate(ctx, userCred, item.GetOwnerId(), item, query, data); err != nil {
  190. logclient.AddActionLogWithContext(ctx, item, logclient.ACT_POST_CREATE_HOOK, err, userCred, false)
  191. }
  192. OpsLog.LogAttachEvent(ctx, master, slave, userCred, jsonutils.Marshal(item))
  193. dispatcher.manager.OnCreateComplete(ctx, []IModel{item}, userCred, nil, query, []jsonutils.JSONObject{data})
  194. return getItemDetails(dispatcher.JointModelManager(), item, ctx, userCred, query)
  195. }
  196. func (dispatcher *DBJointModelDispatcher) Attach(ctx context.Context, id1 string, id2 string, query jsonutils.JSONObject, data jsonutils.JSONObject) (jsonutils.JSONObject, error) {
  197. userCred := fetchUserCredential(ctx)
  198. master, err := fetchItem(dispatcher.JointModelManager().GetMasterManager(), ctx, userCred, id1, query)
  199. if err != nil {
  200. if err == sql.ErrNoRows {
  201. return nil, httperrors.NewResourceNotFoundError2(dispatcher.JointModelManager().GetMasterManager().Keyword(), id1)
  202. } else {
  203. return nil, httperrors.NewGeneralError(err)
  204. }
  205. }
  206. slave, err := fetchItem(dispatcher.JointModelManager().GetSlaveManager(), ctx, userCred, id2, query)
  207. if err != nil {
  208. if err == sql.ErrNoRows {
  209. return nil, httperrors.NewResourceNotFoundError2(dispatcher.JointModelManager().GetSlaveManager().Keyword(), id2)
  210. } else {
  211. return nil, httperrors.NewGeneralError(err)
  212. }
  213. }
  214. _, _, joinItem, err := fetchJointItem(dispatcher.JointModelManager(), ctx, userCred, master.GetId(), slave.GetId(), query)
  215. if err != nil && err != sql.ErrNoRows {
  216. return nil, httperrors.NewGeneralError(err)
  217. }
  218. if joinItem != nil {
  219. return nil, httperrors.NewNotAcceptableError("Object %s %s has attached %s %s", master.KeywordPlural(), master.GetId(), slave.KeywordPlural(), slave.GetId())
  220. }
  221. lockman.LockJointObject(ctx, master, slave)
  222. defer lockman.ReleaseJointObject(ctx, master, slave)
  223. resp, err := attachItems(dispatcher, master.(IStandaloneModel), slave.(IStandaloneModel), ctx, userCred, query, data)
  224. if err == nil {
  225. CallCustomizeNotifyHook(ctx, userCred, ACT_ATTACH, master, slave.GetShortDesc(ctx))
  226. }
  227. return resp, err
  228. }
  229. func (dispatcher *DBJointModelDispatcher) Update(ctx context.Context, id1 string, id2 string, query jsonutils.JSONObject, data jsonutils.JSONObject) (jsonutils.JSONObject, error) {
  230. // 获取用户信息
  231. userCred := fetchUserCredential(ctx)
  232. manager := dispatcher.manager.GetMutableInstance(ctx, userCred, query, data)
  233. // 获取对象与关联表(such as guestdisks_tbl)
  234. master, slave, item, err := fetchJointItem(dispatcher.JointModelManager(), ctx, userCred, id1, id2, query)
  235. if err == sql.ErrNoRows {
  236. if jsonutils.QueryBoolean(query, "auto_create", false) {
  237. queryDict := query.(*jsonutils.JSONDict)
  238. queryDict.Remove("auto_create")
  239. return dispatcher.Attach(ctx, id1, id2, query, data)
  240. }
  241. return nil, httperrors.NewResourceNotFoundError2(manager.Keyword(), id1+"-"+id2)
  242. } else if err != nil {
  243. return nil, httperrors.NewGeneralError(err)
  244. }
  245. // 判断权限
  246. err = isJointObjectRbacAllowed(ctx, item, userCred, policy.PolicyActionUpdate)
  247. if err != nil {
  248. return nil, err
  249. }
  250. // 锁住实例与关联表
  251. lockman.LockJointObject(ctx, master, slave)
  252. defer lockman.ReleaseJointObject(ctx, master, slave)
  253. return updateItem(dispatcher.JointModelManager(), item, ctx, userCred, query, data)
  254. }
  255. func (dispatcher *DBJointModelDispatcher) Detach(ctx context.Context, id1 string, id2 string, query jsonutils.JSONObject, data jsonutils.JSONObject) (jsonutils.JSONObject, error) {
  256. userCred := fetchUserCredential(ctx)
  257. manager := dispatcher.manager.GetMutableInstance(ctx, userCred, query, data)
  258. master, slave, item, err := fetchJointItem(dispatcher.JointModelManager(), ctx, userCred, id1, id2, query)
  259. if err == sql.ErrNoRows {
  260. return nil, httperrors.NewResourceNotFoundError2(manager.Keyword(), id1+"-"+id2)
  261. } else if err != nil {
  262. return nil, httperrors.NewGeneralError(err)
  263. }
  264. err = isObjectRbacAllowed(ctx, master, userCred, policy.PolicyActionPerform, "detach")
  265. if err != nil {
  266. return nil, err
  267. }
  268. err = isObjectRbacAllowed(ctx, slave, userCred, policy.PolicyActionPerform, "detach")
  269. if err != nil {
  270. return nil, err
  271. }
  272. lockman.LockJointObject(ctx, master, slave)
  273. defer lockman.ReleaseJointObject(ctx, master, slave)
  274. obj, err := deleteItem(dispatcher.JointModelManager(), item, ctx, userCred, query, data)
  275. if err == nil {
  276. CallCustomizeNotifyHook(ctx, userCred, ACT_DETACH, master, slave.GetShortDesc(ctx))
  277. OpsLog.LogDetachEvent(ctx, JointMaster(item), JointSlave(item), userCred, jsonutils.Marshal(item))
  278. }
  279. return obj, err
  280. }
  281. func DetachJoint(ctx context.Context, userCred mcclient.TokenCredential, item IJointModel) error {
  282. err := ValidateDeleteCondition(item, ctx, nil)
  283. if err != nil {
  284. return err
  285. }
  286. err = item.Delete(ctx, userCred)
  287. if err == nil {
  288. OpsLog.LogDetachEvent(ctx, JointMaster(item), JointSlave(item), userCred, item.GetShortDesc(ctx))
  289. }
  290. if err := item.GetModelManager().GetExtraHook().AfterPostDelete(ctx, userCred, item, nil); err != nil {
  291. logclient.AddActionLogWithContext(ctx, item, logclient.ACT_POST_DELETE_HOOK, err, userCred, false)
  292. }
  293. return err
  294. }