2022-04-14 13:54:35 -05:00
|
|
|
// Package cuectx provides a single, central CUE context (runtime) and Thema
|
|
|
|
// library that can be used uniformly across Grafana, and related helper
|
|
|
|
// functions for loading Thema lineages.
|
|
|
|
|
|
|
|
package cuectx
|
|
|
|
|
|
|
|
import (
|
|
|
|
"io/fs"
|
|
|
|
"path/filepath"
|
|
|
|
"testing/fstest"
|
|
|
|
|
|
|
|
"cuelang.org/go/cue"
|
|
|
|
"cuelang.org/go/cue/cuecontext"
|
|
|
|
"github.com/grafana/thema"
|
|
|
|
"github.com/grafana/thema/kernel"
|
|
|
|
"github.com/grafana/thema/load"
|
|
|
|
)
|
|
|
|
|
2022-06-06 19:52:44 -05:00
|
|
|
var ctx = cuecontext.New()
|
|
|
|
var lib = thema.NewLibrary(ctx)
|
2022-04-14 13:54:35 -05:00
|
|
|
|
|
|
|
// ProvideCUEContext is a wire service provider of a central cue.Context.
|
|
|
|
func ProvideCUEContext() *cue.Context {
|
|
|
|
return ctx
|
|
|
|
}
|
|
|
|
|
|
|
|
// ProvideThemaLibrary is a wire service provider of a central thema.Library.
|
|
|
|
func ProvideThemaLibrary() thema.Library {
|
|
|
|
return lib
|
|
|
|
}
|
|
|
|
|
|
|
|
// JSONtoCUE attempts to decode the given []byte into a cue.Value, relying on
|
|
|
|
// the central Grafana cue.Context provided in this package.
|
|
|
|
//
|
|
|
|
// The provided path argument determines the name given to the input bytes if
|
|
|
|
// later CUE operations (e.g. Thema validation) produce errors related to the
|
|
|
|
// returned cue.Value.
|
|
|
|
//
|
|
|
|
// This is a convenience function for one-off JSON decoding. It's wasteful to
|
|
|
|
// call it repeatedly. Most use cases use cases should probably prefer making
|
|
|
|
// their own Thema/CUE decoders.
|
|
|
|
func JSONtoCUE(path string, b []byte) (cue.Value, error) {
|
|
|
|
return kernel.NewJSONDecoder(path)(ctx, b)
|
|
|
|
}
|
|
|
|
|
|
|
|
// LoadGrafanaInstancesWithThema loads CUE files containing a lineage
|
|
|
|
// representing some Grafana core model schema. It is expected to be used when
|
|
|
|
// implementing a thema.LineageFactory.
|
|
|
|
//
|
|
|
|
// This function primarily juggles paths to make CUE's loader happy. Provide the
|
|
|
|
// path from the grafana root to the directory containing the lineage.cue. The
|
|
|
|
// lineage.cue file must be the sole contents of the provided fs.FS.
|
|
|
|
//
|
|
|
|
// More details on underlying behavior can be found in the docs for github.com/grafana/thema/load.InstancesWithThema.
|
2022-08-22 11:11:45 -05:00
|
|
|
//
|
|
|
|
// TODO this approach is complicated and confusing, refactor to something understandable
|
|
|
|
func LoadGrafanaInstancesWithThema(path string, cueFS fs.FS, lib thema.Library, opts ...thema.BindOption) (thema.Lineage, error) {
|
2022-04-14 13:54:35 -05:00
|
|
|
prefix := filepath.FromSlash(path)
|
|
|
|
fs, err := prefixWithGrafanaCUE(prefix, cueFS)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
inst, err := load.InstancesWithThema(fs, prefix)
|
|
|
|
|
|
|
|
// Need to trick loading by creating the embedded file and
|
|
|
|
// making it look like a module in the root dir.
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
val := lib.Context().BuildInstance(inst)
|
|
|
|
|
|
|
|
lin, err := thema.BindLineage(val, lib, opts...)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return lin, nil
|
|
|
|
}
|
|
|
|
|
2022-06-06 19:52:44 -05:00
|
|
|
// prefixWithGrafanaCUE constructs an fs.FS that merges the provided fs.FS with one
|
|
|
|
// containing grafana's cue.mod at the root. The provided prefix should be the
|
|
|
|
//
|
|
|
|
// The returned fs.FS is suitable for passing to a CUE loader, such as
|
|
|
|
// cuelang.org/cue/load.Instances or
|
|
|
|
// github.com/grafana/thema/load.InstancesWithThema.
|
2022-04-14 13:54:35 -05:00
|
|
|
func prefixWithGrafanaCUE(prefix string, inputfs fs.FS) (fs.FS, error) {
|
|
|
|
m := fstest.MapFS{
|
2022-05-23 15:00:30 -05:00
|
|
|
// fstest can recognize only forward slashes.
|
|
|
|
filepath.ToSlash(filepath.Join("cue.mod", "module.cue")): &fstest.MapFile{Data: []byte(`module: "github.com/grafana/grafana"`)},
|
2022-04-14 13:54:35 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
prefix = filepath.FromSlash(prefix)
|
2022-06-06 19:52:44 -05:00
|
|
|
err := fs.WalkDir(inputfs, ".", func(path string, d fs.DirEntry, err error) error {
|
2022-04-14 13:54:35 -05:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if d.IsDir() {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2022-08-22 11:11:45 -05:00
|
|
|
b, err := fs.ReadFile(inputfs, path)
|
2022-04-14 13:54:35 -05:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2022-05-23 15:00:30 -05:00
|
|
|
// fstest can recognize only forward slashes.
|
|
|
|
m[filepath.ToSlash(filepath.Join(prefix, path))] = &fstest.MapFile{Data: b}
|
2022-04-14 13:54:35 -05:00
|
|
|
return nil
|
2022-06-06 19:52:44 -05:00
|
|
|
})
|
2022-04-14 13:54:35 -05:00
|
|
|
|
|
|
|
return m, err
|
|
|
|
}
|