mirror of
https://github.com/grafana/grafana.git
synced 2026-03-13 15:29:48 +08:00
* IAM: Add hidden users filtering and improved RBAC mapper for users API - Add StoreWrapper for user resource that filters hidden users on Get/List - Wire up StoreWrapper in the users API group registration - Expand RBAC verb mapping for users to use explicit action translations - Add integration tests for hidden users filtering behavior Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com> * IAM: Fix duplicate user validation and storewrapper context propagation The storewrapper replaced the request context with a service identity (OrgID=0) before invoking createValidation/updateValidation callbacks. Since these callbacks wrap k8s admission webhooks (including the duplicate email/login checks), the validation ran with OrgID=0 causing SearchOrgUsers to return no results, silently passing duplicates through to the DB which then returned a 500 instead of 409. Fix 1 (storewrapper): Add validationWithUserContext and updateValidationWithUserContext helpers that rebind validation callbacks to the original user context before passing them to the inner store. Fix 2 (legacy store): Add toUserConflictError as defense-in-depth that converts SQLite UNIQUE constraint failures on user.email/user.login into proper 409 Conflict API errors in CreateUser and UpdateUser. Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com> * Regen * Use configprovider.ConfigProvider instead of setting.Cfg * Enforce hidden-users restrictions on write operations BeforeCreate, BeforeUpdate, and BeforeDelete in the user StoreWrapper now return HTTP 403 when the target user's login is in the hidden-users list, returning a generic "operation not permitted" message to callers and logging the hidden-user detail server-side via a structured logger. Integration tests are updated to create the user before marking it hidden (so BeforeCreate does not block setup), then verify all four guarded paths (get→404, list filtered, update→403, delete→403) and add a dedicated sub-test that confirms create is blocked once a login is in the hidden list. Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com> * IAM: Add WithPreserveIdentity option to storewrapper Introduces a WithPreserveIdentity() functional option on storewrapper.New() so the users storage path passes the original caller identity through to the inner store instead of replacing it with a service identity. This ensures admission validation (e.g. duplicate email/login checks) runs with the correct OrgID. Adds unit tests for the new option. Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com> * Address feedback * Fix some minor issues * Update pkg/registry/apis/iam/register.go Co-authored-by: Gabriel MABILLE <gamab@users.noreply.github.com> * Address feedback --------- Co-authored-by: Claude Sonnet 4.6 <noreply@anthropic.com> Co-authored-by: Gabriel MABILLE <gamab@users.noreply.github.com>
608 lines
16 KiB
Go
608 lines
16 KiB
Go
package user
|
|
|
|
import (
|
|
"context"
|
|
"testing"
|
|
|
|
"github.com/stretchr/testify/require"
|
|
|
|
"github.com/grafana/authlib/types"
|
|
iamv0alpha1 "github.com/grafana/grafana/apps/iam/pkg/apis/iam/v0alpha1"
|
|
"github.com/grafana/grafana/pkg/apimachinery/identity"
|
|
"github.com/grafana/grafana/pkg/services/org"
|
|
"github.com/grafana/grafana/pkg/storage/unified/resourcepb"
|
|
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
|
|
)
|
|
|
|
func TestValidateOnCreate(t *testing.T) {
|
|
tests := []struct {
|
|
name string
|
|
user *iamv0alpha1.User
|
|
requester *identity.StaticRequester
|
|
searchClient resourcepb.ResourceIndexClient
|
|
expectError bool
|
|
errorContains string
|
|
}{
|
|
{
|
|
name: "valid user creation by grafana admin",
|
|
user: &iamv0alpha1.User{
|
|
Spec: iamv0alpha1.UserSpec{
|
|
Login: "testuser",
|
|
Role: "Viewer",
|
|
},
|
|
},
|
|
requester: &identity.StaticRequester{
|
|
Type: types.TypeUser,
|
|
IsGrafanaAdmin: true,
|
|
},
|
|
searchClient: &FakeUserLegacySearchClient{},
|
|
expectError: false,
|
|
},
|
|
{
|
|
name: "grafana admin creating another grafana admin",
|
|
user: &iamv0alpha1.User{
|
|
Spec: iamv0alpha1.UserSpec{
|
|
Login: "newadmin",
|
|
GrafanaAdmin: true,
|
|
Role: "Viewer",
|
|
},
|
|
},
|
|
requester: &identity.StaticRequester{
|
|
Type: types.TypeUser,
|
|
IsGrafanaAdmin: true,
|
|
},
|
|
searchClient: &FakeUserLegacySearchClient{},
|
|
expectError: false,
|
|
},
|
|
{
|
|
name: "non-admin trying to create a grafana admin",
|
|
user: &iamv0alpha1.User{
|
|
Spec: iamv0alpha1.UserSpec{
|
|
Login: "newadmin",
|
|
GrafanaAdmin: true,
|
|
Role: "Viewer",
|
|
},
|
|
},
|
|
requester: &identity.StaticRequester{
|
|
Type: types.TypeUser,
|
|
IsGrafanaAdmin: false,
|
|
},
|
|
searchClient: &FakeUserLegacySearchClient{},
|
|
expectError: true,
|
|
errorContains: "only grafana admins can create grafana admins",
|
|
},
|
|
{
|
|
name: "user with empty login and email",
|
|
user: &iamv0alpha1.User{
|
|
Spec: iamv0alpha1.UserSpec{
|
|
Role: "Viewer",
|
|
},
|
|
},
|
|
requester: &identity.StaticRequester{
|
|
Type: types.TypeUser,
|
|
IsGrafanaAdmin: false,
|
|
},
|
|
searchClient: &FakeUserLegacySearchClient{},
|
|
expectError: true,
|
|
errorContains: "user must have either login or email",
|
|
},
|
|
{
|
|
name: "user with only login",
|
|
user: &iamv0alpha1.User{
|
|
Spec: iamv0alpha1.UserSpec{
|
|
Login: "testuser",
|
|
Role: "Viewer",
|
|
},
|
|
},
|
|
requester: &identity.StaticRequester{
|
|
Type: types.TypeUser,
|
|
IsGrafanaAdmin: false,
|
|
},
|
|
searchClient: &FakeUserLegacySearchClient{},
|
|
expectError: false,
|
|
},
|
|
{
|
|
name: "user with only email",
|
|
user: &iamv0alpha1.User{
|
|
Spec: iamv0alpha1.UserSpec{
|
|
Email: "test@example",
|
|
Role: "Viewer",
|
|
},
|
|
},
|
|
requester: &identity.StaticRequester{
|
|
Type: types.TypeUser,
|
|
IsGrafanaAdmin: false,
|
|
},
|
|
searchClient: &FakeUserLegacySearchClient{},
|
|
expectError: false,
|
|
},
|
|
{
|
|
name: "user with empty role",
|
|
user: &iamv0alpha1.User{
|
|
Spec: iamv0alpha1.UserSpec{
|
|
Login: "testuser",
|
|
},
|
|
},
|
|
requester: &identity.StaticRequester{
|
|
Type: types.TypeUser,
|
|
IsGrafanaAdmin: false,
|
|
},
|
|
searchClient: &FakeUserLegacySearchClient{},
|
|
expectError: true,
|
|
errorContains: "role is required",
|
|
},
|
|
{
|
|
name: "user with invalid role",
|
|
user: &iamv0alpha1.User{
|
|
Spec: iamv0alpha1.UserSpec{
|
|
Login: "testuser",
|
|
Role: "InvalidRole",
|
|
},
|
|
},
|
|
requester: &identity.StaticRequester{
|
|
Type: types.TypeUser,
|
|
IsGrafanaAdmin: false,
|
|
},
|
|
searchClient: &FakeUserLegacySearchClient{},
|
|
expectError: true,
|
|
errorContains: "invalid role 'InvalidRole'",
|
|
},
|
|
{
|
|
name: "user with valid role",
|
|
user: &iamv0alpha1.User{
|
|
Spec: iamv0alpha1.UserSpec{
|
|
Login: "testuser",
|
|
Role: "Admin",
|
|
},
|
|
},
|
|
requester: &identity.StaticRequester{
|
|
Type: types.TypeUser,
|
|
IsGrafanaAdmin: true,
|
|
},
|
|
searchClient: &FakeUserLegacySearchClient{},
|
|
expectError: false,
|
|
},
|
|
{
|
|
name: "user with existing email",
|
|
user: &iamv0alpha1.User{
|
|
ObjectMeta: metav1.ObjectMeta{
|
|
Name: "userx",
|
|
},
|
|
Spec: iamv0alpha1.UserSpec{
|
|
Email: "existing@example",
|
|
Role: "Viewer",
|
|
},
|
|
},
|
|
requester: &identity.StaticRequester{
|
|
Type: types.TypeUser,
|
|
IsGrafanaAdmin: false,
|
|
},
|
|
searchClient: &FakeUserLegacySearchClient{
|
|
Users: []*org.OrgUserDTO{
|
|
{Email: "existing@example"},
|
|
},
|
|
},
|
|
expectError: true,
|
|
errorContains: "email 'existing@example' is already taken",
|
|
},
|
|
{
|
|
name: "user with existing login",
|
|
user: &iamv0alpha1.User{
|
|
ObjectMeta: metav1.ObjectMeta{
|
|
Name: "userx",
|
|
},
|
|
Spec: iamv0alpha1.UserSpec{
|
|
Login: "existinguser",
|
|
Email: "existinguser@example",
|
|
Role: "Viewer",
|
|
},
|
|
},
|
|
requester: &identity.StaticRequester{
|
|
Type: types.TypeUser,
|
|
IsGrafanaAdmin: false,
|
|
},
|
|
searchClient: &FakeUserLegacySearchClient{
|
|
Users: []*org.OrgUserDTO{
|
|
{Login: "existinguser"},
|
|
},
|
|
},
|
|
expectError: true,
|
|
errorContains: "login 'existinguser' is already taken",
|
|
},
|
|
}
|
|
|
|
for _, tt := range tests {
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
ctx := identity.WithRequester(
|
|
context.Background(),
|
|
tt.requester,
|
|
)
|
|
|
|
err := ValidateOnCreate(ctx, tt.searchClient, tt.user)
|
|
|
|
if tt.expectError {
|
|
require.Error(t, err)
|
|
if tt.errorContains != "" {
|
|
require.Contains(t, err.Error(), tt.errorContains)
|
|
}
|
|
} else {
|
|
require.NoError(t, err)
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestValidateOnUpdate(t *testing.T) {
|
|
tests := []struct {
|
|
name string
|
|
oldUser *iamv0alpha1.User
|
|
newUser *iamv0alpha1.User
|
|
requester *identity.StaticRequester
|
|
searchClient resourcepb.ResourceIndexClient
|
|
expectError bool
|
|
errorContains string
|
|
}{
|
|
{
|
|
name: "un-provisioning a provisioned user",
|
|
oldUser: &iamv0alpha1.User{
|
|
Spec: iamv0alpha1.UserSpec{Login: "testuser", Provisioned: true, Role: "Viewer"},
|
|
},
|
|
newUser: &iamv0alpha1.User{
|
|
Spec: iamv0alpha1.UserSpec{Login: "testuser", Provisioned: false, Role: "Viewer"},
|
|
},
|
|
requester: &identity.StaticRequester{
|
|
Type: types.TypeUser,
|
|
IsGrafanaAdmin: true,
|
|
},
|
|
expectError: true,
|
|
errorContains: "provisioned user cannot be un-provisioned",
|
|
},
|
|
{
|
|
name: "non-service user provisions a user",
|
|
oldUser: &iamv0alpha1.User{
|
|
Spec: iamv0alpha1.UserSpec{Login: "testuser", Provisioned: false, Role: "Viewer"},
|
|
},
|
|
newUser: &iamv0alpha1.User{
|
|
Spec: iamv0alpha1.UserSpec{Login: "testuser", Provisioned: true, Role: "Viewer"},
|
|
},
|
|
requester: &identity.StaticRequester{
|
|
Type: types.TypeUser,
|
|
IsGrafanaAdmin: true,
|
|
},
|
|
expectError: true,
|
|
errorContains: "only service users can provision a user",
|
|
},
|
|
{
|
|
name: "service user provisions a user",
|
|
oldUser: &iamv0alpha1.User{
|
|
Spec: iamv0alpha1.UserSpec{Login: "testuser", Provisioned: false, Role: "Viewer"},
|
|
},
|
|
newUser: &iamv0alpha1.User{
|
|
Spec: iamv0alpha1.UserSpec{Login: "testuser", Provisioned: true, Role: "Viewer"},
|
|
},
|
|
requester: &identity.StaticRequester{
|
|
Type: types.TypeAccessPolicy,
|
|
},
|
|
expectError: false,
|
|
},
|
|
{
|
|
name: "no changes",
|
|
oldUser: &iamv0alpha1.User{
|
|
Spec: iamv0alpha1.UserSpec{Login: "testuser", Role: "Viewer"},
|
|
},
|
|
newUser: &iamv0alpha1.User{
|
|
Spec: iamv0alpha1.UserSpec{Login: "testuser", Role: "Viewer"},
|
|
},
|
|
requester: &identity.StaticRequester{
|
|
Type: types.TypeUser,
|
|
},
|
|
expectError: false,
|
|
},
|
|
{
|
|
name: "update with empty login and email",
|
|
oldUser: &iamv0alpha1.User{
|
|
Spec: iamv0alpha1.UserSpec{Login: "testuser", Role: "Viewer"},
|
|
},
|
|
newUser: &iamv0alpha1.User{
|
|
Spec: iamv0alpha1.UserSpec{Login: "", Email: "", Role: "Viewer"},
|
|
},
|
|
requester: &identity.StaticRequester{
|
|
Type: types.TypeAccessPolicy,
|
|
},
|
|
expectError: true,
|
|
errorContains: "user must have either login or email",
|
|
},
|
|
{
|
|
name: "update with only login",
|
|
oldUser: &iamv0alpha1.User{
|
|
Spec: iamv0alpha1.UserSpec{Email: "test@example", Role: "Viewer"},
|
|
},
|
|
newUser: &iamv0alpha1.User{
|
|
Spec: iamv0alpha1.UserSpec{Login: "testuser", Email: "", Role: "Viewer"},
|
|
},
|
|
requester: &identity.StaticRequester{
|
|
Type: types.TypeAccessPolicy,
|
|
},
|
|
searchClient: &FakeUserLegacySearchClient{},
|
|
expectError: false,
|
|
},
|
|
{
|
|
name: "update with only email",
|
|
oldUser: &iamv0alpha1.User{
|
|
Spec: iamv0alpha1.UserSpec{Login: "testuser", Role: "Viewer"},
|
|
},
|
|
newUser: &iamv0alpha1.User{
|
|
Spec: iamv0alpha1.UserSpec{Login: "", Email: "test@example", Role: "Viewer"},
|
|
},
|
|
requester: &identity.StaticRequester{
|
|
Type: types.TypeAccessPolicy,
|
|
},
|
|
searchClient: &FakeUserLegacySearchClient{},
|
|
expectError: false,
|
|
},
|
|
{
|
|
name: "service user verifies email",
|
|
oldUser: &iamv0alpha1.User{
|
|
Spec: iamv0alpha1.UserSpec{Login: "testuser", EmailVerified: false, Role: "Viewer"},
|
|
},
|
|
newUser: &iamv0alpha1.User{
|
|
Spec: iamv0alpha1.UserSpec{Login: "testuser", EmailVerified: true, Role: "Viewer"},
|
|
},
|
|
requester: &identity.StaticRequester{
|
|
Type: types.TypeAccessPolicy,
|
|
},
|
|
expectError: false,
|
|
},
|
|
{
|
|
name: "non-service user verifies email",
|
|
oldUser: &iamv0alpha1.User{
|
|
Spec: iamv0alpha1.UserSpec{Login: "testuser", EmailVerified: false, Role: "Viewer"},
|
|
},
|
|
newUser: &iamv0alpha1.User{
|
|
Spec: iamv0alpha1.UserSpec{Login: "testuser", EmailVerified: true, Role: "Viewer"},
|
|
},
|
|
requester: &identity.StaticRequester{
|
|
Type: types.TypeUser,
|
|
},
|
|
expectError: true,
|
|
errorContains: "only service users can verify email",
|
|
},
|
|
{
|
|
name: "grafana admin disables user",
|
|
oldUser: &iamv0alpha1.User{
|
|
Spec: iamv0alpha1.UserSpec{Login: "testuser", Disabled: false, Role: "Viewer"},
|
|
},
|
|
newUser: &iamv0alpha1.User{
|
|
Spec: iamv0alpha1.UserSpec{Login: "testuser", Disabled: true, Role: "Viewer"},
|
|
},
|
|
requester: &identity.StaticRequester{
|
|
Type: types.TypeUser,
|
|
IsGrafanaAdmin: true,
|
|
},
|
|
expectError: false,
|
|
},
|
|
{
|
|
name: "non-admin disables user",
|
|
oldUser: &iamv0alpha1.User{
|
|
Spec: iamv0alpha1.UserSpec{Login: "testuser", Disabled: false, Role: "Viewer"},
|
|
},
|
|
newUser: &iamv0alpha1.User{
|
|
Spec: iamv0alpha1.UserSpec{Login: "testuser", Disabled: true, Role: "Viewer"},
|
|
},
|
|
requester: &identity.StaticRequester{
|
|
Type: types.TypeUser,
|
|
IsGrafanaAdmin: false,
|
|
},
|
|
expectError: true,
|
|
errorContains: "only grafana admins can disable or enable a user",
|
|
},
|
|
{
|
|
name: "grafana admin grants admin",
|
|
oldUser: &iamv0alpha1.User{
|
|
Spec: iamv0alpha1.UserSpec{Login: "testuser", GrafanaAdmin: false, Role: "Viewer"},
|
|
},
|
|
newUser: &iamv0alpha1.User{
|
|
Spec: iamv0alpha1.UserSpec{Login: "testuser", GrafanaAdmin: true, Role: "Viewer"},
|
|
},
|
|
requester: &identity.StaticRequester{
|
|
Type: types.TypeUser,
|
|
IsGrafanaAdmin: true,
|
|
},
|
|
expectError: false,
|
|
},
|
|
{
|
|
name: "non-admin grants admin",
|
|
oldUser: &iamv0alpha1.User{
|
|
Spec: iamv0alpha1.UserSpec{Login: "testuser", GrafanaAdmin: false, Role: "Viewer"},
|
|
},
|
|
newUser: &iamv0alpha1.User{
|
|
Spec: iamv0alpha1.UserSpec{Login: "testuser", GrafanaAdmin: true, Role: "Viewer"},
|
|
},
|
|
requester: &identity.StaticRequester{
|
|
Type: types.TypeUser,
|
|
IsGrafanaAdmin: false,
|
|
},
|
|
expectError: true,
|
|
errorContains: "only grafana admins can change grafana admin status",
|
|
},
|
|
{
|
|
name: "update to empty role",
|
|
oldUser: &iamv0alpha1.User{
|
|
Spec: iamv0alpha1.UserSpec{Login: "testuser", Role: "Viewer"},
|
|
},
|
|
newUser: &iamv0alpha1.User{
|
|
Spec: iamv0alpha1.UserSpec{Login: "testuser", Role: ""},
|
|
},
|
|
requester: &identity.StaticRequester{
|
|
Type: types.TypeUser,
|
|
IsGrafanaAdmin: true,
|
|
},
|
|
expectError: true,
|
|
errorContains: "role is required",
|
|
},
|
|
{
|
|
name: "update to invalid role",
|
|
oldUser: &iamv0alpha1.User{
|
|
Spec: iamv0alpha1.UserSpec{Login: "testuser", Role: "Viewer"},
|
|
},
|
|
newUser: &iamv0alpha1.User{
|
|
Spec: iamv0alpha1.UserSpec{Login: "testuser", Role: "InvalidRole"},
|
|
},
|
|
requester: &identity.StaticRequester{
|
|
Type: types.TypeUser,
|
|
IsGrafanaAdmin: true,
|
|
},
|
|
expectError: true,
|
|
errorContains: "invalid role 'InvalidRole'",
|
|
},
|
|
{
|
|
name: "update to valid role",
|
|
oldUser: &iamv0alpha1.User{
|
|
Spec: iamv0alpha1.UserSpec{Login: "testuser", Role: "Editor"},
|
|
},
|
|
newUser: &iamv0alpha1.User{
|
|
Spec: iamv0alpha1.UserSpec{Login: "testuser", Role: "Viewer"},
|
|
},
|
|
requester: &identity.StaticRequester{
|
|
Type: types.TypeUser,
|
|
IsGrafanaAdmin: true,
|
|
},
|
|
expectError: false,
|
|
},
|
|
{
|
|
name: "non-service non-admin user updating role field is allowed",
|
|
oldUser: &iamv0alpha1.User{
|
|
Spec: iamv0alpha1.UserSpec{Login: "testuser", Email: "user@example", Role: "Viewer"},
|
|
},
|
|
newUser: &iamv0alpha1.User{
|
|
Spec: iamv0alpha1.UserSpec{Login: "testuser", Email: "user@example", Role: "Editor"},
|
|
},
|
|
requester: &identity.StaticRequester{
|
|
Type: types.TypeUser,
|
|
IsGrafanaAdmin: false,
|
|
},
|
|
expectError: false,
|
|
},
|
|
{
|
|
name: "non-service non-admin user updating non-role fields is forbidden",
|
|
oldUser: &iamv0alpha1.User{
|
|
Spec: iamv0alpha1.UserSpec{Login: "testuser", Email: "old@example", Role: "Viewer"},
|
|
},
|
|
newUser: &iamv0alpha1.User{
|
|
Spec: iamv0alpha1.UserSpec{Login: "testuser", Email: "new@example", Role: "Viewer"},
|
|
},
|
|
requester: &identity.StaticRequester{
|
|
Type: types.TypeUser,
|
|
IsGrafanaAdmin: false,
|
|
},
|
|
expectError: true,
|
|
errorContains: "updating fields beyond org role requires service identity or grafana admin",
|
|
},
|
|
{
|
|
name: "grafana admin updating non-role fields is allowed",
|
|
oldUser: &iamv0alpha1.User{
|
|
Spec: iamv0alpha1.UserSpec{Login: "testuser", Email: "user@example", Title: "Old Title", Role: "Viewer"},
|
|
},
|
|
newUser: &iamv0alpha1.User{
|
|
Spec: iamv0alpha1.UserSpec{Login: "testuser", Email: "user@example", Title: "New Title", Role: "Viewer"},
|
|
},
|
|
requester: &identity.StaticRequester{
|
|
Type: types.TypeUser,
|
|
IsGrafanaAdmin: true,
|
|
},
|
|
expectError: false,
|
|
},
|
|
{
|
|
name: "update with existing email",
|
|
oldUser: &iamv0alpha1.User{
|
|
ObjectMeta: metav1.ObjectMeta{
|
|
Name: "userx",
|
|
},
|
|
Spec: iamv0alpha1.UserSpec{Email: "one@example", Role: "Viewer"},
|
|
},
|
|
newUser: &iamv0alpha1.User{
|
|
ObjectMeta: metav1.ObjectMeta{
|
|
Name: "userx",
|
|
},
|
|
Spec: iamv0alpha1.UserSpec{Email: "two@example", Role: "Viewer"},
|
|
},
|
|
requester: &identity.StaticRequester{
|
|
Type: types.TypeAccessPolicy,
|
|
IsGrafanaAdmin: true,
|
|
},
|
|
searchClient: &FakeUserLegacySearchClient{
|
|
Users: []*org.OrgUserDTO{
|
|
{Email: "two@example"},
|
|
},
|
|
},
|
|
expectError: true,
|
|
errorContains: "email 'two@example' is already taken",
|
|
},
|
|
{
|
|
name: "update with existing login",
|
|
oldUser: &iamv0alpha1.User{
|
|
ObjectMeta: metav1.ObjectMeta{
|
|
Name: "userx",
|
|
},
|
|
Spec: iamv0alpha1.UserSpec{Login: "one", Role: "Viewer"},
|
|
},
|
|
newUser: &iamv0alpha1.User{
|
|
ObjectMeta: metav1.ObjectMeta{
|
|
Name: "userx",
|
|
},
|
|
Spec: iamv0alpha1.UserSpec{Login: "two", Role: "Viewer"},
|
|
},
|
|
requester: &identity.StaticRequester{
|
|
Type: types.TypeAccessPolicy,
|
|
IsGrafanaAdmin: true,
|
|
},
|
|
searchClient: &FakeUserLegacySearchClient{
|
|
Users: []*org.OrgUserDTO{
|
|
{Name: "other", UID: "uid456", Login: "two"},
|
|
},
|
|
},
|
|
expectError: true,
|
|
errorContains: "login 'two' is already taken",
|
|
},
|
|
{
|
|
name: "update with no change to login or email",
|
|
oldUser: &iamv0alpha1.User{
|
|
Spec: iamv0alpha1.UserSpec{Login: "testuser", Email: "test@example", Role: "Viewer"},
|
|
},
|
|
newUser: &iamv0alpha1.User{
|
|
Spec: iamv0alpha1.UserSpec{Login: "testuser", Email: "test@example", Role: "Editor"},
|
|
},
|
|
requester: &identity.StaticRequester{
|
|
Type: types.TypeUser,
|
|
IsGrafanaAdmin: true,
|
|
},
|
|
searchClient: &FakeUserLegacySearchClient{
|
|
Users: []*org.OrgUserDTO{
|
|
{Login: "testuser", Email: "test@example"},
|
|
},
|
|
},
|
|
expectError: false,
|
|
},
|
|
}
|
|
|
|
for _, tt := range tests {
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
ctx := identity.WithRequester(
|
|
context.Background(),
|
|
tt.requester,
|
|
)
|
|
|
|
err := ValidateOnUpdate(ctx, tt.searchClient, tt.oldUser, tt.newUser)
|
|
|
|
if tt.expectError {
|
|
require.Error(t, err)
|
|
if tt.errorContains != "" {
|
|
require.Contains(t, err.Error(), tt.errorContains)
|
|
}
|
|
} else {
|
|
require.NoError(t, err)
|
|
}
|
|
})
|
|
}
|
|
}
|