Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

rbd: refactor mirroring to work with volume and volumegroup #4720

Merged
merged 3 commits into from
Jul 26, 2024

Conversation

Madhu-1
Copy link
Collaborator

@Madhu-1 Madhu-1 commented Jul 19, 2024

refactor mirroring to work with volume and volumegroup

@Madhu-1 Madhu-1 requested a review from nixpanic July 19, 2024 08:44
@mergify mergify bot added the component/rbd Issues related to RBD label Jul 19, 2024
@Madhu-1
Copy link
Collaborator Author

Madhu-1 commented Jul 19, 2024

@nixpanic can you please check this PR for early review, i still need to do testing, once everything is good i will move this to Ready state.

scripts/golangci.yml.in Outdated Show resolved Hide resolved
internal/csi-addons/rbd/replication.go Outdated Show resolved Hide resolved
internal/csi-addons/rbd/replication.go Outdated Show resolved Hide resolved
internal/rbd/mirror.go Outdated Show resolved Hide resolved
internal/rbd/mirror.go Outdated Show resolved Hide resolved
internal/rbd/types/mirror.go Outdated Show resolved Hide resolved
internal/rbd/types/volume.go Show resolved Hide resolved
internal/rbd/types/volume.go Show resolved Hide resolved
internal/rbd/types/mirror.go Show resolved Hide resolved
Resync() error

// GetMirroringInfo returns the mirroring information of the resource
GetMirroringInfo() (interface{}, error)
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It is really ugly to return an untyped interface. Can that be improved?

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

i tried but it gets more difficult for implementation where i need to have different duplicate checks for both volume and group, that's the reason to settle on this after lot of other considerations

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Maybe add a MirrorInfo interface too?

Copy link
Collaborator Author

@Madhu-1 Madhu-1 Jul 19, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Am not sure adding MirrorInfo interface helps but the problem is that the volumemirror and groupmirror in go-ceph return different structs, if both are returning the same i could have solved this or else I need to own a local struct and fill the required values from both, i can do this if you prefer (this will have lot of duplication logic and maintenance work)

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Maybe, or have a MirrorInfo interface with only some basic function, or completely empty? At least the type can then describe that it is actually one of the go-ceph types.

@Madhu-1 Madhu-1 force-pushed the mirror-vol-grp branch 3 times, most recently from 1899b64 to 7dbb200 Compare July 19, 2024 14:17
@Madhu-1 Madhu-1 requested a review from nixpanic July 19, 2024 14:18
)

//nolint:interfacebloat // more than 10 methods are needed for the interface
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

In #4719 I added the journalledObject interface. Such an approach also prevents the need for skipping the interfacebloat linter.

rbdVol.Destroy(ctx)
}
}()
mgr := rbd.NewManager(req.GetParameters(), req.GetSecrets())
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

note that the CSI-instance id (drivername) needs to be passed to the manager in an upcoming change.

@Madhu-1 Madhu-1 marked this pull request as ready for review July 22, 2024 12:27
@Madhu-1
Copy link
Collaborator Author

Madhu-1 commented Jul 22, 2024

@nixpanic I have tested this code and works as expected, let me know if you want to make any changes you are referring to the manager as part of this PR.

@Madhu-1 Madhu-1 requested review from nixpanic and a team July 22, 2024 12:28
Copy link
Member

@nixpanic nixpanic left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Splitting the commit would be nice.
At least add the new types in a separate commit, and use them in subsequent ones.

The single commit is +488 -300 lines of code, which is rather a lot.

internal/rbd/mirror.go Show resolved Hide resolved
internal/rbd/mirror.go Show resolved Hide resolved
internal/rbd/types/mirror.go Show resolved Hide resolved
internal/rbd/types/mirror.go Show resolved Hide resolved
// IsPrimary returns true if the resource is primary
IsPrimary() bool
// GetState returns the state of the resource
GetState() string
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

consider adding a MirrorState type to replace the string value

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This needs more work, will take it in next update when implementing this for group

