2018-09-07 15:45:28 +00:00
|
|
|
package http
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"context"
|
|
|
|
"encoding/json"
|
|
|
|
"net/http"
|
|
|
|
"path"
|
|
|
|
|
2019-01-11 17:51:15 +00:00
|
|
|
"github.com/influxdata/influxdb"
|
2019-01-18 20:46:37 +00:00
|
|
|
pctx "github.com/influxdata/influxdb/context"
|
2018-09-07 15:45:28 +00:00
|
|
|
"github.com/julienschmidt/httprouter"
|
2018-12-20 16:07:46 +00:00
|
|
|
"go.uber.org/zap"
|
2018-09-07 15:45:28 +00:00
|
|
|
)
|
|
|
|
|
2019-01-16 12:44:17 +00:00
|
|
|
// ScraperBackend is all services and associated parameters required to construct
|
|
|
|
// the ScraperHandler.
|
|
|
|
type ScraperBackend struct {
|
|
|
|
Logger *zap.Logger
|
|
|
|
|
|
|
|
ScraperStorageService influxdb.ScraperTargetStoreService
|
|
|
|
BucketService influxdb.BucketService
|
|
|
|
OrganizationService influxdb.OrganizationService
|
|
|
|
UserService influxdb.UserService
|
|
|
|
UserResourceMappingService influxdb.UserResourceMappingService
|
|
|
|
LabelService influxdb.LabelService
|
|
|
|
}
|
|
|
|
|
|
|
|
// NewScraperBackend returns a new instance of ScraperBackend.
|
|
|
|
func NewScraperBackend(b *APIBackend) *ScraperBackend {
|
|
|
|
return &ScraperBackend{
|
|
|
|
Logger: b.Logger.With(zap.String("handler", "scraper")),
|
|
|
|
|
2019-02-08 07:38:31 +00:00
|
|
|
ScraperStorageService: b.ScraperTargetStoreService,
|
|
|
|
BucketService: b.BucketService,
|
|
|
|
OrganizationService: b.OrganizationService,
|
|
|
|
UserService: b.UserService,
|
|
|
|
UserResourceMappingService: b.UserResourceMappingService,
|
|
|
|
LabelService: b.LabelService,
|
2019-01-16 12:44:17 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-09-07 15:45:28 +00:00
|
|
|
// ScraperHandler represents an HTTP API handler for scraper targets.
|
|
|
|
type ScraperHandler struct {
|
|
|
|
*httprouter.Router
|
2019-01-18 21:06:32 +00:00
|
|
|
Logger *zap.Logger
|
|
|
|
UserService influxdb.UserService
|
|
|
|
UserResourceMappingService influxdb.UserResourceMappingService
|
|
|
|
LabelService influxdb.LabelService
|
|
|
|
ScraperStorageService influxdb.ScraperTargetStoreService
|
|
|
|
BucketService influxdb.BucketService
|
|
|
|
OrganizationService influxdb.OrganizationService
|
2018-09-07 15:45:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
const (
|
2019-01-18 21:06:32 +00:00
|
|
|
targetsPath = "/api/v2/scrapers"
|
|
|
|
targetsIDMembersPath = targetsPath + "/:id/members"
|
|
|
|
targetsIDMembersIDPath = targetsPath + "/:id/members/:userID"
|
|
|
|
targetsIDOwnersPath = targetsPath + "/:id/owners"
|
|
|
|
targetsIDOwnersIDPath = targetsPath + "/:id/owners/:userID"
|
|
|
|
targetsIDLabelsPath = targetsPath + "/:id/labels"
|
|
|
|
targetsIDLabelsIDPath = targetsPath + "/:id/labels/:lid"
|
2018-09-07 15:45:28 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
// NewScraperHandler returns a new instance of ScraperHandler.
|
2019-01-16 12:44:17 +00:00
|
|
|
func NewScraperHandler(b *ScraperBackend) *ScraperHandler {
|
2018-09-07 15:45:28 +00:00
|
|
|
h := &ScraperHandler{
|
2019-01-23 17:11:06 +00:00
|
|
|
Router: NewRouter(),
|
2019-01-16 12:44:17 +00:00
|
|
|
Logger: b.Logger,
|
|
|
|
UserService: b.UserService,
|
|
|
|
UserResourceMappingService: b.UserResourceMappingService,
|
|
|
|
LabelService: b.LabelService,
|
|
|
|
ScraperStorageService: b.ScraperStorageService,
|
|
|
|
BucketService: b.BucketService,
|
|
|
|
OrganizationService: b.OrganizationService,
|
2018-09-07 15:45:28 +00:00
|
|
|
}
|
2019-01-18 21:06:32 +00:00
|
|
|
h.HandlerFunc("POST", targetsPath, h.handlePostScraperTarget)
|
|
|
|
h.HandlerFunc("GET", targetsPath, h.handleGetScraperTargets)
|
|
|
|
h.HandlerFunc("GET", targetsPath+"/:id", h.handleGetScraperTarget)
|
|
|
|
h.HandlerFunc("PATCH", targetsPath+"/:id", h.handlePatchScraperTarget)
|
|
|
|
h.HandlerFunc("DELETE", targetsPath+"/:id", h.handleDeleteScraperTarget)
|
|
|
|
|
2019-02-05 19:27:49 +00:00
|
|
|
memberBackend := MemberBackend{
|
|
|
|
Logger: b.Logger.With(zap.String("handler", "member")),
|
|
|
|
ResourceType: influxdb.ScraperResourceType,
|
|
|
|
UserType: influxdb.Member,
|
|
|
|
UserResourceMappingService: b.UserResourceMappingService,
|
|
|
|
UserService: b.UserService,
|
|
|
|
}
|
|
|
|
h.HandlerFunc("POST", targetsIDMembersPath, newPostMemberHandler(memberBackend))
|
|
|
|
h.HandlerFunc("GET", targetsIDMembersPath, newGetMembersHandler(memberBackend))
|
|
|
|
h.HandlerFunc("DELETE", targetsIDMembersIDPath, newDeleteMemberHandler(memberBackend))
|
|
|
|
|
|
|
|
ownerBackend := MemberBackend{
|
|
|
|
Logger: b.Logger.With(zap.String("handler", "member")),
|
|
|
|
ResourceType: influxdb.ScraperResourceType,
|
|
|
|
UserType: influxdb.Owner,
|
|
|
|
UserResourceMappingService: b.UserResourceMappingService,
|
|
|
|
UserService: b.UserService,
|
|
|
|
}
|
|
|
|
h.HandlerFunc("POST", targetsIDOwnersPath, newPostMemberHandler(ownerBackend))
|
|
|
|
h.HandlerFunc("GET", targetsIDOwnersPath, newGetMembersHandler(ownerBackend))
|
|
|
|
h.HandlerFunc("DELETE", targetsIDOwnersIDPath, newDeleteMemberHandler(ownerBackend))
|
2019-01-18 21:06:32 +00:00
|
|
|
|
2019-02-05 19:27:49 +00:00
|
|
|
labelBackend := &LabelBackend{
|
|
|
|
Logger: b.Logger.With(zap.String("handler", "label")),
|
|
|
|
LabelService: b.LabelService,
|
2019-02-07 23:23:15 +00:00
|
|
|
ResourceType: influxdb.ScraperResourceType,
|
2019-02-05 19:27:49 +00:00
|
|
|
}
|
|
|
|
h.HandlerFunc("GET", targetsIDLabelsPath, newGetLabelsHandler(labelBackend))
|
|
|
|
h.HandlerFunc("POST", targetsIDLabelsPath, newPostLabelHandler(labelBackend))
|
|
|
|
h.HandlerFunc("DELETE", targetsIDLabelsIDPath, newDeleteLabelHandler(labelBackend))
|
2019-01-18 21:06:32 +00:00
|
|
|
|
2018-09-07 15:45:28 +00:00
|
|
|
return h
|
|
|
|
}
|
|
|
|
|
2019-01-18 15:38:28 +00:00
|
|
|
// handlePostScraperTarget is HTTP handler for the POST /api/v2/scrapers route.
|
2018-09-07 15:45:28 +00:00
|
|
|
func (h *ScraperHandler) handlePostScraperTarget(w http.ResponseWriter, r *http.Request) {
|
|
|
|
ctx := r.Context()
|
|
|
|
req, err := decodeScraperTargetAddRequest(ctx, r)
|
|
|
|
if err != nil {
|
|
|
|
EncodeError(ctx, err, w)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2019-01-18 20:46:37 +00:00
|
|
|
auth, err := pctx.GetAuthorizer(ctx)
|
|
|
|
if err != nil {
|
|
|
|
EncodeError(ctx, err, w)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := h.ScraperStorageService.AddTarget(ctx, req, auth.GetUserID()); err != nil {
|
2018-09-07 15:45:28 +00:00
|
|
|
EncodeError(ctx, err, w)
|
|
|
|
return
|
|
|
|
}
|
2019-01-11 17:51:15 +00:00
|
|
|
resp, err := h.newTargetResponse(ctx, *req)
|
|
|
|
if err != nil {
|
|
|
|
EncodeError(ctx, err, w)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if err := encodeResponse(ctx, w, http.StatusCreated, resp); err != nil {
|
2018-12-20 16:07:46 +00:00
|
|
|
logEncodingError(h.Logger, r, err)
|
2018-09-07 15:45:28 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-01-18 15:38:28 +00:00
|
|
|
// handleDeleteScraperTarget is the HTTP handler for the DELETE /api/v2/scrapers/:id route.
|
2018-09-07 15:45:28 +00:00
|
|
|
func (h *ScraperHandler) handleDeleteScraperTarget(w http.ResponseWriter, r *http.Request) {
|
|
|
|
ctx := r.Context()
|
|
|
|
|
|
|
|
id, err := decodeScraperTargetIDRequest(ctx, r)
|
|
|
|
if err != nil {
|
|
|
|
EncodeError(ctx, err, w)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := h.ScraperStorageService.RemoveTarget(ctx, *id); err != nil {
|
|
|
|
EncodeError(ctx, err, w)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2018-10-29 19:10:33 +00:00
|
|
|
w.WriteHeader(http.StatusNoContent)
|
2018-09-07 15:45:28 +00:00
|
|
|
}
|
|
|
|
|
2019-01-18 15:38:28 +00:00
|
|
|
// handlePatchScraperTarget is the HTTP handler for the PATCH /api/v2/scrapers/:id route.
|
2018-09-07 15:45:28 +00:00
|
|
|
func (h *ScraperHandler) handlePatchScraperTarget(w http.ResponseWriter, r *http.Request) {
|
|
|
|
ctx := r.Context()
|
|
|
|
|
|
|
|
update, err := decodeScraperTargetUpdateRequest(ctx, r)
|
|
|
|
if err != nil {
|
|
|
|
EncodeError(ctx, err, w)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2019-01-18 20:46:37 +00:00
|
|
|
auth, err := pctx.GetAuthorizer(ctx)
|
|
|
|
if err != nil {
|
|
|
|
EncodeError(ctx, err, w)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
target, err := h.ScraperStorageService.UpdateTarget(ctx, update, auth.GetUserID())
|
2018-09-07 15:45:28 +00:00
|
|
|
if err != nil {
|
|
|
|
EncodeError(ctx, err, w)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2019-01-11 17:51:15 +00:00
|
|
|
resp, err := h.newTargetResponse(ctx, *target)
|
|
|
|
if err != nil {
|
|
|
|
EncodeError(ctx, err, w)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := encodeResponse(ctx, w, http.StatusOK, resp); err != nil {
|
2018-12-20 16:07:46 +00:00
|
|
|
logEncodingError(h.Logger, r, err)
|
2018-09-07 15:45:28 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (h *ScraperHandler) handleGetScraperTarget(w http.ResponseWriter, r *http.Request) {
|
|
|
|
ctx := r.Context()
|
|
|
|
|
|
|
|
id, err := decodeScraperTargetIDRequest(ctx, r)
|
|
|
|
if err != nil {
|
|
|
|
EncodeError(ctx, err, w)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
target, err := h.ScraperStorageService.GetTargetByID(ctx, *id)
|
|
|
|
if err != nil {
|
|
|
|
EncodeError(ctx, err, w)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2019-01-11 17:51:15 +00:00
|
|
|
resp, err := h.newTargetResponse(ctx, *target)
|
|
|
|
if err != nil {
|
|
|
|
EncodeError(ctx, err, w)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := encodeResponse(ctx, w, http.StatusOK, resp); err != nil {
|
2018-12-20 16:07:46 +00:00
|
|
|
logEncodingError(h.Logger, r, err)
|
2018-09-07 15:45:28 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-01-18 15:38:28 +00:00
|
|
|
// handleGetScraperTargets is the HTTP handler for the GET /api/v2/scrapers route.
|
2018-09-07 15:45:28 +00:00
|
|
|
func (h *ScraperHandler) handleGetScraperTargets(w http.ResponseWriter, r *http.Request) {
|
|
|
|
ctx := r.Context()
|
|
|
|
|
|
|
|
targets, err := h.ScraperStorageService.ListTargets(ctx)
|
|
|
|
if err != nil {
|
|
|
|
EncodeError(ctx, err, w)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2019-01-11 17:51:15 +00:00
|
|
|
resp, err := h.newListTargetsResponse(ctx, targets)
|
|
|
|
if err != nil {
|
|
|
|
EncodeError(ctx, err, w)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := encodeResponse(ctx, w, http.StatusOK, resp); err != nil {
|
2018-12-20 16:07:46 +00:00
|
|
|
logEncodingError(h.Logger, r, err)
|
2018-09-07 15:45:28 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-01-11 17:51:15 +00:00
|
|
|
func decodeScraperTargetUpdateRequest(ctx context.Context, r *http.Request) (*influxdb.ScraperTarget, error) {
|
|
|
|
update := &influxdb.ScraperTarget{}
|
2018-09-07 15:45:28 +00:00
|
|
|
if err := json.NewDecoder(r.Body).Decode(update); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
id, err := decodeScraperTargetIDRequest(ctx, r)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
update.ID = *id
|
|
|
|
return update, nil
|
|
|
|
}
|
|
|
|
|
2019-01-11 17:51:15 +00:00
|
|
|
func decodeScraperTargetAddRequest(ctx context.Context, r *http.Request) (*influxdb.ScraperTarget, error) {
|
|
|
|
req := &influxdb.ScraperTarget{}
|
2018-09-07 15:45:28 +00:00
|
|
|
if err := json.NewDecoder(r.Body).Decode(req); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return req, nil
|
|
|
|
}
|
|
|
|
|
2019-01-11 17:51:15 +00:00
|
|
|
func decodeScraperTargetIDRequest(ctx context.Context, r *http.Request) (*influxdb.ID, error) {
|
2018-09-07 15:45:28 +00:00
|
|
|
params := httprouter.ParamsFromContext(ctx)
|
|
|
|
id := params.ByName("id")
|
|
|
|
if id == "" {
|
2019-01-11 17:51:15 +00:00
|
|
|
return nil, &influxdb.Error{
|
|
|
|
Code: influxdb.EInvalid,
|
2019-01-10 17:39:37 +00:00
|
|
|
Msg: "url missing id",
|
|
|
|
}
|
2018-09-07 15:45:28 +00:00
|
|
|
}
|
|
|
|
|
2019-01-11 17:51:15 +00:00
|
|
|
var i influxdb.ID
|
2018-09-07 15:45:28 +00:00
|
|
|
if err := i.DecodeFromString(id); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return &i, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// ScraperService connects to Influx via HTTP using tokens to manage scraper targets.
|
|
|
|
type ScraperService struct {
|
|
|
|
Addr string
|
|
|
|
Token string
|
|
|
|
InsecureSkipVerify bool
|
2018-12-17 14:07:38 +00:00
|
|
|
// OpPrefix is for update invalid ops
|
|
|
|
OpPrefix string
|
2018-09-07 15:45:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// ListTargets returns a list of all scraper targets.
|
2019-01-11 17:51:15 +00:00
|
|
|
func (s *ScraperService) ListTargets(ctx context.Context) ([]influxdb.ScraperTarget, error) {
|
2019-01-18 21:06:32 +00:00
|
|
|
url, err := newURL(s.Addr, targetsPath)
|
2018-09-07 15:45:28 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
query := url.Query()
|
|
|
|
|
|
|
|
req, err := http.NewRequest("GET", url.String(), nil)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
req.URL.RawQuery = query.Encode()
|
|
|
|
SetToken(s.Token, req)
|
|
|
|
|
|
|
|
hc := newClient(url.Scheme, s.InsecureSkipVerify)
|
|
|
|
resp, err := hc.Do(req)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2019-01-18 22:32:53 +00:00
|
|
|
defer resp.Body.Close()
|
2019-01-24 01:02:37 +00:00
|
|
|
if err := CheckError(resp); err != nil {
|
2018-09-07 15:45:28 +00:00
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
var targetsResp getTargetsResponse
|
|
|
|
if err := json.NewDecoder(resp.Body).Decode(&targetsResp); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2019-01-11 17:51:15 +00:00
|
|
|
targets := make([]influxdb.ScraperTarget, len(targetsResp.Targets))
|
2018-09-07 15:45:28 +00:00
|
|
|
for k, v := range targetsResp.Targets {
|
|
|
|
targets[k] = v.ScraperTarget
|
|
|
|
}
|
|
|
|
|
|
|
|
return targets, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// UpdateTarget updates a single scraper target with changeset.
|
|
|
|
// Returns the new target state after update.
|
2019-01-18 20:46:37 +00:00
|
|
|
func (s *ScraperService) UpdateTarget(ctx context.Context, update *influxdb.ScraperTarget, userID influxdb.ID) (*influxdb.ScraperTarget, error) {
|
2018-10-12 02:57:09 +00:00
|
|
|
if !update.ID.Valid() {
|
2019-01-11 17:51:15 +00:00
|
|
|
return nil, &influxdb.Error{
|
|
|
|
Code: influxdb.EInvalid,
|
|
|
|
Op: s.OpPrefix + influxdb.OpUpdateTarget,
|
2019-02-19 23:47:19 +00:00
|
|
|
Msg: "provided scraper target ID has invalid format",
|
2018-12-17 14:07:38 +00:00
|
|
|
}
|
2018-10-12 02:57:09 +00:00
|
|
|
}
|
2018-09-07 15:45:28 +00:00
|
|
|
url, err := newURL(s.Addr, targetIDPath(update.ID))
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
octets, err := json.Marshal(update)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
req, err := http.NewRequest("PATCH", url.String(), bytes.NewReader(octets))
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
req.Header.Set("Content-Type", "application/json")
|
|
|
|
SetToken(s.Token, req)
|
|
|
|
hc := newClient(url.Scheme, s.InsecureSkipVerify)
|
|
|
|
|
|
|
|
resp, err := hc.Do(req)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2019-01-18 22:32:53 +00:00
|
|
|
defer resp.Body.Close()
|
2018-09-07 15:45:28 +00:00
|
|
|
|
2019-01-24 01:02:37 +00:00
|
|
|
if err := CheckError(resp); err != nil {
|
2018-09-07 15:45:28 +00:00
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
var targetResp targetResponse
|
|
|
|
if err := json.NewDecoder(resp.Body).Decode(&targetResp); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return &targetResp.ScraperTarget, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// AddTarget creates a new scraper target and sets target.ID with the new identifier.
|
2019-01-18 20:46:37 +00:00
|
|
|
func (s *ScraperService) AddTarget(ctx context.Context, target *influxdb.ScraperTarget, userID influxdb.ID) error {
|
2019-01-18 21:06:32 +00:00
|
|
|
url, err := newURL(s.Addr, targetsPath)
|
2018-09-07 15:45:28 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2019-01-10 17:39:37 +00:00
|
|
|
if !target.OrgID.Valid() {
|
2019-01-11 17:51:15 +00:00
|
|
|
return &influxdb.Error{
|
|
|
|
Code: influxdb.EInvalid,
|
2019-02-19 23:47:19 +00:00
|
|
|
Msg: "provided organization ID has invalid format",
|
2019-01-11 17:51:15 +00:00
|
|
|
Op: s.OpPrefix + influxdb.OpAddTarget,
|
2019-01-10 17:39:37 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if !target.BucketID.Valid() {
|
2019-01-11 17:51:15 +00:00
|
|
|
return &influxdb.Error{
|
|
|
|
Code: influxdb.EInvalid,
|
2019-02-19 23:47:19 +00:00
|
|
|
Msg: "provided bucket ID has invalid format",
|
2019-01-11 17:51:15 +00:00
|
|
|
Op: s.OpPrefix + influxdb.OpAddTarget,
|
2019-01-10 17:39:37 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-09-07 15:45:28 +00:00
|
|
|
octets, err := json.Marshal(target)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
req, err := http.NewRequest("POST", url.String(), bytes.NewReader(octets))
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
req.Header.Set("Content-Type", "application/json")
|
|
|
|
SetToken(s.Token, req)
|
|
|
|
|
|
|
|
hc := newClient(url.Scheme, s.InsecureSkipVerify)
|
|
|
|
|
|
|
|
resp, err := hc.Do(req)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2019-01-18 22:32:53 +00:00
|
|
|
defer resp.Body.Close()
|
2018-09-07 15:45:28 +00:00
|
|
|
|
|
|
|
// TODO(jsternberg): Should this check for a 201 explicitly?
|
2019-01-24 01:02:37 +00:00
|
|
|
if err := CheckError(resp); err != nil {
|
2018-09-07 15:45:28 +00:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
targetResp := new(targetResponse)
|
|
|
|
if err := json.NewDecoder(resp.Body).Decode(targetResp); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// RemoveTarget removes a scraper target by ID.
|
2019-01-11 17:51:15 +00:00
|
|
|
func (s *ScraperService) RemoveTarget(ctx context.Context, id influxdb.ID) error {
|
2018-09-07 15:45:28 +00:00
|
|
|
url, err := newURL(s.Addr, targetIDPath(id))
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
req, err := http.NewRequest("DELETE", url.String(), nil)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
SetToken(s.Token, req)
|
|
|
|
|
|
|
|
hc := newClient(url.Scheme, s.InsecureSkipVerify)
|
|
|
|
resp, err := hc.Do(req)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2019-01-18 22:32:53 +00:00
|
|
|
defer resp.Body.Close()
|
2018-10-29 19:10:33 +00:00
|
|
|
|
2019-01-24 01:02:37 +00:00
|
|
|
return CheckErrorStatus(http.StatusNoContent, resp)
|
2018-09-07 15:45:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// GetTargetByID returns a single target by ID.
|
2019-01-11 17:51:15 +00:00
|
|
|
func (s *ScraperService) GetTargetByID(ctx context.Context, id influxdb.ID) (*influxdb.ScraperTarget, error) {
|
2018-09-07 15:45:28 +00:00
|
|
|
url, err := newURL(s.Addr, targetIDPath(id))
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
req, err := http.NewRequest("GET", url.String(), nil)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
SetToken(s.Token, req)
|
|
|
|
|
|
|
|
hc := newClient(url.Scheme, s.InsecureSkipVerify)
|
|
|
|
resp, err := hc.Do(req)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2019-01-18 22:32:53 +00:00
|
|
|
defer resp.Body.Close()
|
2018-09-07 15:45:28 +00:00
|
|
|
|
2019-01-24 01:02:37 +00:00
|
|
|
if err := CheckError(resp); err != nil {
|
2018-09-07 15:45:28 +00:00
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
var targetResp targetResponse
|
|
|
|
if err := json.NewDecoder(resp.Body).Decode(&targetResp); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return &targetResp.ScraperTarget, nil
|
|
|
|
}
|
|
|
|
|
2019-01-11 17:51:15 +00:00
|
|
|
func targetIDPath(id influxdb.ID) string {
|
2019-01-18 21:06:32 +00:00
|
|
|
return path.Join(targetsPath, id.String())
|
2018-09-07 15:45:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
type getTargetsLinks struct {
|
|
|
|
Self string `json:"self"`
|
|
|
|
}
|
|
|
|
|
|
|
|
type getTargetsResponse struct {
|
|
|
|
Links getTargetsLinks `json:"links"`
|
2019-01-23 02:29:08 +00:00
|
|
|
Targets []targetResponse `json:"configurations"`
|
2018-09-07 15:45:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
type targetLinks struct {
|
2019-01-23 01:21:23 +00:00
|
|
|
Self string `json:"self"`
|
|
|
|
Bucket string `json:"bucket,omitempty"`
|
|
|
|
Organization string `json:"organization,omitempty"`
|
2018-09-07 15:45:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
type targetResponse struct {
|
2019-01-11 17:51:15 +00:00
|
|
|
influxdb.ScraperTarget
|
2019-01-23 01:21:23 +00:00
|
|
|
Organization string `json:"organization,omitempty"`
|
|
|
|
Bucket string `json:"bucket,omitempty"`
|
|
|
|
Links targetLinks `json:"links"`
|
2018-09-07 15:45:28 +00:00
|
|
|
}
|
|
|
|
|
2019-01-11 17:51:15 +00:00
|
|
|
func (h *ScraperHandler) newListTargetsResponse(ctx context.Context, targets []influxdb.ScraperTarget) (getTargetsResponse, error) {
|
2018-09-07 15:45:28 +00:00
|
|
|
res := getTargetsResponse{
|
|
|
|
Links: getTargetsLinks{
|
2019-01-18 21:06:32 +00:00
|
|
|
Self: targetsPath,
|
2018-09-07 15:45:28 +00:00
|
|
|
},
|
|
|
|
Targets: make([]targetResponse, 0, len(targets)),
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, target := range targets {
|
2019-01-11 17:51:15 +00:00
|
|
|
resp, err := h.newTargetResponse(ctx, target)
|
|
|
|
if err != nil {
|
|
|
|
return res, err
|
|
|
|
}
|
|
|
|
res.Targets = append(res.Targets, resp)
|
2018-09-07 15:45:28 +00:00
|
|
|
}
|
|
|
|
|
2019-01-11 17:51:15 +00:00
|
|
|
return res, nil
|
2018-09-07 15:45:28 +00:00
|
|
|
}
|
|
|
|
|
2019-01-11 17:51:15 +00:00
|
|
|
func (h *ScraperHandler) newTargetResponse(ctx context.Context, target influxdb.ScraperTarget) (targetResponse, error) {
|
2019-01-23 01:21:23 +00:00
|
|
|
res := targetResponse{
|
2018-09-07 15:45:28 +00:00
|
|
|
Links: targetLinks{
|
|
|
|
Self: targetIDPath(target.ID),
|
|
|
|
},
|
|
|
|
ScraperTarget: target,
|
2019-01-23 01:21:23 +00:00
|
|
|
}
|
|
|
|
bucket, err := h.BucketService.FindBucketByID(ctx, target.BucketID)
|
|
|
|
if err == nil {
|
|
|
|
res.Bucket = bucket.Name
|
|
|
|
res.BucketID = bucket.ID
|
|
|
|
res.Links.Bucket = bucketIDPath(bucket.ID)
|
2019-01-23 01:35:23 +00:00
|
|
|
} else {
|
|
|
|
res.BucketID = influxdb.InvalidID()
|
2019-01-23 01:21:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
org, err := h.OrganizationService.FindOrganizationByID(ctx, target.OrgID)
|
|
|
|
if err == nil {
|
|
|
|
res.Organization = org.Name
|
|
|
|
res.OrgID = org.ID
|
|
|
|
res.Links.Organization = organizationIDPath(org.ID)
|
2019-01-23 01:35:23 +00:00
|
|
|
} else {
|
|
|
|
res.OrgID = influxdb.InvalidID()
|
2019-01-23 01:21:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return res, nil
|
2018-09-07 15:45:28 +00:00
|
|
|
}
|