63383ef545
* Build out barebones Traces editor - Add Traces query type and operation ID prop to query type - Add necessary header types - Update resource picker to appropriately work with traces query type - Build out TracesQueryEditor component - Include logic to retrieve operationId's for AI Workspaces - Add backend route mapping - Update macro to use timestamp as default time field for traces * AzureMonitor: Traces - Response parsing (#65442) * Update FormatAsField component - Add trace ResultFormat type - Generalise FormatAsField component - Add component to TracesQueryEditor - Remove duplicate code in setQueryValue * Add custom filter function to improve performance * Add basic conversion for logs to trace - Add serviceTags converter - Pass through required parameters (queryType and resultFormat) - Appropriately set visualisation * Update parsing to also fill trace tags - Add constant values for each table schema (include legacy mapping for now if needed) - Add constant for list of table tags - Set the foundation for dynamic query building - Update query to build tags value - Appropriately set operationName - Update tagsConverter to filter empty values * Fix lint and test issues * AzureMonitor: Traces - Data links (#65566) * Add portal link for traces - Pull out necessary values (itemId and itemType) - Appropriately construct - Fix ordering * Set default format as value - Also set default visualisation * Fix event schema * Set default formatAsField value * Include logs link on traces results - Adapt config links to allow custom title to be set * Correctly set operationId for query * Update backend types - Include OperationID in query - Pass forward datasource name and UID * Ensure setTime doesn't consistently get called if operationID is defined * Add explore link - Update util functions to allow setting custom datalinks * Fix tests * AzureMonitor: Traces - Query and Editor updates (#66076) * Add initial query - Will query the resource as soon as a resource has been selected - Updates the data links for the query without operationId - Remove initial operationId query and timeRange dependency - Update query building * Add entirely separate traces query property - Update shared types (also including future types for Azure traces) - Update backend log analytics datasource to accept both azureLogAnalytics and azureTraces queries - Update backend specific types - Update frontend datasource for new properties - Update mock query * Update FormatAsField to be entirely generic * Update query building to be done in backend - Add required mappings in backend - Update frontend querying * Fix query and explore data link * Add trace type selection * Better method for setting explore link * Fix operationId updating * Run go mod tidy * Unnecessary changes * Fix tests * AzureMonitor: Traces - Add correlation API support (#65855) Add correlation API support - Add necessary types - Add correlation API request when conditions are met - Update query * Fix property from merge * AzureMonitor: Traces - Filtering (#66303) * Add initial query - Will query the resource as soon as a resource has been selected - Updates the data links for the query without operationId - Remove initial operationId query and timeRange dependency - Update query building * Add entirely separate traces query property - Update shared types (also including future types for Azure traces) - Update backend log analytics datasource to accept both azureLogAnalytics and azureTraces queries - Update backend specific types - Update frontend datasource for new properties - Update mock query * Update FormatAsField to be entirely generic * Update query building to be done in backend - Add required mappings in backend - Update frontend querying * Fix query and explore data link * Add trace type selection * Better method for setting explore link * Fix operationId updating * Run go mod tidy * Unnecessary changes * Fix tests * Start building out Filters component - Configure component to query for Filter property values when a filter property is set - Add setFilters function - Add typing to tablesSchema - Use component in TracesQueryEditor * Update Filters - Asynchronously pull property options - Setup list of Filter components * Update filters component - Remove unused imports - Have local filters state and query filters - Correctly set filters values - Don't update query every time a filter property changes (not performant) * Update properties query - Use current timeRange - Get count to provide informative labels * Reset map when time changes * Add operation selection * Reset filters when property changes * Appropriate label name for empty values * Add filtering to query * Update filter components - Fix rendering issue - Correctly compare and update timeRange - Split out files for simplicity * Add checkbox option to multiselect - Add custom option component - Correctly call onChange - Add variableOptionGroup for template variable selection * Fix adding template vars * Improve labels and refresh labels on query prop changes * AzureMonitor: Traces - Testing (#66474) * Select ds for template variable interpolation * Update az logs ds tests - Add templateVariables test - Add filter test - Update mock - Remove anys * Update QueryEditor test - Update mocks with timeSrv for log analytics datasource - Fix query mock - Use appropriate and consistent selectors * Add TracesQueryEditor test - Update resourcePickerRows mock to include app insights resources - Remove comments and extra new line * Add FormatAsField test - Remove unneeded condition * Update resourcePicker utils test * Don't hide selected options in filters * Fix multi-selection on filters * Add TraceTypeField test - Add test file - Update selectors (remove copy/paste mistake) - Update placeholder text for select and add label * Add basic filters test * Begin filters test * Update filters test * Add final tests and simplify/generalise addFilter helper * Minor update to datasource test * Update macros test * Update selectors in tests * Add response-table-frame tests * Add datasource tests - Use sorting where JSON models are inconsistent - Update filters clause - Dedupe tags - Correct operationId conditions * Don't set a default value for blurInputOnSelect * Simplify datasource test * Update to use CheckGoldenJSON utils - Update with generated frame files - Remove redundant expected frame code - Update all usages * Fix lint * AzureMonitor: Traces feedback (#67292) * Filter traces if the visualisation is set to trace - Update build query logic - Added additional test cases - Return an error if the traces type is set by itself with the trace visualisation - Add descriptions to event types - Update tests * Fix bug for error displaying traces * Update mappings and add error field - Update tests - Remove unnecessary comments * Switch location of Operation ID field * Re-order fields * Update link title * Update label for event type selection * Update correct link title * Update logs datalink to link to Azure Logs in explore * Fix lint
202 lines
5.7 KiB
Go
202 lines
5.7 KiB
Go
package resourcegraph
|
|
|
|
import (
|
|
"context"
|
|
"io"
|
|
"net/http"
|
|
"strings"
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/google/go-cmp/cmp"
|
|
"github.com/google/go-cmp/cmp/cmpopts"
|
|
"github.com/grafana/grafana-azure-sdk-go/azsettings"
|
|
"github.com/grafana/grafana-plugin-sdk-go/backend"
|
|
"github.com/grafana/grafana-plugin-sdk-go/data"
|
|
|
|
"github.com/stretchr/testify/assert"
|
|
"github.com/stretchr/testify/require"
|
|
|
|
"github.com/grafana/grafana/pkg/components/simplejson"
|
|
"github.com/grafana/grafana/pkg/infra/log"
|
|
"github.com/grafana/grafana/pkg/tsdb/azuremonitor/loganalytics"
|
|
"github.com/grafana/grafana/pkg/tsdb/azuremonitor/types"
|
|
)
|
|
|
|
var logger = log.New("test")
|
|
|
|
func TestBuildingAzureResourceGraphQueries(t *testing.T) {
|
|
datasource := &AzureResourceGraphDatasource{}
|
|
fromStart := time.Date(2018, 3, 15, 13, 0, 0, 0, time.UTC).In(time.Local)
|
|
|
|
tests := []struct {
|
|
name string
|
|
queryModel []backend.DataQuery
|
|
timeRange backend.TimeRange
|
|
azureResourceGraphQueries []*AzureResourceGraphQuery
|
|
Err require.ErrorAssertionFunc
|
|
}{
|
|
{
|
|
name: "Query with macros should be interpolated",
|
|
timeRange: backend.TimeRange{
|
|
From: fromStart,
|
|
To: fromStart.Add(34 * time.Minute),
|
|
},
|
|
queryModel: []backend.DataQuery{
|
|
{
|
|
JSON: []byte(`{
|
|
"queryType": "Azure Resource Graph",
|
|
"azureResourceGraph": {
|
|
"query": "resources | where $__contains(name,'res1','res2')",
|
|
"resultFormat": "table"
|
|
}
|
|
}`),
|
|
RefID: "A",
|
|
},
|
|
},
|
|
azureResourceGraphQueries: []*AzureResourceGraphQuery{
|
|
{
|
|
RefID: "A",
|
|
ResultFormat: "table",
|
|
URL: "",
|
|
JSON: []byte(`{
|
|
"queryType": "Azure Resource Graph",
|
|
"azureResourceGraph": {
|
|
"query": "resources | where $__contains(name,'res1','res2')",
|
|
"resultFormat": "table"
|
|
}
|
|
}`),
|
|
InterpolatedQuery: "resources | where ['name'] in ('res1','res2')",
|
|
},
|
|
},
|
|
Err: require.NoError,
|
|
},
|
|
}
|
|
|
|
for _, tt := range tests {
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
queries, err := datasource.buildQueries(logger, tt.queryModel, types.DatasourceInfo{})
|
|
tt.Err(t, err)
|
|
if diff := cmp.Diff(tt.azureResourceGraphQueries, queries, cmpopts.IgnoreUnexported(simplejson.Json{})); diff != "" {
|
|
t.Errorf("Result mismatch (-want +got):\n%s", diff)
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestAzureResourceGraphCreateRequest(t *testing.T) {
|
|
ctx := context.Background()
|
|
url := "http://ds"
|
|
|
|
tests := []struct {
|
|
name string
|
|
expectedURL string
|
|
expectedHeaders http.Header
|
|
Err require.ErrorAssertionFunc
|
|
}{
|
|
{
|
|
name: "creates a request",
|
|
expectedURL: "http://ds/",
|
|
expectedHeaders: http.Header{
|
|
"Content-Type": []string{"application/json"},
|
|
},
|
|
Err: require.NoError,
|
|
},
|
|
}
|
|
|
|
for _, tt := range tests {
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
ds := AzureResourceGraphDatasource{}
|
|
req, err := ds.createRequest(ctx, logger, []byte{}, url)
|
|
tt.Err(t, err)
|
|
if req.URL.String() != tt.expectedURL {
|
|
t.Errorf("Expecting %s, got %s", tt.expectedURL, req.URL.String())
|
|
}
|
|
if !cmp.Equal(req.Header, tt.expectedHeaders) {
|
|
t.Errorf("Unexpected HTTP headers: %v", cmp.Diff(req.Header, tt.expectedHeaders))
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestAddConfigData(t *testing.T) {
|
|
field := data.Field{}
|
|
dataLink := data.DataLink{Title: "View in Azure Portal", TargetBlank: true, URL: "http://ds"}
|
|
frame := data.Frame{
|
|
Fields: []*data.Field{&field},
|
|
}
|
|
frameWithLink := loganalytics.AddConfigLinks(frame, "http://ds", nil)
|
|
expectedFrameWithLink := data.Frame{
|
|
Fields: []*data.Field{
|
|
{
|
|
Config: &data.FieldConfig{
|
|
Links: []data.DataLink{dataLink},
|
|
},
|
|
},
|
|
},
|
|
}
|
|
if !cmp.Equal(frameWithLink, expectedFrameWithLink, data.FrameTestCompareOptions()...) {
|
|
t.Errorf("unexpepcted frame: %v", cmp.Diff(frameWithLink, expectedFrameWithLink, data.FrameTestCompareOptions()...))
|
|
}
|
|
}
|
|
|
|
func TestGetAzurePortalUrl(t *testing.T) {
|
|
clouds := []string{azsettings.AzurePublic, azsettings.AzureChina, azsettings.AzureUSGovernment}
|
|
expectedAzurePortalUrl := map[string]interface{}{
|
|
azsettings.AzurePublic: "https://portal.azure.com",
|
|
azsettings.AzureChina: "https://portal.azure.cn",
|
|
azsettings.AzureUSGovernment: "https://portal.azure.us",
|
|
}
|
|
|
|
for _, cloud := range clouds {
|
|
azurePortalUrl, err := loganalytics.GetAzurePortalUrl(cloud)
|
|
if err != nil {
|
|
t.Errorf("The cloud not supported")
|
|
}
|
|
assert.Equal(t, expectedAzurePortalUrl[cloud], azurePortalUrl)
|
|
}
|
|
}
|
|
|
|
func TestUnmarshalResponse400(t *testing.T) {
|
|
datasource := &AzureResourceGraphDatasource{}
|
|
res, err := datasource.unmarshalResponse(logger, &http.Response{
|
|
StatusCode: 400,
|
|
Status: "400 Bad Request",
|
|
Body: io.NopCloser(strings.NewReader(("Azure Error Message"))),
|
|
})
|
|
|
|
expectedErrMsg := "400 Bad Request. Azure Resource Graph error: Azure Error Message"
|
|
|
|
assert.Equal(t, expectedErrMsg, err.Error())
|
|
assert.Empty(t, res)
|
|
}
|
|
|
|
func TestUnmarshalResponse200Invalid(t *testing.T) {
|
|
datasource := &AzureResourceGraphDatasource{}
|
|
res, err := datasource.unmarshalResponse(logger, &http.Response{
|
|
StatusCode: 200,
|
|
Status: "OK",
|
|
Body: io.NopCloser(strings.NewReader(("Azure Data"))),
|
|
})
|
|
|
|
expectedRes := AzureResourceGraphResponse{}
|
|
expectedErr := "invalid character 'A' looking for beginning of value"
|
|
|
|
assert.Equal(t, expectedErr, err.Error())
|
|
assert.Equal(t, expectedRes, res)
|
|
}
|
|
|
|
func TestUnmarshalResponse200(t *testing.T) {
|
|
datasource := &AzureResourceGraphDatasource{}
|
|
res, err2 := datasource.unmarshalResponse(logger, &http.Response{
|
|
StatusCode: 200,
|
|
Status: "OK",
|
|
Body: io.NopCloser(strings.NewReader("{}")),
|
|
})
|
|
|
|
expectedRes := AzureResourceGraphResponse{}
|
|
|
|
assert.NoError(t, err2)
|
|
assert.Equal(t, expectedRes, res)
|
|
}
|