2018-10-12 01:06:43 +00:00
|
|
|
package http
|
|
|
|
|
|
|
|
import (
|
2019-01-08 12:22:03 +00:00
|
|
|
"bytes"
|
2018-10-12 01:06:43 +00:00
|
|
|
"context"
|
2019-01-08 12:22:03 +00:00
|
|
|
"encoding/json"
|
2019-01-07 18:33:49 +00:00
|
|
|
"fmt"
|
|
|
|
"io/ioutil"
|
|
|
|
"net/http"
|
2018-10-12 01:06:43 +00:00
|
|
|
"net/http/httptest"
|
2019-01-23 17:11:06 +00:00
|
|
|
"os"
|
2018-10-12 01:06:43 +00:00
|
|
|
"testing"
|
|
|
|
|
2019-01-08 00:37:16 +00:00
|
|
|
platform "github.com/influxdata/influxdb"
|
2019-01-18 20:46:37 +00:00
|
|
|
platcontext "github.com/influxdata/influxdb/context"
|
2019-01-23 17:11:06 +00:00
|
|
|
httpMock "github.com/influxdata/influxdb/http/mock"
|
2019-01-08 00:37:16 +00:00
|
|
|
"github.com/influxdata/influxdb/inmem"
|
2019-01-23 17:11:06 +00:00
|
|
|
"github.com/influxdata/influxdb/logger"
|
2019-01-08 00:37:16 +00:00
|
|
|
"github.com/influxdata/influxdb/mock"
|
|
|
|
platformtesting "github.com/influxdata/influxdb/testing"
|
2019-01-07 18:33:49 +00:00
|
|
|
"github.com/julienschmidt/httprouter"
|
2018-10-12 01:06:43 +00:00
|
|
|
)
|
|
|
|
|
2019-01-07 18:33:49 +00:00
|
|
|
const (
|
|
|
|
targetOneIDString = "0000000000000111"
|
2019-01-08 10:54:57 +00:00
|
|
|
targetTwoIDString = "0000000000000222"
|
2019-01-07 18:33:49 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
|
|
|
targetOneID = platformtesting.MustIDBase16(targetOneIDString)
|
2019-01-08 10:54:57 +00:00
|
|
|
targetTwoID = platformtesting.MustIDBase16(targetTwoIDString)
|
2019-01-07 18:33:49 +00:00
|
|
|
)
|
|
|
|
|
2019-01-08 10:54:57 +00:00
|
|
|
func TestService_handleGetScraperTargets(t *testing.T) {
|
|
|
|
type fields struct {
|
2019-01-11 17:51:15 +00:00
|
|
|
ScraperTargetStoreService platform.ScraperTargetStoreService
|
|
|
|
OrganizationService platform.OrganizationService
|
|
|
|
BucketService platform.BucketService
|
2019-01-08 10:54:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
type args struct {
|
|
|
|
queryParams map[string][]string
|
|
|
|
}
|
|
|
|
|
|
|
|
type wants struct {
|
|
|
|
statusCode int
|
|
|
|
contentType string
|
|
|
|
body string
|
|
|
|
}
|
|
|
|
|
|
|
|
tests := []struct {
|
|
|
|
name string
|
|
|
|
fields fields
|
|
|
|
args args
|
|
|
|
wants wants
|
|
|
|
}{
|
|
|
|
{
|
|
|
|
name: "get all scraper targets",
|
|
|
|
fields: fields{
|
2019-01-11 17:51:15 +00:00
|
|
|
OrganizationService: &mock.OrganizationService{
|
|
|
|
FindOrganizationByIDF: func(ctx context.Context, id platform.ID) (*platform.Organization, error) {
|
|
|
|
return &platform.Organization{
|
|
|
|
ID: platformtesting.MustIDBase16("0000000000000211"),
|
|
|
|
Name: "org1",
|
|
|
|
}, nil
|
|
|
|
},
|
|
|
|
},
|
|
|
|
BucketService: &mock.BucketService{
|
|
|
|
FindBucketByIDFn: func(ctx context.Context, id platform.ID) (*platform.Bucket, error) {
|
|
|
|
return &platform.Bucket{
|
|
|
|
ID: platformtesting.MustIDBase16("0000000000000212"),
|
|
|
|
Name: "bucket1",
|
|
|
|
}, nil
|
|
|
|
},
|
|
|
|
},
|
|
|
|
ScraperTargetStoreService: &mock.ScraperTargetStoreService{
|
2019-01-08 10:54:57 +00:00
|
|
|
ListTargetsF: func(ctx context.Context) ([]platform.ScraperTarget, error) {
|
|
|
|
return []platform.ScraperTarget{
|
|
|
|
{
|
2019-01-10 17:39:37 +00:00
|
|
|
ID: targetOneID,
|
|
|
|
Name: "target-1",
|
|
|
|
Type: platform.PrometheusScraperType,
|
|
|
|
URL: "www.one.url",
|
|
|
|
OrgID: platformtesting.MustIDBase16("0000000000000211"),
|
|
|
|
BucketID: platformtesting.MustIDBase16("0000000000000212"),
|
2019-01-08 10:54:57 +00:00
|
|
|
},
|
|
|
|
{
|
2019-01-10 17:39:37 +00:00
|
|
|
ID: targetTwoID,
|
|
|
|
Name: "target-2",
|
|
|
|
Type: platform.PrometheusScraperType,
|
|
|
|
URL: "www.two.url",
|
|
|
|
OrgID: platformtesting.MustIDBase16("0000000000000211"),
|
|
|
|
BucketID: platformtesting.MustIDBase16("0000000000000212"),
|
2019-01-08 10:54:57 +00:00
|
|
|
},
|
|
|
|
}, nil
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
args: args{},
|
|
|
|
wants: wants{
|
|
|
|
statusCode: http.StatusOK,
|
|
|
|
contentType: "application/json; charset=utf-8",
|
|
|
|
body: fmt.Sprintf(
|
|
|
|
`
|
|
|
|
{
|
|
|
|
"links": {
|
2019-01-18 15:38:28 +00:00
|
|
|
"self": "/api/v2/scrapers"
|
2019-01-08 10:54:57 +00:00
|
|
|
},
|
2019-01-23 02:29:08 +00:00
|
|
|
"configurations": [
|
2019-01-08 10:54:57 +00:00
|
|
|
{
|
|
|
|
"id": "%s",
|
|
|
|
"name": "target-1",
|
2019-01-11 17:51:15 +00:00
|
|
|
"bucket": "bucket1",
|
2019-01-10 17:39:37 +00:00
|
|
|
"bucketID": "0000000000000212",
|
2019-01-11 17:51:15 +00:00
|
|
|
"organization": "org1",
|
2019-01-10 17:39:37 +00:00
|
|
|
"orgID": "0000000000000211",
|
2019-01-08 10:54:57 +00:00
|
|
|
"type": "prometheus",
|
|
|
|
"url": "www.one.url",
|
|
|
|
"links": {
|
2019-01-23 01:21:23 +00:00
|
|
|
"bucket": "/api/v2/buckets/0000000000000212",
|
|
|
|
"organization": "/api/v2/orgs/0000000000000211",
|
2019-01-18 15:38:28 +00:00
|
|
|
"self": "/api/v2/scrapers/0000000000000111"
|
2019-01-08 10:54:57 +00:00
|
|
|
}
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"id": "%s",
|
|
|
|
"name": "target-2",
|
2019-01-11 17:51:15 +00:00
|
|
|
"bucket": "bucket1",
|
2019-01-10 17:39:37 +00:00
|
|
|
"bucketID": "0000000000000212",
|
|
|
|
"orgID": "0000000000000211",
|
2019-01-11 17:51:15 +00:00
|
|
|
"organization": "org1",
|
2019-01-08 10:54:57 +00:00
|
|
|
"type": "prometheus",
|
|
|
|
"url": "www.two.url",
|
|
|
|
"links": {
|
2019-01-23 01:21:23 +00:00
|
|
|
"bucket": "/api/v2/buckets/0000000000000212",
|
|
|
|
"organization": "/api/v2/orgs/0000000000000211",
|
2019-01-18 15:38:28 +00:00
|
|
|
"self": "/api/v2/scrapers/0000000000000222"
|
2019-01-08 10:54:57 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
`,
|
|
|
|
targetOneIDString,
|
|
|
|
targetTwoIDString,
|
|
|
|
),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "get all scraper targets when there are none",
|
|
|
|
fields: fields{
|
2019-01-11 17:51:15 +00:00
|
|
|
OrganizationService: &mock.OrganizationService{
|
|
|
|
FindOrganizationByIDF: func(ctx context.Context, id platform.ID) (*platform.Organization, error) {
|
|
|
|
return &platform.Organization{
|
|
|
|
ID: platformtesting.MustIDBase16("0000000000000211"),
|
|
|
|
Name: "org1",
|
|
|
|
}, nil
|
|
|
|
},
|
|
|
|
},
|
|
|
|
BucketService: &mock.BucketService{
|
|
|
|
FindBucketByIDFn: func(ctx context.Context, id platform.ID) (*platform.Bucket, error) {
|
|
|
|
return &platform.Bucket{
|
|
|
|
ID: platformtesting.MustIDBase16("0000000000000212"),
|
|
|
|
Name: "bucket1",
|
|
|
|
}, nil
|
|
|
|
},
|
|
|
|
},
|
|
|
|
ScraperTargetStoreService: &mock.ScraperTargetStoreService{
|
2019-01-08 10:54:57 +00:00
|
|
|
ListTargetsF: func(ctx context.Context) ([]platform.ScraperTarget, error) {
|
|
|
|
return []platform.ScraperTarget{}, nil
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
args: args{},
|
|
|
|
wants: wants{
|
|
|
|
statusCode: http.StatusOK,
|
|
|
|
contentType: "application/json; charset=utf-8",
|
|
|
|
body: `
|
|
|
|
{
|
|
|
|
"links": {
|
2019-01-18 15:38:28 +00:00
|
|
|
"self": "/api/v2/scrapers"
|
2019-01-08 10:54:57 +00:00
|
|
|
},
|
2019-01-23 02:29:08 +00:00
|
|
|
"configurations": []
|
2019-01-08 10:54:57 +00:00
|
|
|
}
|
|
|
|
`,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, tt := range tests {
|
|
|
|
t.Run(tt.name, func(t *testing.T) {
|
2019-01-23 17:11:06 +00:00
|
|
|
h := NewScraperHandler(
|
|
|
|
logger.New(os.Stdout),
|
|
|
|
mock.NewUserService(),
|
|
|
|
&mock.UserResourceMappingService{},
|
|
|
|
mock.NewLabelService(),
|
|
|
|
tt.fields.ScraperTargetStoreService,
|
|
|
|
tt.fields.BucketService,
|
|
|
|
tt.fields.OrganizationService,
|
|
|
|
)
|
2019-01-08 10:54:57 +00:00
|
|
|
|
|
|
|
r := httptest.NewRequest("GET", "http://any.tld", nil)
|
|
|
|
|
|
|
|
qp := r.URL.Query()
|
|
|
|
for k, vs := range tt.args.queryParams {
|
|
|
|
for _, v := range vs {
|
|
|
|
qp.Add(k, v)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
r.URL.RawQuery = qp.Encode()
|
|
|
|
|
|
|
|
w := httptest.NewRecorder()
|
|
|
|
|
|
|
|
h.handleGetScraperTargets(w, r)
|
|
|
|
|
|
|
|
res := w.Result()
|
|
|
|
content := res.Header.Get("Content-Type")
|
|
|
|
body, _ := ioutil.ReadAll(res.Body)
|
|
|
|
|
|
|
|
if res.StatusCode != tt.wants.statusCode {
|
|
|
|
t.Errorf("%q. handleGetScraperTargets() = %v, want %v", tt.name, res.StatusCode, tt.wants.statusCode)
|
|
|
|
}
|
|
|
|
if tt.wants.contentType != "" && content != tt.wants.contentType {
|
|
|
|
t.Errorf("%q. handleGetScraperTargets() = %v, want %v", tt.name, content, tt.wants.contentType)
|
|
|
|
}
|
|
|
|
if eq, diff, _ := jsonEqual(string(body), tt.wants.body); tt.wants.body != "" && !eq {
|
2019-01-23 01:21:23 +00:00
|
|
|
t.Errorf("%q. handleGetScraperTargets() = ***%s***\n\ngot:\n%s\n\nwant:\n%s", tt.name, diff, string(body), tt.wants.body)
|
2019-01-08 10:54:57 +00:00
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
2019-01-07 18:33:49 +00:00
|
|
|
|
|
|
|
func TestService_handleGetScraperTarget(t *testing.T) {
|
|
|
|
type fields struct {
|
2019-01-11 17:51:15 +00:00
|
|
|
OrganizationService platform.OrganizationService
|
|
|
|
BucketService platform.BucketService
|
|
|
|
ScraperTargetStoreService platform.ScraperTargetStoreService
|
2019-01-07 18:33:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
type args struct {
|
|
|
|
id string
|
|
|
|
}
|
|
|
|
|
|
|
|
type wants struct {
|
|
|
|
statusCode int
|
|
|
|
contentType string
|
|
|
|
body string
|
|
|
|
}
|
|
|
|
|
|
|
|
tests := []struct {
|
|
|
|
name string
|
|
|
|
fields fields
|
|
|
|
args args
|
|
|
|
wants wants
|
|
|
|
}{
|
|
|
|
{
|
|
|
|
name: "get a scraper target by id",
|
|
|
|
fields: fields{
|
2019-01-11 17:51:15 +00:00
|
|
|
OrganizationService: &mock.OrganizationService{
|
|
|
|
FindOrganizationByIDF: func(ctx context.Context, id platform.ID) (*platform.Organization, error) {
|
|
|
|
return &platform.Organization{
|
|
|
|
ID: platformtesting.MustIDBase16("0000000000000211"),
|
|
|
|
Name: "org1",
|
|
|
|
}, nil
|
|
|
|
},
|
|
|
|
},
|
|
|
|
BucketService: &mock.BucketService{
|
|
|
|
FindBucketByIDFn: func(ctx context.Context, id platform.ID) (*platform.Bucket, error) {
|
|
|
|
return &platform.Bucket{
|
|
|
|
ID: platformtesting.MustIDBase16("0000000000000212"),
|
|
|
|
Name: "bucket1",
|
|
|
|
}, nil
|
|
|
|
},
|
|
|
|
},
|
|
|
|
ScraperTargetStoreService: &mock.ScraperTargetStoreService{
|
2019-01-07 18:33:49 +00:00
|
|
|
GetTargetByIDF: func(ctx context.Context, id platform.ID) (*platform.ScraperTarget, error) {
|
|
|
|
if id == targetOneID {
|
|
|
|
return &platform.ScraperTarget{
|
2019-01-10 17:39:37 +00:00
|
|
|
ID: targetOneID,
|
|
|
|
Name: "target-1",
|
|
|
|
Type: platform.PrometheusScraperType,
|
|
|
|
URL: "www.some.url",
|
|
|
|
OrgID: platformtesting.MustIDBase16("0000000000000211"),
|
|
|
|
BucketID: platformtesting.MustIDBase16("0000000000000212"),
|
2019-01-07 18:33:49 +00:00
|
|
|
}, nil
|
|
|
|
}
|
2019-01-11 17:51:15 +00:00
|
|
|
return nil, &platform.Error{
|
|
|
|
Code: platform.ENotFound,
|
|
|
|
Msg: "scraper target is not found",
|
|
|
|
}
|
2019-01-07 18:33:49 +00:00
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
args: args{
|
|
|
|
id: targetOneIDString,
|
|
|
|
},
|
|
|
|
wants: wants{
|
|
|
|
statusCode: http.StatusOK,
|
|
|
|
contentType: "application/json; charset=utf-8",
|
2019-01-08 10:54:57 +00:00
|
|
|
body: fmt.Sprintf(
|
|
|
|
`
|
|
|
|
{
|
|
|
|
"id": "%[1]s",
|
|
|
|
"name": "target-1",
|
|
|
|
"type": "prometheus",
|
2019-01-11 17:51:15 +00:00
|
|
|
"url": "www.some.url",
|
|
|
|
"bucket": "bucket1",
|
2019-01-10 17:39:37 +00:00
|
|
|
"bucketID": "0000000000000212",
|
2019-01-11 17:51:15 +00:00
|
|
|
"orgID": "0000000000000211",
|
|
|
|
"organization": "org1",
|
2019-01-08 10:54:57 +00:00
|
|
|
"links": {
|
2019-01-23 01:21:23 +00:00
|
|
|
"bucket": "/api/v2/buckets/0000000000000212",
|
|
|
|
"organization": "/api/v2/orgs/0000000000000211",
|
2019-01-18 15:38:28 +00:00
|
|
|
"self": "/api/v2/scrapers/%[1]s"
|
2019-01-08 10:54:57 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
`,
|
|
|
|
targetOneIDString,
|
|
|
|
),
|
2019-01-07 18:33:49 +00:00
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, tt := range tests {
|
|
|
|
t.Run(tt.name, func(t *testing.T) {
|
2019-01-23 17:11:06 +00:00
|
|
|
h := NewScraperHandler(
|
|
|
|
logger.New(os.Stdout),
|
|
|
|
mock.NewUserService(),
|
|
|
|
&mock.UserResourceMappingService{},
|
|
|
|
mock.NewLabelService(),
|
|
|
|
tt.fields.ScraperTargetStoreService,
|
|
|
|
tt.fields.BucketService,
|
|
|
|
tt.fields.OrganizationService,
|
|
|
|
)
|
2019-01-07 18:33:49 +00:00
|
|
|
|
|
|
|
r := httptest.NewRequest("GET", "http://any.tld", nil)
|
|
|
|
|
|
|
|
r = r.WithContext(context.WithValue(
|
|
|
|
context.Background(),
|
|
|
|
httprouter.ParamsKey,
|
|
|
|
httprouter.Params{
|
|
|
|
{
|
|
|
|
Key: "id",
|
|
|
|
Value: tt.args.id,
|
|
|
|
},
|
|
|
|
}))
|
|
|
|
|
|
|
|
w := httptest.NewRecorder()
|
|
|
|
|
|
|
|
h.handleGetScraperTarget(w, r)
|
|
|
|
|
|
|
|
res := w.Result()
|
|
|
|
content := res.Header.Get("Content-Type")
|
|
|
|
body, _ := ioutil.ReadAll(res.Body)
|
|
|
|
|
|
|
|
if res.StatusCode != tt.wants.statusCode {
|
|
|
|
t.Errorf("%q. handleGetScraperTarget() = %v, want %v", tt.name, res.StatusCode, tt.wants.statusCode)
|
|
|
|
}
|
|
|
|
if tt.wants.contentType != "" && content != tt.wants.contentType {
|
|
|
|
t.Errorf("%q. handleGetScraperTarget() = %v, want %v", tt.name, content, tt.wants.contentType)
|
|
|
|
}
|
|
|
|
if eq, diff, _ := jsonEqual(string(body), tt.wants.body); tt.wants.body != "" && !eq {
|
|
|
|
t.Errorf("%q. handleGetScraperTarget() = ***%s***", tt.name, diff)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-01-08 12:22:03 +00:00
|
|
|
func TestService_handleDeleteScraperTarget(t *testing.T) {
|
|
|
|
type fields struct {
|
|
|
|
Service platform.ScraperTargetStoreService
|
|
|
|
}
|
2019-01-07 18:33:49 +00:00
|
|
|
|
2019-01-08 12:22:03 +00:00
|
|
|
type args struct {
|
|
|
|
id string
|
|
|
|
}
|
2019-01-07 18:33:49 +00:00
|
|
|
|
2019-01-08 12:22:03 +00:00
|
|
|
type wants struct {
|
|
|
|
statusCode int
|
|
|
|
contentType string
|
|
|
|
body string
|
|
|
|
}
|
|
|
|
|
|
|
|
tests := []struct {
|
|
|
|
name string
|
|
|
|
fields fields
|
|
|
|
args args
|
|
|
|
wants wants
|
|
|
|
}{
|
|
|
|
{
|
|
|
|
name: "delete a scraper target by id",
|
|
|
|
fields: fields{
|
|
|
|
Service: &mock.ScraperTargetStoreService{
|
|
|
|
RemoveTargetF: func(ctx context.Context, id platform.ID) error {
|
|
|
|
if id == targetOneID {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
return fmt.Errorf("wrong id")
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
args: args{
|
|
|
|
id: targetOneIDString,
|
|
|
|
},
|
|
|
|
wants: wants{
|
|
|
|
statusCode: http.StatusNoContent,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "scraper target not found",
|
|
|
|
fields: fields{
|
|
|
|
Service: &mock.ScraperTargetStoreService{
|
|
|
|
RemoveTargetF: func(ctx context.Context, id platform.ID) error {
|
|
|
|
return &platform.Error{
|
|
|
|
Code: platform.ENotFound,
|
|
|
|
Msg: platform.ErrScraperTargetNotFound,
|
|
|
|
}
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
args: args{
|
|
|
|
id: targetTwoIDString,
|
|
|
|
},
|
|
|
|
wants: wants{
|
|
|
|
statusCode: http.StatusNotFound,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, tt := range tests {
|
|
|
|
t.Run(tt.name, func(t *testing.T) {
|
2019-01-23 17:11:06 +00:00
|
|
|
h := NewScraperHandler(
|
|
|
|
logger.New(os.Stdout),
|
|
|
|
mock.NewUserService(),
|
|
|
|
&mock.UserResourceMappingService{},
|
|
|
|
mock.NewLabelService(),
|
|
|
|
tt.fields.Service,
|
|
|
|
mock.NewBucketService(),
|
|
|
|
&mock.OrganizationService{},
|
|
|
|
)
|
2019-01-08 12:22:03 +00:00
|
|
|
r := httptest.NewRequest("GET", "http://any.tld", nil)
|
2019-01-07 18:33:49 +00:00
|
|
|
|
2019-01-08 12:22:03 +00:00
|
|
|
r = r.WithContext(context.WithValue(
|
|
|
|
context.Background(),
|
|
|
|
httprouter.ParamsKey,
|
|
|
|
httprouter.Params{
|
|
|
|
{
|
|
|
|
Key: "id",
|
|
|
|
Value: tt.args.id,
|
|
|
|
},
|
|
|
|
}))
|
|
|
|
|
|
|
|
w := httptest.NewRecorder()
|
|
|
|
|
|
|
|
h.handleDeleteScraperTarget(w, r)
|
|
|
|
|
|
|
|
res := w.Result()
|
|
|
|
content := res.Header.Get("Content-Type")
|
|
|
|
body, _ := ioutil.ReadAll(res.Body)
|
|
|
|
|
|
|
|
if res.StatusCode != tt.wants.statusCode {
|
|
|
|
t.Errorf("%q. handleDeleteScraperTarget() = %v, want %v", tt.name, res.StatusCode, tt.wants.statusCode)
|
|
|
|
}
|
|
|
|
if tt.wants.contentType != "" && content != tt.wants.contentType {
|
|
|
|
t.Errorf("%q. handleDeleteScraperTarget() = %v, want %v", tt.name, content, tt.wants.contentType)
|
|
|
|
}
|
|
|
|
if eq, diff, _ := jsonEqual(string(body), tt.wants.body); tt.wants.body != "" && !eq {
|
|
|
|
t.Errorf("%q. handleDeleteScraperTarget() = ***%s***", tt.name, diff)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
2019-01-07 18:33:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func TestService_handlePostScraperTarget(t *testing.T) {
|
2019-01-08 12:22:03 +00:00
|
|
|
type fields struct {
|
2019-01-11 17:51:15 +00:00
|
|
|
OrganizationService platform.OrganizationService
|
|
|
|
BucketService platform.BucketService
|
|
|
|
ScraperTargetStoreService platform.ScraperTargetStoreService
|
2019-01-08 12:22:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
type args struct {
|
|
|
|
target *platform.ScraperTarget
|
|
|
|
}
|
|
|
|
|
|
|
|
type wants struct {
|
|
|
|
statusCode int
|
|
|
|
contentType string
|
|
|
|
body string
|
|
|
|
}
|
|
|
|
|
|
|
|
tests := []struct {
|
|
|
|
name string
|
|
|
|
fields fields
|
|
|
|
args args
|
|
|
|
wants wants
|
|
|
|
}{
|
|
|
|
{
|
|
|
|
name: "create a new scraper target",
|
|
|
|
fields: fields{
|
2019-01-11 17:51:15 +00:00
|
|
|
OrganizationService: &mock.OrganizationService{
|
|
|
|
FindOrganizationByIDF: func(ctx context.Context, id platform.ID) (*platform.Organization, error) {
|
|
|
|
return &platform.Organization{
|
|
|
|
ID: platformtesting.MustIDBase16("0000000000000211"),
|
|
|
|
Name: "org1",
|
|
|
|
}, nil
|
|
|
|
},
|
|
|
|
},
|
|
|
|
BucketService: &mock.BucketService{
|
|
|
|
FindBucketByIDFn: func(ctx context.Context, id platform.ID) (*platform.Bucket, error) {
|
|
|
|
return &platform.Bucket{
|
|
|
|
ID: platformtesting.MustIDBase16("0000000000000212"),
|
|
|
|
Name: "bucket1",
|
|
|
|
}, nil
|
|
|
|
},
|
|
|
|
},
|
|
|
|
ScraperTargetStoreService: &mock.ScraperTargetStoreService{
|
2019-01-18 20:46:37 +00:00
|
|
|
AddTargetF: func(ctx context.Context, st *platform.ScraperTarget, userID platform.ID) error {
|
2019-01-08 12:22:03 +00:00
|
|
|
st.ID = targetOneID
|
|
|
|
return nil
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
args: args{
|
|
|
|
target: &platform.ScraperTarget{
|
2019-01-10 17:39:37 +00:00
|
|
|
Name: "hello",
|
|
|
|
Type: platform.PrometheusScraperType,
|
|
|
|
BucketID: platformtesting.MustIDBase16("0000000000000212"),
|
|
|
|
OrgID: platformtesting.MustIDBase16("0000000000000211"),
|
|
|
|
URL: "www.some.url",
|
2019-01-08 12:22:03 +00:00
|
|
|
},
|
|
|
|
},
|
|
|
|
wants: wants{
|
|
|
|
statusCode: http.StatusCreated,
|
|
|
|
contentType: "application/json; charset=utf-8",
|
|
|
|
body: fmt.Sprintf(
|
|
|
|
`
|
|
|
|
{
|
|
|
|
"id": "%[1]s",
|
|
|
|
"name": "hello",
|
|
|
|
"type": "prometheus",
|
|
|
|
"url": "www.some.url",
|
2019-01-11 17:51:15 +00:00
|
|
|
"orgID": "0000000000000211",
|
|
|
|
"organization": "org1",
|
|
|
|
"bucket": "bucket1",
|
2019-01-10 17:39:37 +00:00
|
|
|
"bucketID": "0000000000000212",
|
2019-01-08 12:22:03 +00:00
|
|
|
"links": {
|
2019-01-23 01:21:23 +00:00
|
|
|
"bucket": "/api/v2/buckets/0000000000000212",
|
|
|
|
"organization": "/api/v2/orgs/0000000000000211",
|
2019-01-18 15:38:28 +00:00
|
|
|
"self": "/api/v2/scrapers/%[1]s"
|
2019-01-08 12:22:03 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
`,
|
|
|
|
targetOneIDString,
|
|
|
|
),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, tt := range tests {
|
|
|
|
t.Run(tt.name, func(t *testing.T) {
|
2019-01-23 17:11:06 +00:00
|
|
|
h := NewScraperHandler(
|
|
|
|
logger.New(os.Stdout),
|
|
|
|
mock.NewUserService(),
|
|
|
|
&mock.UserResourceMappingService{},
|
|
|
|
mock.NewLabelService(),
|
|
|
|
tt.fields.ScraperTargetStoreService,
|
|
|
|
tt.fields.BucketService,
|
|
|
|
tt.fields.OrganizationService,
|
|
|
|
)
|
2019-01-08 12:22:03 +00:00
|
|
|
|
|
|
|
st, err := json.Marshal(tt.args.target)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("failed to unmarshal scraper target: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
r := httptest.NewRequest("GET", "http://any.tld", bytes.NewReader(st))
|
2019-01-18 20:46:37 +00:00
|
|
|
r = r.WithContext(platcontext.SetAuthorizer(r.Context(), &platform.Authorization{}))
|
2019-01-08 12:22:03 +00:00
|
|
|
w := httptest.NewRecorder()
|
|
|
|
|
|
|
|
h.handlePostScraperTarget(w, r)
|
|
|
|
|
|
|
|
res := w.Result()
|
|
|
|
content := res.Header.Get("Content-Type")
|
|
|
|
body, _ := ioutil.ReadAll(res.Body)
|
|
|
|
|
|
|
|
if res.StatusCode != tt.wants.statusCode {
|
|
|
|
t.Errorf("%q. handlePostScraperTarget() = %v, want %v", tt.name, res.StatusCode, tt.wants.statusCode)
|
|
|
|
}
|
|
|
|
if tt.wants.contentType != "" && content != tt.wants.contentType {
|
|
|
|
t.Errorf("%q. handlePostScraperTarget() = %v, want %v", tt.name, content, tt.wants.contentType)
|
|
|
|
}
|
|
|
|
if eq, diff, _ := jsonEqual(string(body), tt.wants.body); tt.wants.body != "" && !eq {
|
|
|
|
t.Errorf("%q. handlePostScraperTarget() = ***%s***", tt.name, diff)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestService_handlePatchScraperTarget(t *testing.T) {
|
2019-01-08 14:07:52 +00:00
|
|
|
type fields struct {
|
2019-01-11 17:51:15 +00:00
|
|
|
BucketService platform.BucketService
|
|
|
|
OrganizationService platform.OrganizationService
|
|
|
|
ScraperTargetStoreService platform.ScraperTargetStoreService
|
2019-01-08 14:07:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
type args struct {
|
|
|
|
id string
|
|
|
|
update *platform.ScraperTarget
|
|
|
|
}
|
|
|
|
|
|
|
|
type wants struct {
|
|
|
|
statusCode int
|
|
|
|
contentType string
|
|
|
|
body string
|
|
|
|
}
|
|
|
|
|
|
|
|
tests := []struct {
|
|
|
|
name string
|
|
|
|
fields fields
|
|
|
|
args args
|
|
|
|
wants wants
|
|
|
|
}{
|
|
|
|
{
|
|
|
|
name: "update a scraper target",
|
|
|
|
fields: fields{
|
2019-01-11 17:51:15 +00:00
|
|
|
OrganizationService: &mock.OrganizationService{
|
|
|
|
FindOrganizationByIDF: func(ctx context.Context, id platform.ID) (*platform.Organization, error) {
|
|
|
|
return &platform.Organization{
|
|
|
|
ID: platformtesting.MustIDBase16("0000000000000211"),
|
|
|
|
Name: "org1",
|
|
|
|
}, nil
|
|
|
|
},
|
|
|
|
},
|
|
|
|
BucketService: &mock.BucketService{
|
|
|
|
FindBucketByIDFn: func(ctx context.Context, id platform.ID) (*platform.Bucket, error) {
|
|
|
|
return &platform.Bucket{
|
|
|
|
ID: platformtesting.MustIDBase16("0000000000000212"),
|
|
|
|
Name: "bucket1",
|
|
|
|
}, nil
|
|
|
|
},
|
|
|
|
},
|
|
|
|
ScraperTargetStoreService: &mock.ScraperTargetStoreService{
|
2019-01-18 20:46:37 +00:00
|
|
|
UpdateTargetF: func(ctx context.Context, t *platform.ScraperTarget, userID platform.ID) (*platform.ScraperTarget, error) {
|
2019-01-08 14:07:52 +00:00
|
|
|
if t.ID == targetOneID {
|
|
|
|
return t, nil
|
|
|
|
}
|
|
|
|
|
2019-01-11 17:51:15 +00:00
|
|
|
return nil, &platform.Error{
|
|
|
|
Code: platform.ENotFound,
|
|
|
|
Msg: "scraper target is not found",
|
|
|
|
}
|
2019-01-08 14:07:52 +00:00
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
args: args{
|
|
|
|
id: targetOneIDString,
|
|
|
|
update: &platform.ScraperTarget{
|
2019-01-10 17:39:37 +00:00
|
|
|
ID: targetOneID,
|
|
|
|
Name: "name",
|
|
|
|
BucketID: platformtesting.MustIDBase16("0000000000000212"),
|
|
|
|
Type: platform.PrometheusScraperType,
|
|
|
|
URL: "www.example.url",
|
|
|
|
OrgID: platformtesting.MustIDBase16("0000000000000211"),
|
2019-01-08 14:07:52 +00:00
|
|
|
},
|
|
|
|
},
|
|
|
|
wants: wants{
|
|
|
|
statusCode: http.StatusOK,
|
|
|
|
contentType: "application/json; charset=utf-8",
|
|
|
|
body: fmt.Sprintf(
|
2019-01-11 17:51:15 +00:00
|
|
|
`{
|
2019-01-08 14:07:52 +00:00
|
|
|
"id":"%[1]s",
|
|
|
|
"name":"name",
|
|
|
|
"type":"prometheus",
|
2019-01-11 17:51:15 +00:00
|
|
|
"url":"www.example.url",
|
|
|
|
"organization": "org1",
|
|
|
|
"orgID":"0000000000000211",
|
|
|
|
"bucket": "bucket1",
|
|
|
|
"bucketID":"0000000000000212",
|
2019-01-08 14:07:52 +00:00
|
|
|
"links":{
|
2019-01-23 01:21:23 +00:00
|
|
|
"bucket": "/api/v2/buckets/0000000000000212",
|
|
|
|
"organization": "/api/v2/orgs/0000000000000211",
|
2019-01-18 15:38:28 +00:00
|
|
|
"self":"/api/v2/scrapers/%[1]s"
|
2019-01-08 14:07:52 +00:00
|
|
|
}
|
2019-01-11 17:51:15 +00:00
|
|
|
}`,
|
2019-01-08 14:07:52 +00:00
|
|
|
targetOneIDString,
|
|
|
|
),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "scraper target not found",
|
|
|
|
fields: fields{
|
2019-01-11 17:51:15 +00:00
|
|
|
OrganizationService: &mock.OrganizationService{
|
|
|
|
FindOrganizationByIDF: func(ctx context.Context, id platform.ID) (*platform.Organization, error) {
|
|
|
|
return &platform.Organization{
|
|
|
|
ID: platformtesting.MustIDBase16("0000000000000211"),
|
|
|
|
Name: "org1",
|
|
|
|
}, nil
|
|
|
|
},
|
|
|
|
},
|
|
|
|
BucketService: &mock.BucketService{
|
|
|
|
FindBucketByIDFn: func(ctx context.Context, id platform.ID) (*platform.Bucket, error) {
|
|
|
|
return &platform.Bucket{
|
|
|
|
ID: platformtesting.MustIDBase16("0000000000000212"),
|
|
|
|
Name: "bucket1",
|
|
|
|
}, nil
|
|
|
|
},
|
|
|
|
},
|
|
|
|
ScraperTargetStoreService: &mock.ScraperTargetStoreService{
|
2019-01-18 20:46:37 +00:00
|
|
|
UpdateTargetF: func(ctx context.Context, upd *platform.ScraperTarget, userID platform.ID) (*platform.ScraperTarget, error) {
|
2019-01-08 14:07:52 +00:00
|
|
|
return nil, &platform.Error{
|
|
|
|
Code: platform.ENotFound,
|
|
|
|
Msg: platform.ErrScraperTargetNotFound,
|
|
|
|
}
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
args: args{
|
|
|
|
id: targetOneIDString,
|
|
|
|
update: &platform.ScraperTarget{
|
2019-01-10 17:39:37 +00:00
|
|
|
ID: targetOneID,
|
|
|
|
Name: "name",
|
|
|
|
BucketID: platformtesting.MustIDBase16("0000000000000212"),
|
|
|
|
Type: platform.PrometheusScraperType,
|
|
|
|
URL: "www.example.url",
|
|
|
|
OrgID: platformtesting.MustIDBase16("0000000000000211"),
|
2019-01-08 14:07:52 +00:00
|
|
|
},
|
|
|
|
},
|
|
|
|
wants: wants{
|
|
|
|
statusCode: http.StatusNotFound,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
2019-01-07 18:33:49 +00:00
|
|
|
|
2019-01-08 14:07:52 +00:00
|
|
|
for _, tt := range tests {
|
|
|
|
t.Run(tt.name, func(t *testing.T) {
|
2019-01-23 17:11:06 +00:00
|
|
|
h := NewScraperHandler(
|
|
|
|
logger.New(os.Stdout),
|
|
|
|
mock.NewUserService(),
|
|
|
|
&mock.UserResourceMappingService{},
|
|
|
|
mock.NewLabelService(),
|
|
|
|
tt.fields.ScraperTargetStoreService,
|
|
|
|
tt.fields.BucketService,
|
|
|
|
tt.fields.OrganizationService,
|
|
|
|
)
|
2019-01-08 14:07:52 +00:00
|
|
|
|
|
|
|
var err error
|
|
|
|
st := make([]byte, 0)
|
|
|
|
if tt.args.update != nil {
|
|
|
|
st, err = json.Marshal(*tt.args.update)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("failed to unmarshal scraper target: %v", err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
r := httptest.NewRequest("GET", "http://any.tld", bytes.NewReader(st))
|
|
|
|
|
|
|
|
r = r.WithContext(context.WithValue(
|
|
|
|
context.Background(),
|
|
|
|
httprouter.ParamsKey,
|
|
|
|
httprouter.Params{
|
|
|
|
{
|
|
|
|
Key: "id",
|
|
|
|
Value: tt.args.id,
|
|
|
|
},
|
|
|
|
}))
|
2019-01-18 20:46:37 +00:00
|
|
|
r = r.WithContext(platcontext.SetAuthorizer(r.Context(), &platform.Authorization{}))
|
2019-01-08 14:07:52 +00:00
|
|
|
w := httptest.NewRecorder()
|
|
|
|
|
|
|
|
h.handlePatchScraperTarget(w, r)
|
|
|
|
|
|
|
|
res := w.Result()
|
|
|
|
content := res.Header.Get("Content-Type")
|
|
|
|
body, _ := ioutil.ReadAll(res.Body)
|
|
|
|
|
|
|
|
if res.StatusCode != tt.wants.statusCode {
|
|
|
|
t.Errorf("%q. handlePatchScraperTarget() = %v, want %v", tt.name, res.StatusCode, tt.wants.statusCode)
|
|
|
|
}
|
|
|
|
if tt.wants.contentType != "" && content != tt.wants.contentType {
|
|
|
|
t.Errorf("%q. handlePatchScraperTarget() = %v, want %v", tt.name, content, tt.wants.contentType)
|
|
|
|
}
|
|
|
|
if eq, diff, _ := jsonEqual(string(body), tt.wants.body); tt.wants.body != "" && !eq {
|
|
|
|
t.Errorf("%q. handlePatchScraperTarget() = ***%s***", tt.name, diff)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
2019-01-07 18:33:49 +00:00
|
|
|
}
|
|
|
|
|
2018-12-17 14:07:38 +00:00
|
|
|
func initScraperService(f platformtesting.TargetFields, t *testing.T) (platform.ScraperTargetStoreService, string, func()) {
|
2018-10-12 01:06:43 +00:00
|
|
|
t.Helper()
|
|
|
|
svc := inmem.NewService()
|
|
|
|
svc.IDGenerator = f.IDGenerator
|
|
|
|
|
|
|
|
ctx := context.Background()
|
|
|
|
for _, target := range f.Targets {
|
|
|
|
if err := svc.PutTarget(ctx, target); err != nil {
|
|
|
|
t.Fatalf("failed to populate scraper targets")
|
|
|
|
}
|
|
|
|
}
|
2019-01-18 20:46:37 +00:00
|
|
|
for _, m := range f.UserResourceMappings {
|
|
|
|
if err := svc.PutUserResourceMapping(ctx, m); err != nil {
|
|
|
|
t.Fatalf("failed to populate user resource mapping")
|
|
|
|
}
|
|
|
|
}
|
2018-10-12 01:06:43 +00:00
|
|
|
|
2019-01-23 17:11:06 +00:00
|
|
|
handler := NewScraperHandler(
|
|
|
|
logger.New(os.Stdout),
|
|
|
|
mock.NewUserService(),
|
|
|
|
&mock.UserResourceMappingService{},
|
|
|
|
mock.NewLabelService(),
|
|
|
|
svc,
|
|
|
|
&mock.BucketService{
|
|
|
|
FindBucketByIDFn: func(ctx context.Context, id platform.ID) (*platform.Bucket, error) {
|
|
|
|
return &platform.Bucket{
|
|
|
|
ID: id,
|
|
|
|
Name: "bucket1",
|
|
|
|
}, nil
|
|
|
|
},
|
2019-01-11 17:51:15 +00:00
|
|
|
},
|
2019-01-23 17:11:06 +00:00
|
|
|
&mock.OrganizationService{
|
|
|
|
FindOrganizationByIDF: func(ctx context.Context, id platform.ID) (*platform.Organization, error) {
|
|
|
|
return &platform.Organization{
|
|
|
|
ID: id,
|
|
|
|
Name: "org1",
|
|
|
|
}, nil
|
|
|
|
},
|
2019-01-11 17:51:15 +00:00
|
|
|
},
|
2019-01-23 17:11:06 +00:00
|
|
|
)
|
2019-01-18 20:46:37 +00:00
|
|
|
userID, _ := platform.IDFromString("020f755c3c082002")
|
2019-01-23 17:11:06 +00:00
|
|
|
server := httptest.NewServer(httpMock.NewAuthMiddlewareHandler(
|
2019-01-18 20:46:37 +00:00
|
|
|
handler, &platform.Authorization{
|
|
|
|
UserID: *userID,
|
|
|
|
Token: "tok",
|
|
|
|
},
|
|
|
|
))
|
|
|
|
client := struct {
|
|
|
|
platform.UserResourceMappingService
|
|
|
|
ScraperService
|
|
|
|
}{
|
|
|
|
UserResourceMappingService: svc,
|
|
|
|
ScraperService: ScraperService{
|
|
|
|
Token: "tok",
|
|
|
|
Addr: server.URL,
|
|
|
|
OpPrefix: inmem.OpPrefix,
|
|
|
|
},
|
2018-10-12 01:06:43 +00:00
|
|
|
}
|
|
|
|
done := server.Close
|
|
|
|
|
2018-12-17 14:07:38 +00:00
|
|
|
return &client, inmem.OpPrefix, done
|
2018-10-12 01:06:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func TestScraperService(t *testing.T) {
|
|
|
|
platformtesting.ScraperService(initScraperService, t)
|
|
|
|
}
|