2018-08-07 20:10:05 +00:00
|
|
|
package http
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"encoding/json"
|
|
|
|
"fmt"
|
|
|
|
"net/http"
|
|
|
|
|
|
|
|
"github.com/influxdata/platform"
|
|
|
|
"github.com/influxdata/platform/kit/errors"
|
|
|
|
"github.com/julienschmidt/httprouter"
|
|
|
|
)
|
|
|
|
|
|
|
|
// ViewHandler is the handler for the view service
|
|
|
|
type ViewHandler struct {
|
|
|
|
*httprouter.Router
|
|
|
|
|
2018-10-09 20:54:26 +00:00
|
|
|
ViewService platform.ViewService
|
|
|
|
UserResourceMappingService platform.UserResourceMappingService
|
2018-08-07 20:10:05 +00:00
|
|
|
}
|
|
|
|
|
2018-10-09 20:46:17 +00:00
|
|
|
const (
|
2018-10-09 20:54:26 +00:00
|
|
|
viewsPath = "/api/v2/views"
|
|
|
|
viewsIDPath = "/api/v2/views/:id"
|
|
|
|
viewsIDMembersPath = "/api/v2/views/:id/members"
|
|
|
|
viewsIDMembersIDPath = "/api/v2/views/:id/members/:userID"
|
|
|
|
viewsIDOwnersPath = "/api/v2/views/:id/owners"
|
|
|
|
viewsIDOwnersIDPath = "/api/v2/views/:id/owners:userID"
|
2018-10-09 20:46:17 +00:00
|
|
|
)
|
|
|
|
|
2018-08-07 20:10:05 +00:00
|
|
|
// NewViewHandler returns a new instance of ViewHandler.
|
2018-10-16 21:49:35 +00:00
|
|
|
func NewViewHandler(mappingService platform.UserResourceMappingService) *ViewHandler {
|
2018-08-07 20:10:05 +00:00
|
|
|
h := &ViewHandler{
|
2018-10-16 22:18:22 +00:00
|
|
|
Router: httprouter.New(),
|
2018-10-16 21:49:35 +00:00
|
|
|
UserResourceMappingService: mappingService,
|
2018-08-07 20:10:05 +00:00
|
|
|
}
|
|
|
|
|
2018-10-09 20:46:17 +00:00
|
|
|
h.HandlerFunc("POST", viewsPath, h.handlePostViews)
|
|
|
|
h.HandlerFunc("GET", viewsPath, h.handleGetViews)
|
2018-10-09 20:54:26 +00:00
|
|
|
|
2018-10-09 20:46:17 +00:00
|
|
|
h.HandlerFunc("GET", viewsIDPath, h.handleGetView)
|
|
|
|
h.HandlerFunc("DELETE", viewsIDPath, h.handleDeleteView)
|
|
|
|
h.HandlerFunc("PATCH", viewsIDPath, h.handlePatchView)
|
2018-10-09 20:54:26 +00:00
|
|
|
|
|
|
|
h.HandlerFunc("POST", viewsIDMembersPath, newPostMemberHandler(h.UserResourceMappingService, platform.ViewResourceType, platform.Member))
|
|
|
|
h.HandlerFunc("GET", viewsIDMembersPath, newGetMembersHandler(h.UserResourceMappingService, platform.Member))
|
|
|
|
h.HandlerFunc("DELETE", viewsIDMembersIDPath, newDeleteMemberHandler(h.UserResourceMappingService, platform.Member))
|
|
|
|
|
|
|
|
h.HandlerFunc("POST", viewsIDOwnersPath, newPostMemberHandler(h.UserResourceMappingService, platform.ViewResourceType, platform.Owner))
|
|
|
|
h.HandlerFunc("GET", viewsIDOwnersPath, newGetMembersHandler(h.UserResourceMappingService, platform.Owner))
|
|
|
|
h.HandlerFunc("DELETE", viewsIDOwnersIDPath, newDeleteMemberHandler(h.UserResourceMappingService, platform.Owner))
|
|
|
|
|
2018-08-07 20:10:05 +00:00
|
|
|
return h
|
|
|
|
}
|
|
|
|
|
|
|
|
type viewLinks struct {
|
|
|
|
Self string `json:"self"`
|
|
|
|
}
|
|
|
|
|
|
|
|
type viewResponse struct {
|
|
|
|
platform.View
|
|
|
|
Links viewLinks `json:"links"`
|
|
|
|
}
|
|
|
|
|
|
|
|
func (r viewResponse) MarshalJSON() ([]byte, error) {
|
|
|
|
props, err := platform.MarshalViewPropertiesJSON(r.Properties)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return json.Marshal(struct {
|
|
|
|
platform.ViewContents
|
|
|
|
Links viewLinks `json:"links"`
|
|
|
|
Properties json.RawMessage `json:"properties"`
|
|
|
|
}{
|
|
|
|
ViewContents: r.ViewContents,
|
|
|
|
Links: r.Links,
|
|
|
|
Properties: props,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func newViewResponse(c *platform.View) viewResponse {
|
|
|
|
return viewResponse{
|
|
|
|
Links: viewLinks{
|
2018-09-26 08:49:19 +00:00
|
|
|
Self: fmt.Sprintf("/api/v2/views/%s", c.ID),
|
2018-08-07 20:10:05 +00:00
|
|
|
},
|
|
|
|
View: *c,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// handleGetViews returns all views within the store.
|
|
|
|
func (h *ViewHandler) handleGetViews(w http.ResponseWriter, r *http.Request) {
|
|
|
|
ctx := r.Context()
|
2018-12-08 01:15:24 +00:00
|
|
|
|
|
|
|
req := decodeGetViewsRequest(ctx, r)
|
|
|
|
|
|
|
|
views, _, err := h.ViewService.FindViews(ctx, req.filter)
|
2018-08-07 20:10:05 +00:00
|
|
|
if err != nil {
|
|
|
|
EncodeError(ctx, errors.InternalErrorf("Error loading views: %v", err), w)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := encodeResponse(ctx, w, http.StatusOK, newGetViewsResponse(views)); err != nil {
|
|
|
|
EncodeError(ctx, err, w)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-12-08 01:15:24 +00:00
|
|
|
type getViewsRequest struct {
|
|
|
|
filter platform.ViewFilter
|
|
|
|
}
|
|
|
|
|
|
|
|
func decodeGetViewsRequest(ctx context.Context, r *http.Request) *getViewsRequest {
|
|
|
|
qp := r.URL.Query()
|
|
|
|
|
|
|
|
return &getViewsRequest{
|
|
|
|
filter: platform.ViewFilter{
|
|
|
|
Types: qp["type"],
|
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-08-07 20:10:05 +00:00
|
|
|
type getViewsLinks struct {
|
|
|
|
Self string `json:"self"`
|
|
|
|
}
|
|
|
|
|
|
|
|
type getViewsResponse struct {
|
|
|
|
Links getViewsLinks `json:"links"`
|
|
|
|
Views []viewResponse `json:"views"`
|
|
|
|
}
|
|
|
|
|
|
|
|
func newGetViewsResponse(views []*platform.View) getViewsResponse {
|
|
|
|
res := getViewsResponse{
|
|
|
|
Links: getViewsLinks{
|
2018-09-26 08:49:19 +00:00
|
|
|
Self: "/api/v2/views",
|
2018-08-07 20:10:05 +00:00
|
|
|
},
|
|
|
|
Views: make([]viewResponse, 0, len(views)),
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, view := range views {
|
|
|
|
res.Views = append(res.Views, newViewResponse(view))
|
|
|
|
}
|
|
|
|
|
|
|
|
return res
|
|
|
|
}
|
|
|
|
|
|
|
|
// handlePostViews creates a new view.
|
|
|
|
func (h *ViewHandler) handlePostViews(w http.ResponseWriter, r *http.Request) {
|
|
|
|
ctx := r.Context()
|
|
|
|
|
|
|
|
req, err := decodePostViewRequest(ctx, r)
|
|
|
|
if err != nil {
|
|
|
|
EncodeError(ctx, err, w)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if err := h.ViewService.CreateView(ctx, req.View); err != nil {
|
|
|
|
EncodeError(ctx, errors.InternalErrorf("Error loading views: %v", err), w)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := encodeResponse(ctx, w, http.StatusCreated, newViewResponse(req.View)); err != nil {
|
|
|
|
EncodeError(ctx, err, w)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
type postViewRequest struct {
|
|
|
|
View *platform.View
|
|
|
|
}
|
|
|
|
|
|
|
|
func decodePostViewRequest(ctx context.Context, r *http.Request) (*postViewRequest, error) {
|
|
|
|
c := &platform.View{}
|
|
|
|
if err := json.NewDecoder(r.Body).Decode(c); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return &postViewRequest{
|
|
|
|
View: c,
|
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// hanldeGetView retrieves a view by ID.
|
|
|
|
func (h *ViewHandler) handleGetView(w http.ResponseWriter, r *http.Request) {
|
|
|
|
ctx := r.Context()
|
|
|
|
|
|
|
|
req, err := decodeGetViewRequest(ctx, r)
|
|
|
|
if err != nil {
|
|
|
|
EncodeError(ctx, err, w)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
view, err := h.ViewService.FindViewByID(ctx, req.ViewID)
|
|
|
|
if err != nil {
|
|
|
|
if err == platform.ErrViewNotFound {
|
|
|
|
err = errors.New(err.Error(), errors.NotFound)
|
|
|
|
}
|
|
|
|
EncodeError(ctx, err, w)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := encodeResponse(ctx, w, http.StatusOK, newViewResponse(view)); err != nil {
|
|
|
|
EncodeError(ctx, err, w)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
type getViewRequest struct {
|
|
|
|
ViewID platform.ID
|
|
|
|
}
|
|
|
|
|
|
|
|
func decodeGetViewRequest(ctx context.Context, r *http.Request) (*getViewRequest, error) {
|
|
|
|
params := httprouter.ParamsFromContext(ctx)
|
|
|
|
id := params.ByName("id")
|
|
|
|
if id == "" {
|
|
|
|
return nil, errors.InvalidDataf("url missing id")
|
|
|
|
}
|
|
|
|
|
|
|
|
var i platform.ID
|
|
|
|
if err := i.DecodeFromString(id); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return &getViewRequest{
|
|
|
|
ViewID: i,
|
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// handleDeleteView removes a view by ID.
|
|
|
|
func (h *ViewHandler) handleDeleteView(w http.ResponseWriter, r *http.Request) {
|
|
|
|
ctx := r.Context()
|
|
|
|
|
|
|
|
req, err := decodeDeleteViewRequest(ctx, r)
|
|
|
|
if err != nil {
|
|
|
|
EncodeError(ctx, err, w)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := h.ViewService.DeleteView(ctx, req.ViewID); err != nil {
|
|
|
|
if err == platform.ErrViewNotFound {
|
|
|
|
err = errors.New(err.Error(), errors.NotFound)
|
|
|
|
}
|
|
|
|
EncodeError(ctx, err, w)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
w.WriteHeader(http.StatusNoContent)
|
|
|
|
}
|
|
|
|
|
|
|
|
type deleteViewRequest struct {
|
|
|
|
ViewID platform.ID
|
|
|
|
}
|
|
|
|
|
|
|
|
func decodeDeleteViewRequest(ctx context.Context, r *http.Request) (*deleteViewRequest, error) {
|
|
|
|
params := httprouter.ParamsFromContext(ctx)
|
|
|
|
id := params.ByName("id")
|
|
|
|
if id == "" {
|
|
|
|
return nil, errors.InvalidDataf("url missing id")
|
|
|
|
}
|
|
|
|
|
|
|
|
var i platform.ID
|
|
|
|
if err := i.DecodeFromString(id); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return &deleteViewRequest{
|
|
|
|
ViewID: i,
|
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// handlePatchView updates a view.
|
|
|
|
func (h *ViewHandler) handlePatchView(w http.ResponseWriter, r *http.Request) {
|
|
|
|
ctx := r.Context()
|
|
|
|
|
|
|
|
req, err := decodePatchViewRequest(ctx, r)
|
|
|
|
if err != nil {
|
|
|
|
EncodeError(ctx, err, w)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
view, err := h.ViewService.UpdateView(ctx, req.ViewID, req.Upd)
|
|
|
|
if err != nil {
|
|
|
|
if err == platform.ErrViewNotFound {
|
|
|
|
err = errors.New(err.Error(), errors.NotFound)
|
|
|
|
}
|
|
|
|
EncodeError(ctx, err, w)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := encodeResponse(ctx, w, http.StatusOK, newViewResponse(view)); err != nil {
|
|
|
|
EncodeError(ctx, err, w)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
type patchViewRequest struct {
|
|
|
|
ViewID platform.ID
|
|
|
|
Upd platform.ViewUpdate
|
|
|
|
}
|
|
|
|
|
|
|
|
func decodePatchViewRequest(ctx context.Context, r *http.Request) (*patchViewRequest, error) {
|
|
|
|
req := &patchViewRequest{}
|
|
|
|
upd := platform.ViewUpdate{}
|
|
|
|
if err := json.NewDecoder(r.Body).Decode(&upd); err != nil {
|
|
|
|
return nil, errors.MalformedDataf(err.Error())
|
|
|
|
}
|
|
|
|
|
|
|
|
req.Upd = upd
|
|
|
|
|
|
|
|
params := httprouter.ParamsFromContext(ctx)
|
|
|
|
id := params.ByName("id")
|
|
|
|
if id == "" {
|
|
|
|
return nil, errors.InvalidDataf("url missing id")
|
|
|
|
}
|
|
|
|
var i platform.ID
|
|
|
|
if err := i.DecodeFromString(id); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
req.ViewID = i
|
|
|
|
|
|
|
|
if err := req.Valid(); err != nil {
|
|
|
|
return nil, errors.MalformedDataf(err.Error())
|
|
|
|
}
|
|
|
|
|
|
|
|
return req, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Valid validates that the view ID is non zero valued and update has expected values set.
|
|
|
|
func (r *patchViewRequest) Valid() error {
|
2018-09-12 16:07:18 +00:00
|
|
|
if !r.ViewID.Valid() {
|
2018-08-07 20:10:05 +00:00
|
|
|
return fmt.Errorf("missing view ID")
|
|
|
|
}
|
|
|
|
|
|
|
|
return r.Upd.Valid()
|
|
|
|
}
|