From f4cf410b8b4cee84b4ced11736482ee30fa6cec8 Mon Sep 17 00:00:00 2001 From: Aline Abler Date: Thu, 8 Feb 2024 14:02:10 +0100 Subject: [PATCH 1/2] Add default organization to users that are in exactly one organization We want to add the usability feature "As a user, I want my default organization to be set automatically when I create my first organization or accept an invitation, so that I can get started right away and don't have to worry about setup." We simplify this to "every user who is in exactly 1 organization and does not have a default configured should have their default organization set to the one organization they are a member of". This will automatically solve the use case above both for creating your first org and for accepting your first invite. --- controller.go | 8 + .../default_organization_controller.go | 87 ++++++++ .../default_organization_controller_test.go | 196 ++++++++++++++++++ 3 files changed, 291 insertions(+) create mode 100644 controllers/default_organization_controller.go create mode 100644 controllers/default_organization_controller_test.go diff --git a/controller.go b/controller.go index cd1bb04c..8b665a92 100644 --- a/controller.go +++ b/controller.go @@ -313,6 +313,14 @@ func setupManager( return nil, err } } + dor := &controllers.DefaultOrganizationReconciler{ + Client: mgr.GetClient(), + Scheme: mgr.GetScheme(), + Recorder: mgr.GetEventRecorderFor("organization-members-controller"), + } + if err = dor.SetupWithManager(mgr); err != nil { + return nil, err + } obenc := &controllers.OrgBillingEntityNameCacheController{ Client: mgr.GetClient(), Scheme: mgr.GetScheme(), diff --git a/controllers/default_organization_controller.go b/controllers/default_organization_controller.go new file mode 100644 index 00000000..cb4cb587 --- /dev/null +++ b/controllers/default_organization_controller.go @@ -0,0 +1,87 @@ +package controllers + +import ( + "context" + + "go.uber.org/multierr" + "k8s.io/apimachinery/pkg/runtime" + "k8s.io/apimachinery/pkg/types" + "k8s.io/client-go/tools/record" + ctrl "sigs.k8s.io/controller-runtime" + "sigs.k8s.io/controller-runtime/pkg/client" + "sigs.k8s.io/controller-runtime/pkg/log" + + controlv1 "github.com/appuio/control-api/apis/v1" +) + +// DefaultOrganizationReconciler reconciles User resources to ensure they have a DefaultOrganization set if applicable. +type DefaultOrganizationReconciler struct { + client.Client + Recorder record.EventRecorder + Scheme *runtime.Scheme +} + +//+kubebuilder:rbac:groups=appuio.io,resources=organizationmembers,verbs=get;list;watch +//+kubebuilder:rbac:groups=appuio.io,resources=users,verbs=get;list;watch;update;patch +//+kubebuilder:rbac:groups=appuio.io,resources=users/status,verbs=get + +// Reconcile reacts on changes of memberships and sets members' default organization if appropriate +func (r *DefaultOrganizationReconciler) Reconcile(ctx context.Context, req ctrl.Request) (ctrl.Result, error) { + log := log.FromContext(ctx) + log.V(1).WithValues("request", req).Info("Reconciling") + + memb := controlv1.OrganizationMembers{} + if err := r.Get(ctx, req.NamespacedName, &memb); err != nil { + return ctrl.Result{}, client.IgnoreNotFound(err) + } + + if !memb.ObjectMeta.DeletionTimestamp.IsZero() { + return ctrl.Result{}, nil + } + + allMemberships := controlv1.OrganizationMembersList{} + if err := r.List(ctx, &allMemberships); err != nil { + return ctrl.Result{}, err + } + + var errGroup error + for _, user := range memb.Status.ResolvedUserRefs { + myOrgs := make([]string, 0) + + for _, membership := range allMemberships.Items { + for _, membershipUser := range membership.Status.ResolvedUserRefs { + if user.Name == membershipUser.Name { + myOrgs = append(myOrgs, membership.Namespace) + break + } + } + } + if len(myOrgs) == 1 { + err := setUserDefaultOrganization(ctx, r.Client, user.Name, myOrgs[0]) + errGroup = multierr.Append(errGroup, err) + } + } + + return ctrl.Result{}, errGroup +} + +func setUserDefaultOrganization(ctx context.Context, c client.Client, userName string, orgName string) error { + user := controlv1.User{} + if err := c.Get(ctx, types.NamespacedName{Name: userName}, &user); err != nil { + return err + } + + if user.Spec.Preferences.DefaultOrganizationRef != "" { + return nil + } + + user.Spec.Preferences.DefaultOrganizationRef = orgName + return c.Update(ctx, &user) +} + +// SetupWithManager sets up the controller with the Manager. +func (r *DefaultOrganizationReconciler) SetupWithManager(mgr ctrl.Manager) error { + return ctrl.NewControllerManagedBy(mgr). + For(&controlv1.OrganizationMembers{}). + Complete(r) +} diff --git a/controllers/default_organization_controller_test.go b/controllers/default_organization_controller_test.go new file mode 100644 index 00000000..238215b9 --- /dev/null +++ b/controllers/default_organization_controller_test.go @@ -0,0 +1,196 @@ +package controllers_test + +import ( + "context" + "errors" + "strings" + "testing" + + "github.com/stretchr/testify/assert" + "github.com/stretchr/testify/require" + + apierrors "k8s.io/apimachinery/pkg/api/errors" + metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" + "k8s.io/apimachinery/pkg/types" + "k8s.io/client-go/tools/record" + ctrl "sigs.k8s.io/controller-runtime" + "sigs.k8s.io/controller-runtime/pkg/client" + + controlv1 "github.com/appuio/control-api/apis/v1" + . "github.com/appuio/control-api/controllers" +) + +var testMemberships1 = controlv1.OrganizationMembers{ + ObjectMeta: metav1.ObjectMeta{ + Name: "members", + Namespace: "foo-gmbh", + }, + Spec: controlv1.OrganizationMembersSpec{ + UserRefs: []controlv1.UserRef{ + {Name: "u1"}, + {Name: "u2"}, + {Name: "u3"}, + }, + }, + Status: controlv1.OrganizationMembersStatus{ + ResolvedUserRefs: []controlv1.UserRef{ + {Name: "u1"}, + {Name: "u2"}, + {Name: "u3"}, + }, + }, +} + +var testMemberships2 = controlv1.OrganizationMembers{ + ObjectMeta: metav1.ObjectMeta{ + Name: "members", + Namespace: "bar-gmbh", + }, + Spec: controlv1.OrganizationMembersSpec{ + UserRefs: []controlv1.UserRef{ + {Name: "u1"}, + }, + }, + Status: controlv1.OrganizationMembersStatus{ + ResolvedUserRefs: []controlv1.UserRef{ + {Name: "u1"}, + }, + }, +} + +var u1 = controlv1.User{ + ObjectMeta: metav1.ObjectMeta{ + Name: "u1", + }, +} +var u2 = controlv1.User{ + ObjectMeta: metav1.ObjectMeta{ + Name: "u2", + }, +} +var u3 = controlv1.User{ + ObjectMeta: metav1.ObjectMeta{ + Name: "u3", + }, +} + +func Test_DefaultOrganizationReconciler_Reconcile_Sucess(t *testing.T) { + ctx := context.Background() + c := prepareTest(t, &testMemberships1, &testMemberships2, &u1, &u2, &u3) + fakeRecorder := record.NewFakeRecorder(3) + + _, err := (&DefaultOrganizationReconciler{ + Client: c, + Scheme: c.Scheme(), + Recorder: fakeRecorder, + }).Reconcile(ctx, ctrl.Request{ + NamespacedName: types.NamespacedName{ + Name: testMemberships1.Name, + Namespace: testMemberships1.Namespace, + }, + }) + require.NoError(t, err) + + user := controlv1.User{} + require.NoError(t, c.Get(context.TODO(), types.NamespacedName{Name: u1.ObjectMeta.Name}, &user)) + assert.Empty(t, user.Spec.Preferences.DefaultOrganizationRef) + + require.NoError(t, c.Get(context.TODO(), types.NamespacedName{Name: u2.ObjectMeta.Name}, &user)) + assert.Equal(t, testMemberships1.ObjectMeta.Namespace, user.Spec.Preferences.DefaultOrganizationRef) + + require.NoError(t, c.Get(context.TODO(), types.NamespacedName{Name: u3.ObjectMeta.Name}, &user)) + assert.Equal(t, testMemberships1.ObjectMeta.Namespace, user.Spec.Preferences.DefaultOrganizationRef) + +} + +func Test_DefaultOrganizationReconciler_Reconcile_NoMembership_Sucess(t *testing.T) { + ctx := context.Background() + c := prepareTest(t, &testMemberships2, &u1, &u2, &u3) + fakeRecorder := record.NewFakeRecorder(3) + + _, err := (&DefaultOrganizationReconciler{ + Client: c, + Scheme: c.Scheme(), + Recorder: fakeRecorder, + }).Reconcile(ctx, ctrl.Request{ + NamespacedName: types.NamespacedName{ + Name: testMemberships2.Name, + Namespace: testMemberships2.Namespace, + }, + }) + require.NoError(t, err) + + user := controlv1.User{} + require.NoError(t, c.Get(context.TODO(), types.NamespacedName{Name: u1.ObjectMeta.Name}, &user)) + assert.Equal(t, testMemberships2.ObjectMeta.Namespace, user.Spec.Preferences.DefaultOrganizationRef) + + require.NoError(t, c.Get(context.TODO(), types.NamespacedName{Name: u2.ObjectMeta.Name}, &user)) + assert.Empty(t, user.Spec.Preferences.DefaultOrganizationRef) + + require.NoError(t, c.Get(context.TODO(), types.NamespacedName{Name: u3.ObjectMeta.Name}, &user)) + assert.Empty(t, user.Spec.Preferences.DefaultOrganizationRef) + +} + +func Test_DefaultOrganizationReconciler_Reconcile_Error(t *testing.T) { + failU4 := controlv1.User{ + ObjectMeta: metav1.ObjectMeta{ + Name: "fail-u4", + }, + } + failMemberships := controlv1.OrganizationMembers{ + ObjectMeta: metav1.ObjectMeta{ + Name: "members", + Namespace: "foo-gmbh", + }, + Spec: controlv1.OrganizationMembersSpec{ + UserRefs: []controlv1.UserRef{ + {Name: "u1"}, + {Name: "u2"}, + {Name: "u3"}, + {Name: "fail-u4"}, + }, + }, + Status: controlv1.OrganizationMembersStatus{ + ResolvedUserRefs: []controlv1.UserRef{ + {Name: "u1"}, + {Name: "u2"}, + {Name: "u3"}, + {Name: "fail-u4"}, + }, + }, + } + ctx := context.Background() + c := failingClient{prepareTest(t, &failMemberships, &failU4, &u1, &u2, &u3)} + fakeRecorder := record.NewFakeRecorder(3) + + _, err := (&DefaultOrganizationReconciler{ + Client: c, + Scheme: c.Scheme(), + Recorder: fakeRecorder, + }).Reconcile(ctx, ctrl.Request{ + NamespacedName: types.NamespacedName{ + Name: failMemberships.Name, + Namespace: failMemberships.Namespace, + }, + }) + require.Error(t, err) + + user := controlv1.User{} + require.NoError(t, c.Get(context.TODO(), types.NamespacedName{Name: u1.ObjectMeta.Name}, &user)) + assert.Equal(t, failMemberships.ObjectMeta.Namespace, user.Spec.Preferences.DefaultOrganizationRef) + + require.NoError(t, c.Get(context.TODO(), types.NamespacedName{Name: u2.ObjectMeta.Name}, &user)) + assert.Equal(t, failMemberships.ObjectMeta.Namespace, user.Spec.Preferences.DefaultOrganizationRef) + + require.NoError(t, c.Get(context.TODO(), types.NamespacedName{Name: u3.ObjectMeta.Name}, &user)) + assert.Equal(t, failMemberships.ObjectMeta.Namespace, user.Spec.Preferences.DefaultOrganizationRef) + +} + +func (c failingClient) Update(ctx context.Context, obj client.Object, opts ...client.UpdateOption) error { + if strings.HasPrefix(obj.GetName(), "fail-") { + return apierrors.NewInternalError(errors.New("ups")) + } + return c.WithWatch.Update(ctx, obj, opts...) +} From eeafd34807e8bae8f79877895e6367abc21866ff Mon Sep 17 00:00:00 2001 From: Aline Abler Date: Fri, 9 Feb 2024 15:25:12 +0100 Subject: [PATCH 2/2] Create user if they do not exist --- .../default_organization_controller.go | 16 ++++++++- .../default_organization_controller_test.go | 33 +++++++++++++++++-- 2 files changed, 46 insertions(+), 3 deletions(-) diff --git a/controllers/default_organization_controller.go b/controllers/default_organization_controller.go index cb4cb587..a879d182 100644 --- a/controllers/default_organization_controller.go +++ b/controllers/default_organization_controller.go @@ -4,6 +4,8 @@ import ( "context" "go.uber.org/multierr" + apierrors "k8s.io/apimachinery/pkg/api/errors" + v1 "k8s.io/apimachinery/pkg/apis/meta/v1" "k8s.io/apimachinery/pkg/runtime" "k8s.io/apimachinery/pkg/types" "k8s.io/client-go/tools/record" @@ -68,7 +70,19 @@ func (r *DefaultOrganizationReconciler) Reconcile(ctx context.Context, req ctrl. func setUserDefaultOrganization(ctx context.Context, c client.Client, userName string, orgName string) error { user := controlv1.User{} if err := c.Get(ctx, types.NamespacedName{Name: userName}, &user); err != nil { - return err + if !apierrors.IsNotFound(err) { + return err + } + return c.Create(ctx, &controlv1.User{ + ObjectMeta: v1.ObjectMeta{ + Name: userName, + }, + Spec: controlv1.UserSpec{ + Preferences: controlv1.UserPreferences{ + DefaultOrganizationRef: orgName, + }, + }, + }) } if user.Spec.Preferences.DefaultOrganizationRef != "" { diff --git a/controllers/default_organization_controller_test.go b/controllers/default_organization_controller_test.go index 238215b9..e483a3bb 100644 --- a/controllers/default_organization_controller_test.go +++ b/controllers/default_organization_controller_test.go @@ -74,7 +74,7 @@ var u3 = controlv1.User{ }, } -func Test_DefaultOrganizationReconciler_Reconcile_Sucess(t *testing.T) { +func Test_DefaultOrganizationReconciler_Reconcile_Success(t *testing.T) { ctx := context.Background() c := prepareTest(t, &testMemberships1, &testMemberships2, &u1, &u2, &u3) fakeRecorder := record.NewFakeRecorder(3) @@ -103,7 +103,7 @@ func Test_DefaultOrganizationReconciler_Reconcile_Sucess(t *testing.T) { } -func Test_DefaultOrganizationReconciler_Reconcile_NoMembership_Sucess(t *testing.T) { +func Test_DefaultOrganizationReconciler_Reconcile_NoMembership_Success(t *testing.T) { ctx := context.Background() c := prepareTest(t, &testMemberships2, &u1, &u2, &u3) fakeRecorder := record.NewFakeRecorder(3) @@ -132,6 +132,35 @@ func Test_DefaultOrganizationReconciler_Reconcile_NoMembership_Sucess(t *testing } +func Test_DefaultOrganizationReconciler_Reconcile_UserNotExist_Success(t *testing.T) { + ctx := context.Background() + c := prepareTest(t, &testMemberships1, &u1) + fakeRecorder := record.NewFakeRecorder(3) + + _, err := (&DefaultOrganizationReconciler{ + Client: c, + Scheme: c.Scheme(), + Recorder: fakeRecorder, + }).Reconcile(ctx, ctrl.Request{ + NamespacedName: types.NamespacedName{ + Name: testMemberships1.Name, + Namespace: testMemberships1.Namespace, + }, + }) + require.NoError(t, err) + + user := controlv1.User{} + require.NoError(t, c.Get(context.TODO(), types.NamespacedName{Name: u1.ObjectMeta.Name}, &user)) + assert.Equal(t, testMemberships1.ObjectMeta.Namespace, user.Spec.Preferences.DefaultOrganizationRef) + + require.NoError(t, c.Get(context.TODO(), types.NamespacedName{Name: u2.ObjectMeta.Name}, &user)) + assert.Equal(t, testMemberships1.ObjectMeta.Namespace, user.Spec.Preferences.DefaultOrganizationRef) + + require.NoError(t, c.Get(context.TODO(), types.NamespacedName{Name: u3.ObjectMeta.Name}, &user)) + assert.Equal(t, testMemberships1.ObjectMeta.Namespace, user.Spec.Preferences.DefaultOrganizationRef) + +} + func Test_DefaultOrganizationReconciler_Reconcile_Error(t *testing.T) { failU4 := controlv1.User{ ObjectMeta: metav1.ObjectMeta{