feat(pkger): add support for combining pkgs

pull/16763/head
Johnny Steenbergen 2020-02-06 12:26:10 -08:00 committed by Johnny Steenbergen
parent 8e51fe0aad
commit 6ce0c3de7f
3 changed files with 127 additions and 5 deletions

View File

@ -363,6 +363,17 @@ func (p *Pkg) applySecrets(secrets map[string]string) {
} }
} }
// Combine combines pkgs together. Is useful when you want to take multiple disparate pkgs
// and compile them into one to take advantage of the parser and service guarantees.
func Combine(pkgs ...*Pkg) (*Pkg, error) {
newPkg := new(Pkg)
for _, p := range pkgs {
newPkg.Objects = append(newPkg.Objects, p.Objects...)
}
return newPkg, newPkg.Validate()
}
type ( type (
validateOpt struct { validateOpt struct {
minResources bool minResources bool

View File

@ -2,6 +2,7 @@ package pkger
import ( import (
"errors" "errors"
"fmt"
"path/filepath" "path/filepath"
"strconv" "strconv"
"strings" "strings"
@ -3409,7 +3410,7 @@ spec:
}) })
t.Run("jsonnet support", func(t *testing.T) { t.Run("jsonnet support", func(t *testing.T) {
pkg := validParsedPkg(t, "testdata/bucket_associates_labels.jsonnet", EncodingJsonnet) pkg := validParsedPkgFromFile(t, "testdata/bucket_associates_labels.jsonnet", EncodingJsonnet)
sum := pkg.Summary() sum := pkg.Summary()
@ -3448,6 +3449,111 @@ spec:
}) })
} }
func TestCombine(t *testing.T) {
newPkgFromYmlStr := func(t *testing.T, pkgStr string) *Pkg {
t.Helper()
return newParsedPkg(t, FromString(pkgStr), EncodingYAML, ValidSkipParseError())
}
associationsEqual := func(t *testing.T, summaryLabels []SummaryLabel, names ...string) {
t.Helper()
require.Len(t, summaryLabels, len(names))
m := make(map[string]bool)
for _, n := range names {
m[n] = true
}
for _, l := range summaryLabels {
if !m[l.Name] {
assert.Fail(t, "did not find label: "+l.Name)
}
delete(m, l.Name)
}
if len(m) > 0 {
var unexpectedLabels []string
for name := range m {
unexpectedLabels = append(unexpectedLabels, name)
}
assert.Failf(t, "additional labels found", "got: %v", unexpectedLabels)
}
}
t.Run("multiple pkgs with associations across files", func(t *testing.T) {
var pkgs []*Pkg
numLabels := 5
for i := 0; i < numLabels; i++ {
pkg := newPkgFromYmlStr(t, fmt.Sprintf(`
apiVersion: %[1]s
kind: Label
metadata:
name: label_%d
`, APIVersion, i))
pkgs = append(pkgs, pkg)
}
pkgs = append(pkgs, newPkgFromYmlStr(t, fmt.Sprintf(`
apiVersion: %[1]s
kind: Bucket
metadata:
name: rucket_1
spec:
associations:
- kind: Label
name: label_1
`, APIVersion)))
pkgs = append(pkgs, newPkgFromYmlStr(t, fmt.Sprintf(`
apiVersion: %[1]s
kind: Bucket
metadata:
name: rucket_2
spec:
associations:
- kind: Label
name: label_2
`, APIVersion)))
pkgs = append(pkgs, newPkgFromYmlStr(t, fmt.Sprintf(`
apiVersion: %[1]s
kind: Bucket
metadata:
name: rucket_3
spec:
associations:
- kind: Label
name: label_1
- kind: Label
name: label_2
`, APIVersion)))
combinedPkg, err := Combine(pkgs...)
require.NoError(t, err)
sum := combinedPkg.Summary()
require.Len(t, sum.Labels, numLabels)
for i := 0; i < numLabels; i++ {
assert.Equal(t, fmt.Sprintf("label_%d", i), sum.Labels[i].Name)
}
require.Len(t, sum.Labels, numLabels)
for i := 0; i < numLabels; i++ {
assert.Equal(t, fmt.Sprintf("label_%d", i), sum.Labels[i].Name)
}
require.Len(t, sum.Buckets, 3)
assert.Equal(t, "rucket_1", sum.Buckets[0].Name)
associationsEqual(t, sum.Buckets[0].LabelAssociations, "label_1")
assert.Equal(t, "rucket_2", sum.Buckets[1].Name)
associationsEqual(t, sum.Buckets[1].LabelAssociations, "label_2")
assert.Equal(t, "rucket_3", sum.Buckets[2].Name)
associationsEqual(t, sum.Buckets[2].LabelAssociations, "label_1", "label_2")
})
}
func Test_IsParseError(t *testing.T) { func Test_IsParseError(t *testing.T) {
tests := []struct { tests := []struct {
name string name string
@ -3679,10 +3785,15 @@ func nextField(t *testing.T, field string) (string, int) {
return "", -1 return "", -1
} }
func validParsedPkg(t *testing.T, path string, encoding Encoding) *Pkg { func validParsedPkgFromFile(t *testing.T, path string, encoding Encoding) *Pkg {
t.Helper()
return newParsedPkg(t, FromFile(path), encoding)
}
func newParsedPkg(t *testing.T, fn ReaderFn, encoding Encoding, opts ...ValidateOptFn) *Pkg {
t.Helper() t.Helper()
pkg, err := Parse(encoding, FromFile(path)) pkg, err := Parse(encoding, fn, opts...)
require.NoError(t, err) require.NoError(t, err)
for _, k := range pkg.Objects { for _, k := range pkg.Objects {
@ -3727,7 +3838,7 @@ func testfileRunner(t *testing.T, path string, testFn func(t *testing.T, pkg *Pk
fn := func(t *testing.T) { fn := func(t *testing.T) {
t.Helper() t.Helper()
pkg := validParsedPkg(t, path+tt.extension, tt.encoding) pkg := validParsedPkgFromFile(t, path+tt.extension, tt.encoding)
if testFn != nil { if testFn != nil {
testFn(t, pkg) testFn(t, pkg)
} }

View File

@ -890,7 +890,7 @@ func (s *Service) dryRunNotificationRules(ctx context.Context, orgID influxdb.ID
if !ok { if !ok {
influxEndpoint, ok := mPkgEndpoints[r.endpointName.String()] influxEndpoint, ok := mPkgEndpoints[r.endpointName.String()]
if !ok { if !ok {
err := fmt.Errorf("failed to find notification endpoint dependency for notification rule %q; endpointName: %q", r.Name(), r.endpointName) err := fmt.Errorf("failed to find notification endpoint %q dependency for notification rule %q", r.endpointName, r.Name())
return nil, &influxdb.Error{Code: influxdb.EUnprocessableEntity, Err: err} return nil, &influxdb.Error{Code: influxdb.EUnprocessableEntity, Err: err}
} }
e = influxEndpoint e = influxEndpoint