2021-02-11 09:59:47 +00:00
|
|
|
/*
|
|
|
|
Copyright 2021 The Ceph-CSI Authors.
|
|
|
|
|
|
|
|
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
|
|
|
|
|
2023-06-02 09:49:22 +00:00
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
2021-02-11 09:59:47 +00:00
|
|
|
|
|
|
|
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 rbd
|
|
|
|
|
|
|
|
import (
|
2021-12-22 03:36:37 +00:00
|
|
|
"context"
|
2021-02-11 09:59:47 +00:00
|
|
|
"fmt"
|
2021-12-22 03:36:37 +00:00
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/ceph/ceph-csi/internal/util"
|
2021-02-11 09:59:47 +00:00
|
|
|
|
|
|
|
librbd "github.com/ceph/go-ceph/rbd"
|
|
|
|
)
|
|
|
|
|
2024-06-06 11:44:08 +00:00
|
|
|
// FlattenMode is used to indicate the flatten mode for an RBD image.
|
|
|
|
type FlattenMode string
|
|
|
|
|
|
|
|
const (
|
|
|
|
// FlattenModeNever indicates that the image should never be flattened.
|
|
|
|
FlattenModeNever FlattenMode = "never"
|
|
|
|
// FlattenModeForce indicates that the image with the parent must be flattened.
|
|
|
|
FlattenModeForce FlattenMode = "force"
|
|
|
|
)
|
|
|
|
|
|
|
|
// HandleParentImageExistence checks the image's parent.
|
|
|
|
// if the parent image does not exist and is not in trash, it returns nil.
|
|
|
|
// if the flattenMode is FlattenModeForce, it flattens the image itself.
|
|
|
|
// if the parent image is in trash, it returns an error.
|
|
|
|
// if the parent image exists and is not enabled for mirroring, it returns an error.
|
|
|
|
func (rv *rbdVolume) HandleParentImageExistence(
|
|
|
|
ctx context.Context,
|
|
|
|
flattenMode FlattenMode,
|
|
|
|
) error {
|
|
|
|
if rv.ParentName == "" && !rv.ParentInTrash {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
if flattenMode == FlattenModeForce {
|
|
|
|
// Delete temp image that exists for volume datasource since
|
|
|
|
// it is no longer required when the live image is flattened.
|
|
|
|
err := rv.DeleteTempImage(ctx)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("failed to delete temporary rbd image: %w", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
err = rv.flattenRbdImage(ctx, true, 0, 0)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if rv.ParentInTrash {
|
|
|
|
return fmt.Errorf("%w: failed to enable mirroring on image %q:"+
|
|
|
|
" parent is in trash",
|
|
|
|
ErrFailedPrecondition, rv)
|
|
|
|
}
|
|
|
|
|
|
|
|
parent, err := rv.getParent()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
parentMirroringInfo, err := parent.GetImageMirroringInfo()
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf(
|
|
|
|
"failed to get mirroring info of parent %q of image %q: %w",
|
|
|
|
parent, rv, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if parentMirroringInfo.State != librbd.MirrorImageEnabled {
|
|
|
|
return fmt.Errorf("%w: failed to enable mirroring on image %q: "+
|
|
|
|
"parent image %q is not enabled for mirroring",
|
|
|
|
ErrFailedPrecondition, rv, parent)
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2023-01-12 09:52:51 +00:00
|
|
|
// EnableImageMirroring enables mirroring on an image.
|
|
|
|
func (ri *rbdImage) EnableImageMirroring(mode librbd.ImageMirrorMode) error {
|
2021-03-17 07:54:19 +00:00
|
|
|
image, err := ri.open()
|
2021-02-11 09:59:47 +00:00
|
|
|
if err != nil {
|
2021-05-07 05:30:37 +00:00
|
|
|
return fmt.Errorf("failed to open image %q with error: %w", ri, err)
|
2021-02-11 09:59:47 +00:00
|
|
|
}
|
|
|
|
defer image.Close()
|
|
|
|
|
|
|
|
err = image.MirrorEnable(mode)
|
|
|
|
if err != nil {
|
2021-05-07 05:30:37 +00:00
|
|
|
return fmt.Errorf("failed to enable mirroring on %q with error: %w", ri, err)
|
2021-02-11 09:59:47 +00:00
|
|
|
}
|
2021-07-22 05:45:17 +00:00
|
|
|
|
2021-02-11 09:59:47 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2023-01-12 09:52:51 +00:00
|
|
|
// DisableImageMirroring disables mirroring on an image.
|
|
|
|
func (ri *rbdImage) DisableImageMirroring(force bool) error {
|
2021-03-17 07:54:19 +00:00
|
|
|
image, err := ri.open()
|
2021-02-11 09:59:47 +00:00
|
|
|
if err != nil {
|
2021-05-07 05:30:37 +00:00
|
|
|
return fmt.Errorf("failed to open image %q with error: %w", ri, err)
|
2021-02-11 09:59:47 +00:00
|
|
|
}
|
|
|
|
defer image.Close()
|
|
|
|
|
|
|
|
err = image.MirrorDisable(force)
|
|
|
|
if err != nil {
|
2021-05-07 05:30:37 +00:00
|
|
|
return fmt.Errorf("failed to disable mirroring on %q with error: %w", ri, err)
|
2021-02-11 09:59:47 +00:00
|
|
|
}
|
2021-07-22 05:45:17 +00:00
|
|
|
|
2021-02-11 09:59:47 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2023-01-12 09:52:51 +00:00
|
|
|
// GetImageMirroringInfo gets mirroring information of an image.
|
|
|
|
func (ri *rbdImage) GetImageMirroringInfo() (*librbd.MirrorImageInfo, error) {
|
2021-03-17 07:54:19 +00:00
|
|
|
image, err := ri.open()
|
2021-02-11 09:59:47 +00:00
|
|
|
if err != nil {
|
2021-05-07 05:30:37 +00:00
|
|
|
return nil, fmt.Errorf("failed to open image %q with error: %w", ri, err)
|
2021-02-11 09:59:47 +00:00
|
|
|
}
|
|
|
|
defer image.Close()
|
|
|
|
|
|
|
|
info, err := image.GetMirrorImageInfo()
|
|
|
|
if err != nil {
|
2021-05-07 05:30:37 +00:00
|
|
|
return nil, fmt.Errorf("failed to get mirroring info of %q with error: %w", ri, err)
|
2021-02-11 09:59:47 +00:00
|
|
|
}
|
2021-07-22 05:45:17 +00:00
|
|
|
|
2021-02-11 09:59:47 +00:00
|
|
|
return info, nil
|
|
|
|
}
|
|
|
|
|
2023-01-12 09:52:51 +00:00
|
|
|
// PromoteImage promotes image to primary.
|
|
|
|
func (ri *rbdImage) PromoteImage(force bool) error {
|
2021-03-17 07:54:19 +00:00
|
|
|
image, err := ri.open()
|
2021-02-11 09:59:47 +00:00
|
|
|
if err != nil {
|
2021-05-07 05:30:37 +00:00
|
|
|
return fmt.Errorf("failed to open image %q with error: %w", ri, err)
|
2021-02-11 09:59:47 +00:00
|
|
|
}
|
|
|
|
defer image.Close()
|
|
|
|
err = image.MirrorPromote(force)
|
|
|
|
if err != nil {
|
2021-05-07 05:30:37 +00:00
|
|
|
return fmt.Errorf("failed to promote image %q with error: %w", ri, err)
|
2021-02-11 09:59:47 +00:00
|
|
|
}
|
2021-07-22 05:45:17 +00:00
|
|
|
|
2021-02-11 09:59:47 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2023-01-12 09:52:51 +00:00
|
|
|
// ForcePromoteImage promotes image to primary with force option with 2 minutes
|
2022-03-29 09:52:27 +00:00
|
|
|
// timeout. If there is no response within 2 minutes,the rbd CLI process will be
|
2021-12-22 03:36:37 +00:00
|
|
|
// killed and an error is returned.
|
2023-01-12 09:52:51 +00:00
|
|
|
func (rv *rbdVolume) ForcePromoteImage(cr *util.Credentials) error {
|
2021-12-22 03:36:37 +00:00
|
|
|
promoteArgs := []string{
|
|
|
|
"mirror", "image", "promote",
|
|
|
|
rv.String(),
|
|
|
|
"--force",
|
|
|
|
"--id", cr.ID,
|
|
|
|
"-m", rv.Monitors,
|
|
|
|
"--keyfile=" + cr.KeyFile,
|
|
|
|
}
|
|
|
|
_, stderr, err := util.ExecCommandWithTimeout(
|
|
|
|
context.TODO(),
|
2022-03-29 09:52:27 +00:00
|
|
|
// 2 minutes timeout as the Replication RPC timeout is 2.5 minutes.
|
|
|
|
2*time.Minute,
|
2021-12-22 03:36:37 +00:00
|
|
|
"rbd",
|
|
|
|
promoteArgs...,
|
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("failed to promote image %q with error: %w", rv, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if stderr != "" {
|
|
|
|
return fmt.Errorf("failed to promote image %q with stderror: %s", rv, stderr)
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2023-01-12 09:52:51 +00:00
|
|
|
// DemoteImage demotes image to secondary.
|
|
|
|
func (ri *rbdImage) DemoteImage() error {
|
2021-03-17 07:54:19 +00:00
|
|
|
image, err := ri.open()
|
2021-02-11 09:59:47 +00:00
|
|
|
if err != nil {
|
2021-05-07 05:30:37 +00:00
|
|
|
return fmt.Errorf("failed to open image %q with error: %w", ri, err)
|
2021-02-11 09:59:47 +00:00
|
|
|
}
|
|
|
|
defer image.Close()
|
|
|
|
err = image.MirrorDemote()
|
|
|
|
if err != nil {
|
2021-05-07 05:30:37 +00:00
|
|
|
return fmt.Errorf("failed to demote image %q with error: %w", ri, err)
|
2021-02-11 09:59:47 +00:00
|
|
|
}
|
2021-07-22 05:45:17 +00:00
|
|
|
|
2021-02-11 09:59:47 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// resyncImage resync image to correct the split-brain.
|
2021-03-17 07:54:19 +00:00
|
|
|
func (ri *rbdImage) resyncImage() error {
|
|
|
|
image, err := ri.open()
|
2021-02-11 09:59:47 +00:00
|
|
|
if err != nil {
|
2021-05-07 05:30:37 +00:00
|
|
|
return fmt.Errorf("failed to open image %q with error: %w", ri, err)
|
2021-02-11 09:59:47 +00:00
|
|
|
}
|
|
|
|
defer image.Close()
|
|
|
|
err = image.MirrorResync()
|
|
|
|
if err != nil {
|
2021-05-07 05:30:37 +00:00
|
|
|
return fmt.Errorf("failed to resync image %q with error: %w", ri, err)
|
2021-02-11 09:59:47 +00:00
|
|
|
}
|
2021-07-22 05:45:17 +00:00
|
|
|
|
2021-02-11 09:59:47 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2023-01-12 09:52:51 +00:00
|
|
|
// GetImageMirroringStatus get the mirroring status of an image.
|
|
|
|
func (ri *rbdImage) GetImageMirroringStatus() (*librbd.GlobalMirrorImageStatus, error) {
|
2021-06-23 13:14:48 +00:00
|
|
|
image, err := ri.open()
|
2021-02-11 09:59:47 +00:00
|
|
|
if err != nil {
|
2021-06-23 13:14:48 +00:00
|
|
|
return nil, fmt.Errorf("failed to open image %q with error: %w", ri, err)
|
2021-02-11 09:59:47 +00:00
|
|
|
}
|
2021-06-23 13:14:48 +00:00
|
|
|
defer image.Close()
|
|
|
|
statusInfo, err := image.GetGlobalMirrorStatus()
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("failed to get image mirroring status %q with error: %w", ri, err)
|
2021-02-11 09:59:47 +00:00
|
|
|
}
|
2021-07-22 05:45:17 +00:00
|
|
|
|
2021-06-23 13:14:48 +00:00
|
|
|
return &statusInfo, nil
|
2021-02-11 09:59:47 +00:00
|
|
|
}
|
2021-08-12 05:17:25 +00:00
|
|
|
|
2023-01-12 09:52:51 +00:00
|
|
|
// GetLocalState returns the local state of the image.
|
|
|
|
func (ri *rbdImage) GetLocalState() (librbd.SiteMirrorImageStatus, error) {
|
2021-08-12 05:17:25 +00:00
|
|
|
localStatus := librbd.SiteMirrorImageStatus{}
|
|
|
|
image, err := ri.open()
|
|
|
|
if err != nil {
|
|
|
|
return localStatus, fmt.Errorf("failed to open image %q with error: %w", ri, err)
|
|
|
|
}
|
|
|
|
defer image.Close()
|
|
|
|
|
|
|
|
statusInfo, err := image.GetGlobalMirrorStatus()
|
|
|
|
if err != nil {
|
|
|
|
return localStatus, fmt.Errorf("failed to get image mirroring status %q with error: %w", ri, err)
|
|
|
|
}
|
|
|
|
localStatus, err = statusInfo.LocalStatus()
|
|
|
|
if err != nil {
|
|
|
|
return localStatus, fmt.Errorf("failed to get local status: %w", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return localStatus, nil
|
|
|
|
}
|