disk_reset_task.go 7.9 KB

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