2018-01-11 17:36:13 +00:00
|
|
|
package server
|
|
|
|
|
|
|
|
import (
|
2018-02-05 19:54:39 +00:00
|
|
|
"context"
|
2018-02-07 02:33:27 +00:00
|
|
|
|
2018-02-05 21:47:44 +00:00
|
|
|
"encoding/json"
|
2018-02-05 19:54:39 +00:00
|
|
|
"fmt"
|
2018-02-05 21:47:44 +00:00
|
|
|
"net/http"
|
2018-01-11 17:36:13 +00:00
|
|
|
"strings"
|
|
|
|
|
2018-02-05 21:47:44 +00:00
|
|
|
"github.com/bouk/httprouter"
|
2018-07-19 20:52:14 +00:00
|
|
|
"github.com/influxdata/platform/chronograf"
|
|
|
|
"github.com/influxdata/platform/chronograf/oauth2"
|
2018-01-11 17:36:13 +00:00
|
|
|
)
|
|
|
|
|
2018-03-23 19:48:34 +00:00
|
|
|
func (s *Service) mapPrincipalToSuperAdmin(p oauth2.Principal) bool {
|
|
|
|
if p.Issuer != "auth0" {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
groups := strings.Split(p.Group, ",")
|
|
|
|
superAdmin := false
|
|
|
|
for _, group := range groups {
|
|
|
|
if group != "" && group == s.SuperAdminProviderGroups.auth0 {
|
|
|
|
superAdmin = true
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return superAdmin
|
|
|
|
}
|
|
|
|
|
2018-02-05 19:54:39 +00:00
|
|
|
func (s *Service) mapPrincipalToRoles(ctx context.Context, p oauth2.Principal) ([]chronograf.Role, error) {
|
|
|
|
mappings, err := s.Store.Mappings(ctx).All(ctx)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
roles := []chronograf.Role{}
|
|
|
|
MappingsLoop:
|
|
|
|
for _, mapping := range mappings {
|
|
|
|
if applyMapping(mapping, p) {
|
|
|
|
org, err := s.Store.Organizations(ctx).Get(ctx, chronograf.OrganizationQuery{ID: &mapping.Organization})
|
|
|
|
if err != nil {
|
2018-02-05 23:20:03 +00:00
|
|
|
continue MappingsLoop
|
2018-02-05 19:54:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
for _, role := range roles {
|
|
|
|
if role.Organization == org.ID {
|
|
|
|
continue MappingsLoop
|
|
|
|
}
|
|
|
|
}
|
|
|
|
roles = append(roles, chronograf.Role{Organization: org.ID, Name: org.DefaultRole})
|
2018-01-11 17:36:13 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-02-05 19:54:39 +00:00
|
|
|
return roles, nil
|
2018-01-11 17:36:13 +00:00
|
|
|
}
|
|
|
|
|
2018-02-05 19:54:39 +00:00
|
|
|
func applyMapping(m chronograf.Mapping, p oauth2.Principal) bool {
|
2018-01-11 17:36:13 +00:00
|
|
|
switch m.Provider {
|
|
|
|
case chronograf.MappingWildcard, p.Issuer:
|
|
|
|
default:
|
2018-02-05 19:54:39 +00:00
|
|
|
return false
|
2018-01-11 17:36:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
switch m.Scheme {
|
|
|
|
case chronograf.MappingWildcard, "oauth2":
|
|
|
|
default:
|
2018-02-05 19:54:39 +00:00
|
|
|
return false
|
2018-01-11 17:36:13 +00:00
|
|
|
}
|
|
|
|
|
2018-02-07 02:33:27 +00:00
|
|
|
if m.ProviderOrganization == chronograf.MappingWildcard {
|
2018-02-05 19:54:39 +00:00
|
|
|
return true
|
2018-01-11 17:36:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
groups := strings.Split(p.Group, ",")
|
|
|
|
|
2018-02-07 02:33:27 +00:00
|
|
|
return matchGroup(m.ProviderOrganization, groups)
|
2018-01-11 17:36:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func matchGroup(match string, groups []string) bool {
|
|
|
|
for _, group := range groups {
|
|
|
|
if match == group {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
2018-02-05 21:47:44 +00:00
|
|
|
type mappingsRequest chronograf.Mapping
|
|
|
|
|
|
|
|
// Valid determines if a mapping request is valid
|
|
|
|
func (m *mappingsRequest) Valid() error {
|
|
|
|
if m.Provider == "" {
|
|
|
|
return fmt.Errorf("mapping must specify provider")
|
|
|
|
}
|
|
|
|
if m.Scheme == "" {
|
|
|
|
return fmt.Errorf("mapping must specify scheme")
|
|
|
|
}
|
2018-02-07 02:33:27 +00:00
|
|
|
if m.ProviderOrganization == "" {
|
2018-02-05 21:47:44 +00:00
|
|
|
return fmt.Errorf("mapping must specify group")
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
type mappingResponse struct {
|
|
|
|
Links selfLinks `json:"links"`
|
2018-02-07 02:33:27 +00:00
|
|
|
chronograf.Mapping
|
2018-02-05 21:47:44 +00:00
|
|
|
}
|
|
|
|
|
2018-02-09 19:42:07 +00:00
|
|
|
func newMappingResponse(m chronograf.Mapping) *mappingResponse {
|
2018-02-07 02:33:27 +00:00
|
|
|
|
2018-02-05 21:47:44 +00:00
|
|
|
return &mappingResponse{
|
|
|
|
Links: selfLinks{
|
2018-02-09 19:42:07 +00:00
|
|
|
Self: fmt.Sprintf("/chronograf/v1/mappings/%s", m.ID),
|
2018-02-05 21:47:44 +00:00
|
|
|
},
|
2018-02-09 19:42:07 +00:00
|
|
|
Mapping: m,
|
2018-02-05 21:47:44 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
type mappingsResponse struct {
|
|
|
|
Links selfLinks `json:"links"`
|
|
|
|
Mappings []*mappingResponse `json:"mappings"`
|
|
|
|
}
|
|
|
|
|
|
|
|
func newMappingsResponse(ms []chronograf.Mapping) *mappingsResponse {
|
|
|
|
mappings := []*mappingResponse{}
|
|
|
|
for _, m := range ms {
|
2018-02-09 19:42:07 +00:00
|
|
|
mappings = append(mappings, newMappingResponse(m))
|
2018-02-05 21:47:44 +00:00
|
|
|
}
|
|
|
|
return &mappingsResponse{
|
|
|
|
Links: selfLinks{
|
|
|
|
Self: "/chronograf/v1/mappings",
|
|
|
|
},
|
|
|
|
Mappings: mappings,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Mappings retrives all mappings
|
|
|
|
func (s *Service) Mappings(w http.ResponseWriter, r *http.Request) {
|
|
|
|
ctx := r.Context()
|
|
|
|
|
|
|
|
mappings, err := s.Store.Mappings(ctx).All(ctx)
|
|
|
|
if err != nil {
|
|
|
|
Error(w, http.StatusInternalServerError, "failed to retrieve mappings from database", s.Logger)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
res := newMappingsResponse(mappings)
|
2018-02-07 02:33:27 +00:00
|
|
|
|
2018-02-05 21:47:44 +00:00
|
|
|
encodeJSON(w, http.StatusOK, res, s.Logger)
|
|
|
|
}
|
|
|
|
|
|
|
|
// NewMapping adds a new mapping
|
|
|
|
func (s *Service) NewMapping(w http.ResponseWriter, r *http.Request) {
|
|
|
|
var req mappingsRequest
|
|
|
|
if err := json.NewDecoder(r.Body).Decode(&req); err != nil {
|
|
|
|
invalidJSON(w, s.Logger)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := req.Valid(); err != nil {
|
|
|
|
invalidData(w, err, s.Logger)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
ctx := r.Context()
|
|
|
|
|
|
|
|
// validate that the organization exists
|
|
|
|
if !s.organizationExists(ctx, req.Organization) {
|
|
|
|
invalidData(w, fmt.Errorf("organization does not exist"), s.Logger)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
mapping := &chronograf.Mapping{
|
2018-02-07 02:33:27 +00:00
|
|
|
Organization: req.Organization,
|
|
|
|
Scheme: req.Scheme,
|
|
|
|
Provider: req.Provider,
|
|
|
|
ProviderOrganization: req.ProviderOrganization,
|
2018-02-05 21:47:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
m, err := s.Store.Mappings(ctx).Add(ctx, mapping)
|
|
|
|
if err != nil {
|
|
|
|
Error(w, http.StatusInternalServerError, "failed to add mapping to database", s.Logger)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2018-02-09 19:42:07 +00:00
|
|
|
cu := newMappingResponse(*m)
|
2018-02-05 21:47:44 +00:00
|
|
|
location(w, cu.Links.Self)
|
|
|
|
encodeJSON(w, http.StatusCreated, cu, s.Logger)
|
|
|
|
}
|
|
|
|
|
|
|
|
// UpdateMapping updates a mapping
|
|
|
|
func (s *Service) UpdateMapping(w http.ResponseWriter, r *http.Request) {
|
|
|
|
var req mappingsRequest
|
|
|
|
if err := json.NewDecoder(r.Body).Decode(&req); err != nil {
|
|
|
|
invalidJSON(w, s.Logger)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := req.Valid(); err != nil {
|
|
|
|
invalidData(w, err, s.Logger)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
ctx := r.Context()
|
|
|
|
|
|
|
|
// validate that the organization exists
|
|
|
|
if !s.organizationExists(ctx, req.Organization) {
|
|
|
|
invalidData(w, fmt.Errorf("organization does not exist"), s.Logger)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
mapping := &chronograf.Mapping{
|
2018-02-07 02:33:27 +00:00
|
|
|
ID: req.ID,
|
|
|
|
Organization: req.Organization,
|
|
|
|
Scheme: req.Scheme,
|
|
|
|
Provider: req.Provider,
|
|
|
|
ProviderOrganization: req.ProviderOrganization,
|
2018-02-05 21:47:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
err := s.Store.Mappings(ctx).Update(ctx, mapping)
|
|
|
|
if err != nil {
|
|
|
|
Error(w, http.StatusInternalServerError, "failed to update mapping in database", s.Logger)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2018-02-09 19:42:07 +00:00
|
|
|
cu := newMappingResponse(*mapping)
|
2018-02-05 21:47:44 +00:00
|
|
|
location(w, cu.Links.Self)
|
|
|
|
encodeJSON(w, http.StatusOK, cu, s.Logger)
|
|
|
|
}
|
|
|
|
|
|
|
|
// RemoveMapping removes a mapping
|
|
|
|
func (s *Service) RemoveMapping(w http.ResponseWriter, r *http.Request) {
|
|
|
|
ctx := r.Context()
|
|
|
|
id := httprouter.GetParamFromContext(ctx, "id")
|
|
|
|
|
|
|
|
m, err := s.Store.Mappings(ctx).Get(ctx, id)
|
|
|
|
if err == chronograf.ErrMappingNotFound {
|
|
|
|
Error(w, http.StatusNotFound, err.Error(), s.Logger)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
Error(w, http.StatusInternalServerError, "failed to retrieve mapping from database", s.Logger)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := s.Store.Mappings(ctx).Delete(ctx, m); err != nil {
|
|
|
|
Error(w, http.StatusInternalServerError, "failed to remove mapping from database", s.Logger)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
w.WriteHeader(http.StatusNoContent)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *Service) organizationExists(ctx context.Context, orgID string) bool {
|
|
|
|
if _, err := s.Store.Organizations(ctx).Get(ctx, chronograf.OrganizationQuery{ID: &orgID}); err != nil {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
return true
|
|
|
|
}
|