Reconcile coremodels, entities, objects under new kind framework (#56492)
* 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>
2022-11-10 14:36:40 -06:00
|
|
|
package codegen
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"path/filepath"
|
|
|
|
"sort"
|
|
|
|
"strings"
|
|
|
|
|
|
|
|
"cuelang.org/go/cue"
|
|
|
|
"cuelang.org/go/cue/errors"
|
|
|
|
"github.com/grafana/codejen"
|
|
|
|
"github.com/grafana/cuetsy/ts"
|
|
|
|
"github.com/grafana/cuetsy/ts/ast"
|
|
|
|
"github.com/grafana/grafana/pkg/kindsys"
|
|
|
|
"github.com/grafana/thema"
|
|
|
|
"github.com/grafana/thema/encoding/typescript"
|
|
|
|
)
|
|
|
|
|
|
|
|
// TSVeneerIndexJenny generates an index.gen.ts file with references to all
|
|
|
|
// generated TS types. Elements with the attribute @grafana(TSVeneer="type") are
|
|
|
|
// exported from a handwritten file, rather than the raw generated types.
|
|
|
|
//
|
|
|
|
// The provided dir is the path, relative to the grafana root, to the directory
|
|
|
|
// that should contain the generated index.
|
|
|
|
//
|
|
|
|
// Implicitly depends on output patterns in TSTypesJenny.
|
|
|
|
// TODO this is wasteful; share-nothing generator model entails re-running the cuetsy gen that TSTypesJenny already did
|
|
|
|
func TSVeneerIndexJenny(dir string) ManyToOne {
|
|
|
|
return &genTSVeneerIndex{
|
|
|
|
dir: dir,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
type genTSVeneerIndex struct {
|
|
|
|
dir string
|
|
|
|
}
|
|
|
|
|
|
|
|
func (gen *genTSVeneerIndex) JennyName() string {
|
|
|
|
return "TSVeneerIndexJenny"
|
|
|
|
}
|
|
|
|
|
2022-11-16 14:08:01 -06:00
|
|
|
func (gen *genTSVeneerIndex) Generate(decls ...*DeclForGen) (*codejen.File, error) {
|
Reconcile coremodels, entities, objects under new kind framework (#56492)
* 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>
2022-11-10 14:36:40 -06:00
|
|
|
tsf := new(ast.File)
|
|
|
|
for _, decl := range decls {
|
|
|
|
if decl.IsRaw() {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
sch := decl.Lineage().Latest()
|
|
|
|
f, err := typescript.GenerateTypes(sch, &typescript.TypeConfig{
|
|
|
|
RootName: decl.Meta.Common().Name,
|
|
|
|
Group: decl.Meta.Common().LineageIsGroup,
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("%s: %w", decl.Meta.Common().Name, err)
|
|
|
|
}
|
|
|
|
elems, err := gen.extractTSIndexVeneerElements(decl, f)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("%s: %w", decl.Meta.Common().Name, err)
|
|
|
|
}
|
|
|
|
tsf.Nodes = append(tsf.Nodes, elems...)
|
|
|
|
}
|
|
|
|
|
|
|
|
return codejen.NewFile(filepath.Join(gen.dir, "index.gen.ts"), []byte(tsf.String()), gen), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (gen *genTSVeneerIndex) extractTSIndexVeneerElements(decl *DeclForGen, tf *ast.File) ([]ast.Decl, error) {
|
|
|
|
lin := decl.Lineage()
|
|
|
|
comm := decl.Meta.Common()
|
|
|
|
|
|
|
|
// Check the root, then walk the tree
|
2022-11-15 07:48:31 -06:00
|
|
|
rootv := lin.Latest().Underlying()
|
Reconcile coremodels, entities, objects under new kind framework (#56492)
* 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>
2022-11-10 14:36:40 -06:00
|
|
|
|
|
|
|
var raw, custom, rawD, customD ast.Idents
|
|
|
|
|
|
|
|
var terr errors.Error
|
|
|
|
visit := func(p cue.Path, wv cue.Value) bool {
|
|
|
|
var name string
|
|
|
|
sels := p.Selectors()
|
|
|
|
switch len(sels) {
|
|
|
|
case 0:
|
|
|
|
name = strings.Title(lin.Name())
|
|
|
|
fallthrough
|
|
|
|
case 1:
|
|
|
|
// Only deal with subpaths that are definitions, for now
|
|
|
|
// TODO incorporate smarts about grouped lineages here
|
|
|
|
if name == "" {
|
|
|
|
if !sels[0].IsDefinition() {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
// It might seem to make sense that we'd strip replaceout the leading # here for
|
|
|
|
// definitions. However, cuetsy's tsast actually has the # still present in its
|
|
|
|
// Ident types, stripping it replaceout on the fly when stringifying.
|
|
|
|
name = sels[0].String()
|
|
|
|
}
|
|
|
|
|
|
|
|
// Search the generated TS AST for the type and default decl nodes
|
|
|
|
pair := findDeclNode(name, tf)
|
|
|
|
if pair.T == nil {
|
|
|
|
// No generated type for this item, skip it
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
cust, perr := getCustomVeneerAttr(wv)
|
|
|
|
if perr != nil {
|
|
|
|
terr = errors.Append(terr, errors.Promote(perr, fmt.Sprintf("%s: ", p.String())))
|
|
|
|
}
|
|
|
|
var has bool
|
|
|
|
for _, tgt := range cust {
|
|
|
|
has = has || tgt.target == "type"
|
|
|
|
}
|
|
|
|
if has {
|
|
|
|
custom = append(custom, *pair.T)
|
|
|
|
if pair.D != nil {
|
|
|
|
customD = append(customD, *pair.D)
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
raw = append(raw, *pair.T)
|
|
|
|
if pair.D != nil {
|
|
|
|
rawD = append(rawD, *pair.D)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
walk(rootv, visit, nil)
|
|
|
|
|
|
|
|
if len(errors.Errors(terr)) != 0 {
|
|
|
|
return nil, terr
|
|
|
|
}
|
|
|
|
|
|
|
|
vpath := fmt.Sprintf("v%v", thema.LatestVersion(lin)[0])
|
|
|
|
if decl.Meta.Common().Maturity.Less(kindsys.MaturityStable) {
|
|
|
|
vpath = "x"
|
|
|
|
}
|
|
|
|
|
|
|
|
ret := make([]ast.Decl, 0)
|
|
|
|
if len(raw) > 0 {
|
|
|
|
ret = append(ret, ast.ExportSet{
|
|
|
|
CommentList: []ast.Comment{ts.CommentFromString(fmt.Sprintf("Raw generated types from %s kind.", comm.Name), 80, false)},
|
|
|
|
TypeOnly: true,
|
|
|
|
Exports: raw,
|
|
|
|
From: ast.Str{Value: fmt.Sprintf("./raw/%s/%s/%s_types.gen", comm.MachineName, vpath, comm.MachineName)},
|
|
|
|
})
|
|
|
|
}
|
|
|
|
if len(rawD) > 0 {
|
|
|
|
ret = append(ret, ast.ExportSet{
|
|
|
|
CommentList: []ast.Comment{ts.CommentFromString(fmt.Sprintf("Raw generated default consts from %s kind.", lin.Name()), 80, false)},
|
|
|
|
TypeOnly: false,
|
|
|
|
Exports: rawD,
|
|
|
|
From: ast.Str{Value: fmt.Sprintf("./raw/%s/%s/%s_types.gen", comm.MachineName, vpath, comm.MachineName)},
|
|
|
|
})
|
|
|
|
}
|
|
|
|
vtfile := fmt.Sprintf("./veneer/%s.types", lin.Name())
|
|
|
|
customstr := fmt.Sprintf(`// The following exported declarations correspond to types in the %s@%s kind's
|
|
|
|
// schema with attribute @grafana(TSVeneer="type").
|
|
|
|
//
|
|
|
|
// The handwritten file for these type and default veneers is expected to be at
|
|
|
|
// %s.ts.
|
|
|
|
// This re-export declaration enforces that the handwritten veneer file exists,
|
|
|
|
// and exports all the symbols in the list.
|
|
|
|
//
|
|
|
|
// TODO generate code such that tsc enforces type compatibility between raw and veneer decls`,
|
|
|
|
lin.Name(), thema.LatestVersion(lin), filepath.ToSlash(filepath.Join(gen.dir, vtfile)))
|
|
|
|
|
|
|
|
customComments := []ast.Comment{{Text: customstr}}
|
|
|
|
if len(custom) > 0 {
|
|
|
|
ret = append(ret, ast.ExportSet{
|
|
|
|
CommentList: customComments,
|
|
|
|
TypeOnly: true,
|
|
|
|
Exports: custom,
|
|
|
|
From: ast.Str{Value: vtfile},
|
|
|
|
})
|
|
|
|
}
|
|
|
|
if len(customD) > 0 {
|
|
|
|
ret = append(ret, ast.ExportSet{
|
|
|
|
CommentList: customComments,
|
|
|
|
TypeOnly: false,
|
|
|
|
Exports: customD,
|
|
|
|
From: ast.Str{Value: vtfile},
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO emit a decl in the index.gen.ts that ensures any custom veneer types are "compatible" with current version raw types
|
|
|
|
return ret, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
type declPair struct {
|
|
|
|
T, D *ast.Ident
|
|
|
|
}
|
|
|
|
|
|
|
|
type tsVeneerAttr struct {
|
|
|
|
target string
|
|
|
|
}
|
|
|
|
|
|
|
|
func findDeclNode(name string, tf *ast.File) declPair {
|
|
|
|
var p declPair
|
|
|
|
for _, decl := range tf.Nodes {
|
|
|
|
// Peer through export keywords
|
|
|
|
if ex, is := decl.(ast.ExportKeyword); is {
|
|
|
|
decl = ex.Decl
|
|
|
|
}
|
|
|
|
|
|
|
|
switch x := decl.(type) {
|
|
|
|
case ast.TypeDecl:
|
|
|
|
if x.Name.Name == name {
|
|
|
|
p.T = &x.Name
|
|
|
|
}
|
|
|
|
case ast.VarDecl:
|
|
|
|
if x.Names.Idents[0].Name == "default"+name {
|
|
|
|
p.D = &x.Names.Idents[0]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return p
|
|
|
|
}
|
|
|
|
|
|
|
|
func walk(v cue.Value, before func(cue.Path, cue.Value) bool, after func(cue.Path, cue.Value)) {
|
|
|
|
innerWalk(cue.MakePath(), v, before, after)
|
|
|
|
}
|
|
|
|
|
|
|
|
func innerWalk(p cue.Path, v cue.Value, before func(cue.Path, cue.Value) bool, after func(cue.Path, cue.Value)) {
|
|
|
|
switch v.Kind() {
|
|
|
|
default:
|
|
|
|
if before != nil && !before(p, v) {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
case cue.StructKind:
|
|
|
|
if before != nil && !before(p, v) {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
iter, err := v.Fields(cue.All())
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
for iter.Next() {
|
|
|
|
innerWalk(appendPath(p, iter.Selector()), iter.Value(), before, after)
|
|
|
|
}
|
|
|
|
if lv := v.LookupPath(cue.MakePath(cue.AnyString)); lv.Exists() {
|
|
|
|
innerWalk(appendPath(p, cue.AnyString), lv, before, after)
|
|
|
|
}
|
|
|
|
case cue.ListKind:
|
|
|
|
if before != nil && !before(p, v) {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
list, err := v.List()
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
for i := 0; list.Next(); i++ {
|
|
|
|
innerWalk(appendPath(p, cue.Index(i)), list.Value(), before, after)
|
|
|
|
}
|
|
|
|
if lv := v.LookupPath(cue.MakePath(cue.AnyIndex)); lv.Exists() {
|
|
|
|
innerWalk(appendPath(p, cue.AnyString), lv, before, after)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if after != nil {
|
|
|
|
after(p, v)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func appendPath(p cue.Path, sel cue.Selector) cue.Path {
|
|
|
|
return cue.MakePath(append(p.Selectors(), sel)...)
|
|
|
|
}
|
|
|
|
|
|
|
|
func getCustomVeneerAttr(v cue.Value) ([]tsVeneerAttr, error) {
|
|
|
|
var attrs []tsVeneerAttr
|
|
|
|
for _, a := range v.Attributes(cue.ValueAttr) {
|
|
|
|
if a.Name() != "grafana" {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
for i := 0; i < a.NumArgs(); i++ {
|
|
|
|
key, av := a.Arg(i)
|
|
|
|
if key != "TSVeneer" {
|
|
|
|
return nil, valError(v, "attribute 'grafana' only allows the arg 'TSVeneer'")
|
|
|
|
}
|
|
|
|
|
|
|
|
aterr := valError(v, "@grafana(TSVeneer=\"x\") requires one or more of the following separated veneer types for x: %s", allowedTSVeneersString())
|
|
|
|
var some bool
|
|
|
|
for _, tgt := range strings.Split(av, "|") {
|
|
|
|
some = true
|
|
|
|
if !allowedTSVeneers[tgt] {
|
|
|
|
return nil, aterr
|
|
|
|
}
|
|
|
|
attrs = append(attrs, tsVeneerAttr{
|
|
|
|
target: tgt,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
if !some {
|
|
|
|
return nil, aterr
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
sort.Slice(attrs, func(i, j int) bool {
|
|
|
|
return attrs[i].target < attrs[j].target
|
|
|
|
})
|
|
|
|
|
|
|
|
return attrs, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
var allowedTSVeneers = map[string]bool{
|
|
|
|
"type": true,
|
|
|
|
}
|
|
|
|
|
|
|
|
func allowedTSVeneersString() string {
|
|
|
|
var list []string
|
|
|
|
for tgt := range allowedTSVeneers {
|
|
|
|
list = append(list, tgt)
|
|
|
|
}
|
|
|
|
sort.Strings(list)
|
|
|
|
|
|
|
|
return strings.Join(list, "|")
|
|
|
|
}
|
|
|
|
|
|
|
|
func valError(v cue.Value, format string, args ...interface{}) error {
|
|
|
|
s := v.Source()
|
|
|
|
if s == nil {
|
|
|
|
return fmt.Errorf(format, args...)
|
|
|
|
}
|
|
|
|
return errors.Newf(s.Pos(), format, args...)
|
|
|
|
}
|