diff --git a/api/core/v1alpha2/virtual_machine.go b/api/core/v1alpha2/virtual_machine.go index d658aa185..20c814d13 100644 --- a/api/core/v1alpha2/virtual_machine.go +++ b/api/core/v1alpha2/virtual_machine.go @@ -116,7 +116,7 @@ type VirtualMachineSpec struct { // * `AlwaysOn` - after creation the VM is always in a running state, even in case of its shutdown by OS means. // * `AlwaysOff` - after creation the VM is always in the off state. // * `Manual` - after creation the VM is switched off, the VM state (switching on/off) is controlled via sub-resources or OS means. -// * `AlwaysOnUnlessStoppedManually` - after creation the VM is always in a running state, even in case of its shutdown by means of the OS, the VM can be shut down using the corresponding subresource. +// * `AlwaysOnUnlessStoppedManually` - after creation the VM is always in a running state. The VM can be shutdown by means of the OS or use the d8 utility: `d8 v stop `. // // +kubebuilder:validation:Enum={AlwaysOn,AlwaysOff,Manual,AlwaysOnUnlessStoppedManually} type RunPolicy string diff --git a/crds/doc-ru-virtualmachines.yaml b/crds/doc-ru-virtualmachines.yaml index 7cc0683f9..b641e4334 100644 --- a/crds/doc-ru-virtualmachines.yaml +++ b/crds/doc-ru-virtualmachines.yaml @@ -475,7 +475,7 @@ spec: * `AlwaysOn` — после создания ВМ всегда находится в работающем состоянии, даже в случае ее выключения средствами ОС. * `AlwaysOff` — после создания ВМ всегда находится в выключенном состоянии. * `Manual` — после создания ВМ выключается, состояние ВМ (включение/выключение) контролируется через API-сервисы или средствами ОС. - * `AlwaysOnUnlessStoppedManually` — после создания ВМ всегда находится в работающем состоянии, даже в случае ее выключения средствами ОС, ВМ может быть выключена с помощью соответствующего API-сервиса. + * `AlwaysOnUnlessStoppedManually` — после создания ВМ всегда находится в работающем состоянии. ВМ можно выключить средствами ОС или воспользоваться утилитой d8: `d8 v stop `. terminationGracePeriodSeconds: description: | Период ожидания после подачи сигнала (SIGTERM) о прекращении работы ВМ, по истечении которого ВМ принудительно завершается. diff --git a/crds/virtualmachines.yaml b/crds/virtualmachines.yaml index f37f86437..6d29edc1a 100644 --- a/crds/virtualmachines.yaml +++ b/crds/virtualmachines.yaml @@ -162,7 +162,7 @@ spec: * `AlwaysOn` - after creation the VM is always in a running state, even in case of its shutdown by OS means. * `AlwaysOff` - after creation the VM is always in the off state. * `Manual` - after creation the VM is switched off, the VM state (switching on/off) is controlled via sub-resources or OS means. - * `AlwaysOnUnlessStoppedManually` - after creation the VM is always in a running state, even in case of its shutdown by means of the OS, the VM can be shut down using the corresponding subresource. + * `AlwaysOnUnlessStoppedManually` - after creation the VM is always in a running state. The VM can be shutdown by means of the OS or use the d8 utility: `d8 v stop `. virtualMachineIPAddressName: type: string diff --git a/images/virtualization-artifact/pkg/controller/kvbuilder/kvvm.go b/images/virtualization-artifact/pkg/controller/kvbuilder/kvvm.go index 3795ecf07..d74815213 100644 --- a/images/virtualization-artifact/pkg/controller/kvbuilder/kvvm.go +++ b/images/virtualization-artifact/pkg/controller/kvbuilder/kvvm.go @@ -128,15 +128,10 @@ func (b *KVVM) SetCPUModel(class *virtv2.VirtualMachineClass) error { func (b *KVVM) SetRunPolicy(runPolicy virtv2.RunPolicy) error { switch runPolicy { - case virtv2.AlwaysOnPolicy: - b.Resource.Spec.RunStrategy = pointer.GetPointer(virtv1.RunStrategyAlways) - case virtv2.AlwaysOffPolicy: - b.Resource.Spec.RunStrategy = pointer.GetPointer(virtv1.RunStrategyHalted) - case virtv2.ManualPolicy: - if !b.ResourceExists { - // initialize only - b.Resource.Spec.RunStrategy = pointer.GetPointer(virtv1.RunStrategyManual) - } + case virtv2.AlwaysOnPolicy, + virtv2.AlwaysOffPolicy, + virtv2.ManualPolicy: + b.Resource.Spec.RunStrategy = pointer.GetPointer(virtv1.RunStrategyManual) case virtv2.AlwaysOnUnlessStoppedManually: if !b.ResourceExists { // initialize only diff --git a/images/virtualization-artifact/pkg/controller/vm/internal/sync_kvvm.go b/images/virtualization-artifact/pkg/controller/vm/internal/sync_kvvm.go index c4bf24dc0..c0d10e59b 100644 --- a/images/virtualization-artifact/pkg/controller/vm/internal/sync_kvvm.go +++ b/images/virtualization-artifact/pkg/controller/vm/internal/sync_kvvm.go @@ -24,14 +24,12 @@ import ( "time" corev1 "k8s.io/api/core/v1" - k8serrors "k8s.io/apimachinery/pkg/api/errors" metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" "k8s.io/client-go/tools/record" virtv1 "kubevirt.io/api/core/v1" "sigs.k8s.io/controller-runtime/pkg/client" "sigs.k8s.io/controller-runtime/pkg/reconcile" - kvvmutil "github.com/deckhouse/virtualization-controller/pkg/common/kvvm" "github.com/deckhouse/virtualization-controller/pkg/common/object" vmutil "github.com/deckhouse/virtualization-controller/pkg/common/vm" "github.com/deckhouse/virtualization-controller/pkg/controller/conditions" @@ -145,17 +143,10 @@ func (h *SyncKvvmHandler) Handle(ctx context.Context, s state.VirtualMachineStat if synced { // 3.1. Changes are applied, consider current spec as last applied. - lastAppliedSpec = ¤t.Spec changed.Status.RestartAwaitingChanges = nil } - // 4. Ensure power state according to the runPolicy. - powerStateSyncErr := h.syncPowerState(ctx, s, kvvm, lastAppliedSpec) - if powerStateSyncErr != nil { - errs = errors.Join(errs, fmt.Errorf("failed to sync powerstate: %w", powerStateSyncErr)) - } - - // 5. Set ConfigurationApplied condition. + // 4. Set ConfigurationApplied condition. switch { case errs != nil: h.recorder.Event(current, corev1.EventTypeWarning, virtv2.ReasonErrVmNotSynced, kvvmSyncErr.Error()) @@ -542,125 +533,3 @@ func (h *SyncKvvmHandler) updateKVVMLastAppliedSpec(ctx context.Context, vm *vir return nil } - -// syncPowerState enforces runPolicy on the underlying KVVM. -func (h *SyncKvvmHandler) syncPowerState(ctx context.Context, s state.VirtualMachineState, kvvm *virtv1.VirtualMachine, effectiveSpec *virtv2.VirtualMachineSpec) error { - log := logger.FromContext(ctx) - - if kvvm == nil { - return nil - } - - kvvmi, err := s.KVVMI(ctx) - if err != nil { - return fmt.Errorf("find the internal virtual machine instance: %w", err) - } - - vmRunPolicy := effectiveSpec.RunPolicy - var shutdownInfo powerstate.ShutdownInfo - s.Shared(func(s *state.Shared) { - shutdownInfo = s.ShutdownInfo - }) - - switch vmRunPolicy { - case virtv2.AlwaysOffPolicy: - if kvvmi != nil { - // Ensure KVVMI is absent. - err = h.client.Delete(ctx, kvvmi) - if err != nil && !k8serrors.IsNotFound(err) { - return fmt.Errorf("force AlwaysOff: delete KVVMI: %w", err) - } - } - err = h.ensureRunStrategy(ctx, kvvm, virtv1.RunStrategyHalted) - case virtv2.AlwaysOnPolicy: - // Power state change reason is not significant for AlwaysOn: - // kubevirt restarts VM via re-creation of KVVMI. - err = h.ensureRunStrategy(ctx, kvvm, virtv1.RunStrategyAlways) - case virtv2.AlwaysOnUnlessStoppedManually: - strategy, _ := kvvm.RunStrategy() - if strategy == virtv1.RunStrategyAlways && kvvmi == nil { - if err = powerstate.StartVM(ctx, h.client, kvvm); err != nil { - return fmt.Errorf("failed to start VM: %w", err) - } - } - if kvvmi != nil && kvvmi.DeletionTimestamp == nil { - if kvvmi.Status.Phase == virtv1.Succeeded { - if shutdownInfo.PodCompleted { - // Request to start new KVVMI if guest was restarted. - // Cleanup KVVMI is enough if VM was stopped from inside. - switch shutdownInfo.Reason { - case powerstate.GuestResetReason: - log.Info("Restart for guest initiated reset") - err = powerstate.SafeRestartVM(ctx, h.client, kvvm, kvvmi) - if err != nil { - return fmt.Errorf("restart VM on guest-reset: %w", err) - } - default: - log.Info("Cleanup Succeeded KVVMI") - err = h.client.Delete(ctx, kvvmi) - if err != nil && !k8serrors.IsNotFound(err) { - return fmt.Errorf("delete Succeeded KVVMI: %w", err) - } - } - } - } - if kvvmi.Status.Phase == virtv1.Failed { - log.Info("Restart on Failed KVVMI", "obj", kvvmi.GetName()) - err = powerstate.SafeRestartVM(ctx, h.client, kvvm, kvvmi) - if err != nil { - return fmt.Errorf("restart VM on failed: %w", err) - } - } - } - - err = h.ensureRunStrategy(ctx, kvvm, virtv1.RunStrategyManual) - case virtv2.ManualPolicy: - // Manual policy requires to handle only guest-reset event. - // All types of shutdown are a final state. - if kvvmi != nil && kvvmi.DeletionTimestamp == nil { - if kvvmi.Status.Phase == virtv1.Succeeded && shutdownInfo.PodCompleted { - // Request to start new KVVMI (with updated settings). - switch shutdownInfo.Reason { - case powerstate.GuestResetReason: - err = powerstate.SafeRestartVM(ctx, h.client, kvvm, kvvmi) - if err != nil { - return fmt.Errorf("restart VM on guest-reset: %w", err) - } - default: - // Cleanup old version of KVVMI. - log.Info("Cleanup Succeeded KVVMI") - err = h.client.Delete(ctx, kvvmi) - if err != nil && !k8serrors.IsNotFound(err) { - return fmt.Errorf("delete Succeeded KVVMI: %w", err) - } - } - } - } - - err = h.ensureRunStrategy(ctx, kvvm, virtv1.RunStrategyManual) - } - - if err != nil { - return fmt.Errorf("enforce runPolicy %s: %w", vmRunPolicy, err) - } - - return nil -} - -func (h *SyncKvvmHandler) ensureRunStrategy(ctx context.Context, kvvm *virtv1.VirtualMachine, desiredRunStrategy virtv1.VirtualMachineRunStrategy) error { - if kvvm == nil { - return nil - } - kvvmRunStrategy := kvvmutil.GetRunStrategy(kvvm) - - if kvvmRunStrategy == desiredRunStrategy { - return nil - } - patch := kvvmutil.PatchRunStrategy(desiredRunStrategy) - err := h.client.Patch(ctx, kvvm, patch) - if err != nil { - return fmt.Errorf("patch KVVM with runStrategy %s: %w", desiredRunStrategy, err) - } - - return nil -} diff --git a/images/virtualization-artifact/pkg/controller/vm/internal/sync_power_state.go b/images/virtualization-artifact/pkg/controller/vm/internal/sync_power_state.go new file mode 100644 index 000000000..0482f6b59 --- /dev/null +++ b/images/virtualization-artifact/pkg/controller/vm/internal/sync_power_state.go @@ -0,0 +1,210 @@ +/* +Copyright 2024 Flant JSC + +Licensed under the Apache License, Version 2.0 (the "License"); +you may not use this file except in compliance with the License. +You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + +Unless required by applicable law or agreed to in writing, software +distributed under the License is distributed on an "AS IS" BASIS, +WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +See the License for the specific language governing permissions and +limitations under the License. +*/ + +package internal + +import ( + "context" + "fmt" + + k8serrors "k8s.io/apimachinery/pkg/api/errors" + virtv1 "kubevirt.io/api/core/v1" + "sigs.k8s.io/controller-runtime/pkg/client" + "sigs.k8s.io/controller-runtime/pkg/reconcile" + + kvvmutil "github.com/deckhouse/virtualization-controller/pkg/common/kvvm" + "github.com/deckhouse/virtualization-controller/pkg/controller/powerstate" + "github.com/deckhouse/virtualization-controller/pkg/controller/vm/internal/state" + "github.com/deckhouse/virtualization-controller/pkg/logger" + virtv2 "github.com/deckhouse/virtualization/api/core/v1alpha2" +) + +const nameSyncPowerStateHandler = "SyncPowerStateHandler" + +func NewSyncPowerStateHandler(client client.Client) *SyncPowerStateHandler { + return &SyncPowerStateHandler{ + client: client, + } +} + +type SyncPowerStateHandler struct { + client client.Client +} + +func (h *SyncPowerStateHandler) Handle(ctx context.Context, s state.VirtualMachineState) (reconcile.Result, error) { + log, ctx := logger.GetHandlerContext(ctx, nameSyncPowerStateHandler) + + if s.VirtualMachine().IsEmpty() { + return reconcile.Result{}, nil + } + + changed := s.VirtualMachine().Changed() + + kvvm, err := s.KVVM(ctx) + if err != nil { + return reconcile.Result{}, fmt.Errorf("find the internal virtual machine: %w", err) + } + + err = h.syncPowerState(ctx, s, kvvm, changed.Spec.RunPolicy) + if err != nil { + err = fmt.Errorf("failed to sync powerstate: %w", err) + log.Error(err.Error()) + } + + return reconcile.Result{}, err +} + +// syncPowerState enforces runPolicy on the underlying KVVM. +func (h *SyncPowerStateHandler) syncPowerState(ctx context.Context, s state.VirtualMachineState, kvvm *virtv1.VirtualMachine, runPolicy virtv2.RunPolicy) error { + log := logger.FromContext(ctx) + + if kvvm == nil { + return nil + } + + kvvmi, err := s.KVVMI(ctx) + if err != nil { + return fmt.Errorf("find the internal virtual machine instance: %w", err) + } + + if runPolicy == virtv2.AlwaysOnUnlessStoppedManually { + if kvvmi != nil { + err = h.ensureRunStrategy(ctx, kvvm, virtv1.RunStrategyManual) + } + } else { + err = h.ensureRunStrategy(ctx, kvvm, virtv1.RunStrategyManual) + } + + if err != nil { + return fmt.Errorf("enforce runPolicy %s: %w", runPolicy, err) + } + + var shutdownInfo powerstate.ShutdownInfo + s.Shared(func(s *state.Shared) { + shutdownInfo = s.ShutdownInfo + }) + + switch runPolicy { + case virtv2.AlwaysOffPolicy: + if kvvmi != nil { + // Ensure KVVMI is absent. + err = h.client.Delete(ctx, kvvmi) + if err != nil && !k8serrors.IsNotFound(err) { + return fmt.Errorf("force AlwaysOff: delete KVVMI: %w", err) + } + } + case virtv2.AlwaysOnPolicy: + if kvvmi == nil { + if err = powerstate.StartVM(ctx, h.client, kvvm); err != nil { + return fmt.Errorf("failed to start VM: %w", err) + } + } + + if kvvmi != nil && kvvmi.DeletionTimestamp == nil { + if kvvmi.Status.Phase == virtv1.Succeeded { + log.Info("Restart for guest initiated reset") + err = powerstate.SafeRestartVM(ctx, h.client, kvvm, kvvmi) + if err != nil { + return fmt.Errorf("restart VM on guest-reset: %w", err) + } + } + + if kvvmi.Status.Phase == virtv1.Failed { + log.Info("Restart on Failed KVVMI", "obj", kvvmi.GetName()) + err = powerstate.SafeRestartVM(ctx, h.client, kvvm, kvvmi) + if err != nil { + return fmt.Errorf("restart VM on failed: %w", err) + } + } + } + case virtv2.AlwaysOnUnlessStoppedManually: + if kvvmi != nil && kvvmi.DeletionTimestamp == nil { + if kvvmi.Status.Phase == virtv1.Succeeded { + if shutdownInfo.PodCompleted { + // Request to start new KVVMI if guest was restarted. + // Cleanup KVVMI is enough if VM was stopped from inside. + switch shutdownInfo.Reason { + case powerstate.GuestResetReason: + log.Info("Restart for guest initiated reset") + err = powerstate.SafeRestartVM(ctx, h.client, kvvm, kvvmi) + if err != nil { + return fmt.Errorf("restart VM on guest-reset: %w", err) + } + default: + log.Info("Cleanup Succeeded KVVMI") + err = h.client.Delete(ctx, kvvmi) + if err != nil && !k8serrors.IsNotFound(err) { + return fmt.Errorf("delete Succeeded KVVMI: %w", err) + } + } + } + } + if kvvmi.Status.Phase == virtv1.Failed { + log.Info("Restart on Failed KVVMI", "obj", kvvmi.GetName()) + err = powerstate.SafeRestartVM(ctx, h.client, kvvm, kvvmi) + if err != nil { + return fmt.Errorf("restart VM on failed: %w", err) + } + } + } + case virtv2.ManualPolicy: + // Manual policy requires to handle only guest-reset event. + // All types of shutdown are a final state. + if kvvmi != nil && kvvmi.DeletionTimestamp == nil { + if kvvmi.Status.Phase == virtv1.Succeeded && shutdownInfo.PodCompleted { + // Request to start new KVVMI (with updated settings). + switch shutdownInfo.Reason { + case powerstate.GuestResetReason: + err = powerstate.SafeRestartVM(ctx, h.client, kvvm, kvvmi) + if err != nil { + return fmt.Errorf("restart VM on guest-reset: %w", err) + } + default: + // Cleanup old version of KVVMI. + log.Info("Cleanup Succeeded KVVMI") + err = h.client.Delete(ctx, kvvmi) + if err != nil && !k8serrors.IsNotFound(err) { + return fmt.Errorf("delete Succeeded KVVMI: %w", err) + } + } + } + } + } + + return nil +} + +func (h *SyncPowerStateHandler) ensureRunStrategy(ctx context.Context, kvvm *virtv1.VirtualMachine, desiredRunStrategy virtv1.VirtualMachineRunStrategy) error { + if kvvm == nil { + return nil + } + kvvmRunStrategy := kvvmutil.GetRunStrategy(kvvm) + + if kvvmRunStrategy == desiredRunStrategy { + return nil + } + patch := kvvmutil.PatchRunStrategy(desiredRunStrategy) + err := h.client.Patch(ctx, kvvm, patch) + if err != nil { + return fmt.Errorf("patch KVVM with runStrategy %s: %w", desiredRunStrategy, err) + } + + return nil +} + +func (h *SyncPowerStateHandler) Name() string { + return nameSyncPowerStateHandler +} diff --git a/images/virtualization-artifact/pkg/controller/vm/vm_controller.go b/images/virtualization-artifact/pkg/controller/vm/vm_controller.go index c683a02c1..10f4dc32f 100644 --- a/images/virtualization-artifact/pkg/controller/vm/vm_controller.go +++ b/images/virtualization-artifact/pkg/controller/vm/vm_controller.go @@ -63,6 +63,7 @@ func SetupController( internal.NewPodHandler(client), internal.NewSizePolicyHandler(), internal.NewSyncKvvmHandler(dvcrSettings, client, recorder), + internal.NewSyncPowerStateHandler(client), internal.NewSyncMetadataHandler(client), internal.NewLifeCycleHandler(client, recorder), internal.NewStatisticHandler(client),