mirror of
https://github.com/ceph/ceph-csi.git
synced 2024-11-22 22:30:23 +00:00
rbd: implement volume group using go-ceph
This adds the required functionality to call the go-ceph API's for the rbd volume group. Signed-off-by: Madhu Rajanna <madhupr007@gmail.com>
This commit is contained in:
parent
86baab7322
commit
7cf5974e73
@ -29,11 +29,13 @@ import (
|
||||
csicommon "github.com/ceph/ceph-csi/internal/csi-common"
|
||||
"github.com/ceph/ceph-csi/internal/rbd"
|
||||
corerbd "github.com/ceph/ceph-csi/internal/rbd"
|
||||
rbd_group "github.com/ceph/ceph-csi/internal/rbd/group"
|
||||
"github.com/ceph/ceph-csi/internal/rbd/types"
|
||||
"github.com/ceph/ceph-csi/internal/util"
|
||||
"github.com/ceph/ceph-csi/internal/util/log"
|
||||
|
||||
librbd "github.com/ceph/go-ceph/rbd"
|
||||
|
||||
"github.com/ceph/go-ceph/rbd/admin"
|
||||
"github.com/csi-addons/spec/lib/go/replication"
|
||||
"google.golang.org/grpc"
|
||||
@ -785,13 +787,14 @@ func getGRPCError(err error) error {
|
||||
}
|
||||
|
||||
errorStatusMap := map[error]codes.Code{
|
||||
corerbd.ErrImageNotFound: codes.NotFound,
|
||||
util.ErrPoolNotFound: codes.NotFound,
|
||||
corerbd.ErrInvalidArgument: codes.InvalidArgument,
|
||||
corerbd.ErrFlattenInProgress: codes.Aborted,
|
||||
corerbd.ErrAborted: codes.Aborted,
|
||||
corerbd.ErrFailedPrecondition: codes.FailedPrecondition,
|
||||
corerbd.ErrUnavailable: codes.Unavailable,
|
||||
corerbd.ErrImageNotFound: codes.NotFound,
|
||||
util.ErrPoolNotFound: codes.NotFound,
|
||||
corerbd.ErrInvalidArgument: codes.InvalidArgument,
|
||||
corerbd.ErrFlattenInProgress: codes.Aborted,
|
||||
corerbd.ErrAborted: codes.Aborted,
|
||||
corerbd.ErrFailedPrecondition: codes.FailedPrecondition,
|
||||
corerbd.ErrUnavailable: codes.Unavailable,
|
||||
rbd_group.ErrRBDGroupUnAvailable: codes.Unavailable,
|
||||
}
|
||||
|
||||
for e, code := range errorStatusMap {
|
||||
|
328
internal/rbd/group/group_mirror.go
Normal file
328
internal/rbd/group/group_mirror.go
Normal file
@ -0,0 +1,328 @@
|
||||
/*
|
||||
Copyright 2024 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 group
|
||||
|
||||
import (
|
||||
"context"
|
||||
"errors"
|
||||
"fmt"
|
||||
"time"
|
||||
|
||||
"github.com/ceph/go-ceph/rados"
|
||||
librbd "github.com/ceph/go-ceph/rbd"
|
||||
"github.com/ceph/go-ceph/rbd/admin"
|
||||
|
||||
"github.com/ceph/ceph-csi/internal/rbd/types"
|
||||
"github.com/ceph/ceph-csi/internal/util"
|
||||
"github.com/ceph/ceph-csi/internal/util/log"
|
||||
)
|
||||
|
||||
var ErrRBDGroupUnAvailable = errors.New("RBD group is unavailable")
|
||||
|
||||
type volumeGroupMirror struct {
|
||||
*volumeGroup
|
||||
}
|
||||
|
||||
func (vg volumeGroupMirror) EnableMirroring(ctx context.Context, mode librbd.ImageMirrorMode) error {
|
||||
name, err := vg.GetName(ctx)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
ioctx, err := vg.GetIOContext(ctx)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
err = librbd.MirrorGroupEnable(ioctx, name, mode)
|
||||
if err != nil {
|
||||
return fmt.Errorf("failed to enable mirroring on volume group %q: %w", vg, err)
|
||||
}
|
||||
|
||||
log.DebugLog(ctx, "mirroring is enabled on the volume group %q", vg)
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
func (vg volumeGroupMirror) DisableMirroring(ctx context.Context, force bool) error {
|
||||
name, err := vg.GetName(ctx)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
ioctx, err := vg.GetIOContext(ctx)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
err = librbd.MirrorGroupDisable(ioctx, name, force)
|
||||
if err != nil && !errors.Is(rados.ErrNotFound, err) {
|
||||
return fmt.Errorf("failed to disable mirroring on volume group %q: %w", vg, err)
|
||||
}
|
||||
|
||||
log.DebugLog(ctx, "mirroring is disabled on the volume group %q", vg)
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
func (vg volumeGroupMirror) Promote(ctx context.Context, force bool) error {
|
||||
name, err := vg.GetName(ctx)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
ioctx, err := vg.GetIOContext(ctx)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
err = librbd.MirrorGroupPromote(ioctx, name, force)
|
||||
if err != nil {
|
||||
return fmt.Errorf("failed to promote volume group %q: %w", vg, err)
|
||||
}
|
||||
|
||||
log.DebugLog(ctx, "volume group %q has been promoted", vg)
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
func (vg volumeGroupMirror) ForcePromote(ctx context.Context, cr *util.Credentials) error {
|
||||
promoteArgs := []string{
|
||||
"mirror", "group", "promote",
|
||||
vg.String(),
|
||||
"--force",
|
||||
"--id", cr.ID,
|
||||
"-m", vg.monitors,
|
||||
"--keyfile=" + cr.KeyFile,
|
||||
}
|
||||
_, stderr, err := util.ExecCommandWithTimeout(
|
||||
ctx,
|
||||
// 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 group %q with error: %w", vg, err)
|
||||
}
|
||||
|
||||
if stderr != "" {
|
||||
return fmt.Errorf("failed to promote group %q with stderror: %s", vg, stderr)
|
||||
}
|
||||
|
||||
log.DebugLog(ctx, "volume group %q has been force promoted", vg)
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
func (vg volumeGroupMirror) Demote(ctx context.Context) error {
|
||||
name, err := vg.GetName(ctx)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
ioctx, err := vg.GetIOContext(ctx)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
err = librbd.MirrorGroupDemote(ioctx, name)
|
||||
if err != nil {
|
||||
return fmt.Errorf("failed to demote volume group %q: %w", vg, err)
|
||||
}
|
||||
|
||||
log.DebugLog(ctx, "volume group %q has been demoted", vg)
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
func (vg volumeGroupMirror) Resync(ctx context.Context) error {
|
||||
name, err := vg.GetName(ctx)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
ioctx, err := vg.GetIOContext(ctx)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
err = librbd.MirrorGroupResync(ioctx, name)
|
||||
if err != nil {
|
||||
return fmt.Errorf("failed to resync volume group %q: %w", vg, err)
|
||||
}
|
||||
|
||||
log.DebugLog(ctx, "issued resync on volume group %q", vg)
|
||||
// If we issued a resync, return a non-final error as image needs to be recreated
|
||||
// locally. Caller retries till RBD syncs an initial version of the image to
|
||||
// report its status in the resync request.
|
||||
return fmt.Errorf("%w: awaiting initial resync due to split brain", ErrRBDGroupUnAvailable)
|
||||
}
|
||||
|
||||
func (vg volumeGroupMirror) GetMirroringInfo(ctx context.Context) (types.MirrorInfo, error) {
|
||||
name, err := vg.GetName(ctx)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
ioctx, err := vg.GetIOContext(ctx)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
info, err := librbd.GetMirrorGroupInfo(ioctx, name)
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("failed to get volume group mirroring info %q: %w", vg, err)
|
||||
}
|
||||
|
||||
return &groupInfo{MirrorGroupInfo: info}, nil
|
||||
}
|
||||
|
||||
func (vg volumeGroupMirror) GetGlobalMirroringStatus(ctx context.Context) (types.GlobalStatus, error) {
|
||||
name, err := vg.GetName(ctx)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
ioctx, err := vg.GetIOContext(ctx)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
statusInfo, err := librbd.GetGlobalMirrorGroupStatus(ioctx, name)
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("failed to get volume group mirroring status %q: %w", vg, err)
|
||||
}
|
||||
|
||||
return globalMirrorGroupStatus{GlobalMirrorGroupStatus: &statusInfo}, nil
|
||||
}
|
||||
|
||||
func (vg volumeGroupMirror) AddSnapshotScheduling(interval admin.Interval, startTime admin.StartTime) error {
|
||||
ls := admin.NewLevelSpec(vg.pool, vg.namespace, "")
|
||||
ra, err := vg.conn.GetRBDAdmin()
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
adminConn := ra.MirrorSnashotSchedule()
|
||||
err = adminConn.Add(ls, interval, startTime)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
// groupInfo is a wrapper around librbd.MirrorGroupInfo that contains the
|
||||
// group mirror info.
|
||||
type groupInfo struct {
|
||||
*librbd.MirrorGroupInfo
|
||||
}
|
||||
|
||||
func (info *groupInfo) GetState() string {
|
||||
return info.State.String()
|
||||
}
|
||||
|
||||
func (info *groupInfo) IsPrimary() bool {
|
||||
return info.Primary
|
||||
}
|
||||
|
||||
// globalMirrorGroupStatus is a wrapper around librbd.GlobalGroupMirrorImageStatus that contains the
|
||||
// global mirror group status.
|
||||
type globalMirrorGroupStatus struct {
|
||||
*librbd.GlobalMirrorGroupStatus
|
||||
}
|
||||
|
||||
func (status globalMirrorGroupStatus) GetState() string {
|
||||
return status.GlobalMirrorGroupStatus.Info.State.String()
|
||||
}
|
||||
|
||||
func (status globalMirrorGroupStatus) IsPrimary() bool {
|
||||
return status.GlobalMirrorGroupStatus.Info.Primary
|
||||
}
|
||||
|
||||
func (status globalMirrorGroupStatus) GetLocalSiteStatus() (types.SiteStatus, error) {
|
||||
s, err := status.GlobalMirrorGroupStatus.LocalStatus()
|
||||
if err != nil {
|
||||
err = fmt.Errorf("failed to get local site status: %w", err)
|
||||
}
|
||||
|
||||
return siteMirrorGroupStatus{
|
||||
SiteMirrorGroupStatus: &s,
|
||||
}, err
|
||||
}
|
||||
|
||||
func (status globalMirrorGroupStatus) GetAllSitesStatus() []types.SiteStatus {
|
||||
var siteStatuses []types.SiteStatus
|
||||
for i := range status.SiteStatuses {
|
||||
siteStatuses = append(siteStatuses, siteMirrorGroupStatus{SiteMirrorGroupStatus: &status.SiteStatuses[i]})
|
||||
}
|
||||
|
||||
return siteStatuses
|
||||
}
|
||||
|
||||
// RemoteStatus returns one SiteMirrorGroupStatus item from the SiteStatuses
|
||||
// slice that corresponds to the remote site's status. If the remote status
|
||||
// is not found than the error ErrNotExist will be returned.
|
||||
func (status globalMirrorGroupStatus) GetRemoteSiteStatus(ctx context.Context) (types.SiteStatus, error) {
|
||||
var (
|
||||
ss librbd.SiteMirrorGroupStatus
|
||||
err error = librbd.ErrNotExist
|
||||
)
|
||||
|
||||
for i := range status.SiteStatuses {
|
||||
log.DebugLog(
|
||||
ctx,
|
||||
"Site status of MirrorUUID: %s, state: %s, description: %s, lastUpdate: %v, up: %t",
|
||||
status.SiteStatuses[i].MirrorUUID,
|
||||
status.SiteStatuses[i].State,
|
||||
status.SiteStatuses[i].Description,
|
||||
status.SiteStatuses[i].LastUpdate,
|
||||
status.SiteStatuses[i].Up)
|
||||
|
||||
if status.SiteStatuses[i].MirrorUUID != "" {
|
||||
ss = status.SiteStatuses[i]
|
||||
err = nil
|
||||
|
||||
break
|
||||
}
|
||||
}
|
||||
|
||||
return siteMirrorGroupStatus{SiteMirrorGroupStatus: &ss}, err
|
||||
}
|
||||
|
||||
// siteMirrorGroupStatus is a wrapper around librbd.SiteMirrorGroupStatus that contains the
|
||||
// site mirror group status.
|
||||
type siteMirrorGroupStatus struct {
|
||||
*librbd.SiteMirrorGroupStatus
|
||||
}
|
||||
|
||||
func (status siteMirrorGroupStatus) GetMirrorUUID() string {
|
||||
return status.MirrorUUID
|
||||
}
|
||||
|
||||
func (status siteMirrorGroupStatus) GetState() string {
|
||||
return status.State.String()
|
||||
}
|
||||
|
||||
func (status siteMirrorGroupStatus) GetDescription() string {
|
||||
return status.Description
|
||||
}
|
||||
|
||||
func (status siteMirrorGroupStatus) IsUP() bool {
|
||||
return status.Up
|
||||
}
|
||||
|
||||
func (status siteMirrorGroupStatus) GetLastUpdate() time.Time {
|
||||
// convert the last update time to UTC
|
||||
return time.Unix(status.LastUpdate, 0).UTC()
|
||||
}
|
Loading…
Reference in New Issue
Block a user