disk_migrate_task.go 9.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272
  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/pkg/errors"
  20. "yunion.io/x/pkg/utils"
  21. "yunion.io/x/onecloud/pkg/apis/compute"
  22. schedapi "yunion.io/x/onecloud/pkg/apis/scheduler"
  23. "yunion.io/x/onecloud/pkg/cloudcommon/db"
  24. "yunion.io/x/onecloud/pkg/cloudcommon/db/taskman"
  25. "yunion.io/x/onecloud/pkg/cloudcommon/notifyclient"
  26. "yunion.io/x/onecloud/pkg/compute/models"
  27. taskutils "yunion.io/x/onecloud/pkg/compute/tasks/utils"
  28. "yunion.io/x/onecloud/pkg/util/logclient"
  29. )
  30. type DiskMigrateTask struct {
  31. taskutils.SSchedTask
  32. }
  33. func init() {
  34. taskman.RegisterTask(DiskMigrateTask{})
  35. }
  36. func (task *DiskMigrateTask) TaskComplete(ctx context.Context, disk *models.SDisk) {
  37. task.SetStageComplete(ctx, nil)
  38. db.OpsLog.LogEvent(disk, db.ACT_MIGRATE, "Migrate success", task.UserCred)
  39. logclient.AddActionLogWithContext(ctx, disk, logclient.ACT_MIGRATE, task.Params, task.UserCred, true)
  40. }
  41. func (task *DiskMigrateTask) markFailed(ctx context.Context, disk *models.SDisk, reason jsonutils.JSONObject) {
  42. disk.SetStatus(ctx, task.UserCred, compute.DISK_MIGRATE_FAIL, reason.String())
  43. db.OpsLog.LogEvent(disk, db.ACT_MIGRATE_FAIL, reason, task.UserCred)
  44. logclient.AddActionLogWithContext(ctx, disk, logclient.ACT_MIGRATE, reason, task.UserCred, false)
  45. notifyclient.NotifySystemErrorWithCtx(ctx, disk.Id, disk.Name, compute.DISK_MIGRATE_FAIL, reason.String())
  46. notifyclient.EventNotify(ctx, task.GetUserCred(), notifyclient.SEventNotifyParam{
  47. Obj: disk,
  48. Action: notifyclient.ActionMigrate,
  49. IsFail: true,
  50. })
  51. }
  52. func (task *DiskMigrateTask) TaskFailed(ctx context.Context, disk *models.SDisk, reason jsonutils.JSONObject) {
  53. task.markFailed(ctx, disk, reason)
  54. task.SetStageFailed(ctx, reason)
  55. }
  56. func (task *DiskMigrateTask) GetSchedParams() (*schedapi.ScheduleInput, error) {
  57. obj := task.GetObject()
  58. disk := obj.(*models.SDisk)
  59. targetStorageId, _ := task.Params.GetString("target_storage_id")
  60. return disk.GetSchedMigrateParams(targetStorageId)
  61. }
  62. func (task *DiskMigrateTask) OnInit(ctx context.Context, obj db.IStandaloneModel, data jsonutils.JSONObject) {
  63. taskutils.StartScheduleObjects(ctx, task, []db.IStandaloneModel{obj})
  64. }
  65. func (task *DiskMigrateTask) OnScheduleFailCallback(ctx context.Context, obj taskutils.IScheduleModel, reason jsonutils.JSONObject, index int) {
  66. // do nothing
  67. }
  68. func (task *DiskMigrateTask) OnScheduleFailed(ctx context.Context, reason jsonutils.JSONObject) {
  69. obj := task.GetObject()
  70. disk := obj.(*models.SDisk)
  71. task.TaskFailed(ctx, disk, reason)
  72. }
  73. func (task *DiskMigrateTask) SaveScheduleResult(ctx context.Context, obj taskutils.IScheduleModel, candidate *schedapi.CandidateResource, index int) {
  74. disk := obj.(*models.SDisk)
  75. targetHostId := candidate.HostId
  76. storageIds := candidate.Disks[0].StorageIds
  77. targetHost := models.HostManager.FetchHostById(targetHostId)
  78. if targetHost == nil {
  79. task.TaskFailed(ctx, disk, jsonutils.NewString("target host not found?"))
  80. return
  81. }
  82. if len(storageIds) == 0 {
  83. task.TaskFailed(ctx, disk, jsonutils.NewString("no target storage found?"))
  84. return
  85. }
  86. var storageId string
  87. for i := range storageIds {
  88. if storageIds[i] != disk.StorageId {
  89. storageId = storageIds[i]
  90. break
  91. }
  92. }
  93. storage := models.StorageManager.FetchStorageById(storageId)
  94. if storage == nil {
  95. task.TaskFailed(ctx, disk, jsonutils.NewString("target storage not found?"))
  96. return
  97. }
  98. task.Params.Set("target_host_id", jsonutils.NewString(targetHostId))
  99. task.Params.Set("target_storage_id", jsonutils.NewString(storage.Id))
  100. task.SetStage("OnStorageCacheImage", nil)
  101. if disk.TemplateId != "" {
  102. format, err := disk.GetCacheImageFormat(ctx)
  103. if err != nil {
  104. task.TaskFailed(ctx, disk, jsonutils.NewString(fmt.Sprintf("disk get cache image format failed %s", err)))
  105. return
  106. }
  107. input := compute.CacheImageInput{
  108. ImageId: disk.GetTemplateId(),
  109. Format: format,
  110. ParentTaskId: task.GetTaskId(),
  111. }
  112. disk.SetStatus(ctx, task.UserCred, compute.DISK_IMAGE_CACHING, "On disk migrate save schedule result")
  113. storagecache := storage.GetStoragecache()
  114. storagecache.StartImageCacheTask(ctx, task.UserCred, input)
  115. } else {
  116. task.OnStorageCacheImage(ctx, disk, nil)
  117. }
  118. }
  119. func (task *DiskMigrateTask) OnStorageCacheImage(ctx context.Context, disk *models.SDisk, data jsonutils.JSONObject) {
  120. disk.SetStatus(ctx, task.UserCred, compute.DISK_MIGRATING, "On disk migrate start migrate")
  121. storage, err := disk.GetStorage()
  122. if err != nil {
  123. task.TaskFailed(ctx, disk, jsonutils.NewString(err.Error()))
  124. return
  125. }
  126. sourceHost, err := storage.GetMasterHost()
  127. if err != nil {
  128. task.TaskFailed(ctx, disk, jsonutils.NewString(err.Error()))
  129. return
  130. }
  131. driver, err := sourceHost.GetHostDriver()
  132. if err != nil {
  133. task.TaskFailed(ctx, disk, jsonutils.NewString(err.Error()))
  134. return
  135. }
  136. ret, err := driver.RequestDiskSrcMigratePrepare(ctx, sourceHost, disk, task)
  137. if err != nil {
  138. task.TaskFailed(ctx, disk, jsonutils.NewString(err.Error()))
  139. return
  140. }
  141. snapshotsUri := fmt.Sprintf("%s/download/snapshots/", sourceHost.ManagerUri)
  142. diskUri := fmt.Sprintf("%s/download/disks/", sourceHost.ManagerUri)
  143. body := jsonutils.NewDict()
  144. if ret != nil {
  145. body.Update(ret.(*jsonutils.JSONDict))
  146. }
  147. snapChain := []string{}
  148. if ret.Contains("disk_snaps_chain") {
  149. err = ret.Unmarshal(&snapChain, "disk_snaps_chain")
  150. if err != nil {
  151. task.TaskFailed(ctx, disk, jsonutils.NewString(errors.Wrap(err, "unmarshal snap chain").Error()))
  152. return
  153. }
  154. }
  155. outChainSnapshotIds := jsonutils.NewArray()
  156. snapshots := models.SnapshotManager.GetDiskSnapshots(disk.Id)
  157. for j := 0; j < len(snapshots); j++ {
  158. if !utils.IsInStringArray(snapshots[j].Id, snapChain) {
  159. outChainSnapshotIds.Add(jsonutils.NewString(snapshots[j].Id))
  160. }
  161. }
  162. body.Set("out_chain_snapshots", outChainSnapshotIds)
  163. body.Set("snapshots_uri", jsonutils.NewString(snapshotsUri))
  164. body.Set("disk_uri", jsonutils.NewString(diskUri))
  165. body.Set("src_storage_id", jsonutils.NewString(disk.StorageId))
  166. if disk.TemplateId != "" {
  167. body.Set("template_id", jsonutils.NewString(disk.TemplateId))
  168. }
  169. targetHostId, _ := task.Params.GetString("target_host_id")
  170. targetHost := models.HostManager.FetchHostById(targetHostId)
  171. targetStorageId, _ := task.Params.GetString("target_storage_id")
  172. targetStorage := models.StorageManager.FetchStorageById(targetStorageId)
  173. task.SetStage("OnDiskMigrate", nil)
  174. targetDriver, err := targetHost.GetHostDriver()
  175. if err != nil {
  176. task.TaskFailed(ctx, disk, jsonutils.NewString(errors.Wrap(err, "GetHostDriver").Error()))
  177. return
  178. }
  179. if err = targetDriver.RequestDiskMigrate(ctx, targetHost, targetStorage, disk, task, body); err != nil {
  180. task.TaskFailed(ctx, disk, jsonutils.NewString(fmt.Sprintf("failed request disk migrate %s", err)))
  181. return
  182. }
  183. }
  184. func (task *DiskMigrateTask) OnDiskMigrate(ctx context.Context, disk *models.SDisk, data jsonutils.JSONObject) {
  185. srcStorage, _ := disk.GetStorage()
  186. srcHost, _ := srcStorage.GetMasterHost()
  187. diskPath, _ := data.GetString("disk_path")
  188. targetStorageId, _ := task.Params.GetString("target_storage_id")
  189. _, err := db.Update(disk, func() error {
  190. //disk.Status = compute.DISK_READY
  191. disk.StorageId = targetStorageId
  192. if diskPath != "" {
  193. disk.AccessPath = diskPath
  194. }
  195. return nil
  196. })
  197. if err != nil {
  198. task.TaskFailed(ctx, disk, jsonutils.NewString(fmt.Sprintf("db failed update disk %s", err)))
  199. return
  200. }
  201. snapshots := models.SnapshotManager.GetDiskSnapshots(disk.Id)
  202. for _, snapshot := range snapshots {
  203. _, err := db.Update(&snapshot, func() error {
  204. snapshot.StorageId = targetStorageId
  205. return nil
  206. })
  207. if err != nil {
  208. task.TaskFailed(ctx, disk, jsonutils.NewString(fmt.Sprintf("db failed update disk snapshot %s %s", snapshot.Id, err)))
  209. return
  210. }
  211. }
  212. task.SetStage("OnDeallocateSourceDisk", nil)
  213. driver, err := srcHost.GetHostDriver()
  214. if err != nil {
  215. task.TaskFailed(ctx, disk, jsonutils.NewString(fmt.Sprintf("GetHostDriver: %v", err)))
  216. return
  217. }
  218. err = driver.RequestDeallocateDiskOnHost(ctx, srcHost, srcStorage, disk, true, task)
  219. if err != nil {
  220. task.TaskFailed(ctx, disk, jsonutils.NewString(fmt.Sprintf("failed deallocate disk on src storage %s", err)))
  221. return
  222. }
  223. }
  224. func (task *DiskMigrateTask) OnDiskMigrateFailed(ctx context.Context, disk *models.SDisk, data jsonutils.JSONObject) {
  225. task.TaskFailed(ctx, disk, data)
  226. }
  227. func (task *DiskMigrateTask) OnDeallocateSourceDisk(ctx context.Context, disk *models.SDisk, data jsonutils.JSONObject) {
  228. db.Update(disk, func() error {
  229. disk.Status = compute.DISK_READY
  230. return nil
  231. })
  232. task.SetStageComplete(ctx, nil)
  233. db.OpsLog.LogEvent(disk, db.ACT_MIGRATE, "OnDeallocateSourceDisk", task.UserCred)
  234. logclient.AddActionLogWithContext(ctx, disk, logclient.ACT_MIGRATE, task.Params, task.UserCred, true)
  235. }
  236. func (task *DiskMigrateTask) OnDeallocateSourceDiskFailed(ctx context.Context, disk *models.SDisk, data jsonutils.JSONObject) {
  237. task.TaskFailed(ctx, disk, data)
  238. }