2019-08-08 21:59:03 +00:00
|
|
|
package http
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"context"
|
|
|
|
"encoding/json"
|
|
|
|
"fmt"
|
2019-12-13 23:32:21 +00:00
|
|
|
"io/ioutil"
|
2019-08-08 21:59:03 +00:00
|
|
|
"net/http"
|
|
|
|
|
2019-11-25 14:22:19 +00:00
|
|
|
"github.com/influxdata/httprouter"
|
2020-04-03 17:39:20 +00:00
|
|
|
"github.com/influxdata/influxdb/v2"
|
|
|
|
pctx "github.com/influxdata/influxdb/v2/context"
|
|
|
|
"github.com/influxdata/influxdb/v2/notification/endpoint"
|
|
|
|
"github.com/influxdata/influxdb/v2/pkg/httpc"
|
2019-08-08 21:59:03 +00:00
|
|
|
"go.uber.org/zap"
|
|
|
|
)
|
|
|
|
|
|
|
|
// NotificationEndpointBackend is all services and associated parameters required to construct
|
|
|
|
// the NotificationEndpointBackendHandler.
|
|
|
|
type NotificationEndpointBackend struct {
|
|
|
|
influxdb.HTTPErrorHandler
|
2019-12-04 23:10:23 +00:00
|
|
|
log *zap.Logger
|
2019-08-08 21:59:03 +00:00
|
|
|
|
|
|
|
NotificationEndpointService influxdb.NotificationEndpointService
|
|
|
|
UserResourceMappingService influxdb.UserResourceMappingService
|
|
|
|
LabelService influxdb.LabelService
|
|
|
|
UserService influxdb.UserService
|
|
|
|
OrganizationService influxdb.OrganizationService
|
|
|
|
}
|
|
|
|
|
|
|
|
// NewNotificationEndpointBackend returns a new instance of NotificationEndpointBackend.
|
2019-12-04 23:10:23 +00:00
|
|
|
func NewNotificationEndpointBackend(log *zap.Logger, b *APIBackend) *NotificationEndpointBackend {
|
2019-08-08 21:59:03 +00:00
|
|
|
return &NotificationEndpointBackend{
|
|
|
|
HTTPErrorHandler: b.HTTPErrorHandler,
|
2019-12-04 23:10:23 +00:00
|
|
|
log: log,
|
2019-08-08 21:59:03 +00:00
|
|
|
|
|
|
|
NotificationEndpointService: b.NotificationEndpointService,
|
|
|
|
UserResourceMappingService: b.UserResourceMappingService,
|
|
|
|
LabelService: b.LabelService,
|
|
|
|
UserService: b.UserService,
|
|
|
|
OrganizationService: b.OrganizationService,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-12-04 23:10:23 +00:00
|
|
|
func (b *NotificationEndpointBackend) Logger() *zap.Logger {
|
|
|
|
return b.log
|
|
|
|
}
|
|
|
|
|
2019-08-08 21:59:03 +00:00
|
|
|
// NotificationEndpointHandler is the handler for the notificationEndpoint service
|
|
|
|
type NotificationEndpointHandler struct {
|
|
|
|
*httprouter.Router
|
|
|
|
influxdb.HTTPErrorHandler
|
2019-12-04 23:10:23 +00:00
|
|
|
log *zap.Logger
|
2019-08-08 21:59:03 +00:00
|
|
|
|
|
|
|
NotificationEndpointService influxdb.NotificationEndpointService
|
|
|
|
UserResourceMappingService influxdb.UserResourceMappingService
|
|
|
|
LabelService influxdb.LabelService
|
|
|
|
UserService influxdb.UserService
|
|
|
|
OrganizationService influxdb.OrganizationService
|
|
|
|
}
|
|
|
|
|
|
|
|
const (
|
2019-12-09 23:54:16 +00:00
|
|
|
prefixNotificationEndpoints = "/api/v2/notificationEndpoints"
|
2019-08-08 21:59:03 +00:00
|
|
|
notificationEndpointsIDPath = "/api/v2/notificationEndpoints/:id"
|
|
|
|
notificationEndpointsIDMembersPath = "/api/v2/notificationEndpoints/:id/members"
|
|
|
|
notificationEndpointsIDMembersIDPath = "/api/v2/notificationEndpoints/:id/members/:userID"
|
|
|
|
notificationEndpointsIDOwnersPath = "/api/v2/notificationEndpoints/:id/owners"
|
|
|
|
notificationEndpointsIDOwnersIDPath = "/api/v2/notificationEndpoints/:id/owners/:userID"
|
|
|
|
notificationEndpointsIDLabelsPath = "/api/v2/notificationEndpoints/:id/labels"
|
|
|
|
notificationEndpointsIDLabelsIDPath = "/api/v2/notificationEndpoints/:id/labels/:lid"
|
|
|
|
)
|
|
|
|
|
|
|
|
// NewNotificationEndpointHandler returns a new instance of NotificationEndpointHandler.
|
2019-12-04 23:10:23 +00:00
|
|
|
func NewNotificationEndpointHandler(log *zap.Logger, b *NotificationEndpointBackend) *NotificationEndpointHandler {
|
2019-08-08 21:59:03 +00:00
|
|
|
h := &NotificationEndpointHandler{
|
|
|
|
Router: NewRouter(b.HTTPErrorHandler),
|
|
|
|
HTTPErrorHandler: b.HTTPErrorHandler,
|
2019-12-04 23:10:23 +00:00
|
|
|
log: log,
|
2019-08-08 21:59:03 +00:00
|
|
|
|
|
|
|
NotificationEndpointService: b.NotificationEndpointService,
|
|
|
|
UserResourceMappingService: b.UserResourceMappingService,
|
|
|
|
LabelService: b.LabelService,
|
|
|
|
UserService: b.UserService,
|
|
|
|
OrganizationService: b.OrganizationService,
|
|
|
|
}
|
2019-12-09 23:54:16 +00:00
|
|
|
h.HandlerFunc("POST", prefixNotificationEndpoints, h.handlePostNotificationEndpoint)
|
|
|
|
h.HandlerFunc("GET", prefixNotificationEndpoints, h.handleGetNotificationEndpoints)
|
2019-08-08 21:59:03 +00:00
|
|
|
h.HandlerFunc("GET", notificationEndpointsIDPath, h.handleGetNotificationEndpoint)
|
|
|
|
h.HandlerFunc("DELETE", notificationEndpointsIDPath, h.handleDeleteNotificationEndpoint)
|
|
|
|
h.HandlerFunc("PUT", notificationEndpointsIDPath, h.handlePutNotificationEndpoint)
|
|
|
|
h.HandlerFunc("PATCH", notificationEndpointsIDPath, h.handlePatchNotificationEndpoint)
|
|
|
|
|
|
|
|
memberBackend := MemberBackend{
|
|
|
|
HTTPErrorHandler: b.HTTPErrorHandler,
|
2019-12-04 23:10:23 +00:00
|
|
|
log: b.log.With(zap.String("handler", "member")),
|
2019-08-08 21:59:03 +00:00
|
|
|
ResourceType: influxdb.NotificationEndpointResourceType,
|
|
|
|
UserType: influxdb.Member,
|
|
|
|
UserResourceMappingService: b.UserResourceMappingService,
|
|
|
|
UserService: b.UserService,
|
|
|
|
}
|
|
|
|
h.HandlerFunc("POST", notificationEndpointsIDMembersPath, newPostMemberHandler(memberBackend))
|
|
|
|
h.HandlerFunc("GET", notificationEndpointsIDMembersPath, newGetMembersHandler(memberBackend))
|
|
|
|
h.HandlerFunc("DELETE", notificationEndpointsIDMembersIDPath, newDeleteMemberHandler(memberBackend))
|
|
|
|
|
|
|
|
ownerBackend := MemberBackend{
|
|
|
|
HTTPErrorHandler: b.HTTPErrorHandler,
|
2019-12-04 23:10:23 +00:00
|
|
|
log: b.log.With(zap.String("handler", "member")),
|
2019-08-08 21:59:03 +00:00
|
|
|
ResourceType: influxdb.NotificationEndpointResourceType,
|
|
|
|
UserType: influxdb.Owner,
|
|
|
|
UserResourceMappingService: b.UserResourceMappingService,
|
|
|
|
UserService: b.UserService,
|
|
|
|
}
|
|
|
|
h.HandlerFunc("POST", notificationEndpointsIDOwnersPath, newPostMemberHandler(ownerBackend))
|
|
|
|
h.HandlerFunc("GET", notificationEndpointsIDOwnersPath, newGetMembersHandler(ownerBackend))
|
|
|
|
h.HandlerFunc("DELETE", notificationEndpointsIDOwnersIDPath, newDeleteMemberHandler(ownerBackend))
|
|
|
|
|
|
|
|
labelBackend := &LabelBackend{
|
|
|
|
HTTPErrorHandler: b.HTTPErrorHandler,
|
2019-12-04 23:10:23 +00:00
|
|
|
log: b.log.With(zap.String("handler", "label")),
|
2019-08-08 21:59:03 +00:00
|
|
|
LabelService: b.LabelService,
|
2020-03-16 10:32:35 +00:00
|
|
|
ResourceType: influxdb.NotificationEndpointResourceType,
|
2019-08-08 21:59:03 +00:00
|
|
|
}
|
2019-09-23 22:55:37 +00:00
|
|
|
h.HandlerFunc("GET", notificationEndpointsIDLabelsPath, newGetLabelsHandler(labelBackend))
|
2019-08-08 21:59:03 +00:00
|
|
|
h.HandlerFunc("POST", notificationEndpointsIDLabelsPath, newPostLabelHandler(labelBackend))
|
|
|
|
h.HandlerFunc("DELETE", notificationEndpointsIDLabelsIDPath, newDeleteLabelHandler(labelBackend))
|
|
|
|
|
|
|
|
return h
|
|
|
|
}
|
|
|
|
|
|
|
|
type notificationEndpointLinks struct {
|
|
|
|
Self string `json:"self"`
|
|
|
|
Labels string `json:"labels"`
|
|
|
|
Members string `json:"members"`
|
|
|
|
Owners string `json:"owners"`
|
|
|
|
}
|
|
|
|
|
2019-12-10 19:27:48 +00:00
|
|
|
type postNotificationEndpointRequest struct {
|
2019-08-08 21:59:03 +00:00
|
|
|
influxdb.NotificationEndpoint
|
2019-12-10 19:27:48 +00:00
|
|
|
Labels []string `json:"labels"`
|
2019-08-08 21:59:03 +00:00
|
|
|
}
|
|
|
|
|
2019-12-10 19:27:48 +00:00
|
|
|
type notificationEndpointResponse struct {
|
2019-09-30 23:15:14 +00:00
|
|
|
influxdb.NotificationEndpoint
|
2019-12-10 19:27:48 +00:00
|
|
|
Labels []influxdb.Label `json:"labels"`
|
|
|
|
Links notificationEndpointLinks `json:"links"`
|
2019-09-30 23:15:14 +00:00
|
|
|
}
|
|
|
|
|
2019-08-08 21:59:03 +00:00
|
|
|
func (resp notificationEndpointResponse) MarshalJSON() ([]byte, error) {
|
|
|
|
b1, err := json.Marshal(resp.NotificationEndpoint)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
b2, err := json.Marshal(struct {
|
|
|
|
Labels []influxdb.Label `json:"labels"`
|
|
|
|
Links notificationEndpointLinks `json:"links"`
|
|
|
|
}{
|
|
|
|
Links: resp.Links,
|
|
|
|
Labels: resp.Labels,
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return []byte(string(b1[:len(b1)-1]) + ", " + string(b2[1:])), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
type notificationEndpointsResponse struct {
|
2019-12-10 19:27:48 +00:00
|
|
|
NotificationEndpoints []notificationEndpointResponse `json:"notificationEndpoints"`
|
|
|
|
Links *influxdb.PagingLinks `json:"links"`
|
2019-08-08 21:59:03 +00:00
|
|
|
}
|
|
|
|
|
2019-12-10 19:27:48 +00:00
|
|
|
func newNotificationEndpointResponse(edp influxdb.NotificationEndpoint, labels []*influxdb.Label) notificationEndpointResponse {
|
|
|
|
res := notificationEndpointResponse{
|
2019-08-08 21:59:03 +00:00
|
|
|
NotificationEndpoint: edp,
|
|
|
|
Links: notificationEndpointLinks{
|
|
|
|
Self: fmt.Sprintf("/api/v2/notificationEndpoints/%s", edp.GetID()),
|
|
|
|
Labels: fmt.Sprintf("/api/v2/notificationEndpoints/%s/labels", edp.GetID()),
|
|
|
|
Members: fmt.Sprintf("/api/v2/notificationEndpoints/%s/members", edp.GetID()),
|
|
|
|
Owners: fmt.Sprintf("/api/v2/notificationEndpoints/%s/owners", edp.GetID()),
|
|
|
|
},
|
|
|
|
Labels: []influxdb.Label{},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, l := range labels {
|
|
|
|
res.Labels = append(res.Labels, *l)
|
|
|
|
}
|
|
|
|
|
|
|
|
return res
|
|
|
|
}
|
|
|
|
|
|
|
|
func newNotificationEndpointsResponse(ctx context.Context, edps []influxdb.NotificationEndpoint, labelService influxdb.LabelService, f influxdb.PagingFilter, opts influxdb.FindOptions) *notificationEndpointsResponse {
|
|
|
|
resp := ¬ificationEndpointsResponse{
|
2019-12-10 19:27:48 +00:00
|
|
|
NotificationEndpoints: make([]notificationEndpointResponse, len(edps)),
|
2020-04-30 14:52:21 +00:00
|
|
|
Links: influxdb.NewPagingLinks(prefixNotificationEndpoints, opts, f, len(edps)),
|
2019-08-08 21:59:03 +00:00
|
|
|
}
|
|
|
|
for i, edp := range edps {
|
2020-03-12 17:51:50 +00:00
|
|
|
labels, _ := labelService.FindResourceLabels(ctx, influxdb.LabelMappingFilter{ResourceID: edp.GetID(), ResourceType: influxdb.NotificationEndpointResourceType})
|
2019-08-08 21:59:03 +00:00
|
|
|
resp.NotificationEndpoints[i] = newNotificationEndpointResponse(edp, labels)
|
|
|
|
}
|
|
|
|
return resp
|
|
|
|
}
|
|
|
|
|
2019-12-10 19:27:48 +00:00
|
|
|
func decodeGetNotificationEndpointRequest(ctx context.Context) (i influxdb.ID, err error) {
|
2019-08-08 21:59:03 +00:00
|
|
|
params := httprouter.ParamsFromContext(ctx)
|
|
|
|
id := params.ByName("id")
|
|
|
|
if id == "" {
|
|
|
|
return i, &influxdb.Error{
|
|
|
|
Code: influxdb.EInvalid,
|
|
|
|
Msg: "url missing id",
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := i.DecodeFromString(id); err != nil {
|
|
|
|
return i, err
|
|
|
|
}
|
|
|
|
return i, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (h *NotificationEndpointHandler) handleGetNotificationEndpoints(w http.ResponseWriter, r *http.Request) {
|
|
|
|
ctx := r.Context()
|
|
|
|
filter, opts, err := decodeNotificationEndpointFilter(ctx, r)
|
|
|
|
if err != nil {
|
2019-12-04 23:10:23 +00:00
|
|
|
h.log.Debug("Failed to decode request", zap.Error(err))
|
2019-08-08 21:59:03 +00:00
|
|
|
h.HandleHTTPError(ctx, err, w)
|
|
|
|
return
|
|
|
|
}
|
2019-12-10 19:27:48 +00:00
|
|
|
edps, _, err := h.NotificationEndpointService.FindNotificationEndpoints(ctx, filter, opts)
|
2019-08-08 21:59:03 +00:00
|
|
|
if err != nil {
|
|
|
|
h.HandleHTTPError(ctx, err, w)
|
|
|
|
return
|
|
|
|
}
|
2019-12-04 23:10:23 +00:00
|
|
|
h.log.Debug("NotificationEndpoints retrieved", zap.String("notificationEndpoints", fmt.Sprint(edps)))
|
2019-08-08 21:59:03 +00:00
|
|
|
|
2019-12-10 19:27:48 +00:00
|
|
|
if err := encodeResponse(ctx, w, http.StatusOK, newNotificationEndpointsResponse(ctx, edps, h.LabelService, filter, opts)); err != nil {
|
2019-12-04 23:10:23 +00:00
|
|
|
logEncodingError(h.log, r, err)
|
2019-08-08 21:59:03 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (h *NotificationEndpointHandler) handleGetNotificationEndpoint(w http.ResponseWriter, r *http.Request) {
|
|
|
|
ctx := r.Context()
|
2019-12-10 19:27:48 +00:00
|
|
|
id, err := decodeGetNotificationEndpointRequest(ctx)
|
2019-08-08 21:59:03 +00:00
|
|
|
if err != nil {
|
|
|
|
h.HandleHTTPError(ctx, err, w)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
edp, err := h.NotificationEndpointService.FindNotificationEndpointByID(ctx, id)
|
|
|
|
if err != nil {
|
|
|
|
h.HandleHTTPError(ctx, err, w)
|
|
|
|
return
|
|
|
|
}
|
2019-12-04 23:10:23 +00:00
|
|
|
h.log.Debug("NotificationEndpoint retrieved", zap.String("notificationEndpoint", fmt.Sprint(edp)))
|
2019-08-08 21:59:03 +00:00
|
|
|
|
2020-03-12 17:51:50 +00:00
|
|
|
labels, err := h.LabelService.FindResourceLabels(ctx, influxdb.LabelMappingFilter{ResourceID: edp.GetID(), ResourceType: influxdb.NotificationEndpointResourceType})
|
2019-08-08 21:59:03 +00:00
|
|
|
if err != nil {
|
|
|
|
h.HandleHTTPError(ctx, err, w)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := encodeResponse(ctx, w, http.StatusOK, newNotificationEndpointResponse(edp, labels)); err != nil {
|
2019-12-04 23:10:23 +00:00
|
|
|
logEncodingError(h.log, r, err)
|
2019-08-08 21:59:03 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-12-10 19:27:48 +00:00
|
|
|
func decodeNotificationEndpointFilter(ctx context.Context, r *http.Request) (influxdb.NotificationEndpointFilter, influxdb.FindOptions, error) {
|
|
|
|
f := influxdb.NotificationEndpointFilter{
|
2019-10-24 17:28:34 +00:00
|
|
|
UserResourceMappingFilter: influxdb.UserResourceMappingFilter{
|
|
|
|
ResourceType: influxdb.NotificationEndpointResourceType,
|
|
|
|
},
|
|
|
|
}
|
2019-08-08 21:59:03 +00:00
|
|
|
|
2020-04-30 14:52:21 +00:00
|
|
|
opts, err := influxdb.DecodeFindOptions(r)
|
2019-08-08 21:59:03 +00:00
|
|
|
if err != nil {
|
2019-12-10 19:27:48 +00:00
|
|
|
return influxdb.NotificationEndpointFilter{}, influxdb.FindOptions{}, err
|
2019-08-08 21:59:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
q := r.URL.Query()
|
|
|
|
if orgIDStr := q.Get("orgID"); orgIDStr != "" {
|
|
|
|
orgID, err := influxdb.IDFromString(orgIDStr)
|
|
|
|
if err != nil {
|
2019-12-10 19:27:48 +00:00
|
|
|
return influxdb.NotificationEndpointFilter{}, influxdb.FindOptions{}, &influxdb.Error{
|
2019-08-08 21:59:03 +00:00
|
|
|
Code: influxdb.EInvalid,
|
|
|
|
Msg: "orgID is invalid",
|
|
|
|
Err: err,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
f.OrgID = orgID
|
|
|
|
} else if orgNameStr := q.Get("org"); orgNameStr != "" {
|
|
|
|
*f.Org = orgNameStr
|
|
|
|
}
|
2019-11-27 16:09:36 +00:00
|
|
|
|
|
|
|
if userID := q.Get("user"); userID != "" {
|
|
|
|
id, err := influxdb.IDFromString(userID)
|
|
|
|
if err != nil {
|
2019-12-10 19:27:48 +00:00
|
|
|
return influxdb.NotificationEndpointFilter{}, influxdb.FindOptions{}, err
|
2019-11-27 16:09:36 +00:00
|
|
|
}
|
|
|
|
f.UserID = *id
|
|
|
|
}
|
|
|
|
|
2019-12-10 19:27:48 +00:00
|
|
|
return f, *opts, err
|
2019-08-08 21:59:03 +00:00
|
|
|
}
|
|
|
|
|
2019-12-10 19:27:48 +00:00
|
|
|
func decodePostNotificationEndpointRequest(r *http.Request) (postNotificationEndpointRequest, error) {
|
2019-12-13 23:32:21 +00:00
|
|
|
b, err := ioutil.ReadAll(r.Body)
|
2019-08-08 21:59:03 +00:00
|
|
|
if err != nil {
|
2019-12-13 23:32:21 +00:00
|
|
|
return postNotificationEndpointRequest{}, &influxdb.Error{
|
2019-08-08 21:59:03 +00:00
|
|
|
Code: influxdb.EInvalid,
|
|
|
|
Err: err,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
defer r.Body.Close()
|
2019-12-13 23:32:21 +00:00
|
|
|
edp, err := endpoint.UnmarshalJSON(b)
|
2019-08-08 21:59:03 +00:00
|
|
|
if err != nil {
|
2019-12-13 23:32:21 +00:00
|
|
|
return postNotificationEndpointRequest{}, &influxdb.Error{
|
2019-08-08 21:59:03 +00:00
|
|
|
Code: influxdb.EInvalid,
|
|
|
|
Err: err,
|
|
|
|
}
|
|
|
|
}
|
2019-09-30 23:15:14 +00:00
|
|
|
|
|
|
|
var dl decodeLabels
|
2019-12-13 23:32:21 +00:00
|
|
|
if err := json.Unmarshal(b, &dl); err != nil {
|
|
|
|
return postNotificationEndpointRequest{}, &influxdb.Error{
|
2019-09-30 23:15:14 +00:00
|
|
|
Code: influxdb.EInvalid,
|
|
|
|
Err: err,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-12-13 23:32:21 +00:00
|
|
|
return postNotificationEndpointRequest{
|
|
|
|
NotificationEndpoint: edp,
|
|
|
|
Labels: dl.Labels,
|
|
|
|
}, nil
|
2019-08-08 21:59:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func decodePutNotificationEndpointRequest(ctx context.Context, r *http.Request) (influxdb.NotificationEndpoint, error) {
|
|
|
|
buf := new(bytes.Buffer)
|
2019-12-10 19:27:48 +00:00
|
|
|
if _, err := buf.ReadFrom(r.Body); err != nil {
|
2019-08-08 21:59:03 +00:00
|
|
|
return nil, &influxdb.Error{
|
|
|
|
Code: influxdb.EInvalid,
|
|
|
|
Err: err,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
defer r.Body.Close()
|
2019-12-10 19:27:48 +00:00
|
|
|
|
2019-08-08 21:59:03 +00:00
|
|
|
edp, err := endpoint.UnmarshalJSON(buf.Bytes())
|
|
|
|
if err != nil {
|
|
|
|
return nil, &influxdb.Error{
|
|
|
|
Code: influxdb.EInvalid,
|
|
|
|
Err: err,
|
|
|
|
}
|
|
|
|
}
|
2019-12-10 19:27:48 +00:00
|
|
|
|
2019-08-08 21:59:03 +00:00
|
|
|
params := httprouter.ParamsFromContext(ctx)
|
2019-12-10 19:27:48 +00:00
|
|
|
i, err := influxdb.IDFromString(params.ByName("id"))
|
|
|
|
if err != nil {
|
2019-08-08 21:59:03 +00:00
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
edp.SetID(*i)
|
|
|
|
return edp, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
type patchNotificationEndpointRequest struct {
|
|
|
|
influxdb.ID
|
|
|
|
Update influxdb.NotificationEndpointUpdate
|
|
|
|
}
|
|
|
|
|
2019-12-10 19:27:48 +00:00
|
|
|
func decodePatchNotificationEndpointRequest(ctx context.Context, r *http.Request) (patchNotificationEndpointRequest, error) {
|
2019-08-08 21:59:03 +00:00
|
|
|
params := httprouter.ParamsFromContext(ctx)
|
2019-12-10 19:27:48 +00:00
|
|
|
id, err := influxdb.IDFromString(params.ByName("id"))
|
|
|
|
if err != nil {
|
|
|
|
return patchNotificationEndpointRequest{}, err
|
2019-08-08 21:59:03 +00:00
|
|
|
}
|
2019-12-10 19:27:48 +00:00
|
|
|
req := patchNotificationEndpointRequest{
|
|
|
|
ID: *id,
|
2019-08-08 21:59:03 +00:00
|
|
|
}
|
|
|
|
|
2019-12-10 19:27:48 +00:00
|
|
|
var upd influxdb.NotificationEndpointUpdate
|
|
|
|
if err := json.NewDecoder(r.Body).Decode(&upd); err != nil {
|
|
|
|
return patchNotificationEndpointRequest{}, &influxdb.Error{
|
2019-08-08 21:59:03 +00:00
|
|
|
Code: influxdb.EInvalid,
|
|
|
|
Msg: err.Error(),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if err := upd.Valid(); err != nil {
|
2019-12-10 19:27:48 +00:00
|
|
|
return patchNotificationEndpointRequest{}, &influxdb.Error{
|
2019-08-08 21:59:03 +00:00
|
|
|
Code: influxdb.EInvalid,
|
|
|
|
Msg: err.Error(),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-12-10 19:27:48 +00:00
|
|
|
req.Update = upd
|
2019-08-08 21:59:03 +00:00
|
|
|
return req, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// handlePostNotificationEndpoint is the HTTP handler for the POST /api/v2/notificationEndpoints route.
|
|
|
|
func (h *NotificationEndpointHandler) handlePostNotificationEndpoint(w http.ResponseWriter, r *http.Request) {
|
|
|
|
ctx := r.Context()
|
2019-12-10 19:27:48 +00:00
|
|
|
edp, err := decodePostNotificationEndpointRequest(r)
|
2019-08-08 21:59:03 +00:00
|
|
|
if err != nil {
|
2019-12-04 23:10:23 +00:00
|
|
|
h.log.Debug("Failed to decode request", zap.Error(err))
|
2019-08-08 21:59:03 +00:00
|
|
|
h.HandleHTTPError(ctx, err, w)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
auth, err := pctx.GetAuthorizer(ctx)
|
|
|
|
if err != nil {
|
|
|
|
h.HandleHTTPError(ctx, err, w)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2019-12-16 17:39:55 +00:00
|
|
|
err = h.NotificationEndpointService.CreateNotificationEndpoint(ctx, edp.NotificationEndpoint, auth.GetUserID())
|
|
|
|
if err != nil {
|
2019-08-08 21:59:03 +00:00
|
|
|
h.HandleHTTPError(ctx, err, w)
|
|
|
|
return
|
|
|
|
}
|
2019-08-28 20:02:17 +00:00
|
|
|
|
2019-09-30 23:15:14 +00:00
|
|
|
labels := h.mapNewNotificationEndpointLabels(ctx, edp.NotificationEndpoint, edp.Labels)
|
|
|
|
|
2019-12-04 23:10:23 +00:00
|
|
|
h.log.Debug("NotificationEndpoint created", zap.String("notificationEndpoint", fmt.Sprint(edp)))
|
2019-08-08 21:59:03 +00:00
|
|
|
|
2019-09-30 23:15:14 +00:00
|
|
|
if err := encodeResponse(ctx, w, http.StatusCreated, newNotificationEndpointResponse(edp, labels)); err != nil {
|
2019-12-04 23:10:23 +00:00
|
|
|
logEncodingError(h.log, r, err)
|
2019-08-08 21:59:03 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-09-30 23:15:14 +00:00
|
|
|
func (h *NotificationEndpointHandler) mapNewNotificationEndpointLabels(ctx context.Context, nre influxdb.NotificationEndpoint, labels []string) []*influxdb.Label {
|
|
|
|
var ls []*influxdb.Label
|
|
|
|
for _, sid := range labels {
|
|
|
|
var lid influxdb.ID
|
|
|
|
err := lid.DecodeFromString(sid)
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
label, err := h.LabelService.FindLabelByID(ctx, lid)
|
|
|
|
if err != nil {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
mapping := influxdb.LabelMapping{
|
|
|
|
LabelID: label.ID,
|
|
|
|
ResourceID: nre.GetID(),
|
|
|
|
ResourceType: influxdb.NotificationEndpointResourceType,
|
|
|
|
}
|
|
|
|
|
|
|
|
err = h.LabelService.CreateLabelMapping(ctx, &mapping)
|
|
|
|
if err != nil {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
ls = append(ls, label)
|
|
|
|
}
|
|
|
|
return ls
|
|
|
|
}
|
|
|
|
|
2019-08-08 21:59:03 +00:00
|
|
|
// handlePutNotificationEndpoint is the HTTP handler for the PUT /api/v2/notificationEndpoints route.
|
|
|
|
func (h *NotificationEndpointHandler) handlePutNotificationEndpoint(w http.ResponseWriter, r *http.Request) {
|
|
|
|
ctx := r.Context()
|
|
|
|
edp, err := decodePutNotificationEndpointRequest(ctx, r)
|
|
|
|
if err != nil {
|
2019-12-04 23:10:23 +00:00
|
|
|
h.log.Debug("Failed to decode request", zap.Error(err))
|
2019-08-08 21:59:03 +00:00
|
|
|
h.HandleHTTPError(ctx, err, w)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
auth, err := pctx.GetAuthorizer(ctx)
|
|
|
|
if err != nil {
|
|
|
|
h.HandleHTTPError(ctx, err, w)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
edp, err = h.NotificationEndpointService.UpdateNotificationEndpoint(ctx, edp.GetID(), edp, auth.GetUserID())
|
|
|
|
if err != nil {
|
|
|
|
h.HandleHTTPError(ctx, err, w)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2020-03-12 17:51:50 +00:00
|
|
|
labels, err := h.LabelService.FindResourceLabels(ctx, influxdb.LabelMappingFilter{ResourceID: edp.GetID(), ResourceType: influxdb.NotificationEndpointResourceType})
|
2019-08-08 21:59:03 +00:00
|
|
|
if err != nil {
|
|
|
|
h.HandleHTTPError(ctx, err, w)
|
|
|
|
return
|
|
|
|
}
|
2019-12-04 23:10:23 +00:00
|
|
|
h.log.Debug("NotificationEndpoint replaced", zap.String("notificationEndpoint", fmt.Sprint(edp)))
|
2019-08-08 21:59:03 +00:00
|
|
|
|
|
|
|
if err := encodeResponse(ctx, w, http.StatusOK, newNotificationEndpointResponse(edp, labels)); err != nil {
|
2019-12-04 23:10:23 +00:00
|
|
|
logEncodingError(h.log, r, err)
|
2019-08-08 21:59:03 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// handlePatchNotificationEndpoint is the HTTP handler for the PATCH /api/v2/notificationEndpoints/:id route.
|
|
|
|
func (h *NotificationEndpointHandler) handlePatchNotificationEndpoint(w http.ResponseWriter, r *http.Request) {
|
|
|
|
ctx := r.Context()
|
|
|
|
req, err := decodePatchNotificationEndpointRequest(ctx, r)
|
|
|
|
if err != nil {
|
2019-12-04 23:10:23 +00:00
|
|
|
h.log.Debug("Failed to decode request", zap.Error(err))
|
2019-08-08 21:59:03 +00:00
|
|
|
h.HandleHTTPError(ctx, err, w)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
edp, err := h.NotificationEndpointService.PatchNotificationEndpoint(ctx, req.ID, req.Update)
|
|
|
|
if err != nil {
|
|
|
|
h.HandleHTTPError(ctx, err, w)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2020-03-12 17:51:50 +00:00
|
|
|
labels, err := h.LabelService.FindResourceLabels(ctx, influxdb.LabelMappingFilter{ResourceID: edp.GetID(), ResourceType: influxdb.NotificationEndpointResourceType})
|
2019-08-08 21:59:03 +00:00
|
|
|
if err != nil {
|
|
|
|
h.HandleHTTPError(ctx, err, w)
|
|
|
|
return
|
|
|
|
}
|
2019-12-04 23:10:23 +00:00
|
|
|
h.log.Debug("NotificationEndpoint patch", zap.String("notificationEndpoint", fmt.Sprint(edp)))
|
2019-08-08 21:59:03 +00:00
|
|
|
|
|
|
|
if err := encodeResponse(ctx, w, http.StatusOK, newNotificationEndpointResponse(edp, labels)); err != nil {
|
2019-12-04 23:10:23 +00:00
|
|
|
logEncodingError(h.log, r, err)
|
2019-08-08 21:59:03 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (h *NotificationEndpointHandler) handleDeleteNotificationEndpoint(w http.ResponseWriter, r *http.Request) {
|
|
|
|
ctx := r.Context()
|
2019-12-10 19:27:48 +00:00
|
|
|
i, err := decodeGetNotificationEndpointRequest(ctx)
|
2019-08-08 21:59:03 +00:00
|
|
|
if err != nil {
|
|
|
|
h.HandleHTTPError(ctx, err, w)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2019-12-13 23:32:21 +00:00
|
|
|
flds, _, err := h.NotificationEndpointService.DeleteNotificationEndpoint(ctx, i)
|
2019-08-28 20:02:17 +00:00
|
|
|
if err != nil {
|
2019-08-08 21:59:03 +00:00
|
|
|
h.HandleHTTPError(ctx, err, w)
|
|
|
|
return
|
|
|
|
}
|
2019-08-28 20:02:17 +00:00
|
|
|
keys := make([]string, len(flds))
|
|
|
|
for k, fld := range flds {
|
|
|
|
if fld.Key == "" {
|
|
|
|
h.HandleHTTPError(ctx, &influxdb.Error{
|
|
|
|
Op: "http/handleDeleteNotificationEndpoint",
|
|
|
|
Msg: "Bad Secret Key in endpoint " + i.String(),
|
|
|
|
}, w)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
keys[k] = fld.Key
|
|
|
|
}
|
2019-12-04 23:10:23 +00:00
|
|
|
h.log.Debug("NotificationEndpoint deleted", zap.String("notificationEndpointID", fmt.Sprint(i)))
|
2019-08-08 21:59:03 +00:00
|
|
|
|
|
|
|
w.WriteHeader(http.StatusNoContent)
|
|
|
|
}
|
2019-12-10 19:27:48 +00:00
|
|
|
|
|
|
|
// NotificationEndpointService is an http client for the influxdb.NotificationEndpointService server implementation.
|
|
|
|
type NotificationEndpointService struct {
|
|
|
|
Client *httpc.Client
|
|
|
|
*UserResourceMappingService
|
|
|
|
*OrganizationService
|
|
|
|
}
|
|
|
|
|
|
|
|
// NewNotificationEndpointService constructs a new http NotificationEndpointService.
|
|
|
|
func NewNotificationEndpointService(client *httpc.Client) *NotificationEndpointService {
|
|
|
|
return &NotificationEndpointService{
|
|
|
|
Client: client,
|
|
|
|
UserResourceMappingService: &UserResourceMappingService{
|
|
|
|
Client: client,
|
|
|
|
},
|
|
|
|
OrganizationService: &OrganizationService{
|
|
|
|
Client: client,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
var _ influxdb.NotificationEndpointService = (*NotificationEndpointService)(nil)
|
|
|
|
|
|
|
|
// FindNotificationEndpointByID returns a single notification endpoint by ID.
|
|
|
|
func (s *NotificationEndpointService) FindNotificationEndpointByID(ctx context.Context, id influxdb.ID) (influxdb.NotificationEndpoint, error) {
|
2020-03-11 14:57:21 +00:00
|
|
|
if !id.Valid() {
|
|
|
|
return nil, fmt.Errorf("invalid ID: please provide a valid ID")
|
|
|
|
}
|
2019-12-10 21:35:23 +00:00
|
|
|
var resp notificationEndpointDecoder
|
2019-12-10 19:27:48 +00:00
|
|
|
err := s.Client.
|
2019-12-09 23:54:16 +00:00
|
|
|
Get(prefixNotificationEndpoints, id.String()).
|
2019-12-10 19:27:48 +00:00
|
|
|
DecodeJSON(&resp).
|
|
|
|
Do(ctx)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2019-12-10 21:35:23 +00:00
|
|
|
return resp.endpoint, nil
|
2019-12-10 19:27:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// FindNotificationEndpoints returns a list of notification endpoints that match filter and the total count of matching notification endpoints.
|
|
|
|
// Additional options provide pagination & sorting.
|
|
|
|
func (s *NotificationEndpointService) FindNotificationEndpoints(ctx context.Context, filter influxdb.NotificationEndpointFilter, opt ...influxdb.FindOptions) ([]influxdb.NotificationEndpoint, int, error) {
|
2020-04-30 14:52:21 +00:00
|
|
|
params := influxdb.FindOptionParams(opt...)
|
2019-12-10 19:27:48 +00:00
|
|
|
if filter.ID != nil {
|
|
|
|
params = append(params, [2]string{"id", filter.ID.String()})
|
|
|
|
}
|
|
|
|
if filter.OrgID != nil {
|
|
|
|
params = append(params, [2]string{"orgID", filter.OrgID.String()})
|
|
|
|
}
|
|
|
|
if filter.Org != nil {
|
|
|
|
params = append(params, [2]string{"org", *filter.Org})
|
|
|
|
}
|
|
|
|
|
2019-12-10 21:35:23 +00:00
|
|
|
var resp struct {
|
|
|
|
Endpoints []notificationEndpointDecoder `json:"notificationEndpoints"`
|
|
|
|
}
|
2019-12-10 19:27:48 +00:00
|
|
|
err := s.Client.
|
2019-12-09 23:54:16 +00:00
|
|
|
Get(prefixNotificationEndpoints).
|
2019-12-10 19:27:48 +00:00
|
|
|
QueryParams(params...).
|
|
|
|
DecodeJSON(&resp).
|
|
|
|
Do(ctx)
|
|
|
|
if err != nil {
|
|
|
|
return nil, 0, err
|
|
|
|
}
|
|
|
|
|
|
|
|
var endpoints []influxdb.NotificationEndpoint
|
2019-12-10 21:35:23 +00:00
|
|
|
for _, e := range resp.Endpoints {
|
|
|
|
endpoints = append(endpoints, e.endpoint)
|
2019-12-10 19:27:48 +00:00
|
|
|
}
|
2019-12-10 21:35:23 +00:00
|
|
|
return endpoints, len(endpoints), nil
|
2019-12-10 19:27:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// CreateNotificationEndpoint creates a new notification endpoint and sets b.ID with the new identifier.
|
|
|
|
// TODO(@jsteenb2): this is unsatisfactory, we have no way of grabbing the new notification endpoint without
|
|
|
|
// serious hacky hackertoning. Put it on the list...
|
|
|
|
func (s *NotificationEndpointService) CreateNotificationEndpoint(ctx context.Context, ne influxdb.NotificationEndpoint, userID influxdb.ID) error {
|
2019-12-10 21:35:23 +00:00
|
|
|
var resp notificationEndpointDecoder
|
2019-12-10 19:27:48 +00:00
|
|
|
err := s.Client.
|
2019-12-12 03:26:02 +00:00
|
|
|
PostJSON(¬ificationEndpointEncoder{ne: ne}, prefixNotificationEndpoints).
|
2019-12-10 19:27:48 +00:00
|
|
|
DecodeJSON(&resp).
|
|
|
|
Do(ctx)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
// :sadpanda:
|
2019-12-10 21:35:23 +00:00
|
|
|
ne.SetID(resp.endpoint.GetID())
|
|
|
|
ne.SetOrgID(resp.endpoint.GetOrgID())
|
2019-12-10 19:27:48 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// UpdateNotificationEndpoint updates a single notification endpoint.
|
|
|
|
// Returns the new notification endpoint after update.
|
2019-12-10 22:51:11 +00:00
|
|
|
func (s *NotificationEndpointService) UpdateNotificationEndpoint(ctx context.Context, id influxdb.ID, ne influxdb.NotificationEndpoint, userID influxdb.ID) (influxdb.NotificationEndpoint, error) {
|
2020-03-11 14:57:21 +00:00
|
|
|
if !id.Valid() {
|
|
|
|
return nil, fmt.Errorf("invalid ID: please provide a valid ID")
|
|
|
|
}
|
2019-12-10 21:35:23 +00:00
|
|
|
var resp notificationEndpointDecoder
|
2019-12-10 19:27:48 +00:00
|
|
|
err := s.Client.
|
2019-12-12 03:26:02 +00:00
|
|
|
PutJSON(¬ificationEndpointEncoder{ne: ne}, prefixNotificationEndpoints, id.String()).
|
2019-12-10 19:27:48 +00:00
|
|
|
DecodeJSON(&resp).
|
|
|
|
Do(ctx)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2019-12-10 21:35:23 +00:00
|
|
|
return resp.endpoint, nil
|
2019-12-10 19:27:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// PatchNotificationEndpoint updates a single notification endpoint with changeset.
|
|
|
|
// Returns the new notification endpoint state after update.
|
|
|
|
func (s *NotificationEndpointService) PatchNotificationEndpoint(ctx context.Context, id influxdb.ID, upd influxdb.NotificationEndpointUpdate) (influxdb.NotificationEndpoint, error) {
|
2020-03-11 14:57:21 +00:00
|
|
|
if !id.Valid() {
|
|
|
|
return nil, fmt.Errorf("invalid ID: please provide a valid ID")
|
|
|
|
}
|
2019-12-10 19:27:48 +00:00
|
|
|
if err := upd.Valid(); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2019-12-10 21:35:23 +00:00
|
|
|
var resp notificationEndpointDecoder
|
2019-12-10 19:27:48 +00:00
|
|
|
err := s.Client.
|
2019-12-12 03:26:02 +00:00
|
|
|
PatchJSON(upd, prefixNotificationEndpoints, id.String()).
|
2019-12-10 19:27:48 +00:00
|
|
|
DecodeJSON(&resp).
|
|
|
|
Do(ctx)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2019-12-10 21:35:23 +00:00
|
|
|
return resp.endpoint, nil
|
2019-12-10 19:27:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// DeleteNotificationEndpoint removes a notification endpoint by ID, returns secret fields, orgID for further deletion.
|
2019-12-10 22:51:11 +00:00
|
|
|
// TODO: axe this delete design, makes little sense in how its currently being done. Right now, as an http client,
|
|
|
|
// I am forced to know how the store handles this and then figure out what the server does in between me and that store,
|
|
|
|
// then see what falls out :flushed... for now returning nothing for secrets, orgID, and only returning an error. This makes
|
|
|
|
// the code/design smell super obvious imo
|
|
|
|
func (s *NotificationEndpointService) DeleteNotificationEndpoint(ctx context.Context, id influxdb.ID) ([]influxdb.SecretField, influxdb.ID, error) {
|
2020-03-11 14:57:21 +00:00
|
|
|
if !id.Valid() {
|
|
|
|
return nil, 0, fmt.Errorf("invalid ID: please provide a valid ID")
|
|
|
|
}
|
2019-12-10 22:51:11 +00:00
|
|
|
err := s.Client.
|
|
|
|
Delete(prefixNotificationEndpoints, id.String()).
|
|
|
|
Do(ctx)
|
|
|
|
return nil, 0, err
|
|
|
|
}
|
|
|
|
|
|
|
|
type notificationEndpointEncoder struct {
|
|
|
|
ne influxdb.NotificationEndpoint
|
|
|
|
}
|
|
|
|
|
|
|
|
func (n *notificationEndpointEncoder) MarshalJSON() ([]byte, error) {
|
|
|
|
b, err := json.Marshal(n.ne)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
ughhh := make(map[string]interface{})
|
|
|
|
if err := json.Unmarshal(b, &ughhh); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
n.ne.BackfillSecretKeys()
|
|
|
|
|
|
|
|
// this makes me queezy and altogether sad
|
|
|
|
fieldMap := map[string]string{
|
|
|
|
"-password": "password",
|
|
|
|
"-routing-key": "routingKey",
|
|
|
|
"-token": "token",
|
|
|
|
"-username": "username",
|
|
|
|
}
|
|
|
|
for _, sec := range n.ne.SecretFields() {
|
|
|
|
var v string
|
|
|
|
if sec.Value != nil {
|
|
|
|
v = *sec.Value
|
|
|
|
}
|
|
|
|
ughhh[fieldMap[sec.Key]] = v
|
|
|
|
}
|
|
|
|
return json.Marshal(ughhh)
|
2019-12-10 19:27:48 +00:00
|
|
|
}
|
2019-12-10 21:35:23 +00:00
|
|
|
|
|
|
|
type notificationEndpointDecoder struct {
|
|
|
|
endpoint influxdb.NotificationEndpoint
|
|
|
|
}
|
|
|
|
|
|
|
|
func (n *notificationEndpointDecoder) UnmarshalJSON(b []byte) error {
|
|
|
|
newEndpoint, err := endpoint.UnmarshalJSON(b)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
n.endpoint = newEndpoint
|
|
|
|
return nil
|
|
|
|
}
|