mirror of
https://github.com/grafana/grafana.git
synced 2025-01-27 16:57:14 -06:00
2e0dc835cf
* Add test for devenv resources * Refactor validation tests for grokkability * Devenv dashboards error-tracking script * Refactor to use cueerrors.Details() * Further test refinement * Close major elements of dashboard schema * Centralize dashboard validation tests General dashboard validation testing belongs in the load package. * Better names for error context on glue CUE code * Fixup validate-resource Do only one of base or dist, and fix copied docs. * Skip the devenv test * Remove test for validateResources * Fix shellcheck * Backend linter Co-authored-by: sam boyer <sdboyer@grafana.com>
177 lines
5.5 KiB
Go
177 lines
5.5 KiB
Go
package load
|
|
|
|
import (
|
|
"path/filepath"
|
|
|
|
"cuelang.org/go/cue"
|
|
"cuelang.org/go/cue/load"
|
|
"github.com/grafana/grafana/pkg/schema"
|
|
)
|
|
|
|
// getBaseScuemata attempts to load the base scuemata family and schema
|
|
// definitions on which all Grafana scuemata rely.
|
|
//
|
|
// TODO probably cache this or something
|
|
func getBaseScuemata(p BaseLoadPaths) (*cue.Instance, error) {
|
|
overlay := make(map[string]load.Source)
|
|
|
|
if err := toOverlay(filepath.Join(prefix, "grafana"), p.BaseCueFS, overlay); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
cfg := &load.Config{
|
|
Overlay: overlay,
|
|
Package: "scuemata",
|
|
// TODO Semantics of loading instances is quite confusing. This 'Dir'
|
|
// field is a case in point. It must be set to "/" in order for the
|
|
// overlay to be searched and have all files loaded in the cue/scuemata
|
|
// directory. (This isn't necessary when loading individual .cue files.)
|
|
// But anchoring a search at root seems like we're begging for
|
|
// vulnerabilities where Grafana can read and print out anything on the
|
|
// filesystem, which can be a disclosure problem, unless we're
|
|
// absolutely sure the search is within a virtual filesystem. Which i'm
|
|
// not.
|
|
//
|
|
// And no, changing the toOverlay() to have a subpath and the
|
|
// load.Instances to mirror that subpath does not allow us to get rid of
|
|
// this "/".
|
|
Dir: prefix,
|
|
}
|
|
return rt.Build(load.Instances([]string{
|
|
filepath.Join(prefix, "grafana", "cue", "scuemata", "scuemata.cue"),
|
|
filepath.Join(prefix, "grafana", "cue", "scuemata", "panel-plugin.cue"),
|
|
}, cfg)[0])
|
|
}
|
|
|
|
func buildGenericScuemata(famval cue.Value) (schema.VersionedCueSchema, error) {
|
|
// TODO verify subsumption by #Family; renders many
|
|
// error checks below unnecessary
|
|
majiter, err := famval.LookupPath(cue.MakePath(cue.Str("lineages"))).List()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
var major int
|
|
var first, lastgvs *genericVersionedSchema
|
|
for majiter.Next() {
|
|
var minor int
|
|
miniter, _ := majiter.Value().List()
|
|
for miniter.Next() {
|
|
gvs := &genericVersionedSchema{
|
|
actual: miniter.Value(),
|
|
major: major,
|
|
minor: minor,
|
|
// This gets overwritten on all but the very final schema
|
|
migration: terminalMigrationFunc,
|
|
}
|
|
|
|
if minor != 0 {
|
|
// TODO Verify that this schema is backwards compat with prior.
|
|
// Create an implicit migration operation on the prior schema.
|
|
lastgvs.migration = implicitMigration(gvs.actual, gvs)
|
|
lastgvs.next = gvs
|
|
} else if major != 0 {
|
|
lastgvs.next = gvs
|
|
// x.0. There should exist an explicit migration definition;
|
|
// load it up and ready it for use, and place it on the final
|
|
// schema in the prior sequence.
|
|
//
|
|
// Also...should at least try to make sure it's pointing at the
|
|
// expected schema, to maintain our invariants?
|
|
|
|
// TODO impl
|
|
} else {
|
|
first = gvs
|
|
}
|
|
lastgvs = gvs
|
|
minor++
|
|
}
|
|
major++
|
|
}
|
|
|
|
return first, nil
|
|
}
|
|
|
|
type genericVersionedSchema struct {
|
|
actual cue.Value
|
|
major int
|
|
minor int
|
|
next *genericVersionedSchema
|
|
migration migrationFunc
|
|
}
|
|
|
|
// Validate checks that the resource is correct with respect to the schema.
|
|
func (gvs *genericVersionedSchema) Validate(r schema.Resource) error {
|
|
name := r.Name
|
|
if name == "" {
|
|
name = "resource"
|
|
}
|
|
rv, err := rt.Compile(name, r.Value)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return gvs.actual.Unify(rv.Value()).Validate(cue.Concrete(true))
|
|
}
|
|
|
|
// CUE returns the cue.Value representing the actual schema.
|
|
func (gvs *genericVersionedSchema) CUE() cue.Value {
|
|
return gvs.actual
|
|
}
|
|
|
|
// Version reports the major and minor versions of the schema.
|
|
func (gvs *genericVersionedSchema) Version() (major int, minor int) {
|
|
return gvs.major, gvs.minor
|
|
}
|
|
|
|
// Returns the next VersionedCueSchema
|
|
func (gvs *genericVersionedSchema) Successor() schema.VersionedCueSchema {
|
|
if gvs.next == nil {
|
|
// Untyped nil, allows `<sch> == nil` checks to work as people expect
|
|
return nil
|
|
}
|
|
return gvs.next
|
|
}
|
|
|
|
// Migrate transforms a resource into a new Resource that is correct with
|
|
// respect to its Successor schema.
|
|
func (gvs *genericVersionedSchema) Migrate(x schema.Resource) (schema.Resource, schema.VersionedCueSchema, error) { // TODO restrict input/return type to concrete
|
|
r, sch, err := gvs.migration(x.Value)
|
|
if err != nil || sch == nil {
|
|
r = x.Value.(cue.Value)
|
|
}
|
|
|
|
return schema.Resource{Value: r}, sch, nil
|
|
}
|
|
|
|
type migrationFunc func(x interface{}) (cue.Value, schema.VersionedCueSchema, error)
|
|
|
|
var terminalMigrationFunc = func(x interface{}) (cue.Value, schema.VersionedCueSchema, error) {
|
|
// TODO send back the input
|
|
return cue.Value{}, nil, nil
|
|
}
|
|
|
|
// panic if called
|
|
// var panicMigrationFunc = func(x interface{}) (cue.Value, schema.VersionedCueSchema, error) {
|
|
// panic("migrations are not yet implemented")
|
|
// }
|
|
|
|
// Creates a func to perform a "migration" that simply unifies the input
|
|
// artifact (which is expected to have already have been validated against an
|
|
// earlier schema) with a later schema.
|
|
func implicitMigration(v cue.Value, next schema.VersionedCueSchema) migrationFunc {
|
|
return func(x interface{}) (cue.Value, schema.VersionedCueSchema, error) {
|
|
w := v.FillPath(cue.Path{}, x)
|
|
// TODO is it possible that migration would be successful, but there
|
|
// still exists some error here? Need to better understand internal CUE
|
|
// erroring rules? seems like incomplete cue.Value may always an Err()?
|
|
//
|
|
// TODO should check concreteness here? Or can we guarantee a priori it
|
|
// can be made concrete simply by looking at the schema, before
|
|
// implicitMigration() is called to create this function?
|
|
if w.Err() != nil {
|
|
return w, nil, w.Err()
|
|
}
|
|
return w, next, w.Err()
|
|
}
|
|
}
|