2017-10-31 20:41:17 +00:00
|
|
|
package server
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
|
|
|
|
"github.com/influxdata/chronograf"
|
2017-10-31 22:07:42 +00:00
|
|
|
"github.com/influxdata/chronograf/noop"
|
2017-10-31 20:41:17 +00:00
|
|
|
"github.com/influxdata/chronograf/organizations"
|
2017-11-03 19:32:59 +00:00
|
|
|
"github.com/influxdata/chronograf/roles"
|
2017-10-31 20:41:17 +00:00
|
|
|
)
|
|
|
|
|
2017-11-01 18:24:40 +00:00
|
|
|
// hasOrganizationContext retrieves organization specified on context
|
|
|
|
// under the organizations.ContextKey
|
2017-10-31 21:40:58 +00:00
|
|
|
func hasOrganizationContext(ctx context.Context) (string, bool) {
|
|
|
|
// prevents panic in case of nil context
|
|
|
|
if ctx == nil {
|
|
|
|
return "", false
|
|
|
|
}
|
2017-11-01 13:49:02 +00:00
|
|
|
orgID, ok := ctx.Value(organizations.ContextKey).(string)
|
2017-10-31 21:40:58 +00:00
|
|
|
// should never happen
|
|
|
|
if !ok {
|
|
|
|
return "", false
|
|
|
|
}
|
|
|
|
if orgID == "" {
|
|
|
|
return "", false
|
|
|
|
}
|
|
|
|
return orgID, true
|
|
|
|
}
|
|
|
|
|
2017-11-03 19:32:59 +00:00
|
|
|
// hasRoleContext retrieves organization specified on context
|
|
|
|
// under the organizations.ContextKey
|
|
|
|
func hasRoleContext(ctx context.Context) (string, bool) {
|
|
|
|
// prevents panic in case of nil context
|
|
|
|
if ctx == nil {
|
|
|
|
return "", false
|
|
|
|
}
|
|
|
|
role, ok := ctx.Value(roles.ContextKey).(string)
|
|
|
|
// should never happen
|
|
|
|
if !ok {
|
|
|
|
return "", false
|
|
|
|
}
|
|
|
|
switch role {
|
2017-11-03 20:32:05 +00:00
|
|
|
case roles.MemberRoleName, roles.ViewerRoleName, roles.EditorRoleName, roles.AdminRoleName:
|
2017-11-03 19:32:59 +00:00
|
|
|
return role, true
|
|
|
|
default:
|
|
|
|
return "", false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-11-10 16:18:06 +00:00
|
|
|
type userContextKey string
|
2017-11-01 13:49:02 +00:00
|
|
|
|
2017-11-10 16:18:06 +00:00
|
|
|
// UserContextKey is the context key for retrieving the user off of context
|
|
|
|
const UserContextKey = userContextKey("user")
|
2017-10-31 21:49:35 +00:00
|
|
|
|
2017-11-09 14:37:15 +00:00
|
|
|
// hasUserContext speficies if the context contains
|
2017-11-10 16:18:06 +00:00
|
|
|
// the UserContextKey and that the value stored there is chronograf.User
|
2017-11-09 14:37:15 +00:00
|
|
|
func hasUserContext(ctx context.Context) (*chronograf.User, bool) {
|
2017-10-31 21:49:35 +00:00
|
|
|
// prevents panic in case of nil context
|
|
|
|
if ctx == nil {
|
2017-11-09 14:37:15 +00:00
|
|
|
return nil, false
|
2017-10-31 21:49:35 +00:00
|
|
|
}
|
2017-11-10 16:18:06 +00:00
|
|
|
u, ok := ctx.Value(UserContextKey).(*chronograf.User)
|
2017-10-31 21:49:35 +00:00
|
|
|
// should never happen
|
|
|
|
if !ok {
|
2017-11-09 14:37:15 +00:00
|
|
|
return nil, false
|
2017-10-31 21:49:35 +00:00
|
|
|
}
|
2017-11-09 14:18:13 +00:00
|
|
|
if u == nil {
|
2017-11-09 14:37:15 +00:00
|
|
|
return nil, false
|
|
|
|
}
|
|
|
|
return u, true
|
|
|
|
}
|
|
|
|
|
|
|
|
// hasSuperAdminContext speficies if the context contains
|
2017-11-10 16:18:06 +00:00
|
|
|
// the UserContextKey user is a super admin
|
2017-11-09 14:37:15 +00:00
|
|
|
func hasSuperAdminContext(ctx context.Context) bool {
|
|
|
|
u, ok := hasUserContext(ctx)
|
|
|
|
if !ok {
|
2017-11-09 14:18:13 +00:00
|
|
|
return false
|
|
|
|
}
|
|
|
|
return u.SuperAdmin
|
2017-10-31 21:49:35 +00:00
|
|
|
}
|
|
|
|
|
2017-11-01 20:43:31 +00:00
|
|
|
// DataStore is collection of resources that are used by the Service
|
2017-11-01 18:24:40 +00:00
|
|
|
// Abstracting this into an interface was useful for isolated testing
|
2017-10-31 20:41:17 +00:00
|
|
|
type DataStore interface {
|
|
|
|
Sources(ctx context.Context) chronograf.SourcesStore
|
|
|
|
Servers(ctx context.Context) chronograf.ServersStore
|
|
|
|
Layouts(ctx context.Context) chronograf.LayoutsStore
|
|
|
|
Users(ctx context.Context) chronograf.UsersStore
|
|
|
|
Organizations(ctx context.Context) chronograf.OrganizationsStore
|
|
|
|
Dashboards(ctx context.Context) chronograf.DashboardsStore
|
2017-12-13 01:06:57 +00:00
|
|
|
Config(ctx context.Context) chronograf.ConfigStore
|
2017-10-31 20:41:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// ensure that Store implements a DataStore
|
|
|
|
var _ DataStore = &Store{}
|
|
|
|
|
2017-11-01 18:24:40 +00:00
|
|
|
// Store implements the DataStore interface
|
2017-10-31 20:41:17 +00:00
|
|
|
type Store struct {
|
|
|
|
SourcesStore chronograf.SourcesStore
|
|
|
|
ServersStore chronograf.ServersStore
|
|
|
|
LayoutsStore chronograf.LayoutsStore
|
|
|
|
UsersStore chronograf.UsersStore
|
|
|
|
DashboardsStore chronograf.DashboardsStore
|
|
|
|
OrganizationsStore chronograf.OrganizationsStore
|
2017-12-13 01:06:57 +00:00
|
|
|
ConfigStore chronograf.ConfigStore
|
2017-10-31 20:41:17 +00:00
|
|
|
}
|
|
|
|
|
2017-11-01 18:24:40 +00:00
|
|
|
// Sources returns a noop.SourcesStore if the context has no organization specified
|
|
|
|
// and a organization.SourcesStore otherwise.
|
2017-10-31 20:41:17 +00:00
|
|
|
func (s *Store) Sources(ctx context.Context) chronograf.SourcesStore {
|
2017-11-08 21:56:34 +00:00
|
|
|
if isServer := hasServerContext(ctx); isServer {
|
|
|
|
return s.SourcesStore
|
|
|
|
}
|
2017-10-31 21:40:58 +00:00
|
|
|
if org, ok := hasOrganizationContext(ctx); ok {
|
2017-11-29 22:32:41 +00:00
|
|
|
return organizations.NewSourcesStore(s.SourcesStore, org)
|
2017-10-31 21:40:58 +00:00
|
|
|
}
|
|
|
|
|
2017-10-31 22:07:42 +00:00
|
|
|
return &noop.SourcesStore{}
|
2017-10-31 20:41:17 +00:00
|
|
|
}
|
|
|
|
|
2017-11-01 18:24:40 +00:00
|
|
|
// Servers returns a noop.ServersStore if the context has no organization specified
|
|
|
|
// and a organization.ServersStore otherwise.
|
2017-10-31 20:41:17 +00:00
|
|
|
func (s *Store) Servers(ctx context.Context) chronograf.ServersStore {
|
2017-11-08 21:56:34 +00:00
|
|
|
if isServer := hasServerContext(ctx); isServer {
|
|
|
|
return s.ServersStore
|
|
|
|
}
|
2017-10-31 21:40:58 +00:00
|
|
|
if org, ok := hasOrganizationContext(ctx); ok {
|
|
|
|
return organizations.NewServersStore(s.ServersStore, org)
|
|
|
|
}
|
|
|
|
|
2017-10-31 22:07:42 +00:00
|
|
|
return &noop.ServersStore{}
|
2017-10-31 20:41:17 +00:00
|
|
|
}
|
|
|
|
|
2017-12-05 22:07:09 +00:00
|
|
|
// Layouts returns all layouts in the underlying layouts store.
|
2017-10-31 20:41:17 +00:00
|
|
|
func (s *Store) Layouts(ctx context.Context) chronograf.LayoutsStore {
|
2017-12-05 22:07:09 +00:00
|
|
|
return s.LayoutsStore
|
2017-10-31 20:41:17 +00:00
|
|
|
}
|
|
|
|
|
2017-11-01 18:24:40 +00:00
|
|
|
// Users returns a chronograf.UsersStore.
|
2017-11-08 21:56:34 +00:00
|
|
|
// If the context is a server context, then the underlying chronograf.UsersStore
|
2017-11-01 18:24:40 +00:00
|
|
|
// is returned.
|
|
|
|
// If there is an organization specified on context, then an organizations.UsersStore
|
|
|
|
// is returned.
|
|
|
|
// If niether are specified, a noop.UsersStore is returned.
|
2017-10-31 20:41:17 +00:00
|
|
|
func (s *Store) Users(ctx context.Context) chronograf.UsersStore {
|
2017-11-08 21:56:34 +00:00
|
|
|
if isServer := hasServerContext(ctx); isServer {
|
2017-10-31 21:49:35 +00:00
|
|
|
return s.UsersStore
|
|
|
|
}
|
2017-10-31 21:40:58 +00:00
|
|
|
if org, ok := hasOrganizationContext(ctx); ok {
|
|
|
|
return organizations.NewUsersStore(s.UsersStore, org)
|
|
|
|
}
|
|
|
|
|
2017-10-31 22:07:42 +00:00
|
|
|
return &noop.UsersStore{}
|
2017-10-31 21:40:58 +00:00
|
|
|
}
|
|
|
|
|
2017-11-01 18:24:40 +00:00
|
|
|
// Dashboards returns a noop.DashboardsStore if the context has no organization specified
|
|
|
|
// and a organization.DashboardsStore otherwise.
|
2017-10-31 21:40:58 +00:00
|
|
|
func (s *Store) Dashboards(ctx context.Context) chronograf.DashboardsStore {
|
2017-11-08 21:56:34 +00:00
|
|
|
if isServer := hasServerContext(ctx); isServer {
|
|
|
|
return s.DashboardsStore
|
|
|
|
}
|
2017-10-31 21:40:58 +00:00
|
|
|
if org, ok := hasOrganizationContext(ctx); ok {
|
|
|
|
return organizations.NewDashboardsStore(s.DashboardsStore, org)
|
|
|
|
}
|
|
|
|
|
2017-10-31 22:07:42 +00:00
|
|
|
return &noop.DashboardsStore{}
|
2017-10-31 20:41:17 +00:00
|
|
|
}
|
|
|
|
|
2017-11-01 20:43:31 +00:00
|
|
|
// Organizations returns the underlying OrganizationsStore.
|
2017-10-31 20:41:17 +00:00
|
|
|
func (s *Store) Organizations(ctx context.Context) chronograf.OrganizationsStore {
|
2017-11-08 21:56:34 +00:00
|
|
|
if isServer := hasServerContext(ctx); isServer {
|
|
|
|
return s.OrganizationsStore
|
|
|
|
}
|
2017-11-10 20:33:28 +00:00
|
|
|
if isSuperAdmin := hasSuperAdminContext(ctx); isSuperAdmin {
|
|
|
|
return s.OrganizationsStore
|
|
|
|
}
|
2017-11-07 22:05:47 +00:00
|
|
|
if org, ok := hasOrganizationContext(ctx); ok {
|
|
|
|
return organizations.NewOrganizationsStore(s.OrganizationsStore, org)
|
|
|
|
}
|
|
|
|
return &noop.OrganizationsStore{}
|
2017-10-31 20:41:17 +00:00
|
|
|
}
|
2017-12-13 01:06:57 +00:00
|
|
|
|
|
|
|
// Config returns the underlying ConfigStore.
|
|
|
|
func (s *Store) Config(ctx context.Context) chronograf.ConfigStore {
|
2017-12-13 18:46:08 +00:00
|
|
|
if isServer := hasServerContext(ctx); isServer {
|
|
|
|
return s.ConfigStore
|
|
|
|
}
|
|
|
|
if isSuperAdmin := hasSuperAdminContext(ctx); isSuperAdmin {
|
|
|
|
return s.ConfigStore
|
|
|
|
}
|
|
|
|
return &noop.ConfigStore{}
|
2017-12-13 01:06:57 +00:00
|
|
|
}
|