internal/rbd/types/volume.go Outdated Show resolved Hide resolved
@Madhu-1 Madhu-1 requested a review from nixpanic July 26, 2024 08:05
internal/rbd/types/mirror.go Outdated Show resolved Hide resolved
internal/rbd/types/mirror.go Outdated Show resolved Hide resolved
@@ -40,4 +42,21 @@ type Volume interface {

// RemoveFromGroup removes the Volume from the VolumeGroup.
RemoveFromGroup(ctx context.Context, vg VolumeGroup) error

// GetPoolName returns the name of the pool where the volume is stored.
GetPoolName() string
Copy link
Member

@nixpanic nixpanic Jul 26, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Maybe Volume should inherit the journalledObject too?

type journalledObject interface {
// GetID returns the CSI-Addons VolumeGroupId of the VolumeGroup.
GetID(ctx context.Context) (string, error)
// GetName returns the name in the backend storage for the VolumeGroup.
GetName(ctx context.Context) (string, error)
// GetPool returns the name of the pool that holds the VolumeGroup.
GetPool(ctx context.Context) (string, error)
// GetClusterID returns the ID of the cluster of the VolumeGroup.
GetClusterID(ctx context.Context) (string, error)
}

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This needs a little more refactoring as journalledObject is specific to volumeGroup now, will address this in follow-up when implementing for group support

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It is only the comments that refer to VolumeGroup. The functions could be for Volume as well.

default:
err = status.Errorf(codes.Internal, err.Error())
}

return nil, err
}
mirror = rbdVol
Copy link
Member

@nixpanic nixpanic Jul 26, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Would it make sense to have rbdVol.ToMirror() that validates everything required for the new Mirror type?

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@nixpanic can you explain a bit more on this one?

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

updated the code, PTAL and see if this matches your expectation

@@ -25,6 +25,8 @@ import (

//nolint:interfacebloat // more than 10 methods are needed for the interface
type Volume interface {
// Mirror interface implementation for Volume
Mirror
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

because you use mirror = rbdVol, I would expect that the Mirror interface is kept seperate. I do not see a need to have it part of the Volume interface. The rbdVolume struct can still implement both interfaces.

@Madhu-1 Madhu-1 requested a review from nixpanic July 26, 2024 09:11
@@ -77,3 +77,7 @@ func (rv *rbdVolume) RemoveFromGroup(ctx context.Context, vg types.VolumeGroup)

return librbd.GroupImageRemove(ioctx, name, rv.ioctx, rv.RbdImageName)
}

func (rv *rbdVolume) ToMirror() types.Mirror {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thanks! Would it make sense to have some checking if mirror attributes are valid?

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Not for now, do you have any specific checks in your mind?

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Not really. Mostly just to have the function match func (*rbdVolume) ToCSI(ctx) (csi.Volume, error)

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

the signature is matching now, it returns nil for now but can be updated later as required.

@Madhu-1 Madhu-1 requested a review from nixpanic July 26, 2024 09:25
@Madhu-1 Madhu-1 requested a review from a team July 26, 2024 09:34
@nixpanic
Copy link
Member

@Mergifyio rebase

Madhu-1 added 3 commits July 26, 2024 09:44
Adding mirror interface for rbd image
and group which can be used to manage
mirroring for both.

Signed-off-by: Madhu Rajanna <[email protected]>
Add helper methods for the rbd
image mirroring.

Signed-off-by: Madhu Rajanna <[email protected]>
Refractoring code to use mirror
interface.

Signed-off-by: Madhu Rajanna <[email protected]>
Copy link
Contributor

mergify bot commented Jul 26, 2024

rebase

✅ Branch has been successfully rebased

@nixpanic nixpanic added the ok-to-test Label to trigger E2E tests label Jul 26, 2024
@ceph-csi-bot
Copy link
Collaborator

/test ci/centos/upgrade-tests-cephfs

@ceph-csi-bot
Copy link
Collaborator

/test ci/centos/upgrade-tests-rbd

@ceph-csi-bot
Copy link
Collaborator

/test ci/centos/k8s-e2e-external-storage/1.27

@ceph-csi-bot
Copy link
Collaborator

/test ci/centos/k8s-e2e-external-storage/1.29

@ceph-csi-bot
Copy link
Collaborator

/test ci/centos/mini-e2e-helm/k8s-1.27

@ceph-csi-bot
Copy link
Collaborator

/test ci/centos/mini-e2e/k8s-1.27

@ceph-csi-bot
Copy link
Collaborator

/test ci/centos/mini-e2e-helm/k8s-1.29

@ceph-csi-bot
Copy link
Collaborator

/test ci/centos/k8s-e2e-external-storage/1.28

@ceph-csi-bot
Copy link
Collaborator

/test ci/centos/mini-e2e/k8s-1.29

@ceph-csi-bot
Copy link
Collaborator

/test ci/centos/mini-e2e-helm/k8s-1.28

@ceph-csi-bot
Copy link
Collaborator

/test ci/centos/mini-e2e/k8s-1.28

@ceph-csi-bot
Copy link
Collaborator

/test ci/centos/k8s-e2e-external-storage/1.30

@ceph-csi-bot
Copy link
Collaborator

/test ci/centos/mini-e2e-helm/k8s-1.30

@ceph-csi-bot
Copy link
Collaborator

/test ci/centos/mini-e2e/k8s-1.30

@ceph-csi-bot ceph-csi-bot removed the ok-to-test Label to trigger E2E tests label Jul 26, 2024
Copy link
Contributor

@iPraveenParihar iPraveenParihar left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

nit

AddSnapshotScheduling(interval admin.Interval, startTime admin.StartTime) error
}

// MirrorImage is the interface for managing mirroring on an RBD image or group of images.
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
// MirrorImage is the interface for managing mirroring on an RBD image or group of images.
// MirrorInfo is the interface for managing mirroring on an RBD image or group of images.

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Will address this in follow up. Thanks @Madhu-1 for catching it.

@mergify mergify bot merged commit b185bfd into ceph:devel Jul 26, 2024
40 checks passed
@@ -219,7 +219,7 @@ func (r *Driver) setupCSIAddonsServer(conf *util.Config) error {
fcs := casrbd.NewFenceControllerServer()
r.cas.RegisterService(fcs)

rcs := casrbd.NewReplicationServer(NewControllerServer(r.cd))
rcs := casrbd.NewReplicationServer(rbd.CSIInstanceID, NewControllerServer(r.cd))
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@Madhu-1 , shouldn't this be

Suggested change
rcs := casrbd.NewReplicationServer(rbd.CSIInstanceID, NewControllerServer(r.cd))
rcs := casrbd.NewReplicationServer(conf.InstanceID, NewControllerServer(r.cd))

?

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

yes, I think so. Can you send a PR for that?

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

conf.InstanceID will not work if its not set, we need to use default in that case we are not updating conf.InstanceID with rbd.CSIInstanceID, that's the reason to use rbd.CSIInstanceID as its gets updated always or it contains the default

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@Madhu-1, we are doing that for InitJournals -

func InitJournals(instance string) {
// Use passed in instance ID, if provided for omap suffix naming
if instance != "" {
CSIInstanceID = instance
}
volJournal = journal.NewCSIVolumeJournal(CSIInstanceID)
snapJournal = journal.NewCSISnapshotJournal(CSIInstanceID)

And conf.InstanceID is passed to InitJournals

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

if the conf.InsatceID is empty its not updated with the default one that we have in the code right rbd.InitJournals(conf.InstanceID)? with this one

CSIInstanceID = "default"

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Apologies, I was mistaken it was other way around.
Thanks @Madhu-1.

Then I guess for VolumeGroupServer we need to use rbd.CSIInstanceID

vgcs := casrbd.NewVolumeGroupServer(conf.InstanceID)

@nixpanic ?

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

rbd.CSIInstanceID seems to be a global value, it is configurable on the commandline, so I think using the configured value is cleaner (all config options should have default values anyway?).

In internal/rbd/driver/driver.go there is rbd.InitJournals(conf.InstanceID), so in the end it is all the same.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

conf.InstanceID default value is empty and rbd.CSIInstanceID is replaced with conf.InstanceID when it is configured in command line.
So using conf.InstanceID when its not configured will lead to creating journal directory csi.groups. ?

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

yeah, I noticed that too. The global rbd.CSIInstanceID could be set to something earlier on, in case the ordering of (seemingly unrelated) initialization of the journals is done later. This is a little fragile, and using global variables is quite ugly in any case. I'm preparing a PR for improving it.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

see #4747

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
component/rbd Issues related to RBD
Projects
None yet
Development

Successfully merging this pull request may close these issues.

5 participants