mirror of
https://github.com/grafana/grafana.git
synced 2024-12-02 05:29:42 -06:00
d7862c50b8
* first pass * add instance manager * fix tests * remove dead code * unexport fields * cleanup * remove ds instance from executor * cleanup * inline im * remove old func * get error working * unexport field * let fe do its magic * fix channel name * revert some tsdb changes * fix annotations * cleanup
563 lines
14 KiB
Go
563 lines
14 KiB
Go
package cloudwatch
|
|
|
|
import (
|
|
"context"
|
|
"encoding/json"
|
|
"fmt"
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/aws/aws-sdk-go/aws"
|
|
"github.com/aws/aws-sdk-go/aws/session"
|
|
"github.com/aws/aws-sdk-go/service/cloudwatchlogs"
|
|
"github.com/aws/aws-sdk-go/service/cloudwatchlogs/cloudwatchlogsiface"
|
|
"github.com/grafana/grafana-plugin-sdk-go/backend"
|
|
"github.com/grafana/grafana-plugin-sdk-go/backend/datasource"
|
|
"github.com/grafana/grafana-plugin-sdk-go/backend/instancemgmt"
|
|
"github.com/grafana/grafana-plugin-sdk-go/data"
|
|
|
|
"github.com/stretchr/testify/assert"
|
|
"github.com/stretchr/testify/require"
|
|
)
|
|
|
|
func TestQuery_DescribeLogGroups(t *testing.T) {
|
|
origNewCWLogsClient := NewCWLogsClient
|
|
t.Cleanup(func() {
|
|
NewCWLogsClient = origNewCWLogsClient
|
|
})
|
|
|
|
var cli FakeCWLogsClient
|
|
|
|
NewCWLogsClient = func(sess *session.Session) cloudwatchlogsiface.CloudWatchLogsAPI {
|
|
return cli
|
|
}
|
|
|
|
t.Run("Empty log group name prefix", func(t *testing.T) {
|
|
cli = FakeCWLogsClient{
|
|
logGroups: cloudwatchlogs.DescribeLogGroupsOutput{
|
|
LogGroups: []*cloudwatchlogs.LogGroup{
|
|
{
|
|
LogGroupName: aws.String("group_a"),
|
|
},
|
|
{
|
|
LogGroupName: aws.String("group_b"),
|
|
},
|
|
{
|
|
LogGroupName: aws.String("group_c"),
|
|
},
|
|
},
|
|
},
|
|
}
|
|
|
|
im := datasource.NewInstanceManager(func(s backend.DataSourceInstanceSettings) (instancemgmt.Instance, error) {
|
|
return datasourceInfo{}, nil
|
|
})
|
|
|
|
executor := newExecutor(nil, im, newTestConfig(), fakeSessionCache{})
|
|
resp, err := executor.QueryData(context.Background(), &backend.QueryDataRequest{
|
|
PluginContext: backend.PluginContext{
|
|
DataSourceInstanceSettings: &backend.DataSourceInstanceSettings{},
|
|
},
|
|
Queries: []backend.DataQuery{
|
|
{
|
|
JSON: json.RawMessage(`{
|
|
"type": "logAction",
|
|
"subtype": "DescribeLogGroups",
|
|
"limit": 50
|
|
}`),
|
|
},
|
|
},
|
|
})
|
|
require.NoError(t, err)
|
|
require.NotNil(t, resp)
|
|
|
|
assert.Equal(t, &backend.QueryDataResponse{Responses: backend.Responses{
|
|
"": backend.DataResponse{
|
|
Frames: data.Frames{
|
|
&data.Frame{
|
|
Name: "logGroups",
|
|
Fields: []*data.Field{
|
|
data.NewField("logGroupName", nil, []*string{
|
|
aws.String("group_a"), aws.String("group_b"), aws.String("group_c"),
|
|
}),
|
|
},
|
|
Meta: &data.FrameMeta{
|
|
PreferredVisualization: "logs",
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
}, resp)
|
|
})
|
|
|
|
t.Run("Non-empty log group name prefix", func(t *testing.T) {
|
|
cli = FakeCWLogsClient{
|
|
logGroups: cloudwatchlogs.DescribeLogGroupsOutput{
|
|
LogGroups: []*cloudwatchlogs.LogGroup{
|
|
{
|
|
LogGroupName: aws.String("group_a"),
|
|
},
|
|
{
|
|
LogGroupName: aws.String("group_b"),
|
|
},
|
|
{
|
|
LogGroupName: aws.String("group_c"),
|
|
},
|
|
},
|
|
},
|
|
}
|
|
|
|
im := datasource.NewInstanceManager(func(s backend.DataSourceInstanceSettings) (instancemgmt.Instance, error) {
|
|
return datasourceInfo{}, nil
|
|
})
|
|
|
|
executor := newExecutor(nil, im, newTestConfig(), fakeSessionCache{})
|
|
resp, err := executor.QueryData(context.Background(), &backend.QueryDataRequest{
|
|
PluginContext: backend.PluginContext{
|
|
DataSourceInstanceSettings: &backend.DataSourceInstanceSettings{},
|
|
},
|
|
Queries: []backend.DataQuery{
|
|
{
|
|
JSON: json.RawMessage(`{
|
|
"type": "logAction",
|
|
"subtype": "DescribeLogGroups",
|
|
"limit": 50,
|
|
"region": "default",
|
|
"logGroupNamePrefix": "g"
|
|
}`),
|
|
},
|
|
},
|
|
})
|
|
require.NoError(t, err)
|
|
require.NotNil(t, resp)
|
|
|
|
assert.Equal(t, &backend.QueryDataResponse{Responses: backend.Responses{
|
|
"": backend.DataResponse{
|
|
Frames: data.Frames{
|
|
&data.Frame{
|
|
Name: "logGroups",
|
|
Fields: []*data.Field{
|
|
data.NewField("logGroupName", nil, []*string{
|
|
aws.String("group_a"), aws.String("group_b"), aws.String("group_c"),
|
|
}),
|
|
},
|
|
Meta: &data.FrameMeta{
|
|
PreferredVisualization: "logs",
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
}, resp)
|
|
})
|
|
}
|
|
|
|
func TestQuery_GetLogGroupFields(t *testing.T) {
|
|
origNewCWLogsClient := NewCWLogsClient
|
|
t.Cleanup(func() {
|
|
NewCWLogsClient = origNewCWLogsClient
|
|
})
|
|
|
|
var cli FakeCWLogsClient
|
|
|
|
NewCWLogsClient = func(sess *session.Session) cloudwatchlogsiface.CloudWatchLogsAPI {
|
|
return cli
|
|
}
|
|
|
|
cli = FakeCWLogsClient{
|
|
logGroupFields: cloudwatchlogs.GetLogGroupFieldsOutput{
|
|
LogGroupFields: []*cloudwatchlogs.LogGroupField{
|
|
{
|
|
Name: aws.String("field_a"),
|
|
Percent: aws.Int64(100),
|
|
},
|
|
{
|
|
Name: aws.String("field_b"),
|
|
Percent: aws.Int64(30),
|
|
},
|
|
{
|
|
Name: aws.String("field_c"),
|
|
Percent: aws.Int64(55),
|
|
},
|
|
},
|
|
},
|
|
}
|
|
|
|
const refID = "A"
|
|
|
|
im := datasource.NewInstanceManager(func(s backend.DataSourceInstanceSettings) (instancemgmt.Instance, error) {
|
|
return datasourceInfo{}, nil
|
|
})
|
|
|
|
executor := newExecutor(nil, im, newTestConfig(), fakeSessionCache{})
|
|
resp, err := executor.QueryData(context.Background(), &backend.QueryDataRequest{
|
|
PluginContext: backend.PluginContext{
|
|
DataSourceInstanceSettings: &backend.DataSourceInstanceSettings{},
|
|
},
|
|
Queries: []backend.DataQuery{
|
|
{
|
|
RefID: refID,
|
|
JSON: json.RawMessage(`{
|
|
"type": "logAction",
|
|
"subtype": "GetLogGroupFields",
|
|
"logGroupName": "group_a",
|
|
"limit": 50
|
|
}`),
|
|
},
|
|
},
|
|
})
|
|
require.NoError(t, err)
|
|
require.NotNil(t, resp)
|
|
|
|
expFrame := &data.Frame{
|
|
Name: refID,
|
|
Fields: []*data.Field{
|
|
data.NewField("name", nil, []*string{
|
|
aws.String("field_a"), aws.String("field_b"), aws.String("field_c"),
|
|
}),
|
|
data.NewField("percent", nil, []*int64{
|
|
aws.Int64(100), aws.Int64(30), aws.Int64(55),
|
|
}),
|
|
},
|
|
Meta: &data.FrameMeta{
|
|
PreferredVisualization: "logs",
|
|
},
|
|
}
|
|
expFrame.RefID = refID
|
|
assert.Equal(t, &backend.QueryDataResponse{Responses: backend.Responses{
|
|
refID: backend.DataResponse{
|
|
Frames: data.Frames{expFrame},
|
|
},
|
|
},
|
|
}, resp)
|
|
}
|
|
|
|
func TestQuery_StartQuery(t *testing.T) {
|
|
origNewCWLogsClient := NewCWLogsClient
|
|
t.Cleanup(func() {
|
|
NewCWLogsClient = origNewCWLogsClient
|
|
})
|
|
|
|
var cli FakeCWLogsClient
|
|
|
|
NewCWLogsClient = func(sess *session.Session) cloudwatchlogsiface.CloudWatchLogsAPI {
|
|
return cli
|
|
}
|
|
|
|
t.Run("invalid time range", func(t *testing.T) {
|
|
cli = FakeCWLogsClient{
|
|
logGroupFields: cloudwatchlogs.GetLogGroupFieldsOutput{
|
|
LogGroupFields: []*cloudwatchlogs.LogGroupField{
|
|
{
|
|
Name: aws.String("field_a"),
|
|
Percent: aws.Int64(100),
|
|
},
|
|
{
|
|
Name: aws.String("field_b"),
|
|
Percent: aws.Int64(30),
|
|
},
|
|
{
|
|
Name: aws.String("field_c"),
|
|
Percent: aws.Int64(55),
|
|
},
|
|
},
|
|
},
|
|
}
|
|
|
|
timeRange := backend.TimeRange{
|
|
From: time.Unix(1584873443, 0),
|
|
To: time.Unix(1584700643, 0),
|
|
}
|
|
|
|
im := datasource.NewInstanceManager(func(s backend.DataSourceInstanceSettings) (instancemgmt.Instance, error) {
|
|
return datasourceInfo{}, nil
|
|
})
|
|
|
|
executor := newExecutor(nil, im, newTestConfig(), fakeSessionCache{})
|
|
_, err := executor.QueryData(context.Background(), &backend.QueryDataRequest{
|
|
PluginContext: backend.PluginContext{
|
|
DataSourceInstanceSettings: &backend.DataSourceInstanceSettings{},
|
|
},
|
|
Queries: []backend.DataQuery{
|
|
{
|
|
TimeRange: timeRange,
|
|
JSON: json.RawMessage(`{
|
|
"type": "logAction",
|
|
"subtype": "StartQuery",
|
|
"limit": 50,
|
|
"region": "default",
|
|
"queryString": "fields @message"
|
|
}`),
|
|
},
|
|
},
|
|
})
|
|
require.Error(t, err)
|
|
|
|
assert.Equal(t, fmt.Errorf("invalid time range: start time must be before end time"), err)
|
|
})
|
|
|
|
t.Run("valid time range", func(t *testing.T) {
|
|
const refID = "A"
|
|
cli = FakeCWLogsClient{
|
|
logGroupFields: cloudwatchlogs.GetLogGroupFieldsOutput{
|
|
LogGroupFields: []*cloudwatchlogs.LogGroupField{
|
|
{
|
|
Name: aws.String("field_a"),
|
|
Percent: aws.Int64(100),
|
|
},
|
|
{
|
|
Name: aws.String("field_b"),
|
|
Percent: aws.Int64(30),
|
|
},
|
|
{
|
|
Name: aws.String("field_c"),
|
|
Percent: aws.Int64(55),
|
|
},
|
|
},
|
|
},
|
|
}
|
|
|
|
timeRange := backend.TimeRange{
|
|
From: time.Unix(1584700643000, 0),
|
|
To: time.Unix(1584873443000, 0),
|
|
}
|
|
|
|
im := datasource.NewInstanceManager(func(s backend.DataSourceInstanceSettings) (instancemgmt.Instance, error) {
|
|
return datasourceInfo{}, nil
|
|
})
|
|
|
|
executor := newExecutor(nil, im, newTestConfig(), fakeSessionCache{})
|
|
resp, err := executor.QueryData(context.Background(), &backend.QueryDataRequest{
|
|
PluginContext: backend.PluginContext{
|
|
DataSourceInstanceSettings: &backend.DataSourceInstanceSettings{},
|
|
},
|
|
Queries: []backend.DataQuery{
|
|
{
|
|
RefID: refID,
|
|
TimeRange: timeRange,
|
|
JSON: json.RawMessage(`{
|
|
"type": "logAction",
|
|
"subtype": "StartQuery",
|
|
"limit": 50,
|
|
"region": "default",
|
|
"queryString": "fields @message"
|
|
}`),
|
|
},
|
|
},
|
|
})
|
|
require.NoError(t, err)
|
|
|
|
expFrame := data.NewFrame(
|
|
refID,
|
|
data.NewField("queryId", nil, []string{"abcd-efgh-ijkl-mnop"}),
|
|
)
|
|
expFrame.RefID = refID
|
|
expFrame.Meta = &data.FrameMeta{
|
|
Custom: map[string]interface{}{
|
|
"Region": "default",
|
|
},
|
|
PreferredVisualization: "logs",
|
|
}
|
|
assert.Equal(t, &backend.QueryDataResponse{Responses: backend.Responses{
|
|
refID: {
|
|
Frames: data.Frames{expFrame},
|
|
},
|
|
},
|
|
}, resp)
|
|
})
|
|
}
|
|
|
|
func TestQuery_StopQuery(t *testing.T) {
|
|
origNewCWLogsClient := NewCWLogsClient
|
|
t.Cleanup(func() {
|
|
NewCWLogsClient = origNewCWLogsClient
|
|
})
|
|
|
|
var cli FakeCWLogsClient
|
|
|
|
NewCWLogsClient = func(sess *session.Session) cloudwatchlogsiface.CloudWatchLogsAPI {
|
|
return cli
|
|
}
|
|
|
|
cli = FakeCWLogsClient{
|
|
logGroupFields: cloudwatchlogs.GetLogGroupFieldsOutput{
|
|
LogGroupFields: []*cloudwatchlogs.LogGroupField{
|
|
{
|
|
Name: aws.String("field_a"),
|
|
Percent: aws.Int64(100),
|
|
},
|
|
{
|
|
Name: aws.String("field_b"),
|
|
Percent: aws.Int64(30),
|
|
},
|
|
{
|
|
Name: aws.String("field_c"),
|
|
Percent: aws.Int64(55),
|
|
},
|
|
},
|
|
},
|
|
}
|
|
|
|
im := datasource.NewInstanceManager(func(s backend.DataSourceInstanceSettings) (instancemgmt.Instance, error) {
|
|
return datasourceInfo{}, nil
|
|
})
|
|
|
|
timeRange := backend.TimeRange{
|
|
From: time.Unix(1584873443, 0),
|
|
To: time.Unix(1584700643, 0),
|
|
}
|
|
|
|
executor := newExecutor(nil, im, newTestConfig(), fakeSessionCache{})
|
|
resp, err := executor.QueryData(context.Background(), &backend.QueryDataRequest{
|
|
PluginContext: backend.PluginContext{
|
|
DataSourceInstanceSettings: &backend.DataSourceInstanceSettings{},
|
|
},
|
|
Queries: []backend.DataQuery{
|
|
{
|
|
TimeRange: timeRange,
|
|
JSON: json.RawMessage(`{
|
|
"type": "logAction",
|
|
"subtype": "StopQuery",
|
|
"queryId": "abcd-efgh-ijkl-mnop"
|
|
}`),
|
|
},
|
|
},
|
|
})
|
|
require.NoError(t, err)
|
|
|
|
expFrame := &data.Frame{
|
|
Name: "StopQueryResponse",
|
|
Fields: []*data.Field{
|
|
data.NewField("success", nil, []bool{true}),
|
|
},
|
|
Meta: &data.FrameMeta{
|
|
PreferredVisualization: "logs",
|
|
},
|
|
}
|
|
assert.Equal(t, &backend.QueryDataResponse{Responses: backend.Responses{
|
|
"": {
|
|
Frames: data.Frames{expFrame},
|
|
},
|
|
},
|
|
}, resp)
|
|
}
|
|
|
|
func TestQuery_GetQueryResults(t *testing.T) {
|
|
origNewCWLogsClient := NewCWLogsClient
|
|
t.Cleanup(func() {
|
|
NewCWLogsClient = origNewCWLogsClient
|
|
})
|
|
|
|
var cli FakeCWLogsClient
|
|
|
|
NewCWLogsClient = func(sess *session.Session) cloudwatchlogsiface.CloudWatchLogsAPI {
|
|
return cli
|
|
}
|
|
|
|
const refID = "A"
|
|
cli = FakeCWLogsClient{
|
|
queryResults: cloudwatchlogs.GetQueryResultsOutput{
|
|
Results: [][]*cloudwatchlogs.ResultField{
|
|
{
|
|
{
|
|
Field: aws.String("@timestamp"),
|
|
Value: aws.String("2020-03-20 10:37:23.000"),
|
|
},
|
|
{
|
|
Field: aws.String("field_b"),
|
|
Value: aws.String("b_1"),
|
|
},
|
|
{
|
|
Field: aws.String("@ptr"),
|
|
Value: aws.String("abcdefg"),
|
|
},
|
|
},
|
|
{
|
|
{
|
|
Field: aws.String("@timestamp"),
|
|
Value: aws.String("2020-03-20 10:40:43.000"),
|
|
},
|
|
{
|
|
Field: aws.String("field_b"),
|
|
Value: aws.String("b_2"),
|
|
},
|
|
{
|
|
Field: aws.String("@ptr"),
|
|
Value: aws.String("hijklmnop"),
|
|
},
|
|
},
|
|
},
|
|
Statistics: &cloudwatchlogs.QueryStatistics{
|
|
BytesScanned: aws.Float64(512),
|
|
RecordsMatched: aws.Float64(256),
|
|
RecordsScanned: aws.Float64(1024),
|
|
},
|
|
Status: aws.String("Complete"),
|
|
},
|
|
}
|
|
|
|
im := datasource.NewInstanceManager(func(s backend.DataSourceInstanceSettings) (instancemgmt.Instance, error) {
|
|
return datasourceInfo{}, nil
|
|
})
|
|
|
|
executor := newExecutor(nil, im, newTestConfig(), fakeSessionCache{})
|
|
resp, err := executor.QueryData(context.Background(), &backend.QueryDataRequest{
|
|
PluginContext: backend.PluginContext{
|
|
DataSourceInstanceSettings: &backend.DataSourceInstanceSettings{},
|
|
},
|
|
Queries: []backend.DataQuery{
|
|
{
|
|
RefID: refID,
|
|
JSON: json.RawMessage(`{
|
|
"type": "logAction",
|
|
"subtype": "GetQueryResults",
|
|
"queryId": "abcd-efgh-ijkl-mnop"
|
|
}`),
|
|
},
|
|
},
|
|
})
|
|
require.NoError(t, err)
|
|
|
|
time1, err := time.Parse("2006-01-02 15:04:05.000", "2020-03-20 10:37:23.000")
|
|
require.NoError(t, err)
|
|
time2, err := time.Parse("2006-01-02 15:04:05.000", "2020-03-20 10:40:43.000")
|
|
require.NoError(t, err)
|
|
expField1 := data.NewField("@timestamp", nil, []*time.Time{
|
|
aws.Time(time1), aws.Time(time2),
|
|
})
|
|
expField1.SetConfig(&data.FieldConfig{DisplayName: "Time"})
|
|
expField2 := data.NewField("field_b", nil, []*string{
|
|
aws.String("b_1"), aws.String("b_2"),
|
|
})
|
|
expFrame := data.NewFrame(refID, expField1, expField2)
|
|
expFrame.RefID = refID
|
|
expFrame.Meta = &data.FrameMeta{
|
|
Custom: map[string]interface{}{
|
|
"Status": "Complete",
|
|
},
|
|
Stats: []data.QueryStat{
|
|
{
|
|
FieldConfig: data.FieldConfig{DisplayName: "Bytes scanned"},
|
|
Value: 512,
|
|
},
|
|
{
|
|
FieldConfig: data.FieldConfig{DisplayName: "Records scanned"},
|
|
Value: 1024,
|
|
},
|
|
{
|
|
FieldConfig: data.FieldConfig{DisplayName: "Records matched"},
|
|
Value: 256,
|
|
},
|
|
},
|
|
PreferredVisualization: "logs",
|
|
}
|
|
|
|
assert.Equal(t, &backend.QueryDataResponse{Responses: backend.Responses{
|
|
refID: {
|
|
Frames: data.Frames{expFrame},
|
|
},
|
|
},
|
|
}, resp)
|
|
}
|