2019-10-23 17:09:04 +00:00
|
|
|
package pkger
|
|
|
|
|
|
|
|
import (
|
2019-11-07 00:45:00 +00:00
|
|
|
"path/filepath"
|
2019-11-14 00:24:05 +00:00
|
|
|
"strconv"
|
2019-11-07 00:45:00 +00:00
|
|
|
"strings"
|
2019-10-23 17:09:04 +00:00
|
|
|
"testing"
|
|
|
|
"time"
|
|
|
|
|
2019-10-30 21:13:42 +00:00
|
|
|
"github.com/influxdata/influxdb"
|
2019-10-23 17:09:04 +00:00
|
|
|
"github.com/stretchr/testify/assert"
|
|
|
|
"github.com/stretchr/testify/require"
|
|
|
|
)
|
|
|
|
|
|
|
|
func TestParse(t *testing.T) {
|
2019-10-26 02:11:47 +00:00
|
|
|
t.Run("pkg has all necessary metadata", func(t *testing.T) {
|
2019-10-23 17:09:04 +00:00
|
|
|
t.Run("has valid metadata and at least 1 resource", func(t *testing.T) {
|
|
|
|
testfileRunner(t, "testdata/bucket", nil)
|
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("malformed required metadata", func(t *testing.T) {
|
2019-11-06 22:41:06 +00:00
|
|
|
tests := []testPkgResourceError{
|
2019-10-23 17:09:04 +00:00
|
|
|
{
|
|
|
|
name: "missing apiVersion",
|
2019-11-06 22:41:06 +00:00
|
|
|
pkgStr: `kind: Package
|
2019-10-23 17:09:04 +00:00
|
|
|
meta:
|
|
|
|
pkgName: first_bucket_package
|
|
|
|
pkgVersion: 1
|
|
|
|
spec:
|
|
|
|
resources:
|
|
|
|
- kind: Bucket
|
|
|
|
name: buck_1
|
|
|
|
retention_period: 1h
|
|
|
|
`,
|
2019-11-06 22:41:06 +00:00
|
|
|
valFields: []string{"apiVersion"},
|
2019-10-23 17:09:04 +00:00
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "apiVersion is invalid version",
|
2019-11-06 22:41:06 +00:00
|
|
|
pkgStr: `apiVersion: 222.2 #invalid apiVersion
|
2019-10-23 17:09:04 +00:00
|
|
|
kind: Package
|
|
|
|
meta:
|
|
|
|
pkgName: first_bucket_package
|
|
|
|
pkgVersion: 1
|
|
|
|
spec:
|
|
|
|
resources:
|
|
|
|
- kind: Bucket
|
|
|
|
name: buck_1
|
|
|
|
retention_period: 1h
|
|
|
|
`,
|
2019-11-06 22:41:06 +00:00
|
|
|
valFields: []string{"apiVersion"},
|
2019-10-23 17:09:04 +00:00
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "missing kind",
|
2019-11-06 22:41:06 +00:00
|
|
|
pkgStr: `apiVersion: 0.1.0
|
2019-10-23 17:09:04 +00:00
|
|
|
meta:
|
|
|
|
pkgName: first_bucket_package
|
|
|
|
pkgVersion: 1
|
|
|
|
spec:
|
|
|
|
resources:
|
|
|
|
- kind: Bucket
|
|
|
|
name: buck_1
|
|
|
|
retention_period: 1h
|
|
|
|
`,
|
2019-11-06 22:41:06 +00:00
|
|
|
valFields: []string{"kind"},
|
2019-10-23 17:09:04 +00:00
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "missing pkgName",
|
2019-11-06 22:41:06 +00:00
|
|
|
pkgStr: `apiVersion: 0.1.0
|
2019-10-23 17:09:04 +00:00
|
|
|
kind: Package
|
|
|
|
meta:
|
|
|
|
pkgVersion: 1
|
|
|
|
spec:
|
|
|
|
resources:
|
|
|
|
- kind: Bucket
|
|
|
|
name: buck_1
|
|
|
|
retention_period: 1h
|
|
|
|
`,
|
2019-11-08 19:33:41 +00:00
|
|
|
valFields: []string{"meta.pkgName"},
|
2019-10-23 17:09:04 +00:00
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "missing pkgVersion",
|
2019-11-06 22:41:06 +00:00
|
|
|
pkgStr: `apiVersion: 0.1.0
|
2019-10-23 17:09:04 +00:00
|
|
|
kind: Package
|
|
|
|
meta:
|
|
|
|
pkgName: foo_name
|
|
|
|
spec:
|
|
|
|
resources:
|
|
|
|
- kind: Bucket
|
|
|
|
name: buck_1
|
|
|
|
retention_period: 1h
|
|
|
|
`,
|
2019-11-08 19:33:41 +00:00
|
|
|
valFields: []string{"meta.pkgVersion"},
|
2019-10-23 17:09:04 +00:00
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "missing multiple",
|
2019-11-06 22:41:06 +00:00
|
|
|
pkgStr: `spec:
|
2019-10-23 17:09:04 +00:00
|
|
|
resources:
|
|
|
|
- kind: Bucket
|
|
|
|
name: buck_1
|
|
|
|
retention_period: 1h
|
|
|
|
`,
|
2019-11-08 19:33:41 +00:00
|
|
|
valFields: []string{"apiVersion", "kind", "meta.pkgVersion", "meta.pkgName"},
|
2019-10-23 17:09:04 +00:00
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, tt := range tests {
|
2019-11-08 19:33:41 +00:00
|
|
|
testPkgErrors(t, KindPackage, tt)
|
2019-10-23 17:09:04 +00:00
|
|
|
}
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
2019-11-06 22:41:06 +00:00
|
|
|
t.Run("pkg with a bucket", func(t *testing.T) {
|
2019-10-24 23:59:01 +00:00
|
|
|
t.Run("with valid bucket pkg should be valid", func(t *testing.T) {
|
2019-10-23 17:09:04 +00:00
|
|
|
testfileRunner(t, "testdata/bucket", func(t *testing.T, pkg *Pkg) {
|
2019-12-06 00:53:00 +00:00
|
|
|
buckets := pkg.Summary().Buckets
|
2019-10-23 17:09:04 +00:00
|
|
|
require.Len(t, buckets, 1)
|
|
|
|
|
|
|
|
actual := buckets[0]
|
2019-12-06 00:53:00 +00:00
|
|
|
expectedBucket := SummaryBucket{
|
|
|
|
Bucket: influxdb.Bucket{
|
|
|
|
Name: "rucket_11",
|
|
|
|
Description: "bucket 1 description",
|
|
|
|
RetentionPeriod: time.Hour,
|
|
|
|
},
|
2019-10-23 17:09:04 +00:00
|
|
|
}
|
2019-12-06 00:53:00 +00:00
|
|
|
assert.Equal(t, expectedBucket, actual)
|
2019-10-23 17:09:04 +00:00
|
|
|
})
|
|
|
|
})
|
|
|
|
|
2019-11-06 22:41:06 +00:00
|
|
|
t.Run("handles bad config", func(t *testing.T) {
|
|
|
|
tests := []testPkgResourceError{
|
2019-10-23 17:09:04 +00:00
|
|
|
{
|
2019-11-06 22:41:06 +00:00
|
|
|
name: "missing name",
|
|
|
|
validationErrs: 1,
|
|
|
|
valFields: []string{"name"},
|
|
|
|
pkgStr: `apiVersion: 0.1.0
|
2019-10-23 17:09:04 +00:00
|
|
|
kind: Package
|
|
|
|
meta:
|
|
|
|
pkgName: first_bucket_package
|
|
|
|
pkgVersion: 1
|
|
|
|
spec:
|
|
|
|
resources:
|
|
|
|
- kind: Bucket
|
|
|
|
retention_period: 1h
|
|
|
|
`,
|
2019-11-06 22:41:06 +00:00
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "mixed valid and missing name",
|
|
|
|
validationErrs: 1,
|
|
|
|
valFields: []string{"name"},
|
|
|
|
pkgStr: `apiVersion: 0.1.0
|
2019-10-23 17:09:04 +00:00
|
|
|
kind: Package
|
|
|
|
meta:
|
|
|
|
pkgName: first_bucket_package
|
|
|
|
pkgVersion: 1
|
|
|
|
spec:
|
|
|
|
resources:
|
|
|
|
- kind: Bucket
|
|
|
|
retention_period: 1h
|
|
|
|
name: valid name
|
|
|
|
- kind: Bucket
|
|
|
|
retention_period: 1h
|
|
|
|
`,
|
2019-11-06 22:41:06 +00:00
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "mixed valid and multiple bad names",
|
|
|
|
resourceErrs: 2,
|
|
|
|
validationErrs: 1,
|
|
|
|
valFields: []string{"name"},
|
|
|
|
pkgStr: `apiVersion: 0.1.0
|
2019-10-23 17:09:04 +00:00
|
|
|
kind: Package
|
|
|
|
meta:
|
|
|
|
pkgName: first_bucket_package
|
|
|
|
pkgVersion: 1
|
|
|
|
spec:
|
|
|
|
resources:
|
|
|
|
- kind: Bucket
|
|
|
|
retention_period: 1h
|
|
|
|
name: valid name
|
|
|
|
- kind: Bucket
|
|
|
|
retention_period: 1h
|
|
|
|
- kind: Bucket
|
|
|
|
retention_period: 1h
|
|
|
|
`,
|
|
|
|
},
|
2019-11-06 22:41:06 +00:00
|
|
|
{
|
|
|
|
name: "duplicate bucket names",
|
|
|
|
resourceErrs: 1,
|
|
|
|
validationErrs: 1,
|
|
|
|
valFields: []string{"name"},
|
|
|
|
pkgStr: `apiVersion: 0.1.0
|
2019-10-23 17:09:04 +00:00
|
|
|
kind: Package
|
|
|
|
meta:
|
|
|
|
pkgName: first_bucket_package
|
|
|
|
pkgVersion: 1
|
|
|
|
spec:
|
|
|
|
resources:
|
|
|
|
- kind: Bucket
|
|
|
|
retention_period: 1h
|
|
|
|
name: valid name
|
|
|
|
- kind: Bucket
|
|
|
|
retention_period: 1h
|
|
|
|
name: valid name
|
2019-11-06 22:41:06 +00:00
|
|
|
`,
|
|
|
|
},
|
|
|
|
}
|
2019-10-23 17:09:04 +00:00
|
|
|
|
2019-11-06 22:41:06 +00:00
|
|
|
for _, tt := range tests {
|
2019-11-08 19:33:41 +00:00
|
|
|
testPkgErrors(t, KindBucket, tt)
|
2019-11-06 22:41:06 +00:00
|
|
|
}
|
2019-10-23 17:09:04 +00:00
|
|
|
})
|
|
|
|
})
|
2019-10-24 23:59:01 +00:00
|
|
|
|
2019-11-06 22:41:06 +00:00
|
|
|
t.Run("pkg with a label", func(t *testing.T) {
|
2019-10-24 23:59:01 +00:00
|
|
|
t.Run("with valid label pkg should be valid", func(t *testing.T) {
|
|
|
|
testfileRunner(t, "testdata/label", func(t *testing.T, pkg *Pkg) {
|
|
|
|
labels := pkg.labels()
|
|
|
|
require.Len(t, labels, 2)
|
|
|
|
|
|
|
|
expectedLabel1 := label{
|
2019-12-03 02:05:10 +00:00
|
|
|
name: "label_1",
|
2019-10-24 23:59:01 +00:00
|
|
|
Description: "label 1 description",
|
|
|
|
Color: "#FFFFFF",
|
|
|
|
}
|
|
|
|
assert.Equal(t, expectedLabel1, *labels[0])
|
|
|
|
|
|
|
|
expectedLabel2 := label{
|
2019-12-03 02:05:10 +00:00
|
|
|
name: "label_2",
|
2019-10-24 23:59:01 +00:00
|
|
|
Description: "label 2 description",
|
|
|
|
Color: "#000000",
|
|
|
|
}
|
|
|
|
assert.Equal(t, expectedLabel2, *labels[1])
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("with missing label name should error", func(t *testing.T) {
|
2019-11-06 22:41:06 +00:00
|
|
|
tests := []testPkgResourceError{
|
2019-10-24 23:59:01 +00:00
|
|
|
{
|
2019-11-06 22:41:06 +00:00
|
|
|
name: "missing name",
|
|
|
|
validationErrs: 1,
|
|
|
|
valFields: []string{"name"},
|
|
|
|
pkgStr: `apiVersion: 0.1.0
|
2019-10-24 23:59:01 +00:00
|
|
|
kind: Package
|
|
|
|
meta:
|
|
|
|
pkgName: first_label_pkg
|
|
|
|
pkgVersion: 1
|
|
|
|
spec:
|
|
|
|
resources:
|
|
|
|
- kind: Label
|
|
|
|
`,
|
|
|
|
},
|
|
|
|
{
|
2019-11-06 22:41:06 +00:00
|
|
|
name: "mixed valid and missing name",
|
|
|
|
validationErrs: 1,
|
|
|
|
valFields: []string{"name"},
|
|
|
|
pkgStr: `apiVersion: 0.1.0
|
2019-10-24 23:59:01 +00:00
|
|
|
kind: Package
|
|
|
|
meta:
|
|
|
|
pkgName: label_pkg
|
|
|
|
pkgVersion: 1
|
|
|
|
spec:
|
|
|
|
resources:
|
|
|
|
- kind: Label
|
|
|
|
name: valid name
|
|
|
|
- kind: Label
|
|
|
|
`,
|
|
|
|
},
|
|
|
|
{
|
2019-11-06 22:41:06 +00:00
|
|
|
name: "multiple labels with missing name",
|
|
|
|
resourceErrs: 2,
|
|
|
|
validationErrs: 1,
|
|
|
|
valFields: []string{"name"},
|
|
|
|
pkgStr: `apiVersion: 0.1.0
|
2019-10-24 23:59:01 +00:00
|
|
|
kind: Package
|
|
|
|
meta:
|
|
|
|
pkgName: label_pkg
|
|
|
|
pkgVersion: 1
|
|
|
|
spec:
|
|
|
|
resources:
|
|
|
|
- kind: Label
|
|
|
|
- kind: Label
|
|
|
|
`,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, tt := range tests {
|
2019-11-08 19:33:41 +00:00
|
|
|
testPkgErrors(t, KindLabel, tt)
|
2019-10-24 23:59:01 +00:00
|
|
|
}
|
|
|
|
})
|
|
|
|
})
|
2019-10-26 02:11:47 +00:00
|
|
|
|
|
|
|
t.Run("pkg with buckets and labels associated", func(t *testing.T) {
|
|
|
|
testfileRunner(t, "testdata/bucket_associates_label", func(t *testing.T, pkg *Pkg) {
|
|
|
|
sum := pkg.Summary()
|
|
|
|
require.Len(t, sum.Labels, 2)
|
|
|
|
|
|
|
|
bkts := sum.Buckets
|
|
|
|
require.Len(t, bkts, 3)
|
|
|
|
|
|
|
|
expectedLabels := []struct {
|
|
|
|
bktName string
|
|
|
|
labels []string
|
|
|
|
}{
|
|
|
|
{
|
|
|
|
bktName: "rucket_1",
|
|
|
|
labels: []string{"label_1"},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
bktName: "rucket_2",
|
|
|
|
labels: []string{"label_2"},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
bktName: "rucket_3",
|
|
|
|
labels: []string{"label_1", "label_2"},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
for i, expected := range expectedLabels {
|
|
|
|
bkt := bkts[i]
|
2019-10-30 21:13:42 +00:00
|
|
|
require.Len(t, bkt.LabelAssociations, len(expected.labels))
|
2019-10-26 02:11:47 +00:00
|
|
|
|
|
|
|
for j, label := range expected.labels {
|
2019-10-30 21:13:42 +00:00
|
|
|
assert.Equal(t, label, bkt.LabelAssociations[j].Name)
|
2019-10-26 02:11:47 +00:00
|
|
|
}
|
|
|
|
}
|
2019-10-30 21:13:42 +00:00
|
|
|
|
|
|
|
expectedMappings := []SummaryLabelMapping{
|
|
|
|
{
|
|
|
|
ResourceName: "rucket_1",
|
|
|
|
LabelName: "label_1",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
ResourceName: "rucket_2",
|
|
|
|
LabelName: "label_2",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
ResourceName: "rucket_3",
|
|
|
|
LabelName: "label_1",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
ResourceName: "rucket_3",
|
|
|
|
LabelName: "label_2",
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
require.Len(t, sum.LabelMappings, len(expectedMappings))
|
|
|
|
for i, expected := range expectedMappings {
|
|
|
|
expected.LabelMapping.ResourceType = influxdb.BucketsResourceType
|
|
|
|
assert.Equal(t, expected, sum.LabelMappings[i])
|
|
|
|
}
|
2019-10-26 02:11:47 +00:00
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("association doesn't exist then provides an error", func(t *testing.T) {
|
2019-11-06 22:41:06 +00:00
|
|
|
tests := []testPkgResourceError{
|
2019-10-26 02:11:47 +00:00
|
|
|
{
|
|
|
|
name: "no labels provided",
|
2019-11-06 22:41:06 +00:00
|
|
|
assErrs: 1,
|
|
|
|
assIdxs: []int{0},
|
|
|
|
pkgStr: `apiVersion: 0.1.0
|
2019-10-26 02:11:47 +00:00
|
|
|
kind: Package
|
|
|
|
meta:
|
|
|
|
pkgName: label_pkg
|
|
|
|
pkgVersion: 1
|
|
|
|
spec:
|
|
|
|
resources:
|
|
|
|
- kind: Bucket
|
|
|
|
name: buck_1
|
|
|
|
associations:
|
|
|
|
- kind: Label
|
|
|
|
name: label_1
|
|
|
|
`,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "mixed found and not found",
|
2019-11-06 22:41:06 +00:00
|
|
|
assErrs: 1,
|
|
|
|
assIdxs: []int{1},
|
|
|
|
pkgStr: `apiVersion: 0.1.0
|
2019-10-26 02:11:47 +00:00
|
|
|
kind: Package
|
|
|
|
meta:
|
|
|
|
pkgName: label_pkg
|
|
|
|
pkgVersion: 1
|
|
|
|
spec:
|
|
|
|
resources:
|
|
|
|
- kind: Label
|
|
|
|
name: label_1
|
|
|
|
- kind: Bucket
|
|
|
|
name: buck_1
|
|
|
|
associations:
|
|
|
|
- kind: Label
|
|
|
|
name: label_1
|
|
|
|
- kind: Label
|
|
|
|
name: unfound label
|
|
|
|
`,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "multiple not found",
|
2019-11-06 22:41:06 +00:00
|
|
|
assErrs: 1,
|
|
|
|
assIdxs: []int{0, 1},
|
|
|
|
pkgStr: `apiVersion: 0.1.0
|
2019-10-26 02:11:47 +00:00
|
|
|
kind: Package
|
|
|
|
meta:
|
|
|
|
pkgName: label_pkg
|
|
|
|
pkgVersion: 1
|
|
|
|
spec:
|
|
|
|
resources:
|
|
|
|
- kind: Label
|
|
|
|
name: label_1
|
|
|
|
- kind: Bucket
|
|
|
|
name: buck_1
|
|
|
|
associations:
|
|
|
|
- kind: Label
|
|
|
|
name: not found 1
|
|
|
|
- kind: Label
|
|
|
|
name: unfound label
|
|
|
|
`,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "duplicate valid nested labels",
|
2019-11-06 22:41:06 +00:00
|
|
|
assErrs: 1,
|
|
|
|
assIdxs: []int{1},
|
|
|
|
pkgStr: `apiVersion: 0.1.0
|
2019-10-26 02:11:47 +00:00
|
|
|
kind: Package
|
|
|
|
meta:
|
|
|
|
pkgName: label_pkg
|
|
|
|
pkgVersion: 1
|
|
|
|
spec:
|
|
|
|
resources:
|
|
|
|
- kind: Label
|
|
|
|
name: label_1
|
|
|
|
- kind: Bucket
|
|
|
|
name: buck_1
|
|
|
|
associations:
|
|
|
|
- kind: Label
|
|
|
|
name: label_1
|
|
|
|
- kind: Label
|
|
|
|
name: label_1
|
|
|
|
`,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, tt := range tests {
|
2019-11-08 19:33:41 +00:00
|
|
|
testPkgErrors(t, KindBucket, tt)
|
2019-10-26 02:11:47 +00:00
|
|
|
}
|
|
|
|
})
|
|
|
|
})
|
2019-10-30 21:13:42 +00:00
|
|
|
|
2019-11-01 18:11:42 +00:00
|
|
|
t.Run("pkg with single dashboard and single chart", func(t *testing.T) {
|
2019-11-12 18:06:53 +00:00
|
|
|
t.Run("single gauge chart", func(t *testing.T) {
|
|
|
|
testfileRunner(t, "testdata/dashboard_gauge", func(t *testing.T, pkg *Pkg) {
|
|
|
|
sum := pkg.Summary()
|
|
|
|
require.Len(t, sum.Dashboards, 1)
|
|
|
|
|
|
|
|
actual := sum.Dashboards[0]
|
|
|
|
assert.Equal(t, "dash_1", actual.Name)
|
|
|
|
assert.Equal(t, "desc1", actual.Description)
|
|
|
|
|
|
|
|
require.Len(t, actual.Charts, 1)
|
|
|
|
actualChart := actual.Charts[0]
|
|
|
|
assert.Equal(t, 3, actualChart.Height)
|
|
|
|
assert.Equal(t, 6, actualChart.Width)
|
|
|
|
assert.Equal(t, 1, actualChart.XPosition)
|
|
|
|
assert.Equal(t, 2, actualChart.YPosition)
|
|
|
|
|
|
|
|
props, ok := actualChart.Properties.(influxdb.GaugeViewProperties)
|
|
|
|
require.True(t, ok)
|
|
|
|
assert.Equal(t, "gauge", props.GetType())
|
|
|
|
assert.Equal(t, "gauge note", props.Note)
|
|
|
|
assert.True(t, props.ShowNoteWhenEmpty)
|
|
|
|
|
|
|
|
require.Len(t, props.Queries, 1)
|
|
|
|
q := props.Queries[0]
|
|
|
|
queryText := `from(bucket: v.bucket) |> range(start: v.timeRangeStart, stop: v.timeRangeStop) |> filter(fn: (r) => r._measurement == "boltdb_writes_total") |> filter(fn: (r) => r._field == "counter")`
|
|
|
|
assert.Equal(t, queryText, q.Text)
|
|
|
|
assert.Equal(t, "advanced", q.EditMode)
|
|
|
|
|
|
|
|
require.Len(t, props.ViewColors, 3)
|
|
|
|
c := props.ViewColors[0]
|
|
|
|
assert.NotZero(t, c.ID)
|
|
|
|
assert.Equal(t, "laser", c.Name)
|
|
|
|
assert.Equal(t, "min", c.Type)
|
|
|
|
assert.Equal(t, "#8F8AF4", c.Hex)
|
|
|
|
assert.Equal(t, 0.0, c.Value)
|
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("handles invalid config", func(t *testing.T) {
|
|
|
|
tests := []testPkgResourceError{
|
|
|
|
{
|
|
|
|
name: "color a gauge type",
|
|
|
|
encoding: EncodingJSON,
|
|
|
|
validationErrs: 1,
|
|
|
|
valFields: []string{"charts[0].colors"},
|
|
|
|
pkgStr: `
|
|
|
|
{
|
|
|
|
"apiVersion": "0.1.0",
|
|
|
|
"kind": "Package",
|
|
|
|
"meta": {
|
|
|
|
"pkgName": "pkg_name",
|
|
|
|
"pkgVersion": "1",
|
|
|
|
"description": "pack description"
|
|
|
|
},
|
|
|
|
"spec": {
|
|
|
|
"resources": [
|
|
|
|
{
|
|
|
|
"kind": "Dashboard",
|
|
|
|
"name": "dash_1",
|
|
|
|
"description": "desc1",
|
|
|
|
"charts": [
|
|
|
|
{
|
|
|
|
"kind": "gauge",
|
|
|
|
"name": "gauge",
|
|
|
|
"prefix": "prefix",
|
|
|
|
"suffix": "suffix",
|
|
|
|
"note": "gauge note",
|
|
|
|
"noteOnEmpty": true,
|
|
|
|
"xPos": 1,
|
|
|
|
"yPos": 2,
|
|
|
|
"width": 6,
|
|
|
|
"height": 3,
|
|
|
|
"decimalPlaces": 1,
|
|
|
|
"xColumn": "_time",
|
|
|
|
"yColumn": "_value",
|
|
|
|
"queries": [
|
|
|
|
{
|
|
|
|
"query": "from(bucket: v.bucket) |> range(start: v.timeRangeStart, stop: v.timeRangeStop) |> filter(fn: (r) => r._measurement == \"boltdb_writes_total\") |> filter(fn: (r) => r._field == \"counter\")"
|
|
|
|
}
|
|
|
|
],
|
|
|
|
"colors": [
|
|
|
|
{
|
|
|
|
"name": "laser",
|
|
|
|
"type": "min",
|
|
|
|
"hex": "#8F8AF4",
|
|
|
|
"value": 0
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"name": "comet",
|
|
|
|
"type": "max",
|
|
|
|
"hex": "#F4CF31",
|
|
|
|
"value": 5000
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
`,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "color mixing a hex value",
|
|
|
|
encoding: EncodingJSON,
|
|
|
|
validationErrs: 1,
|
|
|
|
valFields: []string{"charts[0].colors[0].hex"},
|
|
|
|
pkgStr: `
|
|
|
|
{
|
|
|
|
"apiVersion": "0.1.0",
|
|
|
|
"kind": "Package",
|
|
|
|
"meta": {
|
|
|
|
"pkgName": "pkg_name",
|
|
|
|
"pkgVersion": "1",
|
|
|
|
"description": "pack description"
|
|
|
|
},
|
|
|
|
"spec": {
|
|
|
|
"resources": [
|
|
|
|
{
|
|
|
|
"kind": "Dashboard",
|
|
|
|
"name": "dash_1",
|
|
|
|
"description": "desc1",
|
|
|
|
"charts": [
|
|
|
|
{
|
|
|
|
"kind": "gauge",
|
|
|
|
"name": "gauge",
|
|
|
|
"prefix": "prefix",
|
|
|
|
"suffix": "suffix",
|
|
|
|
"note": "gauge note",
|
|
|
|
"noteOnEmpty": true,
|
|
|
|
"xPos": 1,
|
|
|
|
"yPos": 2,
|
|
|
|
"width": 6,
|
|
|
|
"height": 3,
|
|
|
|
"decimalPlaces": 1,
|
|
|
|
"xColumn": "_time",
|
|
|
|
"yColumn": "_value",
|
|
|
|
"queries": [
|
|
|
|
{
|
|
|
|
"query": "from(bucket: v.bucket) |> range(start: v.timeRangeStart, stop: v.timeRangeStop) |> filter(fn: (r) => r._measurement == \"boltdb_writes_total\") |> filter(fn: (r) => r._field == \"counter\")"
|
|
|
|
}
|
|
|
|
],
|
|
|
|
"colors": [
|
|
|
|
{
|
|
|
|
"name": "laser",
|
|
|
|
"type": "min",
|
|
|
|
"value": 0
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"name": "pool",
|
|
|
|
"type": "threshold",
|
|
|
|
"hex": "#F4CF31",
|
|
|
|
"value": 700
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"name": "comet",
|
|
|
|
"type": "max",
|
|
|
|
"hex": "#F4CF31",
|
|
|
|
"value": 5000
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
`,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "missing a query value",
|
|
|
|
encoding: EncodingJSON,
|
|
|
|
validationErrs: 1,
|
|
|
|
valFields: []string{"charts[0].queries[0].query"},
|
|
|
|
pkgStr: `
|
|
|
|
{
|
|
|
|
"apiVersion": "0.1.0",
|
|
|
|
"kind": "Package",
|
|
|
|
"meta": {
|
|
|
|
"pkgName": "pkg_name",
|
|
|
|
"pkgVersion": "1",
|
|
|
|
"description": "pack description"
|
|
|
|
},
|
|
|
|
"spec": {
|
|
|
|
"resources": [
|
|
|
|
{
|
|
|
|
"kind": "Dashboard",
|
|
|
|
"name": "dash_1",
|
|
|
|
"description": "desc1",
|
|
|
|
"charts": [
|
|
|
|
{
|
|
|
|
"kind": "gauge",
|
|
|
|
"name": "gauge",
|
|
|
|
"prefix": "prefix",
|
|
|
|
"suffix": "suffix",
|
|
|
|
"note": "gauge note",
|
|
|
|
"noteOnEmpty": true,
|
|
|
|
"xPos": 1,
|
|
|
|
"yPos": 2,
|
|
|
|
"width": 6,
|
|
|
|
"height": 3,
|
|
|
|
"decimalPlaces": 1,
|
|
|
|
"xColumn": "_time",
|
|
|
|
"yColumn": "_value",
|
|
|
|
"queries": [
|
|
|
|
{
|
|
|
|
"query": null
|
|
|
|
}
|
|
|
|
],
|
|
|
|
"colors": [
|
|
|
|
{
|
|
|
|
"name": "laser",
|
|
|
|
"type": "min",
|
|
|
|
"hex": "#FFF000",
|
|
|
|
"value": 0
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"name": "pool",
|
|
|
|
"type": "threshold",
|
|
|
|
"hex": "#F4CF31",
|
|
|
|
"value": 700
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"name": "comet",
|
|
|
|
"type": "max",
|
|
|
|
"hex": "#F4CF31",
|
|
|
|
"value": 5000
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
}
|
2019-11-12 20:09:13 +00:00
|
|
|
`,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, tt := range tests {
|
|
|
|
testPkgErrors(t, KindDashboard, tt)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
2019-12-03 22:59:07 +00:00
|
|
|
t.Run("single heatmap chart", func(t *testing.T) {
|
2019-11-12 20:09:13 +00:00
|
|
|
testfileRunner(t, "testdata/dashboard_heatmap", func(t *testing.T, pkg *Pkg) {
|
|
|
|
sum := pkg.Summary()
|
|
|
|
require.Len(t, sum.Dashboards, 1)
|
|
|
|
|
|
|
|
actual := sum.Dashboards[0]
|
|
|
|
assert.Equal(t, "dashboard w/ single heatmap chart", actual.Name)
|
2019-11-15 01:05:21 +00:00
|
|
|
assert.Equal(t, "a dashboard w/ heatmap chart", actual.Description)
|
2019-11-12 20:09:13 +00:00
|
|
|
|
|
|
|
require.Len(t, actual.Charts, 1)
|
|
|
|
actualChart := actual.Charts[0]
|
|
|
|
assert.Equal(t, 3, actualChart.Height)
|
|
|
|
assert.Equal(t, 6, actualChart.Width)
|
|
|
|
assert.Equal(t, 1, actualChart.XPosition)
|
|
|
|
assert.Equal(t, 2, actualChart.YPosition)
|
|
|
|
|
|
|
|
props, ok := actualChart.Properties.(influxdb.HeatmapViewProperties)
|
|
|
|
require.True(t, ok)
|
|
|
|
assert.Equal(t, "heatmap", props.GetType())
|
|
|
|
assert.Equal(t, "heatmap note", props.Note)
|
|
|
|
assert.Equal(t, int32(10), props.BinSize)
|
|
|
|
assert.True(t, props.ShowNoteWhenEmpty)
|
|
|
|
|
2019-11-15 01:05:21 +00:00
|
|
|
assert.Equal(t, []float64{0, 10}, props.XDomain)
|
|
|
|
assert.Equal(t, []float64{0, 100}, props.YDomain)
|
|
|
|
|
2019-11-12 20:09:13 +00:00
|
|
|
require.Len(t, props.Queries, 1)
|
|
|
|
q := props.Queries[0]
|
|
|
|
queryText := `from(bucket: v.bucket) |> range(start: v.timeRangeStart) |> filter(fn: (r) => r._measurement == "mem") |> filter(fn: (r) => r._field == "used_percent") |> aggregateWindow(every: v.windowPeriod, fn: mean, createEmpty: false) |> yield(name: "mean")`
|
|
|
|
assert.Equal(t, queryText, q.Text)
|
|
|
|
assert.Equal(t, "advanced", q.EditMode)
|
|
|
|
|
|
|
|
require.Len(t, props.ViewColors, 12)
|
|
|
|
c := props.ViewColors[0]
|
|
|
|
assert.Equal(t, "#000004", c)
|
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("handles invalid config", func(t *testing.T) {
|
|
|
|
tests := []testPkgResourceError{
|
|
|
|
{
|
|
|
|
name: "a color is missing a hex value",
|
|
|
|
encoding: EncodingJSON,
|
|
|
|
validationErrs: 1,
|
|
|
|
valFields: []string{"charts[0].colors[2].hex"},
|
|
|
|
pkgStr: `
|
|
|
|
{
|
|
|
|
"apiVersion": "0.1.0",
|
|
|
|
"kind": "Package",
|
|
|
|
"meta": {
|
|
|
|
"pkgName": "pkg_name",
|
|
|
|
"pkgVersion": "1",
|
|
|
|
"description": "pack description"
|
|
|
|
},
|
|
|
|
"spec": {
|
|
|
|
"resources": [
|
|
|
|
{
|
|
|
|
"kind": "Dashboard",
|
|
|
|
"name": "dashboard w/ single heatmap chart",
|
2019-11-15 01:05:21 +00:00
|
|
|
"description": "a dashboard w/ heatmap chart",
|
2019-11-12 20:09:13 +00:00
|
|
|
"charts": [
|
|
|
|
{
|
|
|
|
"kind": "heatmap",
|
|
|
|
"name": "heatmap chart",
|
|
|
|
"note": "heatmap note",
|
|
|
|
"noteOnEmpty": true,
|
|
|
|
"xPos": 1,
|
|
|
|
"yPos": 2,
|
|
|
|
"width": 6,
|
|
|
|
"height": 3,
|
|
|
|
"xCol": "_time",
|
|
|
|
"yCol": "_value",
|
|
|
|
"binSize": 10,
|
|
|
|
"queries": [
|
|
|
|
{
|
|
|
|
"query": "from(bucket: v.bucket) |> range(start: v.timeRangeStart) |> filter(fn: (r) => r._measurement == \"mem\") |> filter(fn: (r) => r._field == \"used_percent\") |> aggregateWindow(every: v.windowPeriod, fn: mean, createEmpty: false) |> yield(name: \"mean\")"
|
|
|
|
}
|
|
|
|
],
|
|
|
|
"axes":[
|
|
|
|
{
|
|
|
|
"name": "x",
|
|
|
|
"label": "x_label",
|
|
|
|
"prefix": "x_prefix",
|
|
|
|
"suffix": "x_suffix"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"name": "y",
|
|
|
|
"label": "y_label",
|
|
|
|
"prefix": "y_prefix",
|
|
|
|
"suffix": "y_suffix"
|
|
|
|
}
|
|
|
|
],
|
|
|
|
"colors": [
|
|
|
|
{
|
|
|
|
"hex": "#8F8AF4"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"hex": "#F4CF31"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"hex": null
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
`,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "missing axes",
|
|
|
|
encoding: EncodingJSON,
|
|
|
|
validationErrs: 1,
|
|
|
|
valFields: []string{"charts[0].axes", "charts[0].axes"},
|
|
|
|
pkgStr: `
|
|
|
|
{
|
|
|
|
"apiVersion": "0.1.0",
|
|
|
|
"kind": "Package",
|
|
|
|
"meta": {
|
|
|
|
"pkgName": "pkg_name",
|
|
|
|
"pkgVersion": "1",
|
|
|
|
"description": "pack description"
|
|
|
|
},
|
|
|
|
"spec": {
|
|
|
|
"resources": [
|
|
|
|
{
|
|
|
|
"kind": "Dashboard",
|
|
|
|
"name": "dashboard w/ single heatmap chart",
|
2019-11-15 01:05:21 +00:00
|
|
|
"description": "a dashboard w/ heatmap chart",
|
2019-11-12 20:09:13 +00:00
|
|
|
"charts": [
|
|
|
|
{
|
|
|
|
"kind": "heatmap",
|
|
|
|
"name": "heatmap chart",
|
|
|
|
"note": "heatmap note",
|
|
|
|
"noteOnEmpty": true,
|
|
|
|
"xPos": 1,
|
|
|
|
"yPos": 2,
|
|
|
|
"width": 6,
|
|
|
|
"height": 3,
|
|
|
|
"xCol": "_time",
|
|
|
|
"yCol": "_value",
|
|
|
|
"binSize": 10,
|
|
|
|
"queries": [
|
|
|
|
{
|
|
|
|
"query": "from(bucket: v.bucket) |> range(start: v.timeRangeStart) |> filter(fn: (r) => r._measurement == \"mem\") |> filter(fn: (r) => r._field == \"used_percent\") |> aggregateWindow(every: v.windowPeriod, fn: mean, createEmpty: false) |> yield(name: \"mean\")"
|
|
|
|
}
|
|
|
|
],
|
|
|
|
"colors": [
|
|
|
|
{
|
|
|
|
"hex": "#8F8AF4"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"hex": "#F4CF31"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"hex": "#FFFFFF"
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
`,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "missing a query value",
|
|
|
|
encoding: EncodingJSON,
|
|
|
|
validationErrs: 1,
|
|
|
|
valFields: []string{"charts[0].queries[0].query"},
|
|
|
|
pkgStr: `
|
|
|
|
{
|
|
|
|
"apiVersion": "0.1.0",
|
|
|
|
"kind": "Package",
|
|
|
|
"meta": {
|
|
|
|
"pkgName": "pkg_name",
|
|
|
|
"pkgVersion": "1",
|
|
|
|
"description": "pack description"
|
2019-11-16 20:14:46 +00:00
|
|
|
},
|
|
|
|
"spec": {
|
|
|
|
"resources": [
|
|
|
|
{
|
|
|
|
"kind": "Dashboard",
|
|
|
|
"name": "dashboard w/ single heatmap chart",
|
|
|
|
"description": "a dashboard w/ heatmap chart",
|
|
|
|
"charts": [
|
|
|
|
{
|
|
|
|
"kind": "heatmap",
|
|
|
|
"name": "heatmap chart",
|
|
|
|
"note": "heatmap note",
|
|
|
|
"noteOnEmpty": true,
|
|
|
|
"xPos": 1,
|
|
|
|
"yPos": 2,
|
|
|
|
"width": 6,
|
|
|
|
"height": 3,
|
|
|
|
"xCol": "_time",
|
|
|
|
"yCol": "_value",
|
|
|
|
"binSize": 10,
|
|
|
|
"queries": [
|
|
|
|
{
|
|
|
|
"query": null
|
|
|
|
}
|
|
|
|
],
|
|
|
|
"axes":[
|
|
|
|
{
|
|
|
|
"name": "x",
|
|
|
|
"label": "x_label",
|
|
|
|
"prefix": "x_prefix",
|
|
|
|
"suffix": "x_suffix"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"name": "y",
|
|
|
|
"label": "y_label",
|
|
|
|
"prefix": "y_prefix",
|
|
|
|
"suffix": "y_suffix"
|
|
|
|
}
|
|
|
|
],
|
|
|
|
"colors": [
|
|
|
|
{
|
|
|
|
"hex": "#8F8AF4"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"hex": "#F4CF31"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"hex": "#FFFFFF"
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
`,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, tt := range tests {
|
|
|
|
testPkgErrors(t, KindDashboard, tt)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
2019-12-03 22:59:07 +00:00
|
|
|
t.Run("single histogram chart", func(t *testing.T) {
|
2019-11-16 20:14:46 +00:00
|
|
|
testfileRunner(t, "testdata/dashboard_histogram", func(t *testing.T, pkg *Pkg) {
|
|
|
|
sum := pkg.Summary()
|
|
|
|
require.Len(t, sum.Dashboards, 1)
|
|
|
|
|
|
|
|
actual := sum.Dashboards[0]
|
|
|
|
assert.Equal(t, "dashboard w/ single histogram chart", actual.Name)
|
|
|
|
assert.Equal(t, "a dashboard w/ single histogram chart", actual.Description)
|
|
|
|
|
|
|
|
require.Len(t, actual.Charts, 1)
|
|
|
|
actualChart := actual.Charts[0]
|
|
|
|
assert.Equal(t, 3, actualChart.Height)
|
|
|
|
assert.Equal(t, 6, actualChart.Width)
|
|
|
|
|
|
|
|
props, ok := actualChart.Properties.(influxdb.HistogramViewProperties)
|
|
|
|
require.True(t, ok)
|
|
|
|
assert.Equal(t, "histogram", props.GetType())
|
|
|
|
assert.Equal(t, "histogram note", props.Note)
|
|
|
|
assert.Equal(t, 30, props.BinCount)
|
|
|
|
assert.True(t, props.ShowNoteWhenEmpty)
|
|
|
|
assert.Equal(t, []float64{0, 10}, props.XDomain)
|
|
|
|
assert.Equal(t, []string{}, props.FillColumns)
|
|
|
|
require.Len(t, props.Queries, 1)
|
|
|
|
q := props.Queries[0]
|
|
|
|
queryText := `from(bucket: v.bucket) |> range(start: v.timeRangeStart, stop: v.timeRangeStop) |> filter(fn: (r) => r._measurement == "boltdb_reads_total") |> filter(fn: (r) => r._field == "counter")`
|
|
|
|
assert.Equal(t, queryText, q.Text)
|
|
|
|
assert.Equal(t, "advanced", q.EditMode)
|
|
|
|
|
|
|
|
require.Len(t, props.ViewColors, 3)
|
|
|
|
assert.Equal(t, "#8F8AF4", props.ViewColors[0].Hex)
|
|
|
|
assert.Equal(t, "#F4CF31", props.ViewColors[1].Hex)
|
|
|
|
assert.Equal(t, "#FFFFFF", props.ViewColors[2].Hex)
|
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("handles invalid config", func(t *testing.T) {
|
|
|
|
tests := []testPkgResourceError{
|
|
|
|
{
|
|
|
|
name: "missing x axis",
|
|
|
|
encoding: EncodingJSON,
|
|
|
|
validationErrs: 1,
|
|
|
|
valFields: []string{"charts[0].axes"},
|
|
|
|
pkgStr: `
|
|
|
|
{
|
|
|
|
"apiVersion": "0.1.0",
|
|
|
|
"kind": "Package",
|
|
|
|
"meta": {
|
|
|
|
"pkgName": "pkg_name",
|
|
|
|
"pkgVersion": "1",
|
|
|
|
"description": "pack description"
|
|
|
|
},
|
|
|
|
"spec": {
|
|
|
|
"resources": [
|
|
|
|
{
|
|
|
|
"kind": "Dashboard",
|
|
|
|
"name": "dashboard w/ single histogram chart",
|
|
|
|
"description": "a dashboard w/ single histogram chart",
|
|
|
|
"charts": [
|
|
|
|
{
|
|
|
|
"kind": "histogram",
|
|
|
|
"name": "histogram chart",
|
|
|
|
"note": "histogram note",
|
|
|
|
"noteOnEmpty": true,
|
|
|
|
"xPos": 1,
|
|
|
|
"yPos": 2,
|
|
|
|
"width": 6,
|
|
|
|
"height": 3,
|
|
|
|
"xCol": "_time",
|
|
|
|
"yCol": "_value",
|
|
|
|
"position": "stacked",
|
|
|
|
"binCount": 30,
|
|
|
|
"queries": [
|
|
|
|
{
|
|
|
|
"query": "from(bucket: v.bucket) |> range(start: v.timeRangeStart) |> filter(fn: (r) => r._measurement == \"mem\") |> filter(fn: (r) => r._field == \"used_percent\") |> aggregateWindow(every: v.windowPeriod, fn: mean, createEmpty: false) |> yield(name: \"mean\")"
|
|
|
|
}
|
|
|
|
],
|
|
|
|
"axes":[],
|
|
|
|
"colors": [
|
|
|
|
{
|
|
|
|
"hex": "#8F8AF4"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"hex": "#F4CF31"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"hex": "#FFFFFF"
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
`,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "missing x-axis",
|
|
|
|
encoding: EncodingJSON,
|
|
|
|
validationErrs: 1,
|
|
|
|
valFields: []string{"charts[0].axes", "charts[0].axes"},
|
|
|
|
pkgStr: `
|
|
|
|
{
|
|
|
|
"apiVersion": "0.1.0",
|
|
|
|
"kind": "Package",
|
|
|
|
"meta": {
|
|
|
|
"pkgName": "pkg_name",
|
|
|
|
"pkgVersion": "1",
|
|
|
|
"description": "pack description"
|
|
|
|
},
|
|
|
|
"spec": {
|
|
|
|
"resources": [
|
|
|
|
{
|
|
|
|
"kind": "Dashboard",
|
|
|
|
"name": "dashboard w/ single histogram chart",
|
|
|
|
"description": "a dashboard w/ single histogram chart",
|
|
|
|
"charts": [
|
|
|
|
{
|
|
|
|
"kind": "histogram",
|
|
|
|
"name": "histogram chart",
|
|
|
|
"note": "histogram note",
|
|
|
|
"noteOnEmpty": true,
|
|
|
|
"xPos": 1,
|
|
|
|
"yPos": 2,
|
|
|
|
"width": 6,
|
|
|
|
"height": 3,
|
|
|
|
"xCol": "_time",
|
|
|
|
"yCol": "_value",
|
|
|
|
"position": "stacked",
|
|
|
|
"binCount": 30,
|
|
|
|
"queries": [
|
|
|
|
{
|
|
|
|
"query": "from(bucket: v.bucket) |> range(start: v.timeRangeStart) |> filter(fn: (r) => r._measurement == \"mem\") |> filter(fn: (r) => r._field == \"used_percent\") |> aggregateWindow(every: v.windowPeriod, fn: mean, createEmpty: false) |> yield(name: \"mean\")"
|
|
|
|
}
|
|
|
|
],
|
|
|
|
"axes":[
|
|
|
|
{
|
|
|
|
"name": "y",
|
|
|
|
"label": "y_label",
|
|
|
|
"domain": [0, 10]
|
|
|
|
}
|
|
|
|
],
|
|
|
|
"colors": [
|
|
|
|
{
|
|
|
|
"hex": "#8F8AF4"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"hex": "#F4CF31"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"hex": "#FFFFFF"
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
`,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "missing a query value",
|
|
|
|
encoding: EncodingJSON,
|
|
|
|
validationErrs: 1,
|
|
|
|
valFields: []string{"charts[0].queries[0].query"},
|
|
|
|
pkgStr: `
|
|
|
|
{
|
|
|
|
"apiVersion": "0.1.0",
|
|
|
|
"kind": "Package",
|
|
|
|
"meta": {
|
|
|
|
"pkgName": "pkg_name",
|
|
|
|
"pkgVersion": "1",
|
|
|
|
"description": "pack description"
|
2019-11-12 20:09:13 +00:00
|
|
|
},
|
|
|
|
"spec": {
|
|
|
|
"resources": [
|
|
|
|
{
|
|
|
|
"kind": "Dashboard",
|
|
|
|
"name": "dashboard w/ single heatmap chart",
|
2019-11-15 01:05:21 +00:00
|
|
|
"description": "a dashboard w/ heatmap chart",
|
2019-11-12 20:09:13 +00:00
|
|
|
"charts": [
|
|
|
|
{
|
|
|
|
"kind": "heatmap",
|
|
|
|
"name": "heatmap chart",
|
|
|
|
"note": "heatmap note",
|
|
|
|
"noteOnEmpty": true,
|
|
|
|
"xPos": 1,
|
|
|
|
"yPos": 2,
|
|
|
|
"width": 6,
|
|
|
|
"height": 3,
|
|
|
|
"xCol": "_time",
|
|
|
|
"yCol": "_value",
|
|
|
|
"binSize": 10,
|
|
|
|
"queries": [
|
|
|
|
{
|
|
|
|
"query": null
|
|
|
|
}
|
|
|
|
],
|
|
|
|
"axes":[
|
|
|
|
{
|
|
|
|
"name": "x",
|
|
|
|
"label": "x_label",
|
|
|
|
"prefix": "x_prefix",
|
|
|
|
"suffix": "x_suffix"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"name": "y",
|
|
|
|
"label": "y_label",
|
|
|
|
"prefix": "y_prefix",
|
|
|
|
"suffix": "y_suffix"
|
|
|
|
}
|
|
|
|
],
|
|
|
|
"colors": [
|
|
|
|
{
|
|
|
|
"hex": "#8F8AF4"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"hex": "#F4CF31"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"hex": "#FFFFFF"
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-11-12 18:06:53 +00:00
|
|
|
`,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, tt := range tests {
|
|
|
|
testPkgErrors(t, KindDashboard, tt)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
2019-12-03 22:59:07 +00:00
|
|
|
t.Run("single markdown chart", func(t *testing.T) {
|
2019-11-13 21:30:52 +00:00
|
|
|
testfileRunner(t, "testdata/dashboard_markdown", func(t *testing.T, pkg *Pkg) {
|
|
|
|
sum := pkg.Summary()
|
|
|
|
require.Len(t, sum.Dashboards, 1)
|
|
|
|
|
|
|
|
actual := sum.Dashboards[0]
|
|
|
|
assert.Equal(t, "dashboard w/ single markdown chart", actual.Name)
|
|
|
|
assert.Equal(t, "a dashboard w/ single markdown chart", actual.Description)
|
|
|
|
|
|
|
|
require.Len(t, actual.Charts, 1)
|
|
|
|
actualChart := actual.Charts[0]
|
|
|
|
|
|
|
|
props, ok := actualChart.Properties.(influxdb.MarkdownViewProperties)
|
|
|
|
require.True(t, ok)
|
|
|
|
assert.Equal(t, "markdown", props.GetType())
|
|
|
|
assert.Equal(t, "## markdown note", props.Note)
|
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("handles empty markdown note", func(t *testing.T) {
|
|
|
|
pkgStr := `{
|
|
|
|
"apiVersion": "0.1.0",
|
|
|
|
"kind": "Package",
|
|
|
|
"meta": {
|
|
|
|
"pkgName": "pkg_name",
|
|
|
|
"pkgVersion": "1",
|
|
|
|
"description": "pack description"
|
|
|
|
},
|
|
|
|
"spec": {
|
|
|
|
"resources": [
|
|
|
|
{
|
|
|
|
"kind": "Dashboard",
|
|
|
|
"name": "dashboard w/ single markdown chart",
|
|
|
|
"description": "a dashboard w/ markdown chart",
|
|
|
|
"charts": [
|
|
|
|
{
|
|
|
|
"kind": "markdown",
|
|
|
|
"name": "markdown chart"
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
}`
|
|
|
|
|
|
|
|
pkg, _ := Parse(EncodingJSON, FromString(pkgStr))
|
|
|
|
props, ok := pkg.Summary().Dashboards[0].Charts[0].Properties.(influxdb.MarkdownViewProperties)
|
|
|
|
require.True(t, ok)
|
|
|
|
assert.Equal(t, "", props.Note)
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
2019-11-12 18:06:53 +00:00
|
|
|
t.Run("single scatter chart", func(t *testing.T) {
|
|
|
|
testfileRunner(t, "testdata/dashboard_scatter", func(t *testing.T, pkg *Pkg) {
|
|
|
|
sum := pkg.Summary()
|
|
|
|
require.Len(t, sum.Dashboards, 1)
|
|
|
|
|
|
|
|
actual := sum.Dashboards[0]
|
|
|
|
assert.Equal(t, "dashboard w/ single scatter chart", actual.Name)
|
|
|
|
assert.Equal(t, "a dashboard w/ single scatter chart", actual.Description)
|
|
|
|
|
|
|
|
require.Len(t, actual.Charts, 1)
|
|
|
|
actualChart := actual.Charts[0]
|
|
|
|
assert.Equal(t, 3, actualChart.Height)
|
|
|
|
assert.Equal(t, 6, actualChart.Width)
|
|
|
|
assert.Equal(t, 1, actualChart.XPosition)
|
|
|
|
assert.Equal(t, 2, actualChart.YPosition)
|
|
|
|
|
|
|
|
props, ok := actualChart.Properties.(influxdb.ScatterViewProperties)
|
|
|
|
require.True(t, ok)
|
|
|
|
assert.Equal(t, "scatter note", props.Note)
|
|
|
|
assert.True(t, props.ShowNoteWhenEmpty)
|
|
|
|
|
|
|
|
require.Len(t, props.Queries, 1)
|
|
|
|
q := props.Queries[0]
|
|
|
|
expectedQuery := `from(bucket: v.bucket) |> range(start: v.timeRangeStart) |> filter(fn: (r) => r._measurement == "mem") |> filter(fn: (r) => r._field == "used_percent") |> aggregateWindow(every: v.windowPeriod, fn: mean, createEmpty: false) |> yield(name: "mean")`
|
|
|
|
assert.Equal(t, expectedQuery, q.Text)
|
|
|
|
assert.Equal(t, "advanced", q.EditMode)
|
|
|
|
|
2019-11-15 01:05:21 +00:00
|
|
|
assert.Equal(t, []float64{0, 10}, props.XDomain)
|
|
|
|
assert.Equal(t, []float64{0, 100}, props.YDomain)
|
2019-11-12 18:06:53 +00:00
|
|
|
assert.Equal(t, "x_label", props.XAxisLabel)
|
|
|
|
assert.Equal(t, "y_label", props.YAxisLabel)
|
|
|
|
assert.Equal(t, "x_prefix", props.XPrefix)
|
|
|
|
assert.Equal(t, "y_prefix", props.YPrefix)
|
|
|
|
assert.Equal(t, "x_suffix", props.XSuffix)
|
|
|
|
assert.Equal(t, "y_suffix", props.YSuffix)
|
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("handles invalid config", func(t *testing.T) {
|
|
|
|
tests := []testPkgResourceError{
|
|
|
|
{
|
|
|
|
name: "missing axes",
|
|
|
|
validationErrs: 1,
|
|
|
|
valFields: []string{"charts[0].axes", "charts[0].axes"},
|
|
|
|
encoding: EncodingJSON,
|
|
|
|
pkgStr: `
|
|
|
|
{
|
|
|
|
"apiVersion": "0.1.0",
|
|
|
|
"kind": "Package",
|
|
|
|
"meta": {
|
|
|
|
"pkgName": "pkg_name",
|
|
|
|
"pkgVersion": "1",
|
|
|
|
"description": "pack description"
|
|
|
|
},
|
|
|
|
"spec": {
|
|
|
|
"resources": [
|
|
|
|
{
|
|
|
|
"kind": "Dashboard",
|
|
|
|
"name": "dashboard w/ single scatter chart",
|
|
|
|
"description": "a dashboard w/ single scatter chart",
|
|
|
|
"charts": [
|
|
|
|
{
|
|
|
|
"kind": "scatter",
|
|
|
|
"name": "scatter chart",
|
|
|
|
"note": "scatter note",
|
|
|
|
"noteOnEmpty": true,
|
|
|
|
"xPos": 1,
|
|
|
|
"yPos": 2,
|
|
|
|
"width": 6,
|
|
|
|
"height": 3,
|
|
|
|
"xCol": "_time",
|
|
|
|
"yCol": "_value",
|
|
|
|
"queries": [
|
|
|
|
{
|
|
|
|
"query": "from(bucket: v.bucket) |> range(start: v.timeRangeStart) |> filter(fn: (r) => r._measurement == \"mem\") |> filter(fn: (r) => r._field == \"used_percent\") |> aggregateWindow(every: v.windowPeriod, fn: mean, createEmpty: false) |> yield(name: \"mean\")"
|
|
|
|
}
|
|
|
|
],
|
|
|
|
"colors": [
|
|
|
|
{
|
|
|
|
"hex": "#8F8AF4"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"hex": "#F4CF31"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"hex": "#FFFFFF"
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
}`,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "missing query value",
|
|
|
|
validationErrs: 1,
|
|
|
|
valFields: []string{"charts[0].queries[0].query"},
|
|
|
|
encoding: EncodingJSON,
|
|
|
|
pkgStr: `
|
|
|
|
{
|
|
|
|
"apiVersion": "0.1.0",
|
|
|
|
"kind": "Package",
|
|
|
|
"meta": {
|
|
|
|
"pkgName": "pkg_name",
|
|
|
|
"pkgVersion": "1",
|
|
|
|
"description": "pack description"
|
|
|
|
},
|
|
|
|
"spec": {
|
|
|
|
"resources": [
|
|
|
|
{
|
|
|
|
"kind": "Dashboard",
|
|
|
|
"name": "dashboard w/ single scatter chart",
|
|
|
|
"description": "a dashboard w/ single scatter chart",
|
|
|
|
"charts": [
|
|
|
|
{
|
|
|
|
"kind": "scatter",
|
|
|
|
"name": "scatter chart",
|
|
|
|
"note": "scatter note",
|
|
|
|
"noteOnEmpty": true,
|
|
|
|
"xPos": 1,
|
|
|
|
"yPos": 2,
|
|
|
|
"width": 6,
|
|
|
|
"height": 3,
|
|
|
|
"xCol": "_time",
|
|
|
|
"yCol": "_value",
|
|
|
|
"queries": [
|
|
|
|
{
|
|
|
|
"query": null
|
|
|
|
}
|
|
|
|
],
|
|
|
|
"axes":[
|
|
|
|
{
|
|
|
|
"name": "x",
|
|
|
|
"label": "x_label",
|
|
|
|
"prefix": "x_prefix",
|
|
|
|
"suffix": "x_suffix"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"name": "y",
|
|
|
|
"label": "y_label",
|
|
|
|
"prefix": "y_prefix",
|
|
|
|
"suffix": "y_suffix"
|
|
|
|
}
|
|
|
|
],
|
|
|
|
"colors": [
|
|
|
|
{
|
|
|
|
"hex": "#8F8AF4"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"hex": "#F4CF31"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"hex": "#FFFFFF"
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
`,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "no queries provided",
|
|
|
|
validationErrs: 1,
|
|
|
|
valFields: []string{"charts[0].queries"},
|
|
|
|
encoding: EncodingJSON,
|
|
|
|
pkgStr: `
|
|
|
|
{
|
|
|
|
"apiVersion": "0.1.0",
|
|
|
|
"kind": "Package",
|
|
|
|
"meta": {
|
|
|
|
"pkgName": "pkg_name",
|
|
|
|
"pkgVersion": "1",
|
|
|
|
"description": "pack description"
|
|
|
|
},
|
|
|
|
"spec": {
|
|
|
|
"resources": [
|
|
|
|
{
|
|
|
|
"kind": "Dashboard",
|
|
|
|
"name": "dashboard w/ single scatter chart",
|
|
|
|
"description": "a dashboard w/ single scatter chart",
|
|
|
|
"charts": [
|
|
|
|
{
|
|
|
|
"kind": "scatter",
|
|
|
|
"name": "scatter chart",
|
|
|
|
"note": "scatter note",
|
|
|
|
"noteOnEmpty": true,
|
|
|
|
"xPos": 1,
|
|
|
|
"yPos": 2,
|
|
|
|
"width": 6,
|
|
|
|
"height": 3,
|
|
|
|
"xCol": "_time",
|
|
|
|
"yCol": "_value",
|
|
|
|
"queries": [],
|
|
|
|
"axes":[
|
|
|
|
{
|
|
|
|
"name": "x",
|
|
|
|
"label": "x_label",
|
|
|
|
"prefix": "x_prefix",
|
|
|
|
"suffix": "x_suffix"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"name": "y",
|
|
|
|
"label": "y_label",
|
|
|
|
"prefix": "y_prefix",
|
|
|
|
"suffix": "y_suffix"
|
|
|
|
}
|
|
|
|
],
|
|
|
|
"colors": [
|
|
|
|
{
|
|
|
|
"hex": "#8F8AF4"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"hex": "#F4CF31"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"hex": "#FFFFFF"
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
`,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "no width provided",
|
|
|
|
validationErrs: 1,
|
|
|
|
valFields: []string{"charts[0].width"},
|
|
|
|
encoding: EncodingJSON,
|
|
|
|
pkgStr: `
|
|
|
|
{
|
|
|
|
"apiVersion": "0.1.0",
|
|
|
|
"kind": "Package",
|
|
|
|
"meta": {
|
|
|
|
"pkgName": "pkg_name",
|
|
|
|
"pkgVersion": "1",
|
|
|
|
"description": "pack description"
|
|
|
|
},
|
|
|
|
"spec": {
|
|
|
|
"resources": [
|
|
|
|
{
|
|
|
|
"kind": "Dashboard",
|
|
|
|
"name": "dashboard w/ single scatter chart",
|
|
|
|
"description": "a dashboard w/ single scatter chart",
|
|
|
|
"charts": [
|
|
|
|
{
|
|
|
|
"kind": "scatter",
|
|
|
|
"name": "scatter chart",
|
|
|
|
"note": "scatter note",
|
|
|
|
"noteOnEmpty": true,
|
|
|
|
"xPos": 1,
|
|
|
|
"yPos": 2,
|
|
|
|
"height": 3,
|
|
|
|
"xCol": "_time",
|
|
|
|
"yCol": "_value",
|
|
|
|
"queries": [
|
|
|
|
{
|
|
|
|
"query": "from(bucket: v.bucket) |> range(start: v.timeRangeStart) |> filter(fn: (r) => r._measurement == \"mem\") |> filter(fn: (r) => r._field == \"used_percent\") |> aggregateWindow(every: v.windowPeriod, fn: mean, createEmpty: false) |> yield(name: \"mean\")"
|
|
|
|
}
|
|
|
|
],
|
|
|
|
"axes":[
|
|
|
|
{
|
|
|
|
"name": "x",
|
|
|
|
"label": "x_label",
|
|
|
|
"prefix": "x_prefix",
|
|
|
|
"suffix": "x_suffix"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"name": "y",
|
|
|
|
"label": "y_label",
|
|
|
|
"prefix": "y_prefix",
|
|
|
|
"suffix": "y_suffix"
|
|
|
|
}
|
|
|
|
],
|
|
|
|
"colors": [
|
|
|
|
{
|
|
|
|
"hex": "#8F8AF4"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"hex": "#F4CF31"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"hex": "#FFFFFF"
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
`,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "no height provided",
|
|
|
|
validationErrs: 1,
|
|
|
|
valFields: []string{"charts[0].height"},
|
|
|
|
encoding: EncodingJSON,
|
|
|
|
pkgStr: `
|
|
|
|
{
|
|
|
|
"apiVersion": "0.1.0",
|
|
|
|
"kind": "Package",
|
|
|
|
"meta": {
|
|
|
|
"pkgName": "pkg_name",
|
|
|
|
"pkgVersion": "1",
|
|
|
|
"description": "pack description"
|
|
|
|
},
|
|
|
|
"spec": {
|
|
|
|
"resources": [
|
|
|
|
{
|
|
|
|
"kind": "Dashboard",
|
|
|
|
"name": "dashboard w/ single scatter chart",
|
|
|
|
"description": "a dashboard w/ single scatter chart",
|
|
|
|
"charts": [
|
|
|
|
{
|
|
|
|
"kind": "scatter",
|
|
|
|
"name": "scatter chart",
|
|
|
|
"note": "scatter note",
|
|
|
|
"noteOnEmpty": true,
|
|
|
|
"xPos": 1,
|
|
|
|
"yPos": 2,
|
|
|
|
"width": 6,
|
|
|
|
"xCol": "_time",
|
|
|
|
"yCol": "_value",
|
|
|
|
"queries": [
|
|
|
|
{
|
|
|
|
"query": "from(bucket: v.bucket) |> range(start: v.timeRangeStart) |> filter(fn: (r) => r._measurement == \"mem\") |> filter(fn: (r) => r._field == \"used_percent\") |> aggregateWindow(every: v.windowPeriod, fn: mean, createEmpty: false) |> yield(name: \"mean\")"
|
|
|
|
}
|
|
|
|
],
|
|
|
|
"axes":[
|
|
|
|
{
|
|
|
|
"name": "x",
|
|
|
|
"label": "x_label",
|
|
|
|
"prefix": "x_prefix",
|
|
|
|
"suffix": "x_suffix"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"name": "y",
|
|
|
|
"label": "y_label",
|
|
|
|
"prefix": "y_prefix",
|
|
|
|
"suffix": "y_suffix"
|
|
|
|
}
|
|
|
|
],
|
|
|
|
"colors": [
|
|
|
|
{
|
|
|
|
"hex": "#8F8AF4"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"hex": "#F4CF31"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"hex": "#FFFFFF"
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
`,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "missing hex color",
|
|
|
|
validationErrs: 1,
|
|
|
|
valFields: []string{"charts[0].colors[0].hex"},
|
|
|
|
encoding: EncodingJSON,
|
|
|
|
pkgStr: `
|
|
|
|
{
|
|
|
|
"apiVersion": "0.1.0",
|
|
|
|
"kind": "Package",
|
|
|
|
"meta": {
|
|
|
|
"pkgName": "pkg_name",
|
|
|
|
"pkgVersion": "1",
|
|
|
|
"description": "pack description"
|
|
|
|
},
|
|
|
|
"spec": {
|
|
|
|
"resources": [
|
|
|
|
{
|
|
|
|
"kind": "Dashboard",
|
|
|
|
"name": "dashboard w/ single scatter chart",
|
|
|
|
"description": "a dashboard w/ single scatter chart",
|
|
|
|
"charts": [
|
|
|
|
{
|
|
|
|
"kind": "scatter",
|
|
|
|
"name": "scatter chart",
|
|
|
|
"note": "scatter note",
|
|
|
|
"noteOnEmpty": true,
|
|
|
|
"xPos": 1,
|
|
|
|
"yPos": 2,
|
|
|
|
"width": 6,
|
|
|
|
"height": 3,
|
|
|
|
"xCol": "_time",
|
|
|
|
"yCol": "_value",
|
|
|
|
"queries": [
|
|
|
|
{
|
|
|
|
"query": "from(bucket: v.bucket) |> range(start: v.timeRangeStart) |> filter(fn: (r) => r._measurement == \"mem\") |> filter(fn: (r) => r._field == \"used_percent\") |> aggregateWindow(every: v.windowPeriod, fn: mean, createEmpty: false) |> yield(name: \"mean\")"
|
|
|
|
}
|
|
|
|
],
|
|
|
|
"axes":[
|
|
|
|
{
|
|
|
|
"name": "x",
|
|
|
|
"label": "x_label",
|
|
|
|
"prefix": "x_prefix",
|
|
|
|
"suffix": "x_suffix"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"name": "y",
|
|
|
|
"label": "y_label",
|
|
|
|
"prefix": "y_prefix",
|
|
|
|
"suffix": "y_suffix"
|
|
|
|
}
|
|
|
|
],
|
|
|
|
"colors": [
|
|
|
|
{
|
|
|
|
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"hex": "#F4CF31"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"hex": "#FFFFFF"
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
`,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "missing x axis",
|
|
|
|
validationErrs: 1,
|
|
|
|
valFields: []string{"charts[0].axes"},
|
|
|
|
encoding: EncodingJSON,
|
|
|
|
pkgStr: `
|
|
|
|
{
|
|
|
|
"apiVersion": "0.1.0",
|
|
|
|
"kind": "Package",
|
|
|
|
"meta": {
|
|
|
|
"pkgName": "pkg_name",
|
|
|
|
"pkgVersion": "1",
|
|
|
|
"description": "pack description"
|
|
|
|
},
|
|
|
|
"spec": {
|
|
|
|
"resources": [
|
|
|
|
{
|
|
|
|
"kind": "Dashboard",
|
|
|
|
"name": "dashboard w/ single scatter chart",
|
|
|
|
"description": "a dashboard w/ single scatter chart",
|
|
|
|
"charts": [
|
|
|
|
{
|
|
|
|
"kind": "scatter",
|
|
|
|
"name": "scatter chart",
|
|
|
|
"note": "scatter note",
|
|
|
|
"noteOnEmpty": true,
|
|
|
|
"xPos": 1,
|
|
|
|
"yPos": 2,
|
|
|
|
"width": 6,
|
|
|
|
"height": 3,
|
|
|
|
"xCol": "_time",
|
|
|
|
"yCol": "_value",
|
|
|
|
"queries": [
|
|
|
|
{
|
|
|
|
"query": "from(bucket: v.bucket) |> range(start: v.timeRangeStart) |> filter(fn: (r) => r._measurement == \"mem\") |> filter(fn: (r) => r._field == \"used_percent\") |> aggregateWindow(every: v.windowPeriod, fn: mean, createEmpty: false) |> yield(name: \"mean\")"
|
|
|
|
}
|
|
|
|
],
|
|
|
|
"axes":[
|
|
|
|
{
|
|
|
|
"name": "y",
|
|
|
|
"label": "y_label",
|
|
|
|
"prefix": "y_prefix",
|
|
|
|
"suffix": "y_suffix"
|
|
|
|
}
|
|
|
|
],
|
|
|
|
"colors": [
|
|
|
|
{
|
|
|
|
"hex": "#8F8AF4"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"hex": "#F4CF31"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"hex": "#FFFFFF"
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
`,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "missing y axis",
|
|
|
|
validationErrs: 1,
|
|
|
|
valFields: []string{"charts[0].axes"},
|
|
|
|
encoding: EncodingJSON,
|
|
|
|
pkgStr: `
|
|
|
|
{
|
|
|
|
"apiVersion": "0.1.0",
|
|
|
|
"kind": "Package",
|
|
|
|
"meta": {
|
|
|
|
"pkgName": "pkg_name",
|
|
|
|
"pkgVersion": "1",
|
|
|
|
"description": "pack description"
|
|
|
|
},
|
|
|
|
"spec": {
|
|
|
|
"resources": [
|
|
|
|
{
|
|
|
|
"kind": "Dashboard",
|
|
|
|
"name": "dashboard w/ single scatter chart",
|
|
|
|
"description": "a dashboard w/ single scatter chart",
|
|
|
|
"charts": [
|
|
|
|
{
|
|
|
|
"kind": "scatter",
|
|
|
|
"name": "scatter chart",
|
|
|
|
"note": "scatter note",
|
|
|
|
"noteOnEmpty": true,
|
|
|
|
"xPos": 1,
|
|
|
|
"yPos": 2,
|
|
|
|
"width": 6,
|
|
|
|
"height": 3,
|
|
|
|
"xCol": "_time",
|
|
|
|
"yCol": "_value",
|
|
|
|
"queries": [
|
|
|
|
{
|
|
|
|
"query": "from(bucket: v.bucket) |> range(start: v.timeRangeStart) |> filter(fn: (r) => r._measurement == \"mem\") |> filter(fn: (r) => r._field == \"used_percent\") |> aggregateWindow(every: v.windowPeriod, fn: mean, createEmpty: false) |> yield(name: \"mean\")"
|
|
|
|
}
|
|
|
|
],
|
|
|
|
"axes":[
|
|
|
|
{
|
|
|
|
"name": "y",
|
|
|
|
"label": "y_label",
|
|
|
|
"prefix": "y_prefix",
|
|
|
|
"suffix": "y_suffix"
|
|
|
|
}
|
|
|
|
],
|
|
|
|
"colors": [
|
|
|
|
{
|
|
|
|
"hex": "#8F8AF4"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"hex": "#F4CF31"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"hex": "#FFFFFF"
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
`,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, tt := range tests {
|
|
|
|
testPkgErrors(t, KindDashboard, tt)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
2019-11-01 18:11:42 +00:00
|
|
|
t.Run("single stat chart", func(t *testing.T) {
|
|
|
|
testfileRunner(t, "testdata/dashboard", func(t *testing.T, pkg *Pkg) {
|
|
|
|
sum := pkg.Summary()
|
|
|
|
require.Len(t, sum.Dashboards, 1)
|
|
|
|
|
|
|
|
actual := sum.Dashboards[0]
|
|
|
|
assert.Equal(t, "dash_1", actual.Name)
|
|
|
|
assert.Equal(t, "desc1", actual.Description)
|
|
|
|
|
|
|
|
require.Len(t, actual.Charts, 1)
|
|
|
|
actualChart := actual.Charts[0]
|
|
|
|
assert.Equal(t, 3, actualChart.Height)
|
|
|
|
assert.Equal(t, 6, actualChart.Width)
|
|
|
|
assert.Equal(t, 1, actualChart.XPosition)
|
|
|
|
assert.Equal(t, 2, actualChart.YPosition)
|
|
|
|
|
|
|
|
props, ok := actualChart.Properties.(influxdb.SingleStatViewProperties)
|
|
|
|
require.True(t, ok)
|
2019-11-04 19:16:32 +00:00
|
|
|
assert.Equal(t, "single-stat", props.GetType())
|
2019-11-01 18:11:42 +00:00
|
|
|
assert.Equal(t, "single stat note", props.Note)
|
|
|
|
assert.True(t, props.ShowNoteWhenEmpty)
|
|
|
|
assert.True(t, props.DecimalPlaces.IsEnforced)
|
|
|
|
assert.Equal(t, int32(1), props.DecimalPlaces.Digits)
|
|
|
|
assert.Equal(t, "days", props.Suffix)
|
|
|
|
assert.Equal(t, "sumtin", props.Prefix)
|
|
|
|
|
|
|
|
require.Len(t, props.Queries, 1)
|
|
|
|
q := props.Queries[0]
|
|
|
|
queryText := `from(bucket: v.bucket) |> range(start: v.timeRangeStart) |> filter(fn: (r) => r._measurement == "processes") |> filter(fn: (r) => r._field == "running" or r._field == "blocked") |> aggregateWindow(every: v.windowPeriod, fn: max) |> yield(name: "max")`
|
|
|
|
assert.Equal(t, queryText, q.Text)
|
|
|
|
assert.Equal(t, "advanced", q.EditMode)
|
|
|
|
|
|
|
|
require.Len(t, props.ViewColors, 1)
|
|
|
|
c := props.ViewColors[0]
|
|
|
|
assert.NotZero(t, c.ID)
|
|
|
|
assert.Equal(t, "laser", c.Name)
|
|
|
|
assert.Equal(t, "text", c.Type)
|
|
|
|
assert.Equal(t, "#8F8AF4", c.Hex)
|
|
|
|
assert.Equal(t, 3.0, c.Value)
|
|
|
|
})
|
2019-10-30 21:13:42 +00:00
|
|
|
|
2019-11-01 18:11:42 +00:00
|
|
|
t.Run("handles invalid config", func(t *testing.T) {
|
2019-11-06 22:41:06 +00:00
|
|
|
tests := []testPkgResourceError{
|
2019-11-01 18:11:42 +00:00
|
|
|
{
|
2019-11-06 22:41:06 +00:00
|
|
|
name: "color missing hex value",
|
|
|
|
validationErrs: 1,
|
|
|
|
valFields: []string{"charts[0].colors[0].hex"},
|
|
|
|
pkgStr: `apiVersion: 0.1.0
|
2019-11-01 18:11:42 +00:00
|
|
|
kind: Package
|
|
|
|
meta:
|
|
|
|
pkgName: pkg_name
|
|
|
|
pkgVersion: 1
|
|
|
|
description: pack description
|
|
|
|
spec:
|
|
|
|
resources:
|
|
|
|
- kind: Dashboard
|
|
|
|
name: dash_1
|
|
|
|
description: desc1
|
|
|
|
charts:
|
|
|
|
- kind: Single_Stat
|
|
|
|
name: single stat
|
|
|
|
suffix: days
|
|
|
|
width: 6
|
|
|
|
height: 3
|
|
|
|
shade: true
|
|
|
|
queries:
|
|
|
|
- query: >
|
|
|
|
from(bucket: v.bucket) |> range(start: v.timeRangeStart) |> filter(fn: (r) => r._measurement == "system") |> filter(fn: (r) => r._field == "uptime") |> last() |> map(fn: (r) => ({r with _value: r._value / 86400})) |> yield(name: "last")
|
|
|
|
colors:
|
|
|
|
- name: laser
|
|
|
|
type: text
|
|
|
|
`,
|
|
|
|
},
|
|
|
|
{
|
2019-11-06 22:41:06 +00:00
|
|
|
name: "query missing text value",
|
|
|
|
validationErrs: 1,
|
|
|
|
valFields: []string{"charts[0].queries[0].query"},
|
|
|
|
pkgStr: `apiVersion: 0.1.0
|
2019-11-01 18:11:42 +00:00
|
|
|
kind: Package
|
|
|
|
meta:
|
|
|
|
pkgName: pkg_name
|
|
|
|
pkgVersion: 1
|
|
|
|
description: pack description
|
|
|
|
spec:
|
|
|
|
resources:
|
|
|
|
- kind: Dashboard
|
|
|
|
name: dash_1
|
|
|
|
description: desc1
|
|
|
|
charts:
|
|
|
|
- kind: Single_Stat
|
|
|
|
name: single stat
|
|
|
|
suffix: days
|
|
|
|
width: 6
|
|
|
|
height: 3
|
|
|
|
shade: true
|
|
|
|
queries:
|
|
|
|
- query:
|
|
|
|
colors:
|
|
|
|
- name: laser
|
|
|
|
type: text
|
|
|
|
hex: "#aaa222"
|
|
|
|
`,
|
|
|
|
},
|
|
|
|
{
|
2019-11-06 22:41:06 +00:00
|
|
|
name: "no queries provided",
|
|
|
|
validationErrs: 1,
|
|
|
|
valFields: []string{"charts[0].queries"},
|
|
|
|
pkgStr: `apiVersion: 0.1.0
|
2019-11-01 18:11:42 +00:00
|
|
|
kind: Package
|
|
|
|
meta:
|
|
|
|
pkgName: pkg_name
|
|
|
|
pkgVersion: 1
|
|
|
|
description: pack description
|
|
|
|
spec:
|
|
|
|
resources:
|
|
|
|
- kind: Dashboard
|
|
|
|
name: dash_1
|
|
|
|
description: desc1
|
|
|
|
charts:
|
|
|
|
- kind: Single_Stat
|
|
|
|
name: single stat
|
|
|
|
suffix: days
|
|
|
|
width: 6
|
|
|
|
height: 3
|
|
|
|
shade: true
|
|
|
|
colors:
|
|
|
|
- name: laser
|
|
|
|
type: text
|
|
|
|
hex: "#aaa222"
|
|
|
|
`,
|
|
|
|
},
|
|
|
|
{
|
2019-11-06 22:41:06 +00:00
|
|
|
name: "no width provided",
|
|
|
|
validationErrs: 1,
|
|
|
|
valFields: []string{"charts[0].width"},
|
|
|
|
pkgStr: `apiVersion: 0.1.0
|
2019-11-01 18:11:42 +00:00
|
|
|
kind: Package
|
|
|
|
meta:
|
|
|
|
pkgName: pkg_name
|
|
|
|
pkgVersion: 1
|
|
|
|
description: pack description
|
|
|
|
spec:
|
|
|
|
resources:
|
|
|
|
- kind: Dashboard
|
|
|
|
name: dash_1
|
|
|
|
description: desc1
|
|
|
|
charts:
|
|
|
|
- kind: Single_Stat
|
|
|
|
name: single stat
|
|
|
|
suffix: days
|
|
|
|
height: 3
|
|
|
|
shade: true
|
|
|
|
queries:
|
|
|
|
- query: >
|
|
|
|
from(bucket: v.bucket) |> range(start: v.timeRangeStart) |> filter(fn: (r) => r._measurement == "system") |> filter(fn: (r) => r._field == "uptime") |> last() |> map(fn: (r) => ({r with _value: r._value / 86400})) |> yield(name: "last")
|
|
|
|
colors:
|
|
|
|
- name: laser
|
|
|
|
type: text
|
|
|
|
hex: "#aaa333"
|
|
|
|
`,
|
|
|
|
},
|
|
|
|
{
|
2019-11-06 22:41:06 +00:00
|
|
|
name: "no height provided",
|
|
|
|
validationErrs: 1,
|
|
|
|
valFields: []string{"charts[0].height"},
|
|
|
|
pkgStr: `apiVersion: 0.1.0
|
2019-11-01 18:11:42 +00:00
|
|
|
kind: Package
|
|
|
|
meta:
|
|
|
|
pkgName: pkg_name
|
|
|
|
pkgVersion: 1
|
|
|
|
description: pack description
|
|
|
|
spec:
|
|
|
|
resources:
|
|
|
|
- kind: Dashboard
|
|
|
|
name: dash_1
|
|
|
|
description: desc1
|
|
|
|
charts:
|
|
|
|
- kind: Single_Stat
|
|
|
|
name: single stat
|
|
|
|
suffix: days
|
|
|
|
width: 3
|
|
|
|
shade: true
|
|
|
|
queries:
|
|
|
|
- query: >
|
|
|
|
from(bucket: v.bucket) |> range(start: v.timeRangeStart) |> filter(fn: (r) => r._measurement == "system") |> filter(fn: (r) => r._field == "uptime") |> last() |> map(fn: (r) => ({r with _value: r._value / 86400})) |> yield(name: "last")
|
|
|
|
colors:
|
|
|
|
- name: laser
|
|
|
|
type: text
|
|
|
|
hex: "#aaa333"
|
|
|
|
`,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, tt := range tests {
|
2019-11-08 19:33:41 +00:00
|
|
|
testPkgErrors(t, KindDashboard, tt)
|
2019-11-04 19:16:32 +00:00
|
|
|
}
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("single stat plus line chart", func(t *testing.T) {
|
|
|
|
testfileRunner(t, "testdata/dashboard_single_stat_plus_line", func(t *testing.T, pkg *Pkg) {
|
|
|
|
sum := pkg.Summary()
|
|
|
|
require.Len(t, sum.Dashboards, 1)
|
|
|
|
|
|
|
|
actual := sum.Dashboards[0]
|
|
|
|
assert.Equal(t, "dash_1", actual.Name)
|
|
|
|
assert.Equal(t, "desc1", actual.Description)
|
|
|
|
|
|
|
|
require.Len(t, actual.Charts, 1)
|
|
|
|
actualChart := actual.Charts[0]
|
|
|
|
assert.Equal(t, 3, actualChart.Height)
|
|
|
|
assert.Equal(t, 6, actualChart.Width)
|
|
|
|
assert.Equal(t, 1, actualChart.XPosition)
|
|
|
|
assert.Equal(t, 2, actualChart.YPosition)
|
|
|
|
|
|
|
|
props, ok := actualChart.Properties.(influxdb.LinePlusSingleStatProperties)
|
|
|
|
require.True(t, ok)
|
|
|
|
assert.Equal(t, "single stat plus line note", props.Note)
|
|
|
|
assert.True(t, props.ShowNoteWhenEmpty)
|
|
|
|
assert.True(t, props.DecimalPlaces.IsEnforced)
|
|
|
|
assert.Equal(t, int32(1), props.DecimalPlaces.Digits)
|
|
|
|
assert.Equal(t, "days", props.Suffix)
|
|
|
|
assert.Equal(t, "sumtin", props.Prefix)
|
2019-12-03 22:59:07 +00:00
|
|
|
assert.Equal(t, "overlaid", props.Position)
|
2019-11-04 19:16:32 +00:00
|
|
|
assert.Equal(t, "leg_type", props.Legend.Type)
|
|
|
|
assert.Equal(t, "horizontal", props.Legend.Orientation)
|
|
|
|
|
|
|
|
require.Len(t, props.Queries, 1)
|
|
|
|
q := props.Queries[0]
|
|
|
|
expectedQuery := `from(bucket: v.bucket) |> range(start: v.timeRangeStart) |> filter(fn: (r) => r._measurement == "mem") |> filter(fn: (r) => r._field == "used_percent") |> aggregateWindow(every: v.windowPeriod, fn: mean, createEmpty: false) |> yield(name: "mean")`
|
|
|
|
assert.Equal(t, expectedQuery, q.Text)
|
|
|
|
assert.Equal(t, "advanced", q.EditMode)
|
|
|
|
|
|
|
|
for _, key := range []string{"x", "y"} {
|
|
|
|
xAxis, ok := props.Axes[key]
|
|
|
|
require.True(t, ok, "key="+key)
|
|
|
|
assert.Equal(t, "10", xAxis.Base, "key="+key)
|
|
|
|
assert.Equal(t, key+"_label", xAxis.Label, "key="+key)
|
|
|
|
assert.Equal(t, key+"_prefix", xAxis.Prefix, "key="+key)
|
|
|
|
assert.Equal(t, "linear", xAxis.Scale, "key="+key)
|
|
|
|
assert.Equal(t, key+"_suffix", xAxis.Suffix, "key="+key)
|
|
|
|
}
|
|
|
|
|
|
|
|
require.Len(t, props.ViewColors, 2)
|
|
|
|
c := props.ViewColors[0]
|
|
|
|
assert.NotZero(t, c.ID)
|
|
|
|
assert.Equal(t, "laser", c.Name)
|
|
|
|
assert.Equal(t, "text", c.Type)
|
|
|
|
assert.Equal(t, "#8F8AF4", c.Hex)
|
|
|
|
assert.Equal(t, 3.0, c.Value)
|
|
|
|
|
|
|
|
c = props.ViewColors[1]
|
|
|
|
assert.NotZero(t, c.ID)
|
|
|
|
assert.Equal(t, "android", c.Name)
|
|
|
|
assert.Equal(t, "scale", c.Type)
|
|
|
|
assert.Equal(t, "#F4CF31", c.Hex)
|
|
|
|
assert.Equal(t, 1.0, c.Value)
|
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("handles invalid config", func(t *testing.T) {
|
2019-11-06 22:41:06 +00:00
|
|
|
tests := []testPkgResourceError{
|
2019-11-04 19:16:32 +00:00
|
|
|
{
|
2019-11-06 22:41:06 +00:00
|
|
|
name: "color missing hex value",
|
|
|
|
validationErrs: 1,
|
|
|
|
valFields: []string{"charts[0].colors[0].hex"},
|
|
|
|
pkgStr: `apiVersion: 0.1.0
|
2019-11-04 19:16:32 +00:00
|
|
|
kind: Package
|
|
|
|
meta:
|
|
|
|
pkgName: pkg_name
|
|
|
|
pkgVersion: 1
|
|
|
|
description: pack description
|
|
|
|
spec:
|
|
|
|
resources:
|
|
|
|
- kind: Dashboard
|
|
|
|
name: dash_1
|
|
|
|
description: desc1
|
|
|
|
charts:
|
|
|
|
- kind: Single_Stat_Plus_Line
|
|
|
|
name: single stat plus line
|
|
|
|
width: 6
|
|
|
|
height: 3
|
|
|
|
queries:
|
|
|
|
- query: >
|
|
|
|
from(bucket: v.bucket) |> range(start: v.timeRangeStart) |> filter(fn: (r) => r._measurement == "system") |> filter(fn: (r) => r._field == "uptime") |> last() |> map(fn: (r) => ({r with _value: r._value / 86400})) |> yield(name: "last")
|
|
|
|
colors:
|
|
|
|
- name: laser
|
|
|
|
type: text
|
|
|
|
- name: android
|
|
|
|
type: scale
|
|
|
|
hex: "#F4CF31"
|
|
|
|
axes:
|
|
|
|
- name : "x"
|
|
|
|
label: x_label
|
|
|
|
base: 10
|
|
|
|
scale: linear
|
|
|
|
- name: "y"
|
|
|
|
label: y_label
|
|
|
|
base: 10
|
|
|
|
scale: linear
|
|
|
|
`,
|
|
|
|
},
|
|
|
|
{
|
2019-11-06 22:41:06 +00:00
|
|
|
name: "missing query value",
|
|
|
|
validationErrs: 1,
|
|
|
|
valFields: []string{"charts[0].queries[0].query"},
|
|
|
|
pkgStr: `apiVersion: 0.1.0
|
2019-11-04 19:16:32 +00:00
|
|
|
kind: Package
|
|
|
|
meta:
|
|
|
|
pkgName: pkg_name
|
|
|
|
pkgVersion: 1
|
|
|
|
description: pack description
|
|
|
|
spec:
|
|
|
|
resources:
|
|
|
|
- kind: Dashboard
|
|
|
|
name: dash_1
|
|
|
|
description: desc1
|
|
|
|
charts:
|
|
|
|
- kind: Single_Stat_Plus_Line
|
|
|
|
name: single stat plus line
|
|
|
|
width: 6
|
|
|
|
height: 3
|
|
|
|
queries:
|
|
|
|
- query:
|
|
|
|
colors:
|
|
|
|
- name: laser
|
|
|
|
type: text
|
|
|
|
hex: "#abcabc"
|
|
|
|
- name: android
|
|
|
|
type: scale
|
|
|
|
hex: "#F4CF31"
|
|
|
|
axes:
|
|
|
|
- name : "x"
|
|
|
|
label: x_label
|
|
|
|
base: 10
|
|
|
|
scale: linear
|
|
|
|
- name: "y"
|
|
|
|
label: y_label
|
|
|
|
base: 10
|
|
|
|
scale: linear
|
|
|
|
`,
|
|
|
|
},
|
|
|
|
{
|
2019-11-06 22:41:06 +00:00
|
|
|
name: "no queries provided",
|
|
|
|
validationErrs: 1,
|
|
|
|
valFields: []string{"charts[0].queries"},
|
|
|
|
pkgStr: `apiVersion: 0.1.0
|
2019-11-04 19:16:32 +00:00
|
|
|
kind: Package
|
|
|
|
meta:
|
|
|
|
pkgName: pkg_name
|
|
|
|
pkgVersion: 1
|
|
|
|
description: pack description
|
|
|
|
spec:
|
|
|
|
resources:
|
|
|
|
- kind: Dashboard
|
|
|
|
name: dash_1
|
|
|
|
description: desc1
|
|
|
|
charts:
|
|
|
|
- kind: Single_Stat_Plus_Line
|
|
|
|
name: single stat plus line
|
|
|
|
width: 6
|
|
|
|
height: 3
|
|
|
|
colors:
|
|
|
|
- name: laser
|
|
|
|
type: text
|
|
|
|
hex: "red"
|
|
|
|
- name: android
|
|
|
|
type: scale
|
|
|
|
hex: "#F4CF31"
|
|
|
|
axes:
|
|
|
|
- name : "x"
|
|
|
|
label: x_label
|
|
|
|
base: 10
|
|
|
|
scale: linear
|
|
|
|
- name: "y"
|
|
|
|
label: y_label
|
|
|
|
base: 10
|
|
|
|
scale: linear`,
|
|
|
|
},
|
|
|
|
{
|
2019-11-06 22:41:06 +00:00
|
|
|
name: "no width provided",
|
|
|
|
validationErrs: 1,
|
|
|
|
valFields: []string{"charts[0].width"},
|
|
|
|
pkgStr: `apiVersion: 0.1.0
|
2019-11-04 19:16:32 +00:00
|
|
|
kind: Package
|
|
|
|
meta:
|
|
|
|
pkgName: pkg_name
|
|
|
|
pkgVersion: 1
|
|
|
|
description: pack description
|
|
|
|
spec:
|
|
|
|
resources:
|
|
|
|
- kind: Dashboard
|
|
|
|
name: dash_1
|
|
|
|
description: desc1
|
|
|
|
charts:
|
|
|
|
- kind: Single_Stat_Plus_Line
|
|
|
|
name: single stat plus line
|
|
|
|
height: 3
|
|
|
|
queries:
|
|
|
|
- query: >
|
|
|
|
from(bucket: v.bucket) |> range(start: v.timeRangeStart) |> filter(fn: (r) => r._measurement == "system") |> filter(fn: (r) => r._field == "uptime") |> last() |> map(fn: (r) => ({r with _value: r._value / 86400})) |> yield(name: "last")
|
|
|
|
colors:
|
|
|
|
- name: laser
|
|
|
|
type: text
|
|
|
|
hex: green
|
|
|
|
- name: android
|
|
|
|
type: scale
|
|
|
|
hex: "#F4CF31"
|
|
|
|
axes:
|
|
|
|
- name : "x"
|
|
|
|
label: x_label
|
|
|
|
base: 10
|
|
|
|
scale: linear
|
|
|
|
- name: "y"
|
|
|
|
label: y_label
|
|
|
|
base: 10
|
|
|
|
scale: linear`,
|
|
|
|
},
|
|
|
|
{
|
2019-11-06 22:41:06 +00:00
|
|
|
name: "no height provided",
|
|
|
|
validationErrs: 1,
|
|
|
|
valFields: []string{"charts[0].height"},
|
|
|
|
pkgStr: `apiVersion: 0.1.0
|
2019-11-04 19:16:32 +00:00
|
|
|
kind: Package
|
|
|
|
meta:
|
|
|
|
pkgName: pkg_name
|
|
|
|
pkgVersion: 1
|
|
|
|
description: pack description
|
|
|
|
spec:
|
|
|
|
resources:
|
|
|
|
- kind: Dashboard
|
|
|
|
name: dash_1
|
|
|
|
description: desc1
|
|
|
|
charts:
|
|
|
|
- kind: Single_Stat_Plus_Line
|
|
|
|
name: single stat plus line
|
|
|
|
width: 3
|
|
|
|
queries:
|
|
|
|
- query: >
|
|
|
|
from(bucket: v.bucket) |> range(start: v.timeRangeStart) |> filter(fn: (r) => r._measurement == "system") |> filter(fn: (r) => r._field == "uptime") |> last() |> map(fn: (r) => ({r with _value: r._value / 86400})) |> yield(name: "last")
|
|
|
|
colors:
|
|
|
|
- name: laser
|
|
|
|
type: text
|
|
|
|
hex: green
|
|
|
|
- name: android
|
|
|
|
type: scale
|
|
|
|
hex: "#F4CF31"
|
|
|
|
axes:
|
|
|
|
- name : "x"
|
|
|
|
label: x_label
|
|
|
|
base: 10
|
|
|
|
scale: linear
|
|
|
|
- name: "y"
|
|
|
|
label: y_label
|
|
|
|
base: 10
|
|
|
|
scale: linear`,
|
|
|
|
},
|
|
|
|
{
|
2019-11-06 22:41:06 +00:00
|
|
|
name: "missing text color but has scale color",
|
|
|
|
validationErrs: 1,
|
|
|
|
valFields: []string{"charts[0].colors"},
|
|
|
|
pkgStr: `apiVersion: 0.1.0
|
2019-11-04 19:16:32 +00:00
|
|
|
kind: Package
|
|
|
|
meta:
|
|
|
|
pkgName: pkg_name
|
|
|
|
pkgVersion: 1
|
|
|
|
description: pack description
|
|
|
|
spec:
|
|
|
|
resources:
|
|
|
|
- kind: Dashboard
|
|
|
|
name: dash_1
|
|
|
|
description: desc1
|
|
|
|
charts:
|
|
|
|
- kind: Single_Stat_Plus_Line
|
|
|
|
name: single stat plus line
|
|
|
|
width: 3
|
|
|
|
height: 3
|
|
|
|
queries:
|
|
|
|
- query: >
|
|
|
|
from(bucket: v.bucket) |> range(start: v.timeRangeStart) |> filter(fn: (r) => r._measurement == "system") |> filter(fn: (r) => r._field == "uptime") |> last() |> map(fn: (r) => ({r with _value: r._value / 86400})) |> yield(name: "last")
|
|
|
|
colors:
|
|
|
|
- name: android
|
|
|
|
type: scale
|
|
|
|
hex: "#F4CF31"
|
|
|
|
axes:
|
|
|
|
- name : "x"
|
|
|
|
label: x_label
|
|
|
|
base: 10
|
|
|
|
scale: linear
|
|
|
|
- name: "y"
|
|
|
|
label: y_label
|
|
|
|
base: 10
|
|
|
|
scale: linear`,
|
|
|
|
},
|
|
|
|
{
|
2019-11-06 22:41:06 +00:00
|
|
|
name: "missing x axis",
|
|
|
|
validationErrs: 1,
|
|
|
|
valFields: []string{"charts[0].axes"},
|
|
|
|
pkgStr: `apiVersion: 0.1.0
|
2019-11-04 19:16:32 +00:00
|
|
|
kind: Package
|
|
|
|
meta:
|
|
|
|
pkgName: pkg_name
|
|
|
|
pkgVersion: 1
|
|
|
|
description: pack description
|
|
|
|
spec:
|
|
|
|
resources:
|
|
|
|
- kind: Dashboard
|
|
|
|
name: dash_1
|
|
|
|
description: desc1
|
|
|
|
charts:
|
|
|
|
- kind: Single_Stat_Plus_Line
|
|
|
|
name: single stat plus line
|
|
|
|
width: 3
|
|
|
|
height: 3
|
|
|
|
queries:
|
|
|
|
- query: >
|
|
|
|
from(bucket: v.bucket) |> range(start: v.timeRangeStart) |> filter(fn: (r) => r._measurement == "system") |> filter(fn: (r) => r._field == "uptime") |> last() |> map(fn: (r) => ({r with _value: r._value / 86400})) |> yield(name: "last")
|
|
|
|
colors:
|
|
|
|
- name: first
|
|
|
|
type: text
|
|
|
|
hex: "#aabbaa"
|
|
|
|
- name: android
|
|
|
|
type: scale
|
|
|
|
hex: "#F4CF31"
|
|
|
|
axes:
|
|
|
|
- name: "y"
|
|
|
|
label: y_label
|
|
|
|
base: 10
|
|
|
|
scale: linear`,
|
|
|
|
},
|
|
|
|
{
|
2019-11-06 22:41:06 +00:00
|
|
|
name: "missing y axis",
|
|
|
|
validationErrs: 1,
|
|
|
|
valFields: []string{"charts[0].axes"},
|
|
|
|
pkgStr: `apiVersion: 0.1.0
|
2019-11-04 19:16:32 +00:00
|
|
|
kind: Package
|
|
|
|
meta:
|
|
|
|
pkgName: pkg_name
|
|
|
|
pkgVersion: 1
|
|
|
|
description: pack description
|
|
|
|
spec:
|
|
|
|
resources:
|
|
|
|
- kind: Dashboard
|
|
|
|
name: dash_1
|
|
|
|
description: desc1
|
|
|
|
charts:
|
|
|
|
- kind: Single_Stat_Plus_Line
|
|
|
|
name: single stat plus line
|
|
|
|
width: 3
|
|
|
|
height: 3
|
|
|
|
queries:
|
|
|
|
- query: >
|
|
|
|
from(bucket: v.bucket) |> range(start: v.timeRangeStart) |> filter(fn: (r) => r._measurement == "system") |> filter(fn: (r) => r._field == "uptime") |> last() |> map(fn: (r) => ({r with _value: r._value / 86400})) |> yield(name: "last")
|
|
|
|
colors:
|
|
|
|
- name: first
|
|
|
|
type: text
|
|
|
|
hex: "#aabbaa"
|
|
|
|
- name: android
|
|
|
|
type: scale
|
|
|
|
hex: "#F4CF31"
|
|
|
|
axes:
|
|
|
|
- name: "x"
|
|
|
|
base: 10
|
|
|
|
scale: linear`,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, tt := range tests {
|
2019-11-08 19:33:41 +00:00
|
|
|
testPkgErrors(t, KindDashboard, tt)
|
2019-11-01 18:11:42 +00:00
|
|
|
}
|
|
|
|
})
|
2019-10-30 21:13:42 +00:00
|
|
|
})
|
2019-11-05 19:19:25 +00:00
|
|
|
|
2019-11-12 18:06:53 +00:00
|
|
|
t.Run("single xy chart", func(t *testing.T) {
|
2019-11-06 22:41:06 +00:00
|
|
|
t.Run("xy chart", func(t *testing.T) {
|
|
|
|
testfileRunner(t, "testdata/dashboard_xy", func(t *testing.T, pkg *Pkg) {
|
|
|
|
sum := pkg.Summary()
|
|
|
|
require.Len(t, sum.Dashboards, 1)
|
2019-11-05 19:19:25 +00:00
|
|
|
|
2019-11-06 22:41:06 +00:00
|
|
|
actual := sum.Dashboards[0]
|
|
|
|
assert.Equal(t, "dash_1", actual.Name)
|
|
|
|
assert.Equal(t, "desc1", actual.Description)
|
2019-11-05 19:19:25 +00:00
|
|
|
|
2019-11-06 22:41:06 +00:00
|
|
|
require.Len(t, actual.Charts, 1)
|
|
|
|
actualChart := actual.Charts[0]
|
|
|
|
assert.Equal(t, 3, actualChart.Height)
|
|
|
|
assert.Equal(t, 6, actualChart.Width)
|
|
|
|
assert.Equal(t, 1, actualChart.XPosition)
|
|
|
|
assert.Equal(t, 2, actualChart.YPosition)
|
2019-11-05 19:19:25 +00:00
|
|
|
|
2019-11-06 22:41:06 +00:00
|
|
|
props, ok := actualChart.Properties.(influxdb.XYViewProperties)
|
|
|
|
require.True(t, ok)
|
|
|
|
assert.Equal(t, "xy", props.GetType())
|
|
|
|
assert.Equal(t, true, props.ShadeBelow)
|
|
|
|
assert.Equal(t, "xy chart note", props.Note)
|
|
|
|
assert.True(t, props.ShowNoteWhenEmpty)
|
2019-12-03 22:59:07 +00:00
|
|
|
assert.Equal(t, "stacked", props.Position)
|
2019-11-06 22:41:06 +00:00
|
|
|
|
|
|
|
require.Len(t, props.Queries, 1)
|
|
|
|
q := props.Queries[0]
|
|
|
|
queryText := `from(bucket: v.bucket) |> range(start: v.timeRangeStart, stop: v.timeRangeStop) |> filter(fn: (r) => r._measurement == "boltdb_writes_total") |> filter(fn: (r) => r._field == "counter")`
|
|
|
|
assert.Equal(t, queryText, q.Text)
|
|
|
|
assert.Equal(t, "advanced", q.EditMode)
|
|
|
|
|
|
|
|
require.Len(t, props.ViewColors, 1)
|
|
|
|
c := props.ViewColors[0]
|
|
|
|
assert.NotZero(t, c.ID)
|
|
|
|
assert.Equal(t, "laser", c.Name)
|
|
|
|
assert.Equal(t, "scale", c.Type)
|
|
|
|
assert.Equal(t, "#8F8AF4", c.Hex)
|
|
|
|
assert.Equal(t, 3.0, c.Value)
|
|
|
|
})
|
2019-11-05 19:19:25 +00:00
|
|
|
})
|
|
|
|
|
2019-11-06 22:41:06 +00:00
|
|
|
t.Run("handles invalid config", func(t *testing.T) {
|
|
|
|
tests := []testPkgResourceError{
|
|
|
|
{
|
|
|
|
name: "color missing hex value",
|
|
|
|
encoding: EncodingJSON,
|
|
|
|
validationErrs: 1,
|
|
|
|
valFields: []string{"charts[0].colors[0].hex"},
|
|
|
|
pkgStr: `{
|
2019-11-05 19:19:25 +00:00
|
|
|
"apiVersion": "0.1.0",
|
|
|
|
"kind": "Package",
|
|
|
|
"meta": {
|
|
|
|
"pkgName": "pkg_name",
|
|
|
|
"pkgVersion": "1",
|
|
|
|
"description": "pack description"
|
|
|
|
},
|
|
|
|
"spec": {
|
|
|
|
"resources": [
|
|
|
|
{
|
|
|
|
"kind": "Dashboard",
|
|
|
|
"name": "dash_1",
|
|
|
|
"description": "desc1",
|
|
|
|
"charts": [
|
|
|
|
{
|
|
|
|
"kind": "XY",
|
|
|
|
"name": "xy chart",
|
|
|
|
"prefix": "sumtin",
|
|
|
|
"note": "xy chart note",
|
|
|
|
"noteOnEmpty": true,
|
|
|
|
"xPos": 1,
|
|
|
|
"yPos": 2,
|
|
|
|
"width": 6,
|
|
|
|
"height": 3,
|
|
|
|
"decimalPlaces": 1,
|
|
|
|
"shade": true,
|
|
|
|
"xColumn": "_time",
|
|
|
|
"yColumn": "_value",
|
|
|
|
"legend": {},
|
|
|
|
"queries": [
|
|
|
|
{
|
|
|
|
"query": "from(bucket: v.bucket) |> range(start: v.timeRangeStart, stop: v.timeRangeStop) |> filter(fn: (r) => r._measurement == \"boltdb_writes_total\") |> filter(fn: (r) => r._field == \"counter\")"
|
|
|
|
}
|
|
|
|
],
|
|
|
|
"colors": [
|
|
|
|
{
|
|
|
|
"name": "laser",
|
|
|
|
"type": "scale",
|
|
|
|
"value": 3
|
|
|
|
}
|
|
|
|
],
|
|
|
|
"axes":[
|
|
|
|
{
|
|
|
|
"name": "x",
|
|
|
|
"label": "x_label",
|
|
|
|
"prefix": "x_prefix",
|
|
|
|
"suffix": "x_suffix",
|
|
|
|
"base": "10",
|
|
|
|
"scale": "linear"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"name": "y",
|
|
|
|
"label": "y_label",
|
|
|
|
"prefix": "y_prefix",
|
|
|
|
"suffix": "y_suffix",
|
|
|
|
"base": "10",
|
|
|
|
"scale": "linear"
|
|
|
|
}
|
|
|
|
],
|
|
|
|
"geom": "line"
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
`,
|
2019-11-06 22:41:06 +00:00
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "invalid geom flag",
|
|
|
|
encoding: EncodingJSON,
|
|
|
|
validationErrs: 1,
|
|
|
|
valFields: []string{"charts[0].geom"},
|
|
|
|
pkgStr: `
|
2019-11-05 19:19:25 +00:00
|
|
|
{
|
|
|
|
"apiVersion": "0.1.0",
|
|
|
|
"kind": "Package",
|
|
|
|
"meta": {
|
|
|
|
"pkgName": "pkg_name",
|
|
|
|
"pkgVersion": "1",
|
|
|
|
"description": "pack description"
|
|
|
|
},
|
|
|
|
"spec": {
|
|
|
|
"resources": [
|
|
|
|
{
|
|
|
|
"kind": "Dashboard",
|
|
|
|
"name": "dash_1",
|
|
|
|
"description": "desc1",
|
|
|
|
"charts": [
|
|
|
|
{
|
|
|
|
"kind": "XY",
|
|
|
|
"name": "xy chart",
|
|
|
|
"prefix": "sumtin",
|
|
|
|
"note": "xy chart note",
|
|
|
|
"noteOnEmpty": true,
|
|
|
|
"xPos": 1,
|
|
|
|
"yPos": 2,
|
|
|
|
"width": 6,
|
|
|
|
"height": 3,
|
|
|
|
"decimalPlaces": 1,
|
|
|
|
"shade": true,
|
|
|
|
"xColumn": "_time",
|
|
|
|
"yColumn": "_value",
|
|
|
|
"legend": {},
|
|
|
|
"queries": [
|
|
|
|
{
|
|
|
|
"query": "from(bucket: v.bucket) |> range(start: v.timeRangeStart, stop: v.timeRangeStop) |> filter(fn: (r) => r._measurement == \"boltdb_writes_total\") |> filter(fn: (r) => r._field == \"counter\")"
|
|
|
|
}
|
|
|
|
],
|
|
|
|
"colors": [
|
|
|
|
{
|
|
|
|
"name": "laser",
|
|
|
|
"type": "scale",
|
|
|
|
"value": 3,
|
|
|
|
"hex": "#FFF000"
|
|
|
|
}
|
|
|
|
],
|
|
|
|
"axes":[
|
|
|
|
{
|
|
|
|
"name": "x",
|
|
|
|
"label": "x_label",
|
|
|
|
"prefix": "x_prefix",
|
|
|
|
"suffix": "x_suffix",
|
|
|
|
"base": "10",
|
|
|
|
"scale": "linear"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"name": "y",
|
|
|
|
"label": "y_label",
|
|
|
|
"prefix": "y_prefix",
|
|
|
|
"suffix": "y_suffix",
|
|
|
|
"base": "10",
|
|
|
|
"scale": "linear"
|
|
|
|
}
|
|
|
|
],
|
|
|
|
"geom": "huzzah"
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
`,
|
2019-11-06 22:41:06 +00:00
|
|
|
},
|
|
|
|
}
|
2019-11-05 19:19:25 +00:00
|
|
|
|
2019-11-06 22:41:06 +00:00
|
|
|
for _, tt := range tests {
|
2019-11-08 19:33:41 +00:00
|
|
|
testPkgErrors(t, KindDashboard, tt)
|
2019-11-05 19:19:25 +00:00
|
|
|
}
|
2019-11-06 22:41:06 +00:00
|
|
|
})
|
2019-11-05 19:19:25 +00:00
|
|
|
})
|
2019-11-06 19:22:36 +00:00
|
|
|
})
|
|
|
|
|
2019-10-30 21:13:42 +00:00
|
|
|
t.Run("pkg with dashboard and labels associated", func(t *testing.T) {
|
|
|
|
testfileRunner(t, "testdata/dashboard_associates_label", func(t *testing.T, pkg *Pkg) {
|
|
|
|
sum := pkg.Summary()
|
|
|
|
require.Len(t, sum.Dashboards, 1)
|
|
|
|
|
|
|
|
actual := sum.Dashboards[0]
|
|
|
|
assert.Equal(t, "dash_1", actual.Name)
|
|
|
|
assert.Equal(t, "desc1", actual.Description)
|
|
|
|
|
|
|
|
require.Len(t, actual.LabelAssociations, 1)
|
|
|
|
actualLabel := actual.LabelAssociations[0]
|
|
|
|
assert.Equal(t, "label_1", actualLabel.Name)
|
|
|
|
|
|
|
|
expectedMappings := []SummaryLabelMapping{
|
|
|
|
{
|
|
|
|
ResourceName: "dash_1",
|
|
|
|
LabelName: "label_1",
|
|
|
|
},
|
|
|
|
}
|
|
|
|
require.Len(t, sum.LabelMappings, len(expectedMappings))
|
|
|
|
|
|
|
|
for i, expected := range expectedMappings {
|
|
|
|
expected.LabelMapping.ResourceType = influxdb.DashboardsResourceType
|
|
|
|
assert.Equal(t, expected, sum.LabelMappings[i])
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("association doesn't exist then provides an error", func(t *testing.T) {
|
2019-11-06 22:41:06 +00:00
|
|
|
tests := []testPkgResourceError{
|
2019-10-30 21:13:42 +00:00
|
|
|
{
|
|
|
|
name: "no labels provided",
|
2019-11-06 22:41:06 +00:00
|
|
|
assErrs: 1,
|
|
|
|
assIdxs: []int{0},
|
|
|
|
pkgStr: `apiVersion: 0.1.0
|
2019-10-30 21:13:42 +00:00
|
|
|
kind: Package
|
|
|
|
meta:
|
|
|
|
pkgName: label_pkg
|
|
|
|
pkgVersion: 1
|
|
|
|
spec:
|
|
|
|
resources:
|
|
|
|
- kind: Dashboard
|
|
|
|
name: dash_1
|
|
|
|
associations:
|
|
|
|
- kind: Label
|
|
|
|
name: label_1
|
|
|
|
`,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "mixed found and not found",
|
2019-11-06 22:41:06 +00:00
|
|
|
assErrs: 1,
|
|
|
|
assIdxs: []int{1},
|
|
|
|
pkgStr: `apiVersion: 0.1.0
|
2019-10-30 21:13:42 +00:00
|
|
|
kind: Package
|
|
|
|
meta:
|
|
|
|
pkgName: label_pkg
|
|
|
|
pkgVersion: 1
|
|
|
|
spec:
|
|
|
|
resources:
|
|
|
|
- kind: Label
|
|
|
|
name: label_1
|
|
|
|
- kind: Dashboard
|
|
|
|
name: dash_1
|
|
|
|
associations:
|
|
|
|
- kind: Label
|
|
|
|
name: label_1
|
|
|
|
- kind: Label
|
|
|
|
name: unfound label
|
|
|
|
`,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "multiple not found",
|
2019-11-06 22:41:06 +00:00
|
|
|
assErrs: 1,
|
|
|
|
assIdxs: []int{0, 1},
|
|
|
|
pkgStr: `apiVersion: 0.1.0
|
2019-10-30 21:13:42 +00:00
|
|
|
kind: Package
|
|
|
|
meta:
|
|
|
|
pkgName: label_pkg
|
|
|
|
pkgVersion: 1
|
|
|
|
spec:
|
|
|
|
resources:
|
|
|
|
- kind: Label
|
|
|
|
name: label_1
|
|
|
|
- kind: Dashboard
|
|
|
|
name: dash_1
|
|
|
|
associations:
|
|
|
|
- kind: Label
|
|
|
|
name: not found 1
|
|
|
|
- kind: Label
|
|
|
|
name: unfound label
|
|
|
|
`,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "duplicate valid nested labels",
|
2019-11-06 22:41:06 +00:00
|
|
|
assErrs: 1,
|
|
|
|
assIdxs: []int{1},
|
|
|
|
pkgStr: `apiVersion: 0.1.0
|
2019-10-30 21:13:42 +00:00
|
|
|
kind: Package
|
|
|
|
meta:
|
|
|
|
pkgName: label_pkg
|
|
|
|
pkgVersion: 1
|
|
|
|
spec:
|
|
|
|
resources:
|
|
|
|
- kind: Label
|
|
|
|
name: label_1
|
|
|
|
- kind: Dashboard
|
|
|
|
name: dash_1
|
|
|
|
associations:
|
|
|
|
- kind: Label
|
|
|
|
name: label_1
|
|
|
|
- kind: Label
|
|
|
|
name: label_1
|
|
|
|
`,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, tt := range tests {
|
2019-11-08 19:33:41 +00:00
|
|
|
testPkgErrors(t, KindDashboard, tt)
|
2019-11-06 22:41:06 +00:00
|
|
|
}
|
|
|
|
})
|
|
|
|
})
|
2019-10-30 21:13:42 +00:00
|
|
|
|
2019-12-03 18:22:59 +00:00
|
|
|
t.Run("pkg with telegraf and label associations", func(t *testing.T) {
|
|
|
|
t.Run("with valid fields", func(t *testing.T) {
|
|
|
|
testfileRunner(t, "testdata/telegraf", func(t *testing.T, pkg *Pkg) {
|
|
|
|
sum := pkg.Summary()
|
|
|
|
require.Len(t, sum.TelegrafConfigs, 1)
|
|
|
|
|
|
|
|
actual := sum.TelegrafConfigs[0]
|
|
|
|
assert.Equal(t, "first_tele_config", actual.Name)
|
|
|
|
assert.Equal(t, "desc", actual.Description)
|
|
|
|
|
|
|
|
require.Len(t, actual.LabelAssociations, 1)
|
|
|
|
assert.Equal(t, "label_1", actual.LabelAssociations[0].Name)
|
2019-12-06 00:53:00 +00:00
|
|
|
|
|
|
|
require.Len(t, sum.LabelMappings, 1)
|
|
|
|
expectedMapping := SummaryLabelMapping{
|
|
|
|
ResourceName: "first_tele_config",
|
|
|
|
LabelName: "label_1",
|
|
|
|
LabelMapping: influxdb.LabelMapping{
|
|
|
|
ResourceType: influxdb.TelegrafsResourceType,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
assert.Equal(t, expectedMapping, sum.LabelMappings[0])
|
2019-12-03 18:22:59 +00:00
|
|
|
})
|
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("handles bad config", func(t *testing.T) {
|
|
|
|
tests := []testPkgResourceError{
|
|
|
|
{
|
|
|
|
name: "config missing",
|
|
|
|
validationErrs: 1,
|
|
|
|
valFields: []string{"config"},
|
|
|
|
pkgStr: `apiVersion: 0.1.0
|
|
|
|
kind: Package
|
|
|
|
meta:
|
|
|
|
pkgName: pkg_name
|
|
|
|
pkgVersion: 1
|
|
|
|
description: pack description
|
|
|
|
spec:
|
|
|
|
resources:
|
|
|
|
- kind: Telegraf
|
|
|
|
name: tele_name
|
|
|
|
description: desc
|
|
|
|
`,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, tt := range tests {
|
|
|
|
testPkgErrors(t, KindTelegraf, tt)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
2019-11-06 22:41:06 +00:00
|
|
|
t.Run("pkg with a variable", func(t *testing.T) {
|
|
|
|
t.Run("with valid fields should produce summary", func(t *testing.T) {
|
|
|
|
testfileRunner(t, "testdata/variables", func(t *testing.T, pkg *Pkg) {
|
|
|
|
sum := pkg.Summary()
|
2019-10-30 21:13:42 +00:00
|
|
|
|
2019-11-06 22:41:06 +00:00
|
|
|
require.Len(t, sum.Variables, 4)
|
2019-10-30 21:13:42 +00:00
|
|
|
|
2019-11-06 22:41:06 +00:00
|
|
|
varEquals := func(t *testing.T, name, vType string, vals interface{}, v SummaryVariable) {
|
|
|
|
t.Helper()
|
|
|
|
|
|
|
|
assert.Equal(t, name, v.Name)
|
|
|
|
assert.Equal(t, name+" desc", v.Description)
|
|
|
|
require.NotNil(t, v.Arguments)
|
|
|
|
assert.Equal(t, vType, v.Arguments.Type)
|
|
|
|
assert.Equal(t, vals, v.Arguments.Values)
|
2019-10-30 21:13:42 +00:00
|
|
|
}
|
|
|
|
|
2019-11-06 22:41:06 +00:00
|
|
|
// validates we support all known variable types
|
|
|
|
varEquals(t,
|
|
|
|
"var_const",
|
|
|
|
"constant",
|
|
|
|
influxdb.VariableConstantValues([]string{"first val"}),
|
|
|
|
sum.Variables[0],
|
|
|
|
)
|
|
|
|
|
|
|
|
varEquals(t,
|
|
|
|
"var_map",
|
|
|
|
"map",
|
|
|
|
influxdb.VariableMapValues{"k1": "v1"},
|
|
|
|
sum.Variables[1],
|
|
|
|
)
|
|
|
|
|
|
|
|
varEquals(t,
|
|
|
|
"var_query_1",
|
|
|
|
"query",
|
|
|
|
influxdb.VariableQueryValues{
|
|
|
|
Query: `buckets() |> filter(fn: (r) => r.name !~ /^_/) |> rename(columns: {name: "_value"}) |> keep(columns: ["_value"])`,
|
|
|
|
Language: "flux",
|
|
|
|
},
|
|
|
|
sum.Variables[2],
|
|
|
|
)
|
|
|
|
|
|
|
|
varEquals(t,
|
|
|
|
"var_query_2",
|
|
|
|
"query",
|
|
|
|
influxdb.VariableQueryValues{
|
|
|
|
Query: "an influxql query of sorts",
|
|
|
|
Language: "influxql",
|
|
|
|
},
|
|
|
|
sum.Variables[3],
|
|
|
|
)
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("handles bad config", func(t *testing.T) {
|
|
|
|
tests := []testPkgResourceError{
|
|
|
|
{
|
|
|
|
name: "name missing",
|
|
|
|
validationErrs: 1,
|
|
|
|
valFields: []string{"name"},
|
|
|
|
pkgStr: `apiVersion: 0.1.0
|
|
|
|
kind: Package
|
|
|
|
meta:
|
|
|
|
pkgName: pkg_name
|
|
|
|
pkgVersion: 1
|
|
|
|
description: pack description
|
|
|
|
spec:
|
|
|
|
resources:
|
|
|
|
- kind: Variable
|
|
|
|
type: map
|
|
|
|
values:
|
|
|
|
k1: v1
|
|
|
|
`,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "map var missing values",
|
|
|
|
validationErrs: 1,
|
|
|
|
valFields: []string{"values"},
|
|
|
|
pkgStr: `apiVersion: 0.1.0
|
|
|
|
kind: Package
|
|
|
|
meta:
|
|
|
|
pkgName: pkg_name
|
|
|
|
pkgVersion: 1
|
|
|
|
description: pack description
|
|
|
|
spec:
|
|
|
|
resources:
|
|
|
|
- kind: Variable
|
|
|
|
name: var
|
|
|
|
type: map
|
|
|
|
`,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "const var missing values",
|
|
|
|
validationErrs: 1,
|
|
|
|
valFields: []string{"values"},
|
|
|
|
pkgStr: `apiVersion: 0.1.0
|
|
|
|
kind: Package
|
|
|
|
meta:
|
|
|
|
pkgName: pkg_name
|
|
|
|
pkgVersion: 1
|
|
|
|
description: pack description
|
|
|
|
spec:
|
|
|
|
resources:
|
|
|
|
- kind: Variable
|
|
|
|
name: var
|
|
|
|
type: constant
|
|
|
|
`,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "query var missing query",
|
|
|
|
validationErrs: 1,
|
|
|
|
valFields: []string{"query"},
|
|
|
|
pkgStr: `apiVersion: 0.1.0
|
|
|
|
kind: Package
|
|
|
|
meta:
|
|
|
|
pkgName: pkg_name
|
|
|
|
pkgVersion: 1
|
|
|
|
description: pack description
|
|
|
|
spec:
|
|
|
|
resources:
|
|
|
|
- kind: Variable
|
|
|
|
name: var
|
|
|
|
type: query
|
|
|
|
language: influxql
|
|
|
|
`,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "query var missing query language",
|
|
|
|
validationErrs: 1,
|
|
|
|
valFields: []string{"language"},
|
|
|
|
pkgStr: `apiVersion: 0.1.0
|
|
|
|
kind: Package
|
|
|
|
meta:
|
|
|
|
pkgName: pkg_name
|
|
|
|
pkgVersion: 1
|
|
|
|
description: pack description
|
|
|
|
spec:
|
|
|
|
resources:
|
|
|
|
- kind: Variable
|
|
|
|
name: var
|
|
|
|
type: query
|
|
|
|
query: from(v.bucket) |> count()
|
|
|
|
`,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "query var provides incorrect query language",
|
|
|
|
validationErrs: 1,
|
|
|
|
valFields: []string{"language"},
|
|
|
|
pkgStr: `apiVersion: 0.1.0
|
|
|
|
kind: Package
|
|
|
|
meta:
|
|
|
|
pkgName: pkg_name
|
|
|
|
pkgVersion: 1
|
|
|
|
description: pack description
|
|
|
|
spec:
|
|
|
|
resources:
|
|
|
|
- kind: Variable
|
|
|
|
name: var
|
|
|
|
type: query
|
|
|
|
query: from(v.bucket) |> count()
|
|
|
|
language: wrongo language
|
|
|
|
`,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "duplicate var names",
|
|
|
|
validationErrs: 1,
|
|
|
|
valFields: []string{"name"},
|
|
|
|
pkgStr: `apiVersion: 0.1.0
|
|
|
|
kind: Package
|
|
|
|
meta:
|
|
|
|
pkgName: pkg_name
|
|
|
|
pkgVersion: 1
|
|
|
|
description: pack description
|
|
|
|
spec:
|
|
|
|
resources:
|
|
|
|
- kind: Variable
|
|
|
|
name: var
|
|
|
|
type: query
|
|
|
|
query: from(v.bucket) |> count()
|
|
|
|
language: flux
|
|
|
|
- kind: Variable
|
|
|
|
name: var
|
|
|
|
type: query
|
|
|
|
query: from(v.bucket) |> mean()
|
|
|
|
language: flux
|
|
|
|
`,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, tt := range tests {
|
2019-11-08 19:33:41 +00:00
|
|
|
testPkgErrors(t, KindVariable, tt)
|
2019-10-30 21:13:42 +00:00
|
|
|
}
|
|
|
|
})
|
|
|
|
})
|
2019-11-07 00:45:00 +00:00
|
|
|
|
|
|
|
t.Run("pkg with variable and labels associated", func(t *testing.T) {
|
2019-12-06 00:53:00 +00:00
|
|
|
testfileRunner(t, "testdata/variable_associates_label.yml", func(t *testing.T, pkg *Pkg) {
|
2019-11-07 00:45:00 +00:00
|
|
|
sum := pkg.Summary()
|
|
|
|
require.Len(t, sum.Labels, 1)
|
|
|
|
|
|
|
|
vars := sum.Variables
|
|
|
|
require.Len(t, vars, 1)
|
|
|
|
|
|
|
|
expectedLabelMappings := []struct {
|
|
|
|
varName string
|
|
|
|
labels []string
|
|
|
|
}{
|
|
|
|
{
|
|
|
|
varName: "var_1",
|
|
|
|
labels: []string{"label_1"},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
for i, expected := range expectedLabelMappings {
|
|
|
|
v := vars[i]
|
|
|
|
require.Len(t, v.LabelAssociations, len(expected.labels))
|
|
|
|
|
|
|
|
for j, label := range expected.labels {
|
|
|
|
assert.Equal(t, label, v.LabelAssociations[j].Name)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
expectedMappings := []SummaryLabelMapping{
|
|
|
|
{
|
|
|
|
ResourceName: "var_1",
|
|
|
|
LabelName: "label_1",
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
require.Len(t, sum.LabelMappings, len(expectedMappings))
|
|
|
|
for i, expected := range expectedMappings {
|
|
|
|
expected.LabelMapping.ResourceType = influxdb.VariablesResourceType
|
|
|
|
assert.Equal(t, expected, sum.LabelMappings[i])
|
|
|
|
}
|
|
|
|
})
|
|
|
|
})
|
2019-10-23 17:09:04 +00:00
|
|
|
}
|
|
|
|
|
2019-11-22 01:07:12 +00:00
|
|
|
func Test_PkgValidationErr(t *testing.T) {
|
|
|
|
iPtr := func(i int) *int {
|
|
|
|
return &i
|
|
|
|
}
|
|
|
|
|
|
|
|
compIntSlcs := func(t *testing.T, expected []int, actuals []*int) {
|
|
|
|
t.Helper()
|
|
|
|
|
|
|
|
if len(expected) >= len(actuals) {
|
|
|
|
require.FailNow(t, "expected array is larger than actuals")
|
|
|
|
}
|
|
|
|
|
|
|
|
for i, actual := range actuals {
|
|
|
|
if i == len(expected) {
|
|
|
|
assert.Nil(t, actual)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
assert.Equal(t, expected[i], *actual)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pErr := &parseErr{
|
|
|
|
Resources: []resourceErr{
|
|
|
|
{
|
|
|
|
Kind: KindDashboard.String(),
|
|
|
|
Idx: intPtr(0),
|
|
|
|
ValidationErrs: []validationErr{
|
|
|
|
{
|
|
|
|
Field: "charts",
|
|
|
|
Index: iPtr(1),
|
|
|
|
Nested: []validationErr{
|
|
|
|
{
|
|
|
|
Field: "colors",
|
|
|
|
Index: iPtr(0),
|
|
|
|
Nested: []validationErr{
|
|
|
|
{
|
|
|
|
Field: "hex",
|
|
|
|
Msg: "hex value required",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Field: "kind",
|
|
|
|
Msg: "chart kind must be provided",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
errs := pErr.ValidationErrs()
|
|
|
|
require.Len(t, errs, 2)
|
|
|
|
assert.Equal(t, KindDashboard.String(), errs[0].Kind)
|
|
|
|
assert.Equal(t, []string{"spec.resources", "charts", "colors", "hex"}, errs[0].Fields)
|
|
|
|
compIntSlcs(t, []int{0, 1, 0}, errs[0].Indexes)
|
|
|
|
assert.Equal(t, "hex value required", errs[0].Reason)
|
|
|
|
|
|
|
|
assert.Equal(t, KindDashboard.String(), errs[1].Kind)
|
|
|
|
assert.Equal(t, []string{"spec.resources", "charts", "kind"}, errs[1].Fields)
|
|
|
|
compIntSlcs(t, []int{0, 1}, errs[1].Indexes)
|
|
|
|
assert.Equal(t, "chart kind must be provided", errs[1].Reason)
|
|
|
|
}
|
|
|
|
|
2019-11-06 22:41:06 +00:00
|
|
|
type testPkgResourceError struct {
|
|
|
|
name string
|
|
|
|
encoding Encoding
|
|
|
|
pkgStr string
|
|
|
|
resourceErrs int
|
|
|
|
validationErrs int
|
|
|
|
valFields []string
|
|
|
|
assErrs int
|
|
|
|
assIdxs []int
|
|
|
|
}
|
|
|
|
|
|
|
|
// defaults to yaml encoding if encoding not provided
|
|
|
|
// defaults num resources to 1 if resource errs not provided.
|
2019-11-08 19:33:41 +00:00
|
|
|
func testPkgErrors(t *testing.T, k Kind, tt testPkgResourceError) {
|
2019-11-06 22:41:06 +00:00
|
|
|
t.Helper()
|
|
|
|
encoding := EncodingYAML
|
|
|
|
if tt.encoding != EncodingUnknown {
|
|
|
|
encoding = tt.encoding
|
|
|
|
}
|
|
|
|
|
|
|
|
resErrs := 1
|
|
|
|
if tt.resourceErrs > 0 {
|
|
|
|
resErrs = tt.resourceErrs
|
|
|
|
}
|
|
|
|
|
|
|
|
fn := func(t *testing.T) {
|
|
|
|
t.Helper()
|
|
|
|
|
|
|
|
_, err := Parse(encoding, FromString(tt.pkgStr))
|
|
|
|
require.Error(t, err)
|
|
|
|
|
2019-11-14 00:43:28 +00:00
|
|
|
require.True(t, IsParseErr(err), err)
|
2019-11-06 22:41:06 +00:00
|
|
|
|
2019-11-22 01:07:12 +00:00
|
|
|
pErr := err.(*parseErr)
|
2019-11-06 22:41:06 +00:00
|
|
|
require.Len(t, pErr.Resources, resErrs)
|
|
|
|
|
|
|
|
resErr := pErr.Resources[0]
|
|
|
|
assert.Equal(t, k.String(), resErr.Kind)
|
|
|
|
|
2019-11-14 00:24:05 +00:00
|
|
|
for i, vFail := range resErr.ValidationErrs {
|
|
|
|
if len(tt.valFields) == i {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
expectedField := tt.valFields[i]
|
|
|
|
findErr(t, expectedField, vFail)
|
2019-11-06 22:41:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if tt.assErrs == 0 {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2019-11-14 00:24:05 +00:00
|
|
|
assFails := pErr.Resources[0].AssociationErrs
|
|
|
|
for i, assFail := range assFails {
|
|
|
|
if len(tt.valFields) == i {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
expectedField := tt.valFields[i]
|
|
|
|
findErr(t, expectedField, assFail)
|
2019-11-06 22:41:06 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
t.Run(tt.name, fn)
|
|
|
|
}
|
|
|
|
|
2019-11-22 01:07:12 +00:00
|
|
|
func findErr(t *testing.T, expectedField string, vErr validationErr) validationErr {
|
2019-11-14 00:24:05 +00:00
|
|
|
t.Helper()
|
|
|
|
|
|
|
|
fields := strings.Split(expectedField, ".")
|
|
|
|
if len(fields) == 1 {
|
|
|
|
require.Equal(t, expectedField, vErr.Field)
|
|
|
|
return vErr
|
|
|
|
}
|
|
|
|
|
|
|
|
currentFieldName, idx := nextField(t, fields[0])
|
|
|
|
if idx > -1 {
|
|
|
|
require.NotNil(t, vErr.Index)
|
|
|
|
require.Equal(t, idx, *vErr.Index)
|
|
|
|
}
|
|
|
|
require.Equal(t, currentFieldName, vErr.Field)
|
|
|
|
|
|
|
|
next := strings.Join(fields[1:], ".")
|
|
|
|
nestedField, _ := nextField(t, next)
|
|
|
|
for _, n := range vErr.Nested {
|
|
|
|
if n.Field == nestedField {
|
|
|
|
return findErr(t, next, n)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
assert.Fail(t, "did not find field: "+expectedField)
|
|
|
|
|
|
|
|
return vErr
|
|
|
|
}
|
|
|
|
|
|
|
|
func nextField(t *testing.T, field string) (string, int) {
|
|
|
|
t.Helper()
|
|
|
|
|
|
|
|
fields := strings.Split(field, ".")
|
|
|
|
if len(fields) == 1 && !strings.HasSuffix(fields[0], "]") {
|
|
|
|
return field, -1
|
|
|
|
}
|
|
|
|
parts := strings.Split(fields[0], "[")
|
|
|
|
if len(parts) == 1 {
|
|
|
|
return "", 0
|
|
|
|
}
|
|
|
|
fieldName := parts[0]
|
|
|
|
|
|
|
|
if strIdx := strings.Index(parts[1], "]"); strIdx > -1 {
|
|
|
|
idx, err := strconv.Atoi(parts[1][:strIdx])
|
|
|
|
require.NoError(t, err)
|
|
|
|
return fieldName, idx
|
|
|
|
}
|
|
|
|
return "", -1
|
|
|
|
}
|
|
|
|
|
2019-10-23 17:09:04 +00:00
|
|
|
type baseAsserts struct {
|
|
|
|
version string
|
2019-11-21 00:38:12 +00:00
|
|
|
kind Kind
|
2019-10-23 17:09:04 +00:00
|
|
|
description string
|
|
|
|
metaName string
|
|
|
|
metaVersion string
|
|
|
|
}
|
|
|
|
|
|
|
|
func validParsedPkg(t *testing.T, path string, encoding Encoding, expected baseAsserts) *Pkg {
|
|
|
|
t.Helper()
|
|
|
|
|
|
|
|
pkg, err := Parse(encoding, FromFile(path))
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
require.Equal(t, expected.version, pkg.APIVersion)
|
2019-11-21 00:38:12 +00:00
|
|
|
require.True(t, pkg.Kind.is(expected.kind))
|
2019-10-23 17:09:04 +00:00
|
|
|
require.Equal(t, expected.description, pkg.Metadata.Description)
|
|
|
|
require.Equal(t, expected.metaName, pkg.Metadata.Name)
|
|
|
|
require.Equal(t, expected.metaVersion, pkg.Metadata.Version)
|
2019-11-06 18:02:45 +00:00
|
|
|
require.True(t, pkg.isParsed)
|
2019-10-23 17:09:04 +00:00
|
|
|
|
|
|
|
return pkg
|
|
|
|
}
|
|
|
|
|
|
|
|
func testfileRunner(t *testing.T, path string, testFn func(t *testing.T, pkg *Pkg)) {
|
|
|
|
t.Helper()
|
|
|
|
|
|
|
|
tests := []struct {
|
|
|
|
name string
|
|
|
|
extension string
|
|
|
|
encoding Encoding
|
|
|
|
}{
|
|
|
|
{
|
|
|
|
name: "yaml",
|
2019-11-07 00:45:00 +00:00
|
|
|
extension: ".yml",
|
2019-10-23 17:09:04 +00:00
|
|
|
encoding: EncodingYAML,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "json",
|
2019-11-07 00:45:00 +00:00
|
|
|
extension: ".json",
|
2019-10-23 17:09:04 +00:00
|
|
|
encoding: EncodingJSON,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
2019-11-07 00:45:00 +00:00
|
|
|
ext := filepath.Ext(path)
|
|
|
|
switch ext {
|
|
|
|
case ".yml":
|
|
|
|
tests = tests[:1]
|
|
|
|
case ".json":
|
|
|
|
tests = tests[1:]
|
|
|
|
}
|
|
|
|
|
|
|
|
path = strings.TrimSuffix(path, ext)
|
|
|
|
|
2019-10-23 17:09:04 +00:00
|
|
|
for _, tt := range tests {
|
|
|
|
fn := func(t *testing.T) {
|
|
|
|
t.Helper()
|
|
|
|
|
2019-11-07 00:45:00 +00:00
|
|
|
pkg := validParsedPkg(t, path+tt.extension, tt.encoding, baseAsserts{
|
2019-10-23 17:09:04 +00:00
|
|
|
version: "0.1.0",
|
2019-11-21 00:38:12 +00:00
|
|
|
kind: KindPackage,
|
2019-10-23 17:09:04 +00:00
|
|
|
description: "pack description",
|
|
|
|
metaName: "pkg_name",
|
|
|
|
metaVersion: "1",
|
|
|
|
})
|
|
|
|
if testFn != nil {
|
|
|
|
testFn(t, pkg)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
t.Run(tt.name, fn)
|
|
|
|
}
|
|
|
|
}
|