mirror of
https://github.com/ceph/ceph-csi.git
synced 2025-01-17 18:29:30 +00:00
rbd: add volume locks for reclaimspace operations
This commit adds locks on reclaimspace operations to prevent multiple process executing rbd sparsify/fstrim on same volume. Signed-off-by: Praveen M <m.praveen@ibm.com>
This commit is contained in:
parent
e6540989a5
commit
243a0fd0fb
@ -37,12 +37,13 @@ import (
|
|||||||
// of CSI-addons reclaimspace controller service spec.
|
// of CSI-addons reclaimspace controller service spec.
|
||||||
type ReclaimSpaceControllerServer struct {
|
type ReclaimSpaceControllerServer struct {
|
||||||
*rs.UnimplementedReclaimSpaceControllerServer
|
*rs.UnimplementedReclaimSpaceControllerServer
|
||||||
|
volumeLocks *util.VolumeLocks
|
||||||
}
|
}
|
||||||
|
|
||||||
// NewReclaimSpaceControllerServer creates a new ReclaimSpaceControllerServer which handles
|
// NewReclaimSpaceControllerServer creates a new ReclaimSpaceControllerServer which handles
|
||||||
// the ReclaimSpace Service requests from the CSI-Addons specification.
|
// the ReclaimSpace Service requests from the CSI-Addons specification.
|
||||||
func NewReclaimSpaceControllerServer() *ReclaimSpaceControllerServer {
|
func NewReclaimSpaceControllerServer(volumeLocks *util.VolumeLocks) *ReclaimSpaceControllerServer {
|
||||||
return &ReclaimSpaceControllerServer{}
|
return &ReclaimSpaceControllerServer{volumeLocks: volumeLocks}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (rscs *ReclaimSpaceControllerServer) RegisterService(server grpc.ServiceRegistrar) {
|
func (rscs *ReclaimSpaceControllerServer) RegisterService(server grpc.ServiceRegistrar) {
|
||||||
@ -64,6 +65,13 @@ func (rscs *ReclaimSpaceControllerServer) ControllerReclaimSpace(
|
|||||||
}
|
}
|
||||||
defer cr.DeleteCredentials()
|
defer cr.DeleteCredentials()
|
||||||
|
|
||||||
|
if acquired := rscs.volumeLocks.TryAcquire(volumeID); !acquired {
|
||||||
|
log.ErrorLog(ctx, util.VolumeOperationAlreadyExistsFmt, volumeID)
|
||||||
|
|
||||||
|
return nil, status.Errorf(codes.Aborted, util.VolumeOperationAlreadyExistsFmt, volumeID)
|
||||||
|
}
|
||||||
|
defer rscs.volumeLocks.Release(volumeID)
|
||||||
|
|
||||||
rbdVol, err := rbdutil.GenVolFromVolID(ctx, volumeID, cr, req.GetSecrets())
|
rbdVol, err := rbdutil.GenVolFromVolID(ctx, volumeID, cr, req.GetSecrets())
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, status.Errorf(codes.Aborted, "failed to find volume with ID %q: %s", volumeID, err.Error())
|
return nil, status.Errorf(codes.Aborted, "failed to find volume with ID %q: %s", volumeID, err.Error())
|
||||||
@ -90,12 +98,13 @@ func (rscs *ReclaimSpaceControllerServer) ControllerReclaimSpace(
|
|||||||
// of CSI-addons reclaimspace controller service spec.
|
// of CSI-addons reclaimspace controller service spec.
|
||||||
type ReclaimSpaceNodeServer struct {
|
type ReclaimSpaceNodeServer struct {
|
||||||
*rs.UnimplementedReclaimSpaceNodeServer
|
*rs.UnimplementedReclaimSpaceNodeServer
|
||||||
|
volumeLocks *util.VolumeLocks
|
||||||
}
|
}
|
||||||
|
|
||||||
// NewReclaimSpaceNodeServer creates a new IdentityServer which handles the
|
// NewReclaimSpaceNodeServer creates a new IdentityServer which handles the
|
||||||
// Identity Service requests from the CSI-Addons specification.
|
// Identity Service requests from the CSI-Addons specification.
|
||||||
func NewReclaimSpaceNodeServer() *ReclaimSpaceNodeServer {
|
func NewReclaimSpaceNodeServer(volumeLocks *util.VolumeLocks) *ReclaimSpaceNodeServer {
|
||||||
return &ReclaimSpaceNodeServer{}
|
return &ReclaimSpaceNodeServer{volumeLocks: volumeLocks}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (rsns *ReclaimSpaceNodeServer) RegisterService(server grpc.ServiceRegistrar) {
|
func (rsns *ReclaimSpaceNodeServer) RegisterService(server grpc.ServiceRegistrar) {
|
||||||
@ -116,6 +125,13 @@ func (rsns *ReclaimSpaceNodeServer) NodeReclaimSpace(
|
|||||||
return nil, status.Error(codes.InvalidArgument, "empty volume ID in request")
|
return nil, status.Error(codes.InvalidArgument, "empty volume ID in request")
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if acquired := rsns.volumeLocks.TryAcquire(volumeID); !acquired {
|
||||||
|
log.ErrorLog(ctx, util.VolumeOperationAlreadyExistsFmt, volumeID)
|
||||||
|
|
||||||
|
return nil, status.Errorf(codes.Aborted, util.VolumeOperationAlreadyExistsFmt, volumeID)
|
||||||
|
}
|
||||||
|
defer rsns.volumeLocks.Release(volumeID)
|
||||||
|
|
||||||
// path can either be the staging path on the node, or the volume path
|
// path can either be the staging path on the node, or the volume path
|
||||||
// inside an application container
|
// inside an application container
|
||||||
path := req.GetStagingTargetPath()
|
path := req.GetStagingTargetPath()
|
||||||
|
@ -20,6 +20,8 @@ import (
|
|||||||
"context"
|
"context"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
|
"github.com/ceph/ceph-csi/internal/util"
|
||||||
|
|
||||||
rs "github.com/csi-addons/spec/lib/go/reclaimspace"
|
rs "github.com/csi-addons/spec/lib/go/reclaimspace"
|
||||||
"github.com/stretchr/testify/require"
|
"github.com/stretchr/testify/require"
|
||||||
)
|
)
|
||||||
@ -30,7 +32,7 @@ import (
|
|||||||
func TestControllerReclaimSpace(t *testing.T) {
|
func TestControllerReclaimSpace(t *testing.T) {
|
||||||
t.Parallel()
|
t.Parallel()
|
||||||
|
|
||||||
controller := NewReclaimSpaceControllerServer()
|
controller := NewReclaimSpaceControllerServer(util.NewVolumeLocks())
|
||||||
|
|
||||||
req := &rs.ControllerReclaimSpaceRequest{
|
req := &rs.ControllerReclaimSpaceRequest{
|
||||||
VolumeId: "",
|
VolumeId: "",
|
||||||
@ -47,7 +49,7 @@ func TestControllerReclaimSpace(t *testing.T) {
|
|||||||
func TestNodeReclaimSpace(t *testing.T) {
|
func TestNodeReclaimSpace(t *testing.T) {
|
||||||
t.Parallel()
|
t.Parallel()
|
||||||
|
|
||||||
node := NewReclaimSpaceNodeServer()
|
node := NewReclaimSpaceNodeServer(&util.VolumeLocks{})
|
||||||
|
|
||||||
req := &rs.NodeReclaimSpaceRequest{
|
req := &rs.NodeReclaimSpaceRequest{
|
||||||
VolumeId: "",
|
VolumeId: "",
|
||||||
|
@ -144,10 +144,8 @@ func (r *Driver) Run(conf *util.Config) {
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
log.FatalLogMsg(err.Error())
|
log.FatalLogMsg(err.Error())
|
||||||
}
|
}
|
||||||
r.ns, err = NewNodeServer(r.cd, conf.Vtype, nodeLabels, topology, crushLocationMap)
|
r.ns = NewNodeServer(r.cd, conf.Vtype, nodeLabels, topology, crushLocationMap)
|
||||||
if err != nil {
|
|
||||||
log.FatalLogMsg("failed to start node server, err %v\n", err)
|
|
||||||
}
|
|
||||||
var attr string
|
var attr string
|
||||||
attr, err = rbd.GetKrbdSupportedFeatures()
|
attr, err = rbd.GetKrbdSupportedFeatures()
|
||||||
if err != nil && !errors.Is(err, os.ErrNotExist) {
|
if err != nil && !errors.Is(err, os.ErrNotExist) {
|
||||||
@ -213,7 +211,7 @@ func (r *Driver) setupCSIAddonsServer(conf *util.Config) error {
|
|||||||
r.cas.RegisterService(is)
|
r.cas.RegisterService(is)
|
||||||
|
|
||||||
if conf.IsControllerServer {
|
if conf.IsControllerServer {
|
||||||
rs := casrbd.NewReclaimSpaceControllerServer()
|
rs := casrbd.NewReclaimSpaceControllerServer(r.cs.VolumeLocks)
|
||||||
r.cas.RegisterService(rs)
|
r.cas.RegisterService(rs)
|
||||||
|
|
||||||
fcs := casrbd.NewFenceControllerServer()
|
fcs := casrbd.NewFenceControllerServer()
|
||||||
@ -227,7 +225,7 @@ func (r *Driver) setupCSIAddonsServer(conf *util.Config) error {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if conf.IsNodeServer {
|
if conf.IsNodeServer {
|
||||||
rs := casrbd.NewReclaimSpaceNodeServer()
|
rs := casrbd.NewReclaimSpaceNodeServer(r.ns.VolumeLocks)
|
||||||
r.cas.RegisterService(rs)
|
r.cas.RegisterService(rs)
|
||||||
|
|
||||||
ekr := casrbd.NewEncryptionKeyRotationServer(r.ns.VolumeLocks)
|
ekr := casrbd.NewEncryptionKeyRotationServer(r.ns.VolumeLocks)
|
||||||
|
Loading…
Reference in New Issue
Block a user