2017-03-10 19:24:48 +00:00
|
|
|
package server
|
|
|
|
|
|
|
|
import (
|
2017-10-26 22:01:20 +00:00
|
|
|
"encoding/json"
|
2017-03-10 19:24:48 +00:00
|
|
|
"fmt"
|
|
|
|
"net/http"
|
2017-10-27 17:02:02 +00:00
|
|
|
"strconv"
|
2017-03-10 19:24:48 +00:00
|
|
|
|
|
|
|
"golang.org/x/net/context"
|
|
|
|
|
|
|
|
"github.com/influxdata/chronograf"
|
|
|
|
"github.com/influxdata/chronograf/oauth2"
|
2017-11-01 13:49:02 +00:00
|
|
|
"github.com/influxdata/chronograf/organizations"
|
2017-03-10 19:24:48 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
type meLinks struct {
|
|
|
|
Self string `json:"self"` // Self link mapping to this resource
|
|
|
|
}
|
|
|
|
|
|
|
|
type meResponse struct {
|
|
|
|
*chronograf.User
|
|
|
|
Links meLinks `json:"links"`
|
|
|
|
}
|
|
|
|
|
|
|
|
// If new user response is nil, return an empty meResponse because it
|
|
|
|
// indicates authentication is not needed
|
|
|
|
func newMeResponse(usr *chronograf.User) meResponse {
|
|
|
|
base := "/chronograf/v1/users"
|
|
|
|
name := "me"
|
|
|
|
if usr != nil {
|
2017-04-07 20:32:35 +00:00
|
|
|
name = PathEscape(usr.Name)
|
2017-03-10 19:24:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return meResponse{
|
|
|
|
User: usr,
|
|
|
|
Links: meLinks{
|
|
|
|
Self: fmt.Sprintf("%s/%s", base, name),
|
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-10-27 19:48:51 +00:00
|
|
|
// getUsername not currently used
|
2017-10-16 23:55:09 +00:00
|
|
|
func getUsername(ctx context.Context) (string, error) {
|
2017-03-10 19:24:48 +00:00
|
|
|
principal, err := getPrincipal(ctx)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
if principal.Subject == "" {
|
|
|
|
return "", fmt.Errorf("Token not found")
|
|
|
|
}
|
|
|
|
return principal.Subject, nil
|
|
|
|
}
|
|
|
|
|
2017-10-27 19:48:51 +00:00
|
|
|
// getProvider not currently used
|
2017-10-17 00:30:23 +00:00
|
|
|
func getProvider(ctx context.Context) (string, error) {
|
|
|
|
principal, err := getPrincipal(ctx)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
if principal.Issuer == "" {
|
|
|
|
return "", fmt.Errorf("Token not found")
|
|
|
|
}
|
|
|
|
return principal.Issuer, nil
|
|
|
|
}
|
|
|
|
|
2017-10-19 18:17:40 +00:00
|
|
|
// TODO: This Scheme value is hard-coded temporarily since we only currently
|
|
|
|
// support OAuth2. This hard-coding should be removed whenever we add
|
|
|
|
// support for other authentication schemes.
|
|
|
|
func getScheme(ctx context.Context) (string, error) {
|
2017-10-24 23:17:59 +00:00
|
|
|
return "oauth2", nil
|
2017-10-19 18:17:40 +00:00
|
|
|
}
|
|
|
|
|
2017-03-10 19:24:48 +00:00
|
|
|
func getPrincipal(ctx context.Context) (oauth2.Principal, error) {
|
|
|
|
principal, ok := ctx.Value(oauth2.PrincipalKey).(oauth2.Principal)
|
|
|
|
if !ok {
|
|
|
|
return oauth2.Principal{}, fmt.Errorf("Token not found")
|
|
|
|
}
|
|
|
|
|
|
|
|
return principal, nil
|
|
|
|
}
|
|
|
|
|
2017-10-27 19:48:51 +00:00
|
|
|
func getValidPrincipal(ctx context.Context) (oauth2.Principal, error) {
|
|
|
|
p, err := getPrincipal(ctx)
|
2017-10-26 22:01:20 +00:00
|
|
|
if err != nil {
|
2017-10-27 19:48:51 +00:00
|
|
|
return p, err
|
|
|
|
}
|
|
|
|
if p.Subject == "" {
|
|
|
|
return oauth2.Principal{}, fmt.Errorf("Token not found")
|
|
|
|
}
|
|
|
|
if p.Issuer == "" {
|
|
|
|
return oauth2.Principal{}, fmt.Errorf("Token not found")
|
2017-10-26 22:01:20 +00:00
|
|
|
}
|
2017-10-27 19:48:51 +00:00
|
|
|
return p, nil
|
2017-10-26 22:01:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
type meOrganizationRequest struct {
|
2017-11-01 14:37:32 +00:00
|
|
|
// Organization is the OrganizationID
|
|
|
|
Organization string `json:"organization"`
|
2017-10-26 22:01:20 +00:00
|
|
|
}
|
|
|
|
|
2017-10-27 17:02:02 +00:00
|
|
|
// MeOrganization changes the user's current organization on the JWT and responds
|
|
|
|
// with the same semantics as Me
|
2017-10-26 22:01:20 +00:00
|
|
|
func (s *Service) MeOrganization(auth oauth2.Authenticator) func(http.ResponseWriter, *http.Request) {
|
|
|
|
return func(w http.ResponseWriter, r *http.Request) {
|
|
|
|
ctx := r.Context()
|
|
|
|
principal, err := auth.Validate(ctx, r)
|
|
|
|
if err != nil {
|
|
|
|
s.Logger.Error("Invalid principal")
|
|
|
|
w.WriteHeader(http.StatusForbidden)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
var req meOrganizationRequest
|
|
|
|
if err := json.NewDecoder(r.Body).Decode(&req); err != nil {
|
|
|
|
invalidJSON(w, s.Logger)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2017-10-27 17:02:02 +00:00
|
|
|
// validate that the organization exists
|
2017-11-01 14:37:32 +00:00
|
|
|
orgID, err := strconv.ParseUint(req.Organization, 10, 64)
|
2017-10-27 17:02:02 +00:00
|
|
|
if err != nil {
|
|
|
|
Error(w, http.StatusInternalServerError, err.Error(), s.Logger)
|
|
|
|
return
|
|
|
|
}
|
2017-10-31 20:41:17 +00:00
|
|
|
_, err = s.Store.Organizations(ctx).Get(ctx, chronograf.OrganizationQuery{ID: &orgID})
|
2017-10-27 17:02:02 +00:00
|
|
|
if err != nil {
|
|
|
|
Error(w, http.StatusBadRequest, err.Error(), s.Logger)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2017-10-27 19:48:51 +00:00
|
|
|
p, err := getValidPrincipal(ctx)
|
2017-10-27 17:02:02 +00:00
|
|
|
if err != nil {
|
|
|
|
invalidData(w, err, s.Logger)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
scheme, err := getScheme(ctx)
|
|
|
|
if err != nil {
|
|
|
|
invalidData(w, err, s.Logger)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
// validate that user belongs to organization
|
2017-11-01 14:37:32 +00:00
|
|
|
ctx = context.WithValue(ctx, organizations.ContextKey, req.Organization)
|
2017-10-31 20:41:17 +00:00
|
|
|
_, err = s.Store.Users(ctx).Get(ctx, chronograf.UserQuery{
|
2017-10-27 19:48:51 +00:00
|
|
|
Name: &p.Subject,
|
|
|
|
Provider: &p.Issuer,
|
2017-10-27 17:02:02 +00:00
|
|
|
Scheme: &scheme,
|
|
|
|
})
|
|
|
|
if err == chronograf.ErrUserNotFound {
|
2017-10-27 17:14:14 +00:00
|
|
|
Error(w, http.StatusBadRequest, err.Error(), s.Logger)
|
2017-10-27 17:02:02 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
if err != nil {
|
|
|
|
Error(w, http.StatusBadRequest, err.Error(), s.Logger)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2017-10-26 23:02:29 +00:00
|
|
|
// TODO: change to principal.CurrentOrganization
|
2017-11-01 14:37:32 +00:00
|
|
|
principal.Organization = req.Organization
|
2017-10-26 22:01:20 +00:00
|
|
|
|
|
|
|
if err := auth.Authorize(ctx, w, principal); err != nil {
|
|
|
|
Error(w, http.StatusInternalServerError, err.Error(), s.Logger)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
ctx = context.WithValue(ctx, oauth2.PrincipalKey, principal)
|
|
|
|
|
|
|
|
s.Me(w, r.WithContext(ctx))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-10-16 23:55:09 +00:00
|
|
|
// Me does a findOrCreate based on the username in the context
|
2017-10-10 22:27:58 +00:00
|
|
|
func (s *Service) Me(w http.ResponseWriter, r *http.Request) {
|
2017-03-10 19:24:48 +00:00
|
|
|
ctx := r.Context()
|
2017-10-10 22:27:58 +00:00
|
|
|
if !s.UseAuth {
|
2017-03-10 19:24:48 +00:00
|
|
|
// If there's no authentication, return an empty user
|
|
|
|
res := newMeResponse(nil)
|
2017-10-10 22:27:58 +00:00
|
|
|
encodeJSON(w, http.StatusOK, res, s.Logger)
|
2017-03-10 19:24:48 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2017-10-27 19:48:51 +00:00
|
|
|
p, err := getValidPrincipal(ctx)
|
2017-10-17 00:30:23 +00:00
|
|
|
if err != nil {
|
|
|
|
invalidData(w, err, s.Logger)
|
|
|
|
return
|
|
|
|
}
|
2017-10-19 18:17:40 +00:00
|
|
|
scheme, err := getScheme(ctx)
|
|
|
|
if err != nil {
|
|
|
|
invalidData(w, err, s.Logger)
|
|
|
|
return
|
|
|
|
}
|
2017-11-01 13:49:02 +00:00
|
|
|
ctx = context.WithValue(ctx, organizations.ContextKey, p.Organization)
|
|
|
|
ctx = context.WithValue(ctx, SuperAdminKey, true)
|
2017-10-31 21:49:35 +00:00
|
|
|
|
|
|
|
usr, err := s.Store.Users(ctx).Get(ctx, chronograf.UserQuery{
|
2017-10-27 19:48:51 +00:00
|
|
|
Name: &p.Subject,
|
|
|
|
Provider: &p.Issuer,
|
2017-10-19 18:17:40 +00:00
|
|
|
Scheme: &scheme,
|
|
|
|
})
|
2017-10-18 18:45:33 +00:00
|
|
|
if err != nil && err != chronograf.ErrUserNotFound {
|
2017-10-18 16:34:23 +00:00
|
|
|
unknownErrorWithMessage(w, err, s.Logger)
|
2017-10-17 00:30:23 +00:00
|
|
|
return
|
|
|
|
}
|
2017-03-10 19:24:48 +00:00
|
|
|
|
2017-10-17 00:30:23 +00:00
|
|
|
if usr != nil {
|
2017-10-27 19:48:51 +00:00
|
|
|
usr.CurrentOrganization = p.Organization
|
2017-03-10 19:24:48 +00:00
|
|
|
res := newMeResponse(usr)
|
2017-10-10 22:27:58 +00:00
|
|
|
encodeJSON(w, http.StatusOK, res, s.Logger)
|
2017-03-10 19:24:48 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// Because we didnt find a user, making a new one
|
|
|
|
user := &chronograf.User{
|
2017-10-27 19:48:51 +00:00
|
|
|
Name: p.Subject,
|
|
|
|
Provider: p.Issuer,
|
2017-10-19 18:17:40 +00:00
|
|
|
// TODO: This Scheme value is hard-coded temporarily since we only currently
|
|
|
|
// support OAuth2. This hard-coding should be removed whenever we add
|
|
|
|
// support for other authentication schemes.
|
2017-10-27 19:48:51 +00:00
|
|
|
Scheme: scheme,
|
2017-10-31 23:50:03 +00:00
|
|
|
Roles: []chronograf.Role{
|
|
|
|
{
|
2017-11-01 13:12:19 +00:00
|
|
|
Name: MemberRoleName,
|
2017-11-01 00:58:40 +00:00
|
|
|
// This is the ID of the default organization
|
|
|
|
Organization: "0",
|
2017-10-31 23:50:03 +00:00
|
|
|
},
|
|
|
|
},
|
2017-11-01 13:12:19 +00:00
|
|
|
SuperAdmin: s.firstUser(),
|
2017-03-10 19:24:48 +00:00
|
|
|
}
|
|
|
|
|
2017-10-31 21:49:35 +00:00
|
|
|
newUser, err := s.Store.Users(ctx).Add(ctx, user)
|
2017-03-10 19:24:48 +00:00
|
|
|
if err != nil {
|
|
|
|
msg := fmt.Errorf("error storing user %s: %v", user.Name, err)
|
2017-10-10 22:27:58 +00:00
|
|
|
unknownErrorWithMessage(w, msg, s.Logger)
|
2017-03-10 19:24:48 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2017-10-27 19:48:51 +00:00
|
|
|
newUser.CurrentOrganization = p.Organization
|
2017-03-10 19:24:48 +00:00
|
|
|
res := newMeResponse(newUser)
|
2017-10-10 22:27:58 +00:00
|
|
|
encodeJSON(w, http.StatusOK, res, s.Logger)
|
2017-03-10 19:24:48 +00:00
|
|
|
}
|
2017-11-01 13:12:19 +00:00
|
|
|
|
|
|
|
// TODO(desa): very slow
|
|
|
|
func (s *Service) firstUser() bool {
|
2017-11-01 13:49:02 +00:00
|
|
|
ctx := context.WithValue(context.Background(), SuperAdminKey, true)
|
2017-11-01 13:12:19 +00:00
|
|
|
users, err := s.Store.Users(ctx).All(ctx)
|
|
|
|
if err != nil {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
return len(users) == 0
|
|
|
|
}
|