2017-02-17 19:37:00 +00:00
|
|
|
package enterprise
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
2017-10-18 18:17:42 +00:00
|
|
|
"fmt"
|
2017-02-17 19:37:00 +00:00
|
|
|
|
|
|
|
"github.com/influxdata/chronograf"
|
|
|
|
)
|
|
|
|
|
2017-02-23 22:02:53 +00:00
|
|
|
// UserStore uses a control client operate on Influx Enterprise users
|
|
|
|
type UserStore struct {
|
|
|
|
Ctrl
|
|
|
|
Logger chronograf.Logger
|
|
|
|
}
|
|
|
|
|
2017-02-19 19:47:19 +00:00
|
|
|
// Add creates a new User in Influx Enterprise
|
2017-02-23 22:02:53 +00:00
|
|
|
func (c *UserStore) Add(ctx context.Context, u *chronograf.User) (*chronograf.User, error) {
|
2017-02-17 19:37:00 +00:00
|
|
|
if err := c.Ctrl.CreateUser(ctx, u.Name, u.Passwd); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2017-02-24 07:16:28 +00:00
|
|
|
perms := ToEnterprise(u.Permissions)
|
2017-03-02 19:57:53 +00:00
|
|
|
|
2017-02-24 07:16:28 +00:00
|
|
|
if err := c.Ctrl.SetUserPerms(ctx, u.Name, perms); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2017-03-10 22:52:24 +00:00
|
|
|
for _, role := range u.Roles {
|
|
|
|
if err := c.Ctrl.AddRoleUsers(ctx, role.Name, []string{u.Name}); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
}
|
2017-03-10 19:24:48 +00:00
|
|
|
|
2017-10-18 18:17:42 +00:00
|
|
|
return c.Get(ctx, chronograf.UserQuery{Name: &u.Name})
|
2017-02-17 19:37:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Delete the User from Influx Enterprise
|
2017-02-23 22:02:53 +00:00
|
|
|
func (c *UserStore) Delete(ctx context.Context, u *chronograf.User) error {
|
2017-02-17 19:37:00 +00:00
|
|
|
return c.Ctrl.DeleteUser(ctx, u.Name)
|
|
|
|
}
|
|
|
|
|
2017-11-30 17:55:59 +00:00
|
|
|
// Number of users in Influx
|
|
|
|
func (c *UserStore) Num(ctx context.Context) (int, error) {
|
|
|
|
all, err := c.All(ctx)
|
|
|
|
if err != nil {
|
|
|
|
return 0, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return len(all), nil
|
|
|
|
}
|
|
|
|
|
2017-02-17 19:37:00 +00:00
|
|
|
// Get retrieves a user if name exists.
|
2017-10-18 18:17:42 +00:00
|
|
|
func (c *UserStore) Get(ctx context.Context, q chronograf.UserQuery) (*chronograf.User, error) {
|
|
|
|
if q.Name == nil {
|
|
|
|
return nil, fmt.Errorf("query must specify name")
|
|
|
|
}
|
|
|
|
u, err := c.Ctrl.User(ctx, *q.Name)
|
2017-02-17 19:37:00 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2017-03-02 01:13:44 +00:00
|
|
|
|
|
|
|
ur, err := c.Ctrl.UserRoles(ctx)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2017-10-18 18:17:42 +00:00
|
|
|
role := ur[*q.Name]
|
2017-03-02 01:13:44 +00:00
|
|
|
cr := role.ToChronograf()
|
|
|
|
// For now we are removing all users from a role being returned.
|
|
|
|
for i, r := range cr {
|
|
|
|
r.Users = []chronograf.User{}
|
|
|
|
cr[i] = r
|
|
|
|
}
|
2017-02-17 19:37:00 +00:00
|
|
|
return &chronograf.User{
|
2017-02-17 22:35:56 +00:00
|
|
|
Name: u.Name,
|
2017-02-23 22:02:53 +00:00
|
|
|
Permissions: ToChronograf(u.Permissions),
|
2017-03-02 01:13:44 +00:00
|
|
|
Roles: cr,
|
2017-02-17 19:37:00 +00:00
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Update the user's permissions or roles
|
2017-02-23 22:02:53 +00:00
|
|
|
func (c *UserStore) Update(ctx context.Context, u *chronograf.User) error {
|
2017-02-17 22:35:56 +00:00
|
|
|
// Only allow one type of change at a time. If it is a password
|
|
|
|
// change then do it and return without any changes to permissions
|
|
|
|
if u.Passwd != "" {
|
|
|
|
return c.Ctrl.ChangePassword(ctx, u.Name, u.Passwd)
|
|
|
|
}
|
2017-03-10 22:52:24 +00:00
|
|
|
|
2017-11-30 18:41:50 +00:00
|
|
|
if u.Roles != nil {
|
|
|
|
// Make a list of the roles we want this user to have:
|
|
|
|
want := make([]string, len(u.Roles))
|
|
|
|
for i, r := range u.Roles {
|
|
|
|
want[i] = r.Name
|
|
|
|
}
|
2017-03-10 22:52:24 +00:00
|
|
|
|
2017-11-30 18:41:50 +00:00
|
|
|
// Find the list of all roles this user is currently in
|
|
|
|
userRoles, err := c.UserRoles(ctx)
|
|
|
|
if err != nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
// Make a list of the roles the user currently has
|
|
|
|
roles := userRoles[u.Name]
|
|
|
|
have := make([]string, len(roles.Roles))
|
|
|
|
for i, r := range roles.Roles {
|
|
|
|
have[i] = r.Name
|
|
|
|
}
|
2017-03-10 22:52:24 +00:00
|
|
|
|
2017-11-30 18:41:50 +00:00
|
|
|
// Calculate the roles the user will be removed from and the roles the user
|
|
|
|
// will be added to.
|
|
|
|
revoke, add := Difference(want, have)
|
2017-03-10 22:52:24 +00:00
|
|
|
|
2017-11-30 18:41:50 +00:00
|
|
|
// First, add the user to the new roles
|
|
|
|
for _, role := range add {
|
|
|
|
if err := c.Ctrl.AddRoleUsers(ctx, role, []string{u.Name}); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2017-03-10 22:52:24 +00:00
|
|
|
}
|
|
|
|
|
2017-11-30 18:41:50 +00:00
|
|
|
// ... and now remove the user from an extra roles
|
|
|
|
for _, role := range revoke {
|
|
|
|
if err := c.Ctrl.RemoveRoleUsers(ctx, role, []string{u.Name}); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2017-03-10 22:52:24 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-11-30 18:41:50 +00:00
|
|
|
if u.Permissions != nil {
|
|
|
|
perms := ToEnterprise(u.Permissions)
|
|
|
|
return c.Ctrl.SetUserPerms(ctx, u.Name, perms)
|
|
|
|
}
|
|
|
|
return nil
|
2017-02-17 19:37:00 +00:00
|
|
|
}
|
2017-02-17 21:13:51 +00:00
|
|
|
|
|
|
|
// All is all users in influx
|
2017-02-23 22:02:53 +00:00
|
|
|
func (c *UserStore) All(ctx context.Context) ([]chronograf.User, error) {
|
2017-02-17 21:13:51 +00:00
|
|
|
all, err := c.Ctrl.Users(ctx, nil)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2017-03-02 01:13:44 +00:00
|
|
|
ur, err := c.Ctrl.UserRoles(ctx)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2017-02-17 21:13:51 +00:00
|
|
|
res := make([]chronograf.User, len(all.Users))
|
|
|
|
for i, user := range all.Users {
|
2017-03-02 01:13:44 +00:00
|
|
|
role := ur[user.Name]
|
|
|
|
cr := role.ToChronograf()
|
|
|
|
// For now we are removing all users from a role being returned.
|
|
|
|
for i, r := range cr {
|
|
|
|
r.Users = []chronograf.User{}
|
|
|
|
cr[i] = r
|
|
|
|
}
|
|
|
|
|
2017-02-17 21:13:51 +00:00
|
|
|
res[i] = chronograf.User{
|
2017-02-17 22:35:56 +00:00
|
|
|
Name: user.Name,
|
2017-02-23 22:02:53 +00:00
|
|
|
Permissions: ToChronograf(user.Permissions),
|
2017-03-02 01:13:44 +00:00
|
|
|
Roles: cr,
|
2017-02-17 21:13:51 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return res, nil
|
|
|
|
}
|
2017-02-17 22:35:56 +00:00
|
|
|
|
2017-02-23 22:02:53 +00:00
|
|
|
// ToEnterprise converts chronograf permission shape to enterprise
|
|
|
|
func ToEnterprise(perms chronograf.Permissions) Permissions {
|
2017-02-17 22:35:56 +00:00
|
|
|
res := Permissions{}
|
|
|
|
for _, perm := range perms {
|
|
|
|
if perm.Scope == chronograf.AllScope {
|
|
|
|
// Enterprise uses empty string as the key for all databases
|
|
|
|
res[""] = perm.Allowed
|
|
|
|
} else {
|
|
|
|
res[perm.Name] = perm.Allowed
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return res
|
|
|
|
}
|
|
|
|
|
2017-02-23 22:02:53 +00:00
|
|
|
// ToChronograf converts enterprise permissions shape to chronograf shape
|
|
|
|
func ToChronograf(perms Permissions) chronograf.Permissions {
|
2017-02-17 22:35:56 +00:00
|
|
|
res := chronograf.Permissions{}
|
|
|
|
for db, perm := range perms {
|
|
|
|
// Enterprise uses empty string as the key for all databases
|
|
|
|
if db == "" {
|
|
|
|
res = append(res, chronograf.Permission{
|
|
|
|
Scope: chronograf.AllScope,
|
|
|
|
Allowed: perm,
|
|
|
|
})
|
|
|
|
} else {
|
|
|
|
res = append(res, chronograf.Permission{
|
|
|
|
Scope: chronograf.DBScope,
|
|
|
|
Name: db,
|
|
|
|
Allowed: perm,
|
|
|
|
})
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return res
|
|
|
|
}
|