guest_delete_task.go 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363
  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 guest
  15. import (
  16. "context"
  17. "fmt"
  18. "yunion.io/x/cloudmux/pkg/cloudprovider"
  19. "yunion.io/x/jsonutils"
  20. "yunion.io/x/log"
  21. "yunion.io/x/pkg/errors"
  22. "yunion.io/x/pkg/utils"
  23. api "yunion.io/x/onecloud/pkg/apis/compute"
  24. "yunion.io/x/onecloud/pkg/cloudcommon/db"
  25. "yunion.io/x/onecloud/pkg/cloudcommon/db/taskman"
  26. "yunion.io/x/onecloud/pkg/cloudcommon/notifyclient"
  27. "yunion.io/x/onecloud/pkg/compute/models"
  28. "yunion.io/x/onecloud/pkg/compute/options"
  29. "yunion.io/x/onecloud/pkg/util/logclient"
  30. )
  31. type BaseGuestDeleteTask struct {
  32. SGuestBaseTask
  33. }
  34. var (
  35. STORAGEIDS = "storage_ids"
  36. )
  37. func init() {
  38. taskman.RegisterTask(BaseGuestDeleteTask{})
  39. }
  40. func (deleteTask *BaseGuestDeleteTask) OnInit(ctx context.Context, obj db.IStandaloneModel, data jsonutils.JSONObject) {
  41. guest := obj.(*models.SGuest)
  42. host, _ := guest.GetHost()
  43. if guest.Hypervisor == api.HYPERVISOR_BAREMETAL && host != nil && host.HostType != api.HOST_TYPE_BAREMETAL {
  44. // if a fake server for converted hypervisor, then just skip stop
  45. deleteTask.OnGuestStopComplete(ctx, guest, data)
  46. return
  47. }
  48. drv, err := guest.GetDriver()
  49. if err != nil {
  50. deleteTask.OnGuestStopComplete(ctx, guest, data)
  51. return
  52. }
  53. if len(guest.BackupHostId) > 0 {
  54. deleteTask.SetStage("OnMasterHostStopGuestComplete", nil)
  55. if err := drv.RequestStopGuestForDelete(ctx, guest, nil, deleteTask); err != nil {
  56. log.Errorf("RequestStopGuestForDelete fail %s", err)
  57. deleteTask.OnMasterHostStopGuestComplete(ctx, guest, nil)
  58. }
  59. } else {
  60. deleteTask.SetStage("OnGuestStopComplete", nil)
  61. if err := drv.RequestStopGuestForDelete(ctx, guest, nil, deleteTask); err != nil {
  62. log.Errorf("RequestStopGuestForDelete fail %s", err)
  63. deleteTask.OnGuestStopComplete(ctx, guest, nil)
  64. }
  65. }
  66. }
  67. func (deleteTask *BaseGuestDeleteTask) OnMasterHostStopGuestComplete(ctx context.Context, guest *models.SGuest, data jsonutils.JSONObject) {
  68. deleteTask.SetStage("OnGuestStopComplete", nil)
  69. host := models.HostManager.FetchHostById(guest.BackupHostId)
  70. drv, err := guest.GetDriver()
  71. if err != nil {
  72. deleteTask.OnGuestStopComplete(ctx, guest, nil)
  73. return
  74. }
  75. err = drv.RequestStopGuestForDelete(ctx, guest, host, deleteTask)
  76. if err != nil {
  77. log.Errorf("RequestStopGuestForDelete fail %s", err)
  78. deleteTask.OnGuestStopComplete(ctx, guest, nil)
  79. }
  80. }
  81. func (deleteTask *BaseGuestDeleteTask) OnMasterHostStopGuestCompleteFailed(ctx context.Context, guest *models.SGuest, data jsonutils.JSONObject) {
  82. deleteTask.OnGuestStopComplete(ctx, guest, nil) // ignore stop error
  83. }
  84. func (deleteTask *BaseGuestDeleteTask) StartDeleteGuestSnapshots(ctx context.Context, guest *models.SGuest) {
  85. guest.StartDeleteGuestSnapshots(ctx, deleteTask.UserCred, deleteTask.GetTaskId())
  86. }
  87. func (deleteTask *BaseGuestDeleteTask) OnGuestStopComplete(ctx context.Context, guest *models.SGuest, data jsonutils.JSONObject) {
  88. if jsonutils.QueryBoolean(deleteTask.Params, "delete_snapshots", false) {
  89. deleteTask.SetStage("OnStartEipDissociate", nil)
  90. guest.StartDeleteGuestSnapshots(ctx, deleteTask.UserCred, deleteTask.Id)
  91. return
  92. }
  93. deleteTask.OnStartEipDissociate(ctx, guest, data)
  94. }
  95. func (deleteTask *BaseGuestDeleteTask) OnGuestStopCompleteFailed(ctx context.Context, guest *models.SGuest, err jsonutils.JSONObject) {
  96. if len(guest.ExternalId) > 0 {
  97. _, e := guest.GetIVM(ctx)
  98. if errors.Cause(e) == cloudprovider.ErrNotFound {
  99. deleteTask.Params.Set("override_pending_delete", jsonutils.JSONTrue)
  100. }
  101. }
  102. deleteTask.OnGuestStopComplete(ctx, guest, err) // ignore stop error
  103. }
  104. func (deleteTask *BaseGuestDeleteTask) OnStartEipDissociateFailed(ctx context.Context, guest *models.SGuest, data jsonutils.JSONObject) {
  105. log.Errorf("Delete guest snapshots faield: %s", data)
  106. deleteTask.OnStartEipDissociate(ctx, guest, nil)
  107. }
  108. func (deleteTask *BaseGuestDeleteTask) OnStartEipDissociate(ctx context.Context, guest *models.SGuest, data jsonutils.JSONObject) {
  109. if sourceGuestId := guest.GetMetadata(ctx, api.SERVER_META_CONVERT_FROM_ESXI, deleteTask.UserCred); len(sourceGuestId) > 0 {
  110. sourceGuest := models.GuestManager.FetchGuestById(sourceGuestId)
  111. if sourceGuest != nil &&
  112. sourceGuest.GetMetadata(ctx, api.SERVER_META_CONVERTED_SERVER, deleteTask.UserCred) == guest.Id {
  113. sourceGuest.RemoveMetadata(ctx, api.SERVER_META_CONVERTED_SERVER, deleteTask.UserCred)
  114. sourceGuest.StartSyncstatus(ctx, deleteTask.UserCred, "")
  115. }
  116. }
  117. eip, _ := guest.GetEipOrPublicIp()
  118. if eip != nil && eip.Mode != api.EIP_MODE_INSTANCE_PUBLICIP {
  119. // detach floating EIP only
  120. if jsonutils.QueryBoolean(deleteTask.Params, "purge", false) {
  121. // purge locally
  122. eip.Dissociate(ctx, deleteTask.UserCred)
  123. deleteTask.OnEipDissociateComplete(ctx, guest, nil)
  124. } else {
  125. deleteTask.SetStage("OnEipDissociateComplete", nil)
  126. autoDelete := jsonutils.QueryBoolean(deleteTask.GetParams(), "delete_eip", false)
  127. eip.StartEipDissociateTask(ctx, deleteTask.UserCred, autoDelete, deleteTask.GetTaskId())
  128. }
  129. } else {
  130. deleteTask.OnEipDissociateComplete(ctx, guest, nil)
  131. }
  132. }
  133. func (deleteTask *BaseGuestDeleteTask) OnEipDissociateCompleteFailed(ctx context.Context, obj db.IStandaloneModel, err jsonutils.JSONObject) {
  134. guest := obj.(*models.SGuest)
  135. deleteTask.OnFailed(ctx, guest, err)
  136. }
  137. func (deleteTask *BaseGuestDeleteTask) OnEipDissociateComplete(ctx context.Context, obj db.IStandaloneModel, data jsonutils.JSONObject) {
  138. deleteTask.SetStage("OnDiskDetachComplete", nil)
  139. deleteTask.OnDiskDetachComplete(ctx, obj, data)
  140. }
  141. // remove detachable disks
  142. func (deleteTask *BaseGuestDeleteTask) OnDiskDetachComplete(ctx context.Context, obj db.IStandaloneModel, data jsonutils.JSONObject) {
  143. log.Debugf("OnDiskDetachComplete")
  144. guest := obj.(*models.SGuest)
  145. guestdisksOrigin, _ := guest.GetGuestDisks()
  146. var guestdisks []models.SGuestdisk
  147. // clean dirty data
  148. for i := range guestdisksOrigin {
  149. if guestdisksOrigin[i].GetDisk() == nil {
  150. guestdisksOrigin[i].Detach(ctx, deleteTask.UserCred)
  151. } else {
  152. guestdisks = append(guestdisks, guestdisksOrigin[i])
  153. }
  154. }
  155. if len(guestdisks) == 0 {
  156. // on guest disks detached
  157. deleteTask.doClearGPUDevicesComplete(ctx, guest)
  158. return
  159. }
  160. // detach last detachable disk
  161. lastDisk := guestdisks[len(guestdisks)-1].GetDisk()
  162. deleteDisks := jsonutils.QueryBoolean(deleteTask.Params, "delete_disks", false)
  163. if deleteDisks {
  164. lastDisk.SetAutoDelete(lastDisk, deleteTask.GetUserCred(), true)
  165. }
  166. log.Debugf("lastDisk IsDetachable?? %v", lastDisk.IsDetachable())
  167. if !lastDisk.IsDetachable() {
  168. // no more disk need detach
  169. deleteTask.doClearGPUDevicesComplete(ctx, guest)
  170. return
  171. }
  172. purge := jsonutils.QueryBoolean(deleteTask.Params, "purge", false)
  173. guest.StartGuestDetachdiskTask(ctx, deleteTask.UserCred, lastDisk, true, deleteTask.GetTaskId(), purge, false)
  174. }
  175. func (deleteTask *BaseGuestDeleteTask) OnDiskDetachCompleteFailed(ctx context.Context, obj db.IStandaloneModel, err jsonutils.JSONObject) {
  176. guest := obj.(*models.SGuest)
  177. deleteTask.OnFailed(ctx, guest, err)
  178. }
  179. // clean gpu devices
  180. func (deleteTask *BaseGuestDeleteTask) doClearGPUDevicesComplete(ctx context.Context, guest *models.SGuest) {
  181. log.Debugf("doClearGPUDevicesComplete")
  182. models.IsolatedDeviceManager.ReleaseGPUDevicesOfGuest(ctx, guest, deleteTask.UserCred)
  183. if jsonutils.QueryBoolean(deleteTask.Params, "purge", false) {
  184. deleteTask.OnSyncConfigComplete(ctx, guest, nil)
  185. } else {
  186. deleteTask.SetStage("OnSyncConfigComplete", nil)
  187. guest.StartSyncTaskWithoutSyncstatus(ctx, deleteTask.UserCred, false, deleteTask.GetTaskId())
  188. }
  189. }
  190. func (deleteTask *BaseGuestDeleteTask) OnSyncConfigComplete(ctx context.Context, obj db.IStandaloneModel, data jsonutils.JSONObject) {
  191. guest := obj.(*models.SGuest)
  192. // try to leave all groups
  193. guest.LeaveAllGroups(ctx, deleteTask.UserCred)
  194. // cleanup tap services and flows
  195. guest.CleanTapRecords(ctx, deleteTask.UserCred)
  196. isPurge := jsonutils.QueryBoolean(deleteTask.Params, "purge", false)
  197. overridePendingDelete := jsonutils.QueryBoolean(deleteTask.Params, "override_pending_delete", false)
  198. if options.Options.EnablePendingDelete && !isPurge && !overridePendingDelete {
  199. if guest.PendingDeleted {
  200. deleteTask.SetStageComplete(ctx, nil)
  201. return
  202. }
  203. log.Debugf("XXXXXXX Do guest pending delete... XXXXXXX")
  204. // pending detach
  205. guest.PendingDetachScalingGroup()
  206. guestStatus, _ := deleteTask.Params.GetString("guest_status")
  207. if !utils.IsInStringArray(guestStatus, []string{
  208. api.VM_SCHEDULE_FAILED, api.VM_NETWORK_FAILED,
  209. api.VM_CREATE_FAILED, api.VM_DEVICE_FAILED}) {
  210. deleteTask.StartPendingDeleteGuest(ctx, guest)
  211. return
  212. }
  213. }
  214. log.Debugf("XXXXXXX Do real delete on guest ... XXXXXXX")
  215. deleteTask.doStartDeleteGuest(ctx, guest)
  216. }
  217. func (deleteTask *BaseGuestDeleteTask) OnSyncConfigCompleteFailed(ctx context.Context, obj db.IStandaloneModel, err jsonutils.JSONObject) {
  218. // guest := obj.(*models.SGuest)
  219. // deleteTask.OnFailed(ctx, guest, err)
  220. deleteTask.OnSyncConfigComplete(ctx, obj, err) // ignore sync config failed error
  221. }
  222. func (deleteTask *BaseGuestDeleteTask) OnGuestDeleteFailed(ctx context.Context, obj db.IStandaloneModel, err jsonutils.JSONObject) {
  223. guest := obj.(*models.SGuest)
  224. deleteTask.OnFailed(ctx, guest, err)
  225. }
  226. func (deleteTask *BaseGuestDeleteTask) doStartDeleteGuest(ctx context.Context, obj db.IStandaloneModel) {
  227. guest := obj.(*models.SGuest)
  228. guest.SetStatus(ctx, deleteTask.UserCred, api.VM_DELETING, "delete server after stop")
  229. db.OpsLog.LogEvent(guest, db.ACT_DELOCATING, guest.GetShortDesc(ctx), deleteTask.UserCred)
  230. deleteTask.StartDeleteGuest(ctx, guest)
  231. }
  232. func (deleteTask *BaseGuestDeleteTask) StartPendingDeleteGuest(ctx context.Context, guest *models.SGuest) {
  233. guest.DoPendingDelete(ctx, deleteTask.UserCred)
  234. deleteTask.SetStage("OnPendingDeleteComplete", nil)
  235. guest.StartSyncstatus(ctx, deleteTask.UserCred, deleteTask.GetTaskId())
  236. }
  237. func (deleteTask *BaseGuestDeleteTask) OnPendingDeleteComplete(ctx context.Context, obj db.IStandaloneModel, data jsonutils.JSONObject) {
  238. deleteTask.SetStageComplete(ctx, jsonutils.NewDict())
  239. }
  240. func (deleteTask *BaseGuestDeleteTask) OnPendingDeleteCompleteFailed(ctx context.Context, obj db.IStandaloneModel, data jsonutils.JSONObject) {
  241. deleteTask.OnPendingDeleteComplete(ctx, obj, nil)
  242. }
  243. func (deleteTask *BaseGuestDeleteTask) StartDeleteGuest(ctx context.Context, guest *models.SGuest) {
  244. // Temporary storageids to sync capacityUsed after delete
  245. {
  246. storages, _ := guest.GetStorages()
  247. storageIds := make([]string, len(storages))
  248. for i := range storages {
  249. storageIds[i] = storages[i].GetId()
  250. }
  251. deleteTask.Params.Set(STORAGEIDS, jsonutils.NewStringArray(storageIds))
  252. }
  253. // No snapshot
  254. deleteTask.SetStage("OnGuestDetachDisksComplete", nil)
  255. drv, err := guest.GetDriver()
  256. if err != nil {
  257. deleteTask.OnGuestDeleteFailed(ctx, guest, jsonutils.NewString(err.Error()))
  258. return
  259. }
  260. drv.RequestDetachDisksFromGuestForDelete(ctx, guest, deleteTask)
  261. }
  262. func (deleteTask *BaseGuestDeleteTask) OnGuestDetachDisksComplete(ctx context.Context, obj db.IStandaloneModel, data jsonutils.JSONObject) {
  263. guest := obj.(*models.SGuest)
  264. deleteTask.DoDeleteGuest(ctx, guest)
  265. }
  266. func (deleteTask *BaseGuestDeleteTask) OnGuestDetachDisksCompleteFailed(ctx context.Context, obj db.IStandaloneModel, data jsonutils.JSONObject) {
  267. deleteTask.OnGuestDeleteFailed(ctx, obj, data)
  268. }
  269. func (deleteTask *BaseGuestDeleteTask) DoDeleteGuest(ctx context.Context, guest *models.SGuest) {
  270. models.IsolatedDeviceManager.ReleaseDevicesOfGuest(ctx, guest, deleteTask.UserCred)
  271. host, _ := guest.GetHost()
  272. if guest.IsPrepaidRecycle() {
  273. err := host.BorrowIpAddrsFromGuest(ctx, deleteTask.UserCred, guest)
  274. if err != nil {
  275. msg := fmt.Sprintf("host.BorrowIpAddrsFromGuest fail %s", err)
  276. log.Errorf("%v", msg)
  277. deleteTask.OnGuestDeleteFailed(ctx, guest, jsonutils.NewString(msg))
  278. return
  279. }
  280. deleteTask.OnGuestDeleteComplete(ctx, guest, nil)
  281. } else if (host == nil || !host.GetEnabled()) && jsonutils.QueryBoolean(deleteTask.Params, "purge", false) {
  282. deleteTask.OnGuestDeleteComplete(ctx, guest, nil)
  283. } else {
  284. deleteTask.SetStage("OnGuestDeleteComplete", nil)
  285. guest.StartUndeployGuestTask(ctx, deleteTask.UserCred, deleteTask.GetTaskId(), "")
  286. }
  287. }
  288. func (deleteTask *BaseGuestDeleteTask) OnFailed(ctx context.Context, guest *models.SGuest, err jsonutils.JSONObject) {
  289. guest.SetStatus(ctx, deleteTask.UserCred, api.VM_DELETE_FAIL, err.String())
  290. db.OpsLog.LogEvent(guest, db.ACT_DELOCATE_FAIL, err, deleteTask.UserCred)
  291. logclient.AddActionLogWithStartable(deleteTask, guest, logclient.ACT_DELOCATE, err, deleteTask.UserCred, false)
  292. notifyclient.EventNotify(ctx, deleteTask.GetUserCred(), notifyclient.SEventNotifyParam{
  293. Obj: guest,
  294. Action: notifyclient.ActionDelete,
  295. IsFail: true,
  296. })
  297. deleteTask.SetStageFailed(ctx, err)
  298. }
  299. func (deleteTask *BaseGuestDeleteTask) OnGuestDeleteCompleteFailed(ctx context.Context, obj db.IStandaloneModel, err jsonutils.JSONObject) {
  300. guest := obj.(*models.SGuest)
  301. deleteTask.OnFailed(ctx, guest, err)
  302. }
  303. func (deleteTask *BaseGuestDeleteTask) OnGuestDeleteComplete(ctx context.Context, obj db.IStandaloneModel, data jsonutils.JSONObject) {
  304. guest := obj.(*models.SGuest)
  305. guest.RevokeAllNetworkSecgroups(ctx, deleteTask.UserCred)
  306. guest.DetachAllNetworks(ctx, deleteTask.UserCred)
  307. guest.EjectAllIso(deleteTask.UserCred)
  308. guest.EjectAllVfd(deleteTask.UserCred)
  309. guest.DeleteEip(ctx, deleteTask.UserCred)
  310. drv, _ := guest.GetDriver()
  311. if drv != nil {
  312. drv.OnDeleteGuestFinalCleanup(ctx, guest, deleteTask.UserCred)
  313. }
  314. deleteTask.DeleteGuest(ctx, guest)
  315. }
  316. func (deleteTask *BaseGuestDeleteTask) DeleteGuest(ctx context.Context, guest *models.SGuest) {
  317. data := jsonutils.NewDict()
  318. data.Set("real_delete", jsonutils.JSONTrue)
  319. deleteTask.SetStageComplete(ctx, data)
  320. }
  321. func (deleteTask *BaseGuestDeleteTask) NotifyServerDeleted(ctx context.Context, guest *models.SGuest) {
  322. guest.EventNotify(ctx, deleteTask.UserCred, notifyclient.ActionPendingDelete)
  323. }