disk_reset_task.go 8.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246
  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 disk
  15. import (
  16. "context"
  17. "fmt"
  18. "yunion.io/x/jsonutils"
  19. "yunion.io/x/log"
  20. "yunion.io/x/pkg/errors"
  21. api "yunion.io/x/onecloud/pkg/apis/compute"
  22. "yunion.io/x/onecloud/pkg/cloudcommon/db"
  23. "yunion.io/x/onecloud/pkg/cloudcommon/db/taskman"
  24. "yunion.io/x/onecloud/pkg/cloudcommon/notifyclient"
  25. "yunion.io/x/onecloud/pkg/compute/models"
  26. "yunion.io/x/onecloud/pkg/util/logclient"
  27. )
  28. type DiskResetTask struct {
  29. SDiskBaseTask
  30. }
  31. func init() {
  32. taskman.RegisterTask(DiskResetTask{})
  33. taskman.RegisterTask(DiskCleanUpSnapshotsTask{})
  34. }
  35. func (self *DiskResetTask) getSnapshot() (*models.SSnapshot, error) {
  36. snapshotId, err := self.Params.GetString("snapshot_id")
  37. if err != nil {
  38. return nil, errors.Wrap(err, "Get snapshotId")
  39. }
  40. snapshot, err := models.SnapshotManager.FetchById(snapshotId)
  41. if err != nil {
  42. return nil, errors.Wrapf(err, "SnapshotManager.FetchById(%s)", snapshotId)
  43. }
  44. return snapshot.(*models.SSnapshot), nil
  45. }
  46. func (self *DiskResetTask) TaskFailed(ctx context.Context, disk *models.SDisk, reason error) {
  47. disk.SetStatus(ctx, self.UserCred, api.DISK_READY, "")
  48. logclient.AddActionLogWithStartable(self, disk, logclient.ACT_RESET_DISK, reason, self.UserCred, false)
  49. notifyclient.EventNotify(ctx, self.UserCred, notifyclient.SEventNotifyParam{
  50. Obj: disk,
  51. Action: notifyclient.ActionReset,
  52. IsFail: true,
  53. })
  54. snapshot, _ := self.getSnapshot()
  55. if snapshot != nil {
  56. logclient.AddActionLogWithStartable(self, snapshot, logclient.ACT_RESET_DISK, reason, self.UserCred, false)
  57. }
  58. self.SetStageFailed(ctx, jsonutils.Marshal(reason))
  59. guests := disk.GetGuests()
  60. if len(guests) == 1 {
  61. guests[0].SetStatus(ctx, self.UserCred, api.VM_DISK_RESET_FAIL, reason.Error())
  62. }
  63. }
  64. func (self *DiskResetTask) TaskCompleted(ctx context.Context, disk *models.SDisk, data *jsonutils.JSONDict) {
  65. notifyclient.EventNotify(ctx, self.GetUserCred(), notifyclient.SEventNotifyParam{
  66. Obj: disk,
  67. Action: notifyclient.ActionReset,
  68. })
  69. guests := disk.GetGuests()
  70. if jsonutils.QueryBoolean(self.Params, "auto_start", false) {
  71. if len(guests) == 1 {
  72. self.SetStage("OnStartGuest", nil)
  73. guests[0].StartGueststartTask(ctx, self.UserCred, nil, self.GetTaskId())
  74. }
  75. } else {
  76. if len(guests) == 1 && !self.IsSubtask() {
  77. guests[0].SetStatus(ctx, self.UserCred, api.VM_READY, "")
  78. }
  79. // data不能为空指针,否则会导致AddActionLog抛空指针异常
  80. if data == nil {
  81. data = jsonutils.NewDict()
  82. }
  83. logclient.AddActionLogWithStartable(self, disk, logclient.ACT_RESET_DISK, data, self.UserCred, true)
  84. snapshot, _ := self.getSnapshot()
  85. if snapshot != nil {
  86. logclient.AddActionLogWithStartable(self, snapshot, logclient.ACT_RESET_DISK, data, self.UserCred, true)
  87. }
  88. self.SetStageComplete(ctx, data)
  89. }
  90. }
  91. func (self *DiskResetTask) OnStartGuest(ctx context.Context, disk *models.SDisk, data jsonutils.JSONObject) {
  92. if data == nil {
  93. data = jsonutils.NewDict()
  94. }
  95. logclient.AddActionLogWithStartable(self, disk, logclient.ACT_RESET_DISK, data, self.UserCred, true)
  96. snapshot, _ := self.getSnapshot()
  97. if snapshot != nil {
  98. logclient.AddActionLogWithStartable(self, snapshot, logclient.ACT_RESET_DISK, data, self.UserCred, true)
  99. }
  100. self.SetStageComplete(ctx, nil)
  101. }
  102. func (self *DiskResetTask) OnInit(ctx context.Context, obj db.IStandaloneModel, data jsonutils.JSONObject) {
  103. disk := obj.(*models.SDisk)
  104. guest := disk.GetGuest()
  105. var host *models.SHost
  106. if guest == nil {
  107. storage, err := disk.GetStorage()
  108. if err != nil {
  109. self.TaskFailed(ctx, disk, errors.Wrapf(err, "disk.GetStorage"))
  110. return
  111. }
  112. host, err = storage.GetMasterHost()
  113. if err != nil {
  114. self.TaskFailed(ctx, disk, errors.Wrapf(err, "storage.GetMasterHost"))
  115. return
  116. }
  117. } else {
  118. var err error
  119. host, err = guest.GetHost()
  120. if err != nil {
  121. self.TaskFailed(ctx, disk, errors.Wrapf(err, "guest.GetHost"))
  122. return
  123. }
  124. }
  125. self.RequestResetDisk(ctx, disk, host)
  126. }
  127. func (self *DiskResetTask) RequestResetDisk(ctx context.Context, disk *models.SDisk, host *models.SHost) {
  128. snapshot, err := self.getSnapshot()
  129. if err != nil {
  130. self.TaskFailed(ctx, disk, errors.Wrap(err, "getSnapshot"))
  131. return
  132. }
  133. params := snapshot.GetRegionDriver().GetDiskResetParams(snapshot)
  134. self.SetStage("OnRequestResetDisk", nil)
  135. driver, err := host.GetHostDriver()
  136. if err != nil {
  137. self.TaskFailed(ctx, disk, errors.Wrap(err, "GetHostDriver"))
  138. return
  139. }
  140. err = driver.RequestResetDisk(ctx, host, disk, params, self)
  141. if err != nil {
  142. self.TaskFailed(ctx, disk, errors.Wrap(err, "RequestResetDisk"))
  143. }
  144. }
  145. func (self *DiskResetTask) OnRequestResetDiskFailed(ctx context.Context, disk *models.SDisk, data jsonutils.JSONObject) {
  146. self.TaskFailed(ctx, disk, fmt.Errorf("%s", data.String()))
  147. }
  148. func (self *DiskResetTask) OnRequestResetDisk(ctx context.Context, disk *models.SDisk, data jsonutils.JSONObject) {
  149. snapshot, err := self.getSnapshot()
  150. if err != nil {
  151. self.TaskFailed(ctx, disk, errors.Wrap(err, "getSnapshot"))
  152. return
  153. }
  154. err = snapshot.GetRegionDriver().OnDiskReset(ctx, self.UserCred, disk, snapshot, data)
  155. if err != nil {
  156. self.TaskFailed(ctx, disk, errors.Wrap(err, "OnDiskReset"))
  157. return
  158. }
  159. disk.SetStatus(ctx, self.UserCred, api.DISK_READY, "")
  160. self.TaskCompleted(ctx, disk, nil)
  161. }
  162. type DiskCleanUpSnapshotsTask struct {
  163. SDiskBaseTask
  164. }
  165. func (self *DiskCleanUpSnapshotsTask) OnInit(ctx context.Context, obj db.IStandaloneModel, data jsonutils.JSONObject) {
  166. disk := obj.(*models.SDisk)
  167. self.StartCleanUpSnapshots(ctx, disk)
  168. }
  169. func (self *DiskCleanUpSnapshotsTask) StartCleanUpSnapshots(ctx context.Context, disk *models.SDisk) {
  170. db.OpsLog.LogEvent(disk, db.ACT_DISK_CLEAN_UP_SNAPSHOTS,
  171. fmt.Sprintf("start clean up disk snapshots: %s", self.Params.String()), self.UserCred)
  172. var host *models.SHost
  173. guests := disk.GetGuests()
  174. if len(guests) == 1 {
  175. host, _ = guests[0].GetHost()
  176. } else {
  177. self.SetStageFailed(ctx, jsonutils.NewString("Disk can't get guest"))
  178. return
  179. }
  180. self.SetStage("OnCleanUpSnapshots", nil)
  181. driver, err := host.GetHostDriver()
  182. if err != nil {
  183. self.SetStageFailed(ctx, jsonutils.NewString(errors.Wrapf(err, "GetHostDriver").Error()))
  184. return
  185. }
  186. err = driver.RequestCleanUpDiskSnapshots(ctx, host, disk, self.Params, self)
  187. if err != nil {
  188. self.SetStageFailed(ctx, jsonutils.NewString(err.Error()))
  189. }
  190. }
  191. func (self *DiskCleanUpSnapshotsTask) OnCleanUpSnapshots(ctx context.Context, disk *models.SDisk, data jsonutils.JSONObject) {
  192. convertSnapshots, _ := self.Params.GetArray("convert_snapshots")
  193. for i := 0; i < len(convertSnapshots); i++ {
  194. snapshot_id, _ := convertSnapshots[i].GetString()
  195. iSnapshot, err := models.SnapshotManager.FetchById(snapshot_id)
  196. if err != nil {
  197. log.Errorf("OnCleanUpSnapshots Fetch snapshot by id(%s) error:%s", snapshot_id, err.Error())
  198. continue
  199. }
  200. snapshot := iSnapshot.(*models.SSnapshot)
  201. db.Update(snapshot, func() error {
  202. snapshot.OutOfChain = true
  203. return nil
  204. })
  205. }
  206. deleteSnapshots, _ := self.Params.GetArray("delete_snapshots")
  207. for i := 0; i < len(deleteSnapshots); i++ {
  208. snapshot_id, _ := deleteSnapshots[i].GetString()
  209. iSnapshot, err := models.SnapshotManager.FetchById(snapshot_id)
  210. if err != nil {
  211. log.Errorf("OnCleanUpSnapshots Fetch snapshot by id(%s) error:%s", snapshot_id, err.Error())
  212. continue
  213. }
  214. snapshot := iSnapshot.(*models.SSnapshot)
  215. snapshot.RealDelete(ctx, self.UserCred)
  216. }
  217. self.SetStageComplete(ctx, nil)
  218. }
  219. func (self *DiskCleanUpSnapshotsTask) OnCleanUpSnapshotsFailed(ctx context.Context, disk *models.SDisk, data jsonutils.JSONObject) {
  220. db.OpsLog.LogEvent(disk, db.ACT_DISK_CLEAN_UP_SNAPSHOTS_FAIL, data, self.UserCred)
  221. self.SetStageFailed(ctx, data)
  222. }