mirror of
https://github.com/ceph/ceph-csi.git
synced 2025-01-08 12:59:30 +00:00
f90408be4d
Increase the timeout to 2 minutes to give enough time for rollback to complete. As rollback is performed by the force-promote command it, at times, may take more than a minute (based on dirty blocks that need to be rolled back approximately) to rollback. The added extra 1 minute is useful though to avoid multiple calls to complete the rollback and in extremely corner cases to avoid failures in the first instance of the call when the mirror watcher is not yet removed (post scaling down the RBD mirror instance) Signed-off-by: Madhu Rajanna <madhupr007@gmail.com>
186 lines
4.9 KiB
Go
186 lines
4.9 KiB
Go
/*
|
|
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
|
|
|
|
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 rbd
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"time"
|
|
|
|
"github.com/ceph/ceph-csi/internal/util"
|
|
|
|
librbd "github.com/ceph/go-ceph/rbd"
|
|
)
|
|
|
|
// enableImageMirroring enables mirroring on an image.
|
|
func (ri *rbdImage) enableImageMirroring(mode librbd.ImageMirrorMode) error {
|
|
image, err := ri.open()
|
|
if err != nil {
|
|
return fmt.Errorf("failed to open image %q with error: %w", ri, err)
|
|
}
|
|
defer image.Close()
|
|
|
|
err = image.MirrorEnable(mode)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to enable mirroring on %q with error: %w", ri, err)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// disableImageMirroring disables mirroring on an image.
|
|
func (ri *rbdImage) disableImageMirroring(force bool) error {
|
|
image, err := ri.open()
|
|
if err != nil {
|
|
return fmt.Errorf("failed to open image %q with error: %w", ri, err)
|
|
}
|
|
defer image.Close()
|
|
|
|
err = image.MirrorDisable(force)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to disable mirroring on %q with error: %w", ri, err)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// getImageMirroringInfo gets mirroring information of an image.
|
|
func (ri *rbdImage) getImageMirroringInfo() (*librbd.MirrorImageInfo, error) {
|
|
image, err := ri.open()
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to open image %q with error: %w", ri, err)
|
|
}
|
|
defer image.Close()
|
|
|
|
info, err := image.GetMirrorImageInfo()
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to get mirroring info of %q with error: %w", ri, err)
|
|
}
|
|
|
|
return info, nil
|
|
}
|
|
|
|
// promoteImage promotes image to primary.
|
|
func (ri *rbdImage) promoteImage(force bool) error {
|
|
image, err := ri.open()
|
|
if err != nil {
|
|
return fmt.Errorf("failed to open image %q with error: %w", ri, err)
|
|
}
|
|
defer image.Close()
|
|
err = image.MirrorPromote(force)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to promote image %q with error: %w", ri, err)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// forcePromoteImage promotes image to primary with force option with 2 minutes
|
|
// timeout. If there is no response within 2 minutes,the rbd CLI process will be
|
|
// killed and an error is returned.
|
|
func (rv *rbdVolume) forcePromoteImage(cr *util.Credentials) error {
|
|
promoteArgs := []string{
|
|
"mirror", "image", "promote",
|
|
rv.String(),
|
|
"--force",
|
|
"--id", cr.ID,
|
|
"-m", rv.Monitors,
|
|
"--keyfile=" + cr.KeyFile,
|
|
}
|
|
_, stderr, err := util.ExecCommandWithTimeout(
|
|
context.TODO(),
|
|
// 2 minutes timeout as the Replication RPC timeout is 2.5 minutes.
|
|
2*time.Minute,
|
|
"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
|
|
}
|
|
|
|
// demoteImage demotes image to secondary.
|
|
func (ri *rbdImage) demoteImage() error {
|
|
image, err := ri.open()
|
|
if err != nil {
|
|
return fmt.Errorf("failed to open image %q with error: %w", ri, err)
|
|
}
|
|
defer image.Close()
|
|
err = image.MirrorDemote()
|
|
if err != nil {
|
|
return fmt.Errorf("failed to demote image %q with error: %w", ri, err)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// resyncImage resync image to correct the split-brain.
|
|
func (ri *rbdImage) resyncImage() error {
|
|
image, err := ri.open()
|
|
if err != nil {
|
|
return fmt.Errorf("failed to open image %q with error: %w", ri, err)
|
|
}
|
|
defer image.Close()
|
|
err = image.MirrorResync()
|
|
if err != nil {
|
|
return fmt.Errorf("failed to resync image %q with error: %w", ri, err)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// getImageMirroingStatus get the mirroring status of an image.
|
|
func (ri *rbdImage) getImageMirroringStatus() (*librbd.GlobalMirrorImageStatus, error) {
|
|
image, err := ri.open()
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to open image %q with error: %w", ri, err)
|
|
}
|
|
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)
|
|
}
|
|
|
|
return &statusInfo, nil
|
|
}
|
|
|
|
// getLocalState returns the local state of the image.
|
|
func (ri *rbdImage) getLocalState() (librbd.SiteMirrorImageStatus, error) {
|
|
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
|
|
}
|