ha_guest_start_task.go 5.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157
  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/jsonutils"
  19. api "yunion.io/x/onecloud/pkg/apis/compute"
  20. "yunion.io/x/onecloud/pkg/cloudcommon/db"
  21. "yunion.io/x/onecloud/pkg/cloudcommon/db/taskman"
  22. "yunion.io/x/onecloud/pkg/compute/models"
  23. )
  24. func init() {
  25. taskman.RegisterTask(HAGuestStartTask{})
  26. }
  27. type HAGuestStartTask struct {
  28. GuestStartTask
  29. }
  30. func (self *HAGuestStartTask) OnInit(
  31. ctx context.Context, obj db.IStandaloneModel, data jsonutils.JSONObject,
  32. ) {
  33. guest := obj.(*models.SGuest)
  34. host := models.HostManager.FetchHostById(guest.BackupHostId)
  35. if host.HostStatus != api.HOST_ONLINE {
  36. guest.SetGuestBackupMirrorJobFailed(ctx, self.UserCred)
  37. // request start master guest
  38. self.GuestStartTask.OnInit(ctx, guest, nil)
  39. } else {
  40. self.RequestStopBackupGuest(ctx, guest)
  41. }
  42. }
  43. func (self *HAGuestStartTask) RequestStopBackupGuest(ctx context.Context, guest *models.SGuest) {
  44. host := models.HostManager.FetchHostById(guest.BackupHostId)
  45. self.SetStage("OnBackupGuestStopComplete", nil)
  46. guest.SetStatus(ctx, self.UserCred, api.VM_BACKUP_STOPING, "HAGuestStartTask")
  47. drv, err := guest.GetDriver()
  48. if err != nil {
  49. guest.SetStatus(ctx, self.UserCred, api.VM_BACKUP_START_FAILED, err.Error())
  50. self.SetStageFailed(ctx, nil)
  51. return
  52. }
  53. err = drv.RequestStopOnHost(ctx, guest, host, self, false)
  54. if err != nil {
  55. guest.SetStatus(ctx, self.UserCred, api.VM_BACKUP_START_FAILED, err.Error())
  56. self.SetStageFailed(ctx, nil)
  57. }
  58. }
  59. func (self *HAGuestStartTask) OnBackupGuestStopComplete(
  60. ctx context.Context, guest *models.SGuest, data jsonutils.JSONObject,
  61. ) {
  62. db.OpsLog.LogEvent(guest, db.ACT_STARTING, nil, self.UserCred)
  63. self.RequestStartBacking(ctx, guest)
  64. }
  65. func (self *HAGuestStartTask) OnBackupGuestStopCompleteFailed(
  66. ctx context.Context, guest *models.SGuest, data jsonutils.JSONObject,
  67. ) {
  68. guest.SetStatus(ctx, self.UserCred, api.VM_BACKUP_START_FAILED, data.String())
  69. self.SetStageFailed(ctx, data)
  70. }
  71. func (self *HAGuestStartTask) RequestStartBacking(ctx context.Context, guest *models.SGuest) {
  72. self.SetStage("OnStartBackupGuestComplete", nil)
  73. host := models.HostManager.FetchHostById(guest.BackupHostId)
  74. guest.SetStatus(ctx, self.UserCred, api.VM_BACKUP_STARTING, "")
  75. if !guest.IsGuestBackupMirrorJobReady(ctx, self.UserCred) {
  76. hostMaster := models.HostManager.FetchHostById(guest.HostId)
  77. self.Params.Set("block_ready", jsonutils.JSONFalse)
  78. diskUri := fmt.Sprintf("%s/disks", hostMaster.GetFetchUrl(true))
  79. self.Params.Set("disk_uri", jsonutils.NewString(diskUri))
  80. } else {
  81. self.Params.Set("block_ready", jsonutils.JSONTrue)
  82. }
  83. drv, err := guest.GetDriver()
  84. if err != nil {
  85. self.OnStartCompleteFailed(ctx, guest, jsonutils.NewString(err.Error()))
  86. return
  87. }
  88. err = drv.RequestStartOnHost(ctx, guest, host, self.UserCred, self)
  89. if err != nil {
  90. self.OnStartCompleteFailed(ctx, guest, jsonutils.NewString(err.Error()))
  91. return
  92. }
  93. }
  94. func (self *HAGuestStartTask) OnStartBackupGuestComplete(
  95. ctx context.Context, guest *models.SGuest, data jsonutils.JSONObject,
  96. ) {
  97. if data != nil && !jsonutils.QueryBoolean(data, "is_running", false) {
  98. nbdServerPort, err := data.Int("nbd_server_port")
  99. if err == nil {
  100. backupHost := models.HostManager.FetchHostById(guest.BackupHostId)
  101. nbdServerUri := fmt.Sprintf("nbd:%s:%d", backupHost.AccessIp, nbdServerPort)
  102. guest.SetMetadata(ctx, "backup_nbd_server_uri", nbdServerUri, self.UserCred)
  103. } else {
  104. self.OnStartCompleteFailed(ctx, guest,
  105. jsonutils.NewString("Start backup guest result missing nbd_server_port"))
  106. return
  107. }
  108. }
  109. if err := guest.ResetGuestQuorumChildIndex(ctx, self.UserCred); err != nil {
  110. self.OnStartBackupGuestCompleteFailed(ctx, guest, jsonutils.NewString(fmt.Sprintf("failed reset quorum child index: %s", err)))
  111. return
  112. }
  113. guest.SetBackupGuestStatus(self.UserCred, api.VM_RUNNING, "on start backup guest complete")
  114. self.RequestStart(ctx, guest)
  115. }
  116. func (self *HAGuestStartTask) OnStartBackupGuestCompleteFailed(
  117. ctx context.Context, guest *models.SGuest, data jsonutils.JSONObject,
  118. ) {
  119. guest.SetBackupGuestStatus(self.UserCred, api.VM_START_FAILED, data.String())
  120. self.OnStartCompleteFailed(ctx, guest, data)
  121. }
  122. func (self *HAGuestStartTask) OnStartComplete(ctx context.Context, guest *models.SGuest, data jsonutils.JSONObject) {
  123. if !guest.IsGuestBackupMirrorJobReady(ctx, self.UserCred) {
  124. drv, err := guest.GetDriver()
  125. if err != nil {
  126. guest.SetGuestBackupMirrorJobFailed(ctx, self.UserCred)
  127. guest.SetBackupGuestStatus(self.UserCred, api.VM_BLOCK_STREAM_FAIL, err.Error())
  128. return
  129. }
  130. if err := drv.RequestSlaveBlockStreamDisks(ctx, guest, self); err != nil {
  131. guest.SetGuestBackupMirrorJobFailed(ctx, self.UserCred)
  132. guest.SetBackupGuestStatus(self.UserCred, api.VM_BLOCK_STREAM_FAIL, err.Error())
  133. } else {
  134. guest.SetGuestBackupMirrorJobInProgress(ctx, self.UserCred)
  135. guest.SetBackupGuestStatus(self.UserCred, api.VM_BLOCK_STREAM, "on RequestSlaveBlockStreamDisks")
  136. }
  137. }
  138. self.GuestStartTask.OnStartComplete(ctx, guest, data)
  139. }