-
Notifications
You must be signed in to change notification settings - Fork 29
/
Copy pathauth.go
231 lines (191 loc) · 6.3 KB
/
auth.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
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
package joe
import (
"errors"
"fmt"
"strings"
"go.uber.org/zap"
)
// ErrNotAllowed is returned if the user is not allowed access to a specific scope.
const ErrNotAllowed = Error("not allowed")
// permissionKeyPrefix is the key prefix in the Storage that all permission keys have.
const permissionKeyPrefix = "joe.permissions."
// Auth implements logic to add user authorization checks to your bot.
type Auth struct {
logger *zap.Logger
store *Storage
}
// NewAuth creates a new Auth instance.
func NewAuth(logger *zap.Logger, store *Storage) *Auth {
return &Auth{
logger: logger,
store: store,
}
}
// CheckPermission checks if a user has permissions to access a resource under a
// given scope. If the user is not permitted access this function returns
// ErrNotAllowed.
//
// Scopes are interpreted in a hierarchical way where scope A can contain scope B
// if A is a prefix to B. For example, you can check if a user is allowed to
// read or write from the "Example" API by checking the "api.example.read" or
// "api.example.write" scope. When you grant the scope to a user you can now
// either decide only to grant the very specific "api.example.read" scope which
// means the user will not have write permissions or you can allow people
// write-only access via "api.example.write".
//
// Alternatively you can also grant any access to the Example API via "api.example"
// which includes both the read and write scope beneath it. If you choose to, you
// could also allow even more general access to everything in the api via the
// "api" scope. The empty scope "" cannot be granted and will thus always return
// an error in the permission check.
func (a *Auth) CheckPermission(scope, userID string) error {
key := a.permissionsKey(userID)
permissions, err := a.loadPermissions(key)
if err != nil {
return err
}
a.logger.Debug("Checking user permissions",
zap.String("requested_scope", scope),
zap.String("user_id", userID),
)
for _, p := range permissions {
if strings.HasPrefix(scope, p) {
return nil
}
}
return ErrNotAllowed
}
// Users returns a list of user IDs having one or more permission scopes.
func (a *Auth) Users() ([]string, error) {
a.logger.Debug("Retrieving all user IDs from storage")
keys, err := a.store.Keys()
if err != nil {
return nil, fmt.Errorf("failed to load permissions: %w", err)
}
var userIDs []string
for _, key := range keys {
if strings.HasPrefix(key, permissionKeyPrefix) {
userID := strings.TrimPrefix(key, permissionKeyPrefix)
userIDs = append(userIDs, userID)
}
}
return userIDs, nil
}
// UserPermissions returns all permission scopes for a specific user.
func (a *Auth) UserPermissions(userID string) ([]string, error) {
a.logger.Debug("Retrieving user permissions from storage",
zap.String("user_id", userID),
)
key := a.permissionsKey(userID)
permissions, err := a.loadPermissions(key)
if err != nil {
return nil, err
}
return permissions, nil
}
func (a *Auth) loadPermissions(key string) ([]string, error) {
var permissions []string
ok, err := a.store.Get(key, &permissions)
if err != nil {
return nil, fmt.Errorf("failed to load user permissions: %w", err)
}
if !ok {
return nil, nil
}
return permissions, nil
}
// Grant adds a permission scope to the given user. When a scope was granted
// to a specific user it can be checked later via CheckPermission(…).
// The returned boolean indicates whether the scope was actually added (i.e. true)
// or the user already had the granted scope (false).
//
// Note that granting a scope is an idempotent operations so granting the same
// scope multiple times is a safe operation and will not change the internal
// permissions that are written to the Memory.
//
// The empty scope cannot be granted and trying to do so will result in an error.
// If you want to grant access to all scopes you should prefix them with a
// common scope such as "root." or "api.".
func (a *Auth) Grant(scope, userID string) (bool, error) {
if scope == "" {
return false, errors.New("scope cannot be empty")
}
key := a.permissionsKey(userID)
oldPermissions, err := a.loadPermissions(key)
if err != nil {
return false, err
}
newPermissions := make([]string, 0, len(oldPermissions)+1)
for _, p := range oldPermissions {
if strings.HasPrefix(scope, p) {
// The user already has this or a scope that "contains" it
return false, nil
}
if !strings.HasPrefix(p, scope) {
newPermissions = append(newPermissions, p)
}
}
a.logger.Info("Granting user permission",
zap.String("userID", userID),
zap.String("scope", scope),
)
newPermissions = append(newPermissions, scope)
err = a.updatePermissions(key, newPermissions)
return true, err
}
// Revoke removes a previously granted permission from a user. If the user does
// not currently have the revoked scope this function returns false and no error.
//
// If you are trying to revoke a permission but the user was previously granted
// a scope that contains the revoked scope this function returns an error.
func (a *Auth) Revoke(scope, userID string) (bool, error) {
if scope == "" {
return false, errors.New("scope cannot be empty")
}
key := a.permissionsKey(userID)
oldPermissions, err := a.loadPermissions(key)
if err != nil {
return false, err
}
if len(oldPermissions) == 0 {
return false, nil
}
var revoked bool
newPermissions := make([]string, 0, len(oldPermissions))
for _, p := range oldPermissions {
if p == scope {
revoked = true
continue
}
if strings.HasPrefix(scope, p) {
return false, fmt.Errorf("cannot revoke scope %q because the user still has the more general scope %q", scope, p)
}
newPermissions = append(newPermissions, p)
}
if !revoked {
return false, nil
}
a.logger.Info("Revoking user permission",
zap.String("userID", userID),
zap.String("scope", scope),
)
if len(newPermissions) == 0 {
_, err := a.store.Delete(key)
if err != nil {
return false, fmt.Errorf("failed to delete last user permission: %w", err)
}
return true, nil
}
err = a.updatePermissions(key, newPermissions)
return true, err
}
func (a *Auth) updatePermissions(key string, permissions []string) error {
err := a.store.Set(key, permissions)
if err != nil {
return fmt.Errorf("failed to update user permissions: %w", err)
}
return nil
}
func (a *Auth) permissionsKey(userID string) string {
return permissionKeyPrefix + userID
}