mirror of
https://github.com/grafana/grafana.git
synced 2025-02-25 18:55:37 -06:00
* Update thema to latest * Deal with s/Library/*Runtime/ * Commit new, working results of codegen * We like pointers now * Always take runtime arg for NewBase() * Sketchy handwavy pass at entity meta framework * Little nibbles * Update pkg/framework/coremodel/entityframework.cue Co-authored-by: Artur Wierzbicki <wierzbicki.artur.94@gmail.com> * Move file into new framework location * Introduce loaders, Go code * Complete rename to kind * Flesh out framework, add svg/dashboard examples * Cruft removal * Remove generated kind go files from gitignore * Refine maturity concept, add SlotKind * Update embed and go deps * Export PrefixWithGrafanaCUE * Make the loader actually work, holy crap * Many small tweaks to type.cue * Add Apache 2 licensing exceptions for kinds * Add new kinds dir, start of generator * Roll back to earlier oapi-codegen * Introduce new grafana-specific CUE loaders * Introduce new tidy code generators framework * Catch up kind framework with tinkering * Add slices for the generators * Add write/verify step to main generator * Many renames * Split up kind framework cue files * Use kind.Decl within generated kinds * Create kind.SomeDecl wrapper type to cache lineages * Better names again * Get one generated implemented, hopefully * Copy dashboard schema into new kind.cue * Small fixes to make the initial gen work * Put svg kind in its new home * Add generated Go dashboard type * More renames and cleanups * Add base kind registry and generator * Stop blacklisting *_gen.go files This is not the Go best practice, anyway. All we actually want to ignore for enterprise is generated wire files. * Change codegen output directories pkg/kind -> pkg/kinds pkg/registry/kindreg -> pkg/registry/corekind * Rename pkg/framework/kind to pkg/kindsys * Add core structured kind generator * Add plural and machine names to kind spec * Copy playlist over to kind system * Consolidate kindsys files * Add raw kind generator * Update CODEOWNERS for kind framework * Touch up comments a bit * More docs tweaks * Remove generated types to reduce noise for review * Split each generator into its own file * Rename Slot kind to Composable kind * Add handwavy types for customkind loading * Guard against init calls to framework loader * First pass at doc on extending the kind system * Improve attribute example in docs * Fix wire imports * Add basic TS types generator * Fix composable kind category def * No need for a separate file with generate directive * Catch dashboard schema up * Rename generator types to something saner and generic * Make version configurable in ts/go generators * Add CommonMeta to ease property access * Add kindsys prop indicating whether lineage is group * Put all kind categories back in a single file * Finish with kindsys group props * Refactor maturity progression per discussion - Replace "committed" with "merged" - All kindcats can use all maturity levels, at least for now * Convert ts veneer index generator to modular system * Move over to new jennywrites framework * Strip down old coremodel generator * Use public version of jennywrites * Pull latest thema * Commit generated Go types * Add header injection postprocessor * Move sdboyer/jennywrites to grafana/codejen * Tweak header output * Remove dashboard and playlist coremodels * Fix up backend dashboards devenv test * Fix TS import patterns to new gen filename * Update internal imports, remove coremodel registry * Fix compilation errors, wire generation * Export and replace the prefix dropper * More Go struct and field name changes * Last name fixes, hopefully * Fix lint errors * Last lint error Co-authored-by: Artur Wierzbicki <wierzbicki.artur.94@gmail.com>
199 lines
6.6 KiB
Go
199 lines
6.6 KiB
Go
// Package cuectx provides a single, central ["cuelang.org/go/cue".Context] and
|
|
// ["github.com/grafana/thema".Runtime] that can be used uniformly across
|
|
// Grafana, and related helper functions for loading Thema lineages.
|
|
|
|
package cuectx
|
|
|
|
import (
|
|
"fmt"
|
|
"io/fs"
|
|
"path/filepath"
|
|
"testing/fstest"
|
|
|
|
"cuelang.org/go/cue"
|
|
"cuelang.org/go/cue/build"
|
|
"cuelang.org/go/cue/cuecontext"
|
|
"github.com/grafana/grafana"
|
|
"github.com/grafana/thema"
|
|
"github.com/grafana/thema/load"
|
|
"github.com/grafana/thema/vmux"
|
|
"github.com/yalue/merged_fs"
|
|
)
|
|
|
|
var ctx = cuecontext.New()
|
|
var rt = thema.NewRuntime(ctx)
|
|
|
|
// GrafanaCUEContext returns Grafana's singleton instance of [cue.Context].
|
|
//
|
|
// All code within grafana/grafana that needs a *cue.Context should get it
|
|
// from this function, when one was not otherwise provided.
|
|
func GrafanaCUEContext() *cue.Context {
|
|
return ctx
|
|
}
|
|
|
|
// GrafanaThemaRuntime returns Grafana's singleton instance of [thema.Runtime].
|
|
//
|
|
// All code within grafana/grafana that needs a *thema.Runtime should get it
|
|
// from this function, when one was not otherwise provided.
|
|
func GrafanaThemaRuntime() *thema.Runtime {
|
|
return rt
|
|
}
|
|
|
|
// 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 should probably prefer making
|
|
// their own Thema/CUE decoders.
|
|
func JSONtoCUE(path string, b []byte) (cue.Value, error) {
|
|
return vmux.NewJSONEndec(path).Decode(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.
|
|
//
|
|
// TODO this approach is complicated and confusing, refactor to something understandable
|
|
func LoadGrafanaInstancesWithThema(path string, cueFS fs.FS, rt *thema.Runtime, opts ...thema.BindOption) (thema.Lineage, error) {
|
|
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 := rt.Context().BuildInstance(inst)
|
|
|
|
lin, err := thema.BindLineage(val, rt, opts...)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return lin, nil
|
|
}
|
|
|
|
// prefixWithGrafanaCUE constructs an fs.FS that merges the provided fs.FS with
|
|
// the embedded FS containing Grafana's core CUE files, [grafana.CueSchemaFS].
|
|
// The provided prefix should be the relative path from the grafana repository
|
|
// root to the directory root of the provided inputfs.
|
|
//
|
|
// The returned fs.FS is suitable for passing to a CUE loader, such as [load.InstancesWithThema].
|
|
func prefixWithGrafanaCUE(prefix string, inputfs fs.FS) (fs.FS, error) {
|
|
m := fstest.MapFS{
|
|
// fstest can recognize only forward slashes.
|
|
// filepath.ToSlash(filepath.Join("cue.mod", "module.cue")): &fstest.MapFile{Data: []byte(`module: "github.com/grafana/grafana"`)},
|
|
}
|
|
|
|
prefix = filepath.FromSlash(prefix)
|
|
err := fs.WalkDir(inputfs, ".", func(path string, d fs.DirEntry, err error) error {
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if d.IsDir() {
|
|
return nil
|
|
}
|
|
|
|
b, err := fs.ReadFile(inputfs, path)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
// fstest can recognize only forward slashes.
|
|
m[filepath.ToSlash(filepath.Join(prefix, path))] = &fstest.MapFile{Data: b}
|
|
return nil
|
|
})
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return merged_fs.NewMergedFS(m, grafana.CueSchemaFS), nil
|
|
}
|
|
|
|
// BuildGrafanaInstance wraps [load.InstancesWithThema] to load a
|
|
// [*build.Instance] corresponding to a particular path within the
|
|
// github.com/grafana/grafana CUE module, then builds that into a [cue.Value],
|
|
// checks it for errors and returns.
|
|
//
|
|
// This allows resolution of imports within the grafana or thema CUE modules to
|
|
// work correctly and consistently by relying on the embedded FS at
|
|
// [grafana.CueSchemaFS] and [thema.CueFS].
|
|
//
|
|
// relpath should be a relative path path within [grafana.CueSchemaFS] to be
|
|
// loaded. Optionally, the caller may provide an additional fs.FS via the
|
|
// overlay parameter, which will be merged with [grafana.CueSchemaFS] at
|
|
// relpath, and loaded.
|
|
//
|
|
// pkg, if non-empty, is set as the value of
|
|
// ["cuelang.org/go/cue/load".Config.Package]. If the CUE package to be loaded
|
|
// is the same as the parent directory name, it should be omitted.
|
|
//
|
|
// NOTE this function will be removed in favor of a more generic loader
|
|
func BuildGrafanaInstance(relpath string, pkg string, ctx *cue.Context, overlay fs.FS) (cue.Value, error) {
|
|
// notes about how this crap needs to work
|
|
//
|
|
// Within grafana/grafana, need:
|
|
// - pass in an fs.FS that, in its root, contains the .cue files to load
|
|
// - has no cue.mod
|
|
// - gets prefixed with the appropriate path within grafana/grafana
|
|
// - and merged with all the other .cue files from grafana/grafana
|
|
if ctx == nil {
|
|
ctx = GrafanaCUEContext()
|
|
}
|
|
relpath = filepath.ToSlash(relpath)
|
|
|
|
var v cue.Value
|
|
var f fs.FS = grafana.CueSchemaFS
|
|
var err error
|
|
if overlay != nil {
|
|
f, err = prefixWithGrafanaCUE(relpath, overlay)
|
|
if err != nil {
|
|
return v, err
|
|
}
|
|
}
|
|
|
|
var bi *build.Instance
|
|
if pkg != "" {
|
|
bi, err = load.InstancesWithThema(f, relpath, load.Package(pkg))
|
|
} else {
|
|
bi, err = load.InstancesWithThema(f, relpath)
|
|
}
|
|
if err != nil {
|
|
return v, err
|
|
}
|
|
|
|
v = ctx.BuildInstance(bi)
|
|
if v.Err() != nil {
|
|
return v, fmt.Errorf("%s not a valid CUE instance: %w", relpath, v.Err())
|
|
}
|
|
return v, nil
|
|
}
|
|
|
|
// TODO docs
|
|
// NOTE this function will be removed in favor of a more generic loader
|
|
func LoadInstanceWithGrafana(ifs fs.FS, prefix string) (*build.Instance, error) {
|
|
// notes about how this crap needs to work
|
|
//
|
|
// Need a prefixing instance loader that:
|
|
// - can take multiple fs.FS, each one representing a CUE module (nesting?)
|
|
// - reconcile at most one of the provided fs with cwd
|
|
// - behavior must differ depending on whether cwd is in a cue module
|
|
// - behavior should(?) be controllable depending on
|
|
|
|
panic("TODO")
|
|
}
|