pulumi/pkg/encoding/convert.go
joeduffy 370b0a1406 Implement property binding and typechecking
This is an initial pass at property binding.  For all stack instantiations,
we must verify that the set of properties supplied are correct.  We also must
remember the bound property information so that code-generation has all of
the information it needs to generate correct code (including capability refs).

This entails:

* Ensuring required properties are provided.

* Expanding missing properties that have Default values.

* Type-checking that supplied properties are of the right type.

* Expanding property values into AST literal nodes.

To do this requires a third AST pass in the semantic analysis part of the
compiler.  In the 1st pass, dependencies aren't even known yet; in the 2nd
pass, dependencies have not yet been bound; therefore, we need a 3rd pass,
which can depend on the full binding information for the transitive closure
of AST nodes and dependencies to have been populated with types.

There are a few loose ends in here:

* We don't yet validate top-level stack properties.

* We don't yet validate top-level stack base type properties.

* We don't yet support complex schema property types.

* We don't yet support even "simple" complex property types, like `[ string ]`.

* We don't yet support strongly typed capability property types (just `service`).

That said, I am going to turn to writing a few tests for the basic cases, and then
resume to finishing this afterwards (tracked by marapongo/mu#25).
2016-12-02 13:23:18 -08:00

28 lines
714 B
Go

// Copyright 2016 Marapongo, Inc. All rights reserved.
package encoding
// StringSlice checks a weakly typed interface ptr to see if it's a []string; if yes, the resulting converted array
// is returned with a "true"; otherwise, nil with a "false" is returned.
func StringSlice(i interface{}) ([]string, bool) {
// First try a direct conversion.
if s, ok := i.([]string); ok {
return s, true
}
// Otherwise, see if it's an untyped array, and then convert each element.
if a, ok := i.([]interface{}); ok {
ss := make([]string, 0, len(a))
for _, e := range a {
if s, ok := e.(string); ok {
ss = append(ss, s)
} else {
return nil, false
}
}
return ss, true
}
return nil, false
}