2019-10-23 17:09:04 +00:00
|
|
|
package pkger
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"encoding/json"
|
|
|
|
"errors"
|
|
|
|
"fmt"
|
|
|
|
"io"
|
|
|
|
"io/ioutil"
|
|
|
|
"sort"
|
|
|
|
"strconv"
|
|
|
|
"strings"
|
|
|
|
"time"
|
|
|
|
|
2019-11-01 18:11:42 +00:00
|
|
|
"github.com/influxdata/influxdb"
|
2019-10-23 17:09:04 +00:00
|
|
|
"gopkg.in/yaml.v3"
|
|
|
|
)
|
|
|
|
|
|
|
|
// ReaderFn is used for functional inputs to abstract the individual
|
|
|
|
// entrypoints for the reader itself.
|
|
|
|
type ReaderFn func() (io.Reader, error)
|
|
|
|
|
|
|
|
// Encoding describes the encoding for the raw package data. The
|
|
|
|
// encoding determines how the raw data is parsed.
|
|
|
|
type Encoding int
|
|
|
|
|
|
|
|
// encoding types
|
|
|
|
const (
|
2019-11-05 01:40:42 +00:00
|
|
|
EncodingUnknown Encoding = iota
|
|
|
|
EncodingYAML
|
2019-10-23 17:09:04 +00:00
|
|
|
EncodingJSON
|
|
|
|
)
|
|
|
|
|
2019-11-05 01:40:42 +00:00
|
|
|
// String provides the string representation of the encoding.
|
|
|
|
func (e Encoding) String() string {
|
|
|
|
switch e {
|
|
|
|
case EncodingJSON:
|
|
|
|
return "json"
|
|
|
|
case EncodingYAML:
|
|
|
|
return "yaml"
|
|
|
|
default:
|
|
|
|
return "unknown"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-10-30 17:55:13 +00:00
|
|
|
// ErrInvalidEncoding indicates the encoding is invalid type for the parser.
|
|
|
|
var ErrInvalidEncoding = errors.New("invalid encoding provided")
|
|
|
|
|
2019-10-23 17:09:04 +00:00
|
|
|
// Parse parses a pkg defined by the encoding and readerFns. As of writing this
|
|
|
|
// we can parse both a YAML and JSON format of the Pkg model.
|
|
|
|
func Parse(encoding Encoding, readerFn ReaderFn) (*Pkg, error) {
|
|
|
|
r, err := readerFn()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
switch encoding {
|
|
|
|
case EncodingYAML:
|
|
|
|
return parseYAML(r)
|
|
|
|
case EncodingJSON:
|
|
|
|
return parseJSON(r)
|
|
|
|
default:
|
2019-10-30 17:55:13 +00:00
|
|
|
return nil, ErrInvalidEncoding
|
2019-10-23 17:09:04 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// FromFile reads a file from disk and provides a reader from it.
|
|
|
|
func FromFile(filePath string) ReaderFn {
|
|
|
|
return func() (io.Reader, error) {
|
|
|
|
// not using os.Open to avoid having to deal with closing the file in here
|
|
|
|
b, err := ioutil.ReadFile(filePath)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return bytes.NewBuffer(b), nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// FromReader simply passes the reader along. Useful when consuming
|
|
|
|
// this from an HTTP request body. There are a number of other useful
|
|
|
|
// places for this functional input.
|
|
|
|
func FromReader(r io.Reader) ReaderFn {
|
|
|
|
return func() (io.Reader, error) {
|
|
|
|
return r, nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// FromString parses a pkg from a raw string value. This is very useful
|
|
|
|
// in tests.
|
|
|
|
func FromString(s string) ReaderFn {
|
|
|
|
return func() (io.Reader, error) {
|
|
|
|
return strings.NewReader(s), nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func parseYAML(r io.Reader) (*Pkg, error) {
|
|
|
|
return parse(yaml.NewDecoder(r))
|
|
|
|
}
|
|
|
|
|
|
|
|
func parseJSON(r io.Reader) (*Pkg, error) {
|
|
|
|
return parse(json.NewDecoder(r))
|
|
|
|
}
|
|
|
|
|
|
|
|
type decoder interface {
|
|
|
|
Decode(interface{}) error
|
|
|
|
}
|
|
|
|
|
|
|
|
func parse(dec decoder) (*Pkg, error) {
|
|
|
|
var pkg Pkg
|
2019-10-30 17:55:13 +00:00
|
|
|
if err := dec.Decode(&pkg); err != nil {
|
2019-10-23 17:09:04 +00:00
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2019-11-05 01:40:42 +00:00
|
|
|
if err := pkg.Validate(); err != nil {
|
|
|
|
return nil, err
|
2019-10-23 17:09:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return &pkg, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Pkg is the model for a package. The resources are more generic that one might
|
|
|
|
// expect at first glance. This was done on purpose. The way json/yaml/toml or
|
|
|
|
// w/e scripting you want to use, can have very different ways of parsing. The
|
|
|
|
// different parsers are limited for the parsers that do not come from the std
|
|
|
|
// lib (looking at you yaml/v2). This allows us to parse it and leave the matching
|
|
|
|
// to another power, the graphing of the package is handled within itself.
|
|
|
|
type Pkg struct {
|
|
|
|
APIVersion string `yaml:"apiVersion" json:"apiVersion"`
|
|
|
|
Kind string `yaml:"kind" json:"kind"`
|
|
|
|
Metadata Metadata `yaml:"meta" json:"meta"`
|
|
|
|
Spec struct {
|
|
|
|
Resources []Resource `yaml:"resources" json:"resources"`
|
|
|
|
} `yaml:"spec" json:"spec"`
|
|
|
|
|
2019-10-30 21:13:42 +00:00
|
|
|
mLabels map[string]*label
|
|
|
|
mBuckets map[string]*bucket
|
|
|
|
mDashboards map[string]*dashboard
|
2019-11-06 22:41:06 +00:00
|
|
|
mVariables map[string]*variable
|
2019-10-28 22:23:40 +00:00
|
|
|
|
2019-11-06 18:02:45 +00:00
|
|
|
isVerified bool // dry run has verified pkg resources with existing resources
|
|
|
|
isParsed bool // indicates the pkg has been parsed and all resources graphed accordingly
|
2019-10-23 17:09:04 +00:00
|
|
|
}
|
|
|
|
|
2019-10-30 21:13:42 +00:00
|
|
|
// Summary returns a package Summary that describes all the resources and
|
2019-10-23 17:09:04 +00:00
|
|
|
// associations the pkg contains. It is very useful for informing users of
|
|
|
|
// the changes that will take place when this pkg would be applied.
|
2019-10-26 02:11:47 +00:00
|
|
|
func (p *Pkg) Summary() Summary {
|
2019-10-23 17:09:04 +00:00
|
|
|
var sum Summary
|
|
|
|
|
2019-10-30 21:13:42 +00:00
|
|
|
for _, b := range p.buckets() {
|
2019-10-28 22:23:40 +00:00
|
|
|
sum.Buckets = append(sum.Buckets, b.summarize())
|
2019-10-23 17:09:04 +00:00
|
|
|
}
|
2019-10-30 21:13:42 +00:00
|
|
|
|
|
|
|
for _, d := range p.dashboards() {
|
|
|
|
sum.Dashboards = append(sum.Dashboards, d.summarize())
|
|
|
|
}
|
2019-10-23 17:09:04 +00:00
|
|
|
|
2019-11-06 22:41:06 +00:00
|
|
|
for _, l := range p.labels() {
|
|
|
|
sum.Labels = append(sum.Labels, l.summarize())
|
|
|
|
}
|
|
|
|
|
2019-10-28 22:23:40 +00:00
|
|
|
for _, m := range p.labelMappings() {
|
2019-10-30 17:55:13 +00:00
|
|
|
sum.LabelMappings = append(sum.LabelMappings, SummaryLabelMapping{
|
2019-10-28 22:23:40 +00:00
|
|
|
ResourceName: m.ResourceName,
|
|
|
|
LabelName: m.LabelName,
|
|
|
|
LabelMapping: m.LabelMapping,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2019-11-06 22:41:06 +00:00
|
|
|
for _, v := range p.variables() {
|
|
|
|
sum.Variables = append(sum.Variables, v.summarize())
|
|
|
|
}
|
|
|
|
|
2019-10-23 17:09:04 +00:00
|
|
|
return sum
|
|
|
|
}
|
|
|
|
|
2019-11-05 01:40:42 +00:00
|
|
|
// Validate will graph all resources and validate every thing is in a useful form.
|
|
|
|
func (p *Pkg) Validate() error {
|
|
|
|
setupFns := []func() error{
|
|
|
|
p.validMetadata,
|
|
|
|
p.validResources,
|
|
|
|
p.graphResources,
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, fn := range setupFns {
|
|
|
|
if err := fn(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
2019-11-06 18:02:45 +00:00
|
|
|
|
|
|
|
p.isParsed = true
|
2019-11-05 01:40:42 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2019-10-26 02:11:47 +00:00
|
|
|
func (p *Pkg) buckets() []*bucket {
|
|
|
|
buckets := make([]*bucket, 0, len(p.mBuckets))
|
|
|
|
for _, b := range p.mBuckets {
|
2019-10-23 17:09:04 +00:00
|
|
|
buckets = append(buckets, b)
|
|
|
|
}
|
|
|
|
|
|
|
|
sort.Slice(buckets, func(i, j int) bool {
|
|
|
|
return buckets[i].Name < buckets[j].Name
|
|
|
|
})
|
|
|
|
|
|
|
|
return buckets
|
|
|
|
}
|
|
|
|
|
2019-10-26 02:11:47 +00:00
|
|
|
func (p *Pkg) labels() []*label {
|
|
|
|
labels := make([]*label, 0, len(p.mLabels))
|
|
|
|
for _, b := range p.mLabels {
|
2019-10-24 23:59:01 +00:00
|
|
|
labels = append(labels, b)
|
|
|
|
}
|
|
|
|
|
|
|
|
sort.Slice(labels, func(i, j int) bool {
|
|
|
|
return labels[i].Name < labels[j].Name
|
|
|
|
})
|
|
|
|
|
|
|
|
return labels
|
|
|
|
}
|
|
|
|
|
2019-10-30 21:13:42 +00:00
|
|
|
func (p *Pkg) dashboards() []*dashboard {
|
|
|
|
dashes := make([]*dashboard, 0, len(p.mDashboards))
|
|
|
|
for _, d := range p.mDashboards {
|
|
|
|
dashes = append(dashes, d)
|
|
|
|
}
|
|
|
|
|
|
|
|
sort.Slice(dashes, func(i, j int) bool {
|
|
|
|
return dashes[i].Name < dashes[j].Name
|
|
|
|
})
|
|
|
|
|
|
|
|
return dashes
|
|
|
|
}
|
|
|
|
|
2019-11-06 22:41:06 +00:00
|
|
|
func (p *Pkg) variables() []*variable {
|
|
|
|
vars := make([]*variable, 0, len(p.mVariables))
|
|
|
|
for _, v := range p.mVariables {
|
|
|
|
vars = append(vars, v)
|
|
|
|
}
|
|
|
|
|
|
|
|
sort.Slice(vars, func(i, j int) bool {
|
|
|
|
return vars[i].Name < vars[j].Name
|
|
|
|
})
|
|
|
|
|
|
|
|
return vars
|
|
|
|
}
|
|
|
|
|
2019-10-26 02:11:47 +00:00
|
|
|
// labelMappings returns the mappings that will be created for
|
|
|
|
// valid pairs of labels and resources of which all have IDs.
|
|
|
|
// If a resource does not exist yet, a label mapping will not
|
|
|
|
// be returned for it.
|
2019-10-30 17:55:13 +00:00
|
|
|
func (p *Pkg) labelMappings() []SummaryLabelMapping {
|
|
|
|
var mappings []SummaryLabelMapping
|
2019-10-28 22:23:40 +00:00
|
|
|
for _, l := range p.mLabels {
|
|
|
|
mappings = append(mappings, l.mappingSummary()...)
|
2019-10-26 02:11:47 +00:00
|
|
|
}
|
|
|
|
|
2019-10-30 21:13:42 +00:00
|
|
|
// sort by res type ASC, then res name ASC, then label name ASC
|
|
|
|
sort.Slice(mappings, func(i, j int) bool {
|
|
|
|
n, m := mappings[i], mappings[j]
|
|
|
|
if n.ResourceType < m.ResourceType {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
if n.ResourceType > m.ResourceType {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
if n.ResourceName < m.ResourceName {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
if n.ResourceName > m.ResourceName {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
return n.LabelName < m.LabelName
|
|
|
|
})
|
|
|
|
|
2019-10-26 02:11:47 +00:00
|
|
|
return mappings
|
|
|
|
}
|
|
|
|
|
|
|
|
func (p *Pkg) validMetadata() error {
|
2019-10-23 17:09:04 +00:00
|
|
|
var failures []*failure
|
2019-10-26 02:11:47 +00:00
|
|
|
if p.APIVersion != "0.1.0" {
|
2019-10-23 17:09:04 +00:00
|
|
|
failures = append(failures, &failure{
|
2019-10-26 02:11:47 +00:00
|
|
|
Field: "apiVersion",
|
|
|
|
Msg: "must be version 0.1.0",
|
2019-10-23 17:09:04 +00:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2019-10-26 02:11:47 +00:00
|
|
|
mKind := kind(strings.TrimSpace(strings.ToLower(p.Kind)))
|
2019-10-23 17:09:04 +00:00
|
|
|
if mKind != kindPackage {
|
|
|
|
failures = append(failures, &failure{
|
2019-10-26 02:11:47 +00:00
|
|
|
Field: "kind",
|
|
|
|
Msg: `must be of kind "Package"`,
|
2019-10-23 17:09:04 +00:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2019-10-26 02:11:47 +00:00
|
|
|
if p.Metadata.Version == "" {
|
2019-10-23 17:09:04 +00:00
|
|
|
failures = append(failures, &failure{
|
2019-10-26 02:11:47 +00:00
|
|
|
Field: "pkgVersion",
|
|
|
|
Msg: "version is required",
|
2019-10-23 17:09:04 +00:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2019-10-26 02:11:47 +00:00
|
|
|
if p.Metadata.Name == "" {
|
2019-10-23 17:09:04 +00:00
|
|
|
failures = append(failures, &failure{
|
2019-10-26 02:11:47 +00:00
|
|
|
Field: "pkgName",
|
|
|
|
Msg: "must be at least 1 char",
|
2019-10-23 17:09:04 +00:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(failures) == 0 {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
res := errResource{
|
2019-11-06 22:41:06 +00:00
|
|
|
Kind: kindPackage.String(),
|
2019-10-23 17:09:04 +00:00
|
|
|
Idx: -1,
|
|
|
|
}
|
|
|
|
for _, f := range failures {
|
2019-10-26 02:11:47 +00:00
|
|
|
res.ValidationFails = append(res.ValidationFails, struct {
|
2019-10-23 17:09:04 +00:00
|
|
|
Field string
|
|
|
|
Msg string
|
|
|
|
}{
|
2019-10-26 02:11:47 +00:00
|
|
|
Field: f.Field,
|
|
|
|
Msg: f.Msg,
|
2019-10-23 17:09:04 +00:00
|
|
|
})
|
|
|
|
}
|
|
|
|
var err ParseErr
|
|
|
|
err.append(res)
|
|
|
|
return &err
|
|
|
|
}
|
|
|
|
|
2019-10-30 17:55:13 +00:00
|
|
|
func (p *Pkg) validResources() error {
|
|
|
|
if len(p.Spec.Resources) > 0 {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
res := errResource{
|
2019-11-01 18:11:42 +00:00
|
|
|
Kind: "Package",
|
2019-10-30 17:55:13 +00:00
|
|
|
Idx: -1,
|
|
|
|
}
|
|
|
|
res.ValidationFails = append(res.ValidationFails, struct {
|
|
|
|
Field string
|
|
|
|
Msg string
|
|
|
|
}{Field: "resources", Msg: "at least 1 resource must be provided"})
|
|
|
|
var err ParseErr
|
|
|
|
err.append(res)
|
|
|
|
return &err
|
|
|
|
}
|
|
|
|
|
2019-10-26 02:11:47 +00:00
|
|
|
func (p *Pkg) graphResources() error {
|
2019-10-23 17:09:04 +00:00
|
|
|
graphFns := []func() error{
|
2019-10-26 02:11:47 +00:00
|
|
|
// labels are first to validate associations with other resources
|
|
|
|
p.graphLabels,
|
2019-11-06 22:41:06 +00:00
|
|
|
p.graphVariables,
|
2019-10-26 02:11:47 +00:00
|
|
|
p.graphBuckets,
|
2019-10-30 21:13:42 +00:00
|
|
|
p.graphDashboards,
|
2019-10-23 17:09:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
for _, fn := range graphFns {
|
|
|
|
if err := fn(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2019-10-26 02:11:47 +00:00
|
|
|
func (p *Pkg) graphBuckets() error {
|
|
|
|
p.mBuckets = make(map[string]*bucket)
|
|
|
|
return p.eachResource(kindBucket, func(r Resource) []failure {
|
2019-10-23 17:09:04 +00:00
|
|
|
if r.Name() == "" {
|
2019-10-26 02:11:47 +00:00
|
|
|
return []failure{{
|
|
|
|
Field: "name",
|
|
|
|
Msg: "must be a string of at least 2 chars in length",
|
|
|
|
}}
|
2019-10-23 17:09:04 +00:00
|
|
|
}
|
|
|
|
|
2019-10-26 02:11:47 +00:00
|
|
|
if _, ok := p.mBuckets[r.Name()]; ok {
|
|
|
|
return []failure{{
|
|
|
|
Field: "name",
|
|
|
|
Msg: "duplicate name: " + r.Name(),
|
|
|
|
}}
|
2019-10-23 17:09:04 +00:00
|
|
|
}
|
2019-10-26 02:11:47 +00:00
|
|
|
|
|
|
|
bkt := &bucket{
|
2019-10-23 17:09:04 +00:00
|
|
|
Name: r.Name(),
|
|
|
|
Description: r.stringShort("description"),
|
|
|
|
RetentionPeriod: r.duration("retention_period"),
|
|
|
|
}
|
|
|
|
|
2019-11-01 18:11:42 +00:00
|
|
|
failures := p.parseNestedLabels(r, func(l *label) error {
|
2019-10-30 21:13:42 +00:00
|
|
|
bkt.labels = append(bkt.labels, l)
|
|
|
|
p.mLabels[l.Name].setBucketMapping(bkt, false)
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
if len(failures) > 0 {
|
|
|
|
return failures
|
2019-10-26 02:11:47 +00:00
|
|
|
}
|
|
|
|
sort.Slice(bkt.labels, func(i, j int) bool {
|
|
|
|
return bkt.labels[i].Name < bkt.labels[j].Name
|
|
|
|
})
|
|
|
|
|
|
|
|
p.mBuckets[r.Name()] = bkt
|
|
|
|
|
|
|
|
return failures
|
2019-10-23 17:09:04 +00:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2019-10-26 02:11:47 +00:00
|
|
|
func (p *Pkg) graphLabels() error {
|
|
|
|
p.mLabels = make(map[string]*label)
|
|
|
|
return p.eachResource(kindLabel, func(r Resource) []failure {
|
2019-10-24 23:59:01 +00:00
|
|
|
if r.Name() == "" {
|
2019-10-26 02:11:47 +00:00
|
|
|
return []failure{{
|
|
|
|
Field: "name",
|
|
|
|
Msg: "must be a string of at least 2 chars in length",
|
|
|
|
}}
|
2019-10-24 23:59:01 +00:00
|
|
|
}
|
|
|
|
|
2019-10-26 02:11:47 +00:00
|
|
|
if _, ok := p.mLabels[r.Name()]; ok {
|
|
|
|
return []failure{{
|
|
|
|
Field: "name",
|
|
|
|
Msg: "duplicate name: " + r.Name(),
|
|
|
|
}}
|
2019-10-24 23:59:01 +00:00
|
|
|
}
|
2019-10-26 02:11:47 +00:00
|
|
|
p.mLabels[r.Name()] = &label{
|
2019-10-24 23:59:01 +00:00
|
|
|
Name: r.Name(),
|
|
|
|
Color: r.stringShort("color"),
|
|
|
|
Description: r.stringShort("description"),
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2019-10-30 21:13:42 +00:00
|
|
|
func (p *Pkg) graphDashboards() error {
|
|
|
|
p.mDashboards = make(map[string]*dashboard)
|
|
|
|
return p.eachResource(kindDashboard, func(r Resource) []failure {
|
|
|
|
if r.Name() == "" {
|
|
|
|
return []failure{{
|
|
|
|
Field: "name",
|
|
|
|
Msg: "must be a string of at least 2 chars in length",
|
|
|
|
}}
|
|
|
|
}
|
|
|
|
|
|
|
|
if _, ok := p.mDashboards[r.Name()]; ok {
|
|
|
|
return []failure{{
|
|
|
|
Field: "name",
|
|
|
|
Msg: "duplicate name: " + r.Name(),
|
|
|
|
}}
|
|
|
|
}
|
|
|
|
|
|
|
|
dash := &dashboard{
|
|
|
|
Name: r.Name(),
|
|
|
|
Description: r.stringShort("description"),
|
|
|
|
}
|
|
|
|
|
2019-11-01 18:11:42 +00:00
|
|
|
failures := p.parseNestedLabels(r, func(l *label) error {
|
2019-10-30 21:13:42 +00:00
|
|
|
dash.labels = append(dash.labels, l)
|
2019-11-01 18:11:42 +00:00
|
|
|
p.mLabels[l.Name].setDashboardMapping(dash)
|
2019-10-30 21:13:42 +00:00
|
|
|
return nil
|
|
|
|
})
|
|
|
|
sort.Slice(dash.labels, func(i, j int) bool {
|
|
|
|
return dash.labels[i].Name < dash.labels[j].Name
|
|
|
|
})
|
|
|
|
|
2019-11-01 18:11:42 +00:00
|
|
|
for i, cr := range r.slcResource("charts") {
|
|
|
|
ch, fails := parseChart(cr)
|
|
|
|
if fails != nil {
|
|
|
|
for _, f := range fails {
|
|
|
|
failures = append(failures, failure{
|
|
|
|
Field: fmt.Sprintf("charts[%d].%s", i, f.Field),
|
|
|
|
Msg: f.Msg,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
dash.Charts = append(dash.Charts, ch)
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(failures) > 0 {
|
|
|
|
return failures
|
|
|
|
}
|
|
|
|
|
2019-10-30 21:13:42 +00:00
|
|
|
p.mDashboards[r.Name()] = dash
|
|
|
|
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2019-11-06 22:41:06 +00:00
|
|
|
func (p *Pkg) graphVariables() error {
|
|
|
|
p.mVariables = make(map[string]*variable)
|
|
|
|
return p.eachResource(kindVariable, func(r Resource) []failure {
|
|
|
|
if r.Name() == "" {
|
|
|
|
return []failure{{
|
|
|
|
Field: "name",
|
|
|
|
Msg: "must be provided",
|
|
|
|
}}
|
|
|
|
}
|
|
|
|
|
|
|
|
if _, ok := p.mVariables[r.Name()]; ok {
|
|
|
|
return []failure{{
|
|
|
|
Field: "name",
|
|
|
|
Msg: "duplicate name: " + r.Name(),
|
|
|
|
}}
|
|
|
|
}
|
|
|
|
|
|
|
|
newVar := &variable{
|
|
|
|
Name: r.Name(),
|
|
|
|
Description: r.stringShort("description"),
|
|
|
|
Type: strings.ToLower(r.stringShort("type")),
|
|
|
|
Query: strings.TrimSpace(r.stringShort("query")),
|
|
|
|
Language: strings.ToLower(strings.TrimSpace(r.stringShort("language"))),
|
|
|
|
ConstValues: r.slcStr("values"),
|
|
|
|
MapValues: r.mapStrStr("values"),
|
|
|
|
}
|
|
|
|
|
|
|
|
p.mVariables[r.Name()] = newVar
|
|
|
|
|
|
|
|
// here we set the var on the var map and return fails
|
|
|
|
// reaons for this is we could end up providing bad
|
|
|
|
// errors to the user if we dont' set it b/c of a bad
|
|
|
|
// query or something, and its being referenced by a
|
|
|
|
// dashboard or something. The var exists, its just
|
|
|
|
// invalid. So the mapping is correct. So we keep this
|
|
|
|
// to validate that mapping is correct, and return fails
|
|
|
|
// to indicate fails from the var.
|
|
|
|
return newVar.valid()
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2019-10-26 02:11:47 +00:00
|
|
|
func (p *Pkg) eachResource(resourceKind kind, fn func(r Resource) []failure) error {
|
2019-10-23 17:09:04 +00:00
|
|
|
var parseErr ParseErr
|
2019-10-26 02:11:47 +00:00
|
|
|
for i, r := range p.Spec.Resources {
|
2019-10-23 17:09:04 +00:00
|
|
|
k, err := r.kind()
|
|
|
|
if err != nil {
|
|
|
|
parseErr.append(errResource{
|
2019-11-01 18:11:42 +00:00
|
|
|
Kind: k.String(),
|
2019-10-23 17:09:04 +00:00
|
|
|
Idx: i,
|
2019-10-26 02:11:47 +00:00
|
|
|
ValidationFails: []struct {
|
2019-10-23 17:09:04 +00:00
|
|
|
Field string
|
|
|
|
Msg string
|
|
|
|
}{
|
|
|
|
{
|
|
|
|
Field: "kind",
|
|
|
|
Msg: err.Error(),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
})
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if k != resourceKind {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2019-10-26 02:11:47 +00:00
|
|
|
if failures := fn(r); failures != nil {
|
|
|
|
err := errResource{
|
2019-11-01 18:11:42 +00:00
|
|
|
Kind: resourceKind.String(),
|
2019-10-23 17:09:04 +00:00
|
|
|
Idx: i,
|
2019-10-26 02:11:47 +00:00
|
|
|
}
|
|
|
|
for _, f := range failures {
|
|
|
|
if f.fromAssociation {
|
|
|
|
err.AssociationFails = append(err.AssociationFails, struct {
|
|
|
|
Field string
|
|
|
|
Msg string
|
|
|
|
Index int
|
2019-11-01 18:11:42 +00:00
|
|
|
}{Field: f.Field, Msg: f.Msg, Index: f.assIndex})
|
2019-10-26 02:11:47 +00:00
|
|
|
continue
|
|
|
|
}
|
|
|
|
err.ValidationFails = append(err.ValidationFails, struct {
|
2019-10-23 17:09:04 +00:00
|
|
|
Field string
|
|
|
|
Msg string
|
2019-10-26 02:11:47 +00:00
|
|
|
}{Field: f.Field, Msg: f.Msg})
|
|
|
|
}
|
|
|
|
parseErr.append(err)
|
2019-10-23 17:09:04 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(parseErr.Resources) > 0 {
|
|
|
|
return &parseErr
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2019-11-01 18:11:42 +00:00
|
|
|
func (p *Pkg) parseNestedLabels(r Resource, fn func(lb *label) error) []failure {
|
2019-10-30 21:13:42 +00:00
|
|
|
nestedLabels := make(map[string]*label)
|
|
|
|
|
|
|
|
var failures []failure
|
|
|
|
for i, nr := range r.nestedAssociations() {
|
2019-11-01 18:11:42 +00:00
|
|
|
fail := p.parseNestedLabel(i, nr, func(l *label) error {
|
2019-10-30 21:13:42 +00:00
|
|
|
if _, ok := nestedLabels[l.Name]; ok {
|
|
|
|
return fmt.Errorf("duplicate nested label: %q", l.Name)
|
|
|
|
}
|
|
|
|
nestedLabels[l.Name] = l
|
|
|
|
|
|
|
|
return fn(l)
|
|
|
|
})
|
|
|
|
if fail != nil {
|
|
|
|
failures = append(failures, *fail)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return failures
|
|
|
|
}
|
|
|
|
|
2019-11-01 18:11:42 +00:00
|
|
|
func (p *Pkg) parseNestedLabel(idx int, nr Resource, fn func(lb *label) error) *failure {
|
2019-10-26 02:11:47 +00:00
|
|
|
k, err := nr.kind()
|
|
|
|
if err != nil {
|
|
|
|
return &failure{
|
|
|
|
Field: "kind",
|
|
|
|
Msg: err.Error(),
|
|
|
|
fromAssociation: true,
|
2019-11-01 18:11:42 +00:00
|
|
|
assIndex: idx,
|
2019-10-26 02:11:47 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if k != kindLabel {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
lb, found := p.mLabels[nr.Name()]
|
|
|
|
if !found {
|
|
|
|
return &failure{
|
|
|
|
Field: "associations",
|
|
|
|
Msg: fmt.Sprintf("label %q does not exist in pkg", nr.Name()),
|
|
|
|
fromAssociation: true,
|
2019-11-01 18:11:42 +00:00
|
|
|
assIndex: idx,
|
2019-10-26 02:11:47 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := fn(lb); err != nil {
|
|
|
|
return &failure{
|
|
|
|
Field: "associations",
|
|
|
|
Msg: err.Error(),
|
|
|
|
fromAssociation: true,
|
2019-11-01 18:11:42 +00:00
|
|
|
assIndex: idx,
|
2019-10-26 02:11:47 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2019-11-01 18:11:42 +00:00
|
|
|
func parseChart(r Resource) (chart, []failure) {
|
|
|
|
ck, err := r.chartKind()
|
|
|
|
if err != nil {
|
|
|
|
return chart{}, []failure{{
|
|
|
|
Field: "kind",
|
|
|
|
Msg: err.Error(),
|
|
|
|
}}
|
|
|
|
}
|
|
|
|
|
|
|
|
c := chart{
|
|
|
|
Kind: ck,
|
|
|
|
Name: r.Name(),
|
|
|
|
Prefix: r.stringShort("prefix"),
|
|
|
|
Suffix: r.stringShort("suffix"),
|
|
|
|
Note: r.stringShort("note"),
|
|
|
|
NoteOnEmpty: r.boolShort("noteOnEmpty"),
|
|
|
|
Shade: r.boolShort("shade"),
|
|
|
|
XCol: r.stringShort("xCol"),
|
|
|
|
YCol: r.stringShort("yCol"),
|
|
|
|
XPos: r.intShort("xPos"),
|
|
|
|
YPos: r.intShort("yPos"),
|
|
|
|
Height: r.intShort("height"),
|
|
|
|
Width: r.intShort("width"),
|
2019-11-05 19:19:25 +00:00
|
|
|
Geom: r.stringShort("geom"),
|
2019-11-01 18:11:42 +00:00
|
|
|
}
|
|
|
|
|
2019-11-06 22:41:06 +00:00
|
|
|
if leg, ok := ifaceToResource(r["legend"]); ok {
|
2019-11-04 19:16:32 +00:00
|
|
|
c.Legend.Type = leg.stringShort("type")
|
|
|
|
c.Legend.Orientation = leg.stringShort("orientation")
|
|
|
|
}
|
|
|
|
|
2019-11-01 18:11:42 +00:00
|
|
|
if dp, ok := r.int("decimalPlaces"); ok {
|
|
|
|
c.EnforceDecimals = true
|
|
|
|
c.DecimalPlaces = dp
|
|
|
|
}
|
|
|
|
|
|
|
|
var failures []failure
|
|
|
|
for _, rq := range r.slcResource("queries") {
|
|
|
|
c.Queries = append(c.Queries, query{
|
2019-11-04 19:16:32 +00:00
|
|
|
Query: strings.TrimSpace(rq.stringShort("query")),
|
2019-11-01 18:11:42 +00:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, rc := range r.slcResource("colors") {
|
|
|
|
c.Colors = append(c.Colors, &color{
|
|
|
|
id: influxdb.ID(int(time.Now().UnixNano())).String(),
|
|
|
|
Name: rc.Name(),
|
|
|
|
Type: rc.stringShort("type"),
|
|
|
|
Hex: rc.stringShort("hex"),
|
|
|
|
Value: rc.float64Short("value"),
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2019-11-04 19:16:32 +00:00
|
|
|
for _, ra := range r.slcResource("axes") {
|
|
|
|
c.Axes = append(c.Axes, axis{
|
|
|
|
Base: ra.stringShort("base"),
|
|
|
|
Label: ra.stringShort("label"),
|
|
|
|
Name: ra.Name(),
|
|
|
|
Prefix: ra.stringShort("prefix"),
|
|
|
|
Scale: ra.stringShort("scale"),
|
|
|
|
Suffix: ra.stringShort("suffix"),
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2019-11-01 18:11:42 +00:00
|
|
|
if fails := c.validProperties(); len(fails) > 0 {
|
|
|
|
failures = append(failures, fails...)
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(failures) > 0 {
|
|
|
|
return chart{}, failures
|
|
|
|
}
|
|
|
|
|
|
|
|
return c, nil
|
|
|
|
}
|
|
|
|
|
2019-10-23 17:09:04 +00:00
|
|
|
// Resource is a pkger Resource kind. It can be one of any of
|
|
|
|
// available kinds that are supported.
|
|
|
|
type Resource map[string]interface{}
|
|
|
|
|
2019-11-01 18:11:42 +00:00
|
|
|
func (r Resource) Name() string {
|
|
|
|
return strings.TrimSpace(r.stringShort("name"))
|
|
|
|
}
|
|
|
|
|
2019-10-23 17:09:04 +00:00
|
|
|
func (r Resource) kind() (kind, error) {
|
|
|
|
resKind, ok := r.string("kind")
|
|
|
|
if !ok {
|
|
|
|
return kindUnknown, errors.New("no kind provided")
|
|
|
|
}
|
|
|
|
|
|
|
|
newKind := kind(strings.TrimSpace(strings.ToLower(resKind)))
|
|
|
|
if newKind == kindUnknown {
|
|
|
|
return kindUnknown, errors.New("invalid kind")
|
|
|
|
}
|
2019-11-06 22:41:06 +00:00
|
|
|
if !kinds[newKind] {
|
|
|
|
return newKind, errors.New("unsupported kind provided")
|
|
|
|
}
|
2019-10-23 17:09:04 +00:00
|
|
|
|
|
|
|
return newKind, nil
|
|
|
|
}
|
|
|
|
|
2019-11-05 22:08:30 +00:00
|
|
|
func (r Resource) chartKind() (chartKind, error) {
|
2019-11-01 18:11:42 +00:00
|
|
|
ck, _ := r.kind()
|
2019-11-05 22:08:30 +00:00
|
|
|
chartKind := chartKind(ck)
|
2019-11-01 18:11:42 +00:00
|
|
|
if !chartKind.ok() {
|
2019-11-05 22:08:30 +00:00
|
|
|
return chartKindUnknown, errors.New("invalid chart kind provided: " + string(chartKind))
|
2019-11-01 18:11:42 +00:00
|
|
|
}
|
|
|
|
return chartKind, nil
|
2019-10-23 17:09:04 +00:00
|
|
|
}
|
|
|
|
|
2019-10-26 02:11:47 +00:00
|
|
|
func (r Resource) nestedAssociations() []Resource {
|
|
|
|
v, ok := r["associations"]
|
2019-10-23 17:09:04 +00:00
|
|
|
if !ok {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
ifaces, ok := v.([]interface{})
|
|
|
|
if !ok {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
var resources []Resource
|
2019-10-26 02:11:47 +00:00
|
|
|
for _, iface := range ifaces {
|
2019-11-06 22:41:06 +00:00
|
|
|
newRes, ok := ifaceToResource(iface)
|
2019-10-23 17:09:04 +00:00
|
|
|
if !ok {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
resources = append(resources, newRes)
|
|
|
|
}
|
|
|
|
|
|
|
|
return resources
|
|
|
|
}
|
|
|
|
|
2019-11-01 18:11:42 +00:00
|
|
|
func (r Resource) bool(key string) (bool, bool) {
|
|
|
|
b, ok := r[key].(bool)
|
|
|
|
return b, ok
|
|
|
|
}
|
|
|
|
|
|
|
|
func (r Resource) boolShort(key string) bool {
|
|
|
|
b, _ := r.bool(key)
|
|
|
|
return b
|
|
|
|
}
|
|
|
|
|
2019-10-23 17:09:04 +00:00
|
|
|
func (r Resource) duration(key string) time.Duration {
|
|
|
|
dur, _ := time.ParseDuration(r.stringShort(key))
|
|
|
|
return dur
|
|
|
|
}
|
|
|
|
|
2019-11-01 18:11:42 +00:00
|
|
|
func (r Resource) float64(key string) (float64, bool) {
|
|
|
|
f, ok := r[key].(float64)
|
|
|
|
if ok {
|
|
|
|
return f, true
|
|
|
|
}
|
|
|
|
|
|
|
|
i, ok := r[key].(int)
|
|
|
|
if ok {
|
|
|
|
return float64(i), true
|
|
|
|
}
|
|
|
|
return 0, false
|
|
|
|
}
|
|
|
|
|
|
|
|
func (r Resource) float64Short(key string) float64 {
|
|
|
|
f, _ := r.float64(key)
|
|
|
|
return f
|
|
|
|
}
|
|
|
|
|
|
|
|
func (r Resource) int(key string) (int, bool) {
|
|
|
|
i, ok := r[key].(int)
|
|
|
|
if ok {
|
|
|
|
return i, true
|
|
|
|
}
|
|
|
|
|
|
|
|
f, ok := r[key].(float64)
|
|
|
|
if ok {
|
|
|
|
return int(f), true
|
|
|
|
}
|
|
|
|
return 0, false
|
|
|
|
}
|
|
|
|
|
|
|
|
func (r Resource) intShort(key string) int {
|
|
|
|
i, _ := r.int(key)
|
|
|
|
return i
|
|
|
|
}
|
|
|
|
|
2019-10-23 17:09:04 +00:00
|
|
|
func (r Resource) string(key string) (string, bool) {
|
2019-11-06 22:41:06 +00:00
|
|
|
return ifaceToStr(r[key])
|
2019-10-23 17:09:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (r Resource) stringShort(key string) string {
|
|
|
|
s, _ := r.string(key)
|
|
|
|
return s
|
|
|
|
}
|
|
|
|
|
2019-11-01 18:11:42 +00:00
|
|
|
func (r Resource) slcResource(key string) []Resource {
|
|
|
|
v, ok := r[key]
|
|
|
|
if !ok {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
iFaceSlc, ok := v.([]interface{})
|
|
|
|
if !ok {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
var newResources []Resource
|
|
|
|
for _, iFace := range iFaceSlc {
|
2019-11-06 22:41:06 +00:00
|
|
|
r, ok := ifaceToResource(iFace)
|
2019-11-01 18:11:42 +00:00
|
|
|
if !ok {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
newResources = append(newResources, r)
|
|
|
|
}
|
|
|
|
|
|
|
|
return newResources
|
|
|
|
}
|
|
|
|
|
2019-11-06 22:41:06 +00:00
|
|
|
func (r Resource) slcStr(key string) []string {
|
|
|
|
v, ok := r[key]
|
|
|
|
if !ok {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
iFaceSlc, ok := v.([]interface{})
|
|
|
|
if !ok {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
var out []string
|
|
|
|
for _, iface := range iFaceSlc {
|
|
|
|
s, ok := ifaceToStr(iface)
|
|
|
|
if !ok {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
out = append(out, s)
|
|
|
|
}
|
|
|
|
|
|
|
|
return out
|
|
|
|
}
|
|
|
|
|
|
|
|
func (r Resource) mapStrStr(key string) map[string]string {
|
|
|
|
res, ok := ifaceToResource(r[key])
|
|
|
|
if !ok {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
m := make(map[string]string)
|
|
|
|
for k, v := range res {
|
|
|
|
s, ok := ifaceToStr(v)
|
|
|
|
if !ok {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
m[k] = s
|
|
|
|
}
|
|
|
|
return m
|
|
|
|
}
|
|
|
|
|
|
|
|
func ifaceToResource(i interface{}) (Resource, bool) {
|
|
|
|
if i == nil {
|
|
|
|
return nil, false
|
|
|
|
}
|
|
|
|
|
2019-10-26 02:11:47 +00:00
|
|
|
res, ok := i.(Resource)
|
|
|
|
if ok {
|
|
|
|
return res, true
|
|
|
|
}
|
|
|
|
|
|
|
|
if m, ok := i.(map[string]interface{}); ok {
|
|
|
|
return m, true
|
|
|
|
}
|
|
|
|
|
2019-10-23 17:09:04 +00:00
|
|
|
m, ok := i.(map[interface{}]interface{})
|
|
|
|
if !ok {
|
|
|
|
return nil, false
|
|
|
|
}
|
|
|
|
|
|
|
|
newRes := make(Resource)
|
|
|
|
for k, v := range m {
|
|
|
|
s, ok := k.(string)
|
|
|
|
if !ok {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
newRes[s] = v
|
|
|
|
}
|
|
|
|
return newRes, true
|
|
|
|
}
|
|
|
|
|
2019-11-06 22:41:06 +00:00
|
|
|
func ifaceToStr(v interface{}) (string, bool) {
|
|
|
|
if v == nil {
|
|
|
|
return "", false
|
|
|
|
}
|
|
|
|
|
|
|
|
if s, ok := v.(string); ok {
|
|
|
|
return s, true
|
|
|
|
}
|
|
|
|
|
|
|
|
if i, ok := v.(int); ok {
|
|
|
|
return strconv.Itoa(i), true
|
|
|
|
}
|
|
|
|
|
|
|
|
if f, ok := v.(float64); ok {
|
|
|
|
return strconv.FormatFloat(f, 'f', -1, 64), true
|
|
|
|
}
|
|
|
|
|
|
|
|
return "", false
|
|
|
|
}
|
|
|
|
|
2019-10-23 17:09:04 +00:00
|
|
|
// ParseErr is a error from parsing the given package. The ParseErr
|
|
|
|
// provides a list of resources that failed and all validations
|
|
|
|
// that failed for that resource. A resource can multiple errors,
|
|
|
|
// and a ParseErr can have multiple resources which themselves can
|
|
|
|
// have multiple validation failures.
|
|
|
|
type ParseErr struct {
|
|
|
|
Resources []struct {
|
2019-11-01 18:11:42 +00:00
|
|
|
Kind string
|
2019-10-26 02:11:47 +00:00
|
|
|
Idx int
|
|
|
|
ValidationFails []struct {
|
|
|
|
Field string
|
|
|
|
Msg string
|
|
|
|
}
|
|
|
|
AssociationFails []struct {
|
2019-10-23 17:09:04 +00:00
|
|
|
Field string
|
|
|
|
Msg string
|
2019-10-26 02:11:47 +00:00
|
|
|
Index int
|
2019-10-23 17:09:04 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Error implements the error interface.
|
|
|
|
func (e *ParseErr) Error() string {
|
|
|
|
var errMsg []string
|
|
|
|
for _, r := range e.Resources {
|
2019-10-26 02:11:47 +00:00
|
|
|
resIndex := strconv.Itoa(r.Idx)
|
2019-10-23 17:09:04 +00:00
|
|
|
if r.Idx == -1 {
|
|
|
|
resIndex = "root"
|
|
|
|
}
|
2019-11-01 18:11:42 +00:00
|
|
|
err := fmt.Sprintf("resource_index=%s resource_kind=%q", resIndex, r.Kind)
|
2019-10-23 17:09:04 +00:00
|
|
|
errMsg = append(errMsg, err)
|
2019-10-26 02:11:47 +00:00
|
|
|
for _, f := range r.ValidationFails {
|
2019-10-23 17:09:04 +00:00
|
|
|
// for time being we go to new line and indent them (mainly for CLI)
|
|
|
|
// other callers (i.e. HTTP client) can inspect the resource and print it out
|
|
|
|
// or we provide a format option of sorts. We'll see
|
2019-10-26 02:11:47 +00:00
|
|
|
errMsg = append(errMsg, fmt.Sprintf("\terr_type=%q field=%q reason=%q", "validation", f.Field, f.Msg))
|
|
|
|
}
|
|
|
|
for _, f := range r.AssociationFails {
|
|
|
|
errMsg = append(errMsg, fmt.Sprintf("\terr_type=%q field=%q association_index=%d reason=%q", "association", f.Field, f.Index, f.Msg))
|
2019-10-23 17:09:04 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return strings.Join(errMsg, "\n")
|
|
|
|
}
|
|
|
|
|
|
|
|
func (e *ParseErr) append(err errResource) {
|
|
|
|
e.Resources = append(e.Resources, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// IsParseErr inspects a given error to determine if it is
|
|
|
|
// a ParseErr. If a ParseErr it is, it will return it along
|
|
|
|
// with the confirmation boolean. If the error is not a ParseErr
|
|
|
|
// it will return nil values for the ParseErr, making it unsafe
|
|
|
|
// to use.
|
|
|
|
func IsParseErr(err error) (*ParseErr, bool) {
|
|
|
|
pErr, ok := err.(*ParseErr)
|
|
|
|
return pErr, ok
|
|
|
|
}
|
|
|
|
|
|
|
|
type errResource struct {
|
2019-11-01 18:11:42 +00:00
|
|
|
Kind string
|
2019-10-26 02:11:47 +00:00
|
|
|
Idx int
|
|
|
|
ValidationFails []struct {
|
|
|
|
Field string
|
|
|
|
Msg string
|
|
|
|
}
|
|
|
|
AssociationFails []struct {
|
2019-10-23 17:09:04 +00:00
|
|
|
Field string
|
|
|
|
Msg string
|
2019-10-26 02:11:47 +00:00
|
|
|
Index int
|
2019-10-23 17:09:04 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
type failure struct {
|
2019-10-26 02:11:47 +00:00
|
|
|
Field, Msg string
|
|
|
|
fromAssociation bool
|
2019-11-01 18:11:42 +00:00
|
|
|
assIndex int
|
2019-10-23 17:09:04 +00:00
|
|
|
}
|