mirror of
https://github.com/mcuadros/ascode
synced 2024-11-22 17:02:03 +01:00
nested: nested blocks and tests
Signed-off-by: Máximo Cuadros <mcuadros@gmail.com>
This commit is contained in:
parent
07d0555a7e
commit
7d56f1762c
60
main_test.go
Normal file
60
main_test.go
Normal file
@ -0,0 +1,60 @@
|
||||
package main
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
"io/ioutil"
|
||||
"log"
|
||||
"testing"
|
||||
|
||||
"go.starlark.net/resolve"
|
||||
"go.starlark.net/starlark"
|
||||
"go.starlark.net/starlarktest"
|
||||
)
|
||||
|
||||
func init() {
|
||||
// The tests make extensive use of these not-yet-standard features.
|
||||
resolve.AllowLambda = true
|
||||
resolve.AllowNestedDef = true
|
||||
resolve.AllowFloat = true
|
||||
resolve.AllowSet = true
|
||||
}
|
||||
|
||||
func TestProvider(t *testing.T) {
|
||||
test(t, "testdata/provider.star")
|
||||
}
|
||||
|
||||
func TestNestedBlock(t *testing.T) {
|
||||
test(t, "testdata/nested.star")
|
||||
}
|
||||
|
||||
func test(t *testing.T, filename string) {
|
||||
log.SetOutput(ioutil.Discard)
|
||||
thread := &starlark.Thread{Load: load}
|
||||
starlarktest.SetReporter(thread, t)
|
||||
|
||||
provider := starlark.NewBuiltin("provider", func(thread *starlark.Thread, fn *starlark.Builtin, args starlark.Tuple, kwargs []starlark.Tuple) (starlark.Value, error) {
|
||||
name := args.Index(0).(starlark.String)
|
||||
version := args.Index(1).(starlark.String)
|
||||
return NewProviderInstance(&PluginManager{".providers"}, string(name), string(version))
|
||||
})
|
||||
|
||||
predeclared := starlark.StringDict{
|
||||
"provider": provider,
|
||||
}
|
||||
|
||||
if _, err := starlark.ExecFile(thread, filename, nil, predeclared); err != nil {
|
||||
if err, ok := err.(*starlark.EvalError); ok {
|
||||
t.Fatal(err.Backtrace())
|
||||
}
|
||||
t.Fatal(err)
|
||||
}
|
||||
}
|
||||
|
||||
// load implements the 'load' operation as used in the evaluator tests.
|
||||
func load(thread *starlark.Thread, module string) (starlark.StringDict, error) {
|
||||
if module == "assert.star" {
|
||||
return starlarktest.LoadAssertModule()
|
||||
}
|
||||
|
||||
return nil, fmt.Errorf("load not implemented")
|
||||
}
|
64
nested.go
Normal file
64
nested.go
Normal file
@ -0,0 +1,64 @@
|
||||
package main
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
|
||||
"github.com/hashicorp/terraform/configs/configschema"
|
||||
|
||||
"go.starlark.net/starlark"
|
||||
)
|
||||
|
||||
type NestedBlock struct {
|
||||
typ string
|
||||
block *configschema.Block
|
||||
*PointerList
|
||||
}
|
||||
|
||||
func NewNestedBlock(typ string, block *configschema.Block, refs *PointerList) *NestedBlock {
|
||||
return &NestedBlock{typ: typ, block: block, PointerList: refs}
|
||||
}
|
||||
|
||||
// String honors the starlark.Value interface.
|
||||
func (r *NestedBlock) String() string {
|
||||
return fmt.Sprintf("%s", r.typ)
|
||||
}
|
||||
|
||||
// Type honors the starlark.Value interface.
|
||||
func (r *NestedBlock) Type() string {
|
||||
return fmt.Sprintf("%s_collection", r.typ)
|
||||
}
|
||||
|
||||
// Truth honors the starlark.Value interface.
|
||||
func (r *NestedBlock) Truth() starlark.Bool {
|
||||
return true // even when empty
|
||||
}
|
||||
|
||||
// Freeze honors the starlark.Value interface.
|
||||
func (r *NestedBlock) Freeze() {}
|
||||
|
||||
// Hash honors the starlark.Value interface.
|
||||
func (r *NestedBlock) Hash() (uint32, error) { return 42, nil }
|
||||
|
||||
// Name honors the starlark.Callable interface.
|
||||
func (b *NestedBlock) Name() string {
|
||||
return b.typ
|
||||
}
|
||||
|
||||
// CallInternal honors the starlark.Callable interface.
|
||||
func (b *NestedBlock) CallInternal(thread *starlark.Thread, args starlark.Tuple, kwargs []starlark.Tuple) (starlark.Value, error) {
|
||||
var name starlark.String
|
||||
if len(args) != 0 {
|
||||
name = args.Index(0).(starlark.String)
|
||||
}
|
||||
|
||||
resource, err := MakeResourceInstance(string(name), b.typ, b.block, kwargs)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
if err := b.PointerList.Append(resource); err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
return resource, nil
|
||||
}
|
33
provider.go
33
provider.go
@ -4,7 +4,7 @@ import (
|
||||
"fmt"
|
||||
"strings"
|
||||
|
||||
"github.com/hashicorp/terraform/configs/configschema"
|
||||
"github.com/hashicorp/terraform/plugin/discovery"
|
||||
|
||||
"github.com/hashicorp/terraform/plugin"
|
||||
"github.com/hashicorp/terraform/providers"
|
||||
@ -14,14 +14,14 @@ import (
|
||||
type ProviderInstance struct {
|
||||
name string
|
||||
provider *plugin.GRPCProvider
|
||||
meta discovery.PluginMeta
|
||||
|
||||
dataSources *MapSchemaIntance
|
||||
resources *MapSchemaIntance
|
||||
}
|
||||
|
||||
func NewProviderInstance(pm *PluginManager, name string) (*ProviderInstance, error) {
|
||||
cli := pm.Get(name, "")
|
||||
|
||||
func NewProviderInstance(pm *PluginManager, name, version string) (*ProviderInstance, error) {
|
||||
cli, meta := pm.Get(name, version)
|
||||
rpc, err := cli.Client()
|
||||
if err != nil {
|
||||
return nil, err
|
||||
@ -39,33 +39,12 @@ func NewProviderInstance(pm *PluginManager, name string) (*ProviderInstance, err
|
||||
return &ProviderInstance{
|
||||
name: name,
|
||||
provider: provider,
|
||||
meta: meta,
|
||||
dataSources: NewMapSchemaInstance(name, response.DataSources),
|
||||
resources: NewMapSchemaInstance(name, response.ResourceTypes),
|
||||
}, nil
|
||||
}
|
||||
|
||||
func computeNestedBlocks(s map[string]providers.Schema) map[string]*configschema.NestedBlock {
|
||||
blks := make(map[string]*configschema.NestedBlock)
|
||||
for k, block := range s {
|
||||
for n, nested := range block.Block.BlockTypes {
|
||||
key := k + "_" + n
|
||||
doComputeNestedBlocks(key, nested, blks)
|
||||
}
|
||||
}
|
||||
|
||||
return blks
|
||||
}
|
||||
|
||||
func doComputeNestedBlocks(name string, b *configschema.NestedBlock, list map[string]*configschema.NestedBlock) {
|
||||
list[name] = b
|
||||
for k, block := range b.BlockTypes {
|
||||
key := name + "_" + k
|
||||
list[key] = block
|
||||
|
||||
doComputeNestedBlocks(key, block, list)
|
||||
}
|
||||
}
|
||||
|
||||
func (t *ProviderInstance) String() string {
|
||||
return fmt.Sprintf("provider(%q)", t.name)
|
||||
}
|
||||
@ -80,6 +59,8 @@ func (t *ProviderInstance) Hash() (uint32, error) { return 1, nil }
|
||||
func (t *ProviderInstance) Name() string { return t.name }
|
||||
func (s *ProviderInstance) Attr(name string) (starlark.Value, error) {
|
||||
switch name {
|
||||
case "version":
|
||||
return starlark.String(s.meta.Version), nil
|
||||
case "data":
|
||||
return s.dataSources, nil
|
||||
case "resource":
|
||||
|
30
testdata/nested.star
vendored
Normal file
30
testdata/nested.star
vendored
Normal file
@ -0,0 +1,30 @@
|
||||
load("assert.star", "assert")
|
||||
|
||||
p = provider("aws", "2.13.0")
|
||||
d = p.data.ami("foo")
|
||||
|
||||
assert.eq(type(d.filter), "filter_collection")
|
||||
|
||||
bar = d.filter(name="bar", values=["qux"])
|
||||
|
||||
assert.eq(type(bar), "filter")
|
||||
assert.eq(bar.name, "bar")
|
||||
assert.eq(bar.values, ["qux"])
|
||||
|
||||
assert.eq(len(d.filter), 1)
|
||||
assert.eq(d.filter[0], bar)
|
||||
|
||||
qux = d.filter()
|
||||
qux.name = "qux"
|
||||
qux.values = ["bar"]
|
||||
|
||||
assert.eq(qux.name, "qux")
|
||||
assert.eq(qux.values, ["bar"])
|
||||
|
||||
assert.eq(len(d.filter), 2)
|
||||
assert.eq(d.filter[1], qux)
|
||||
|
||||
d.filter[1].values = ["baz"]
|
||||
assert.eq(qux.values, ["baz"])
|
||||
|
||||
assert.ne(d.filter[0], d.filter[1])
|
Loading…
Reference in New Issue
Block a user