forked from openshift/origin
-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathstrategy_claim.go
78 lines (62 loc) · 2.39 KB
/
strategy_claim.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
package identitymapper
import (
"context"
"fmt"
userapi "github.com/openshift/api/user/v1"
userclient "github.com/openshift/client-go/user/clientset/versioned/typed/user/v1"
kerrs "k8s.io/apimachinery/pkg/api/errors"
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
"k8s.io/apimachinery/pkg/util/sets"
)
var _ = UserForNewIdentityGetter(&StrategyClaim{})
// StrategyClaim associates a new identity with a user with the identity's preferred username
// if no other identities are already associated with the user
type StrategyClaim struct {
user userclient.UserInterface
initializer Initializer
}
type claimError struct {
User *userapi.User
Identity *userapi.Identity
}
func IsClaimError(err error) bool {
_, ok := err.(claimError)
return ok
}
func NewClaimError(user *userapi.User, identity *userapi.Identity) error {
return claimError{User: user, Identity: identity}
}
func (c claimError) Error() string {
return fmt.Sprintf("user %q cannot be claimed by identity %q because it is already mapped to %v", c.User.Name, c.Identity.Name, c.User.Identities)
}
func NewStrategyClaim(user userclient.UserInterface, initializer Initializer) UserForNewIdentityGetter {
return &StrategyClaim{user, initializer}
}
func (s *StrategyClaim) UserForNewIdentity(ctx context.Context, preferredUserName string, identity *userapi.Identity) (*userapi.User, error) {
persistedUser, err := s.user.Get(preferredUserName, metav1.GetOptions{})
switch {
case kerrs.IsNotFound(err):
// CreateUser a new user, propagating any "already exists" errors
desiredUser := &userapi.User{}
desiredUser.Name = preferredUserName
desiredUser.Identities = []string{identity.Name}
s.initializer.InitializeUser(identity, desiredUser)
return s.user.Create(desiredUser)
case err == nil:
// If the existing user already references our identity, we're done
if sets.NewString(persistedUser.Identities...).Has(identity.Name) {
return persistedUser, nil
}
// If this user has no other identities, claim, initialize, and update
if len(persistedUser.Identities) == 0 {
persistedUser.Identities = []string{identity.Name}
s.initializer.InitializeUser(identity, persistedUser)
return s.user.Update(persistedUser)
}
// Otherwise another identity has already claimed this user, return an error
return nil, NewClaimError(persistedUser, identity)
default:
// Fail on errors other than "not found"
return nil, err
}
}