This repository has been archived by the owner on Jul 4, 2024. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 33
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
feat: added metadata properties substitutions support (#16)
Signed-off-by: Eugene Yarshevich <[email protected]>
- Loading branch information
Showing
3 changed files
with
243 additions
and
51 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,99 @@ | ||
/* | ||
Apache Score | ||
Copyright 2022 The Apache Software Foundation | ||
This product includes software developed at | ||
The Apache Software Foundation (http://www.apache.org/). | ||
*/ | ||
package helm | ||
|
||
import ( | ||
"fmt" | ||
"log" | ||
"os" | ||
|
||
"github.com/mitchellh/mapstructure" | ||
|
||
score "github.com/score-spec/score-go/types" | ||
) | ||
|
||
// templatesContext ia an utility type that provides a context for '${...}' templates substitution | ||
type templatesContext map[string]string | ||
|
||
// buildContext initializes a new templatesContext instance | ||
func buildContext(metadata score.WorkloadMeta, resources score.ResourcesSpecs, values map[string]interface{}) (templatesContext, error) { | ||
var ctx = make(map[string]string) | ||
|
||
var metadataMap = make(map[string]interface{}) | ||
if decoder, err := mapstructure.NewDecoder(&mapstructure.DecoderConfig{ | ||
TagName: "json", | ||
Result: &metadataMap, | ||
}); err != nil { | ||
return nil, err | ||
} else { | ||
decoder.Decode(metadata) | ||
for key, val := range metadataMap { | ||
var ref = fmt.Sprintf("metadata.%s", key) | ||
if _, exists := ctx[ref]; exists { | ||
return nil, fmt.Errorf("ambiguous property reference '%s'", ref) | ||
} | ||
ctx[ref] = fmt.Sprintf("%v", val) | ||
} | ||
} | ||
|
||
for resName, res := range resources { | ||
ctx[fmt.Sprintf("resources.%s", resName)] = resName | ||
|
||
for propName, prop := range res.Properties { | ||
var ref = fmt.Sprintf("resources.%s.%s", resName, propName) | ||
if _, exists := ctx[ref]; exists { | ||
return nil, fmt.Errorf("ambiguous property reference '%s'", ref) | ||
} | ||
|
||
// Use the default value provided (if any) | ||
var val = fmt.Sprintf("%v", prop.Default) | ||
|
||
// Override the default value for the property (if provided) | ||
if src, ok := values[resName]; ok { | ||
if srcMap, ok := src.(map[string]interface{}); ok { | ||
if v, ok := srcMap[propName]; ok { | ||
val = fmt.Sprintf("%v", v) | ||
} | ||
} | ||
} | ||
|
||
ctx[ref] = val | ||
} | ||
} | ||
|
||
return ctx, nil | ||
} | ||
|
||
// Substitute replaces all matching '${...}' templates in a source string | ||
func (context templatesContext) Substitute(src string) string { | ||
return os.Expand(src, context.mapVar) | ||
} | ||
|
||
// MapVar replaces objects and properties references with corresponding values | ||
// Returns an empty string if the reference can't be resolved | ||
func (context templatesContext) mapVar(ref string) string { | ||
if ref == "" { | ||
return "" | ||
} | ||
|
||
// NOTE: os.Expand(..) would invoke a callback function with "$" as an argument for escaped sequences. | ||
// "$${abc}" is treated as "$$" pattern and "{abc}" static text. | ||
// The first segment (pattern) would trigger a callback function call. | ||
// By returning "$" value we would ensure that escaped sequences would remain in the source text. | ||
// For example "$${abc}" would result in "${abc}" after os.Expand(..) call. | ||
if ref == "$" { | ||
return ref | ||
} | ||
|
||
if res, ok := context[ref]; ok { | ||
return res | ||
} | ||
|
||
log.Printf("Warning: Can not resolve '%s'. Resource or property is not declared.", ref) | ||
return "" | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,139 @@ | ||
/* | ||
Apache Score | ||
Copyright 2022 The Apache Software Foundation | ||
This product includes software developed at | ||
The Apache Software Foundation (http://www.apache.org/). | ||
*/ | ||
package helm | ||
|
||
import ( | ||
"testing" | ||
|
||
score "github.com/score-spec/score-go/types" | ||
assert "github.com/stretchr/testify/assert" | ||
) | ||
|
||
func TestBuildContext(t *testing.T) { | ||
var meta = score.WorkloadMeta{ | ||
Name: "test-name", | ||
} | ||
|
||
var resources = score.ResourcesSpecs{ | ||
"env": score.ResourceSpec{ | ||
Type: "environment", | ||
Properties: map[string]score.ResourcePropertySpec{ | ||
"DEBUG": {Required: false, Default: true}, | ||
}, | ||
}, | ||
"db": score.ResourceSpec{ | ||
Type: "postgres", | ||
Properties: map[string]score.ResourcePropertySpec{ | ||
"host": {Required: true, Default: "."}, | ||
"port": {Required: true, Default: "5342"}, | ||
"name": {Required: true}, | ||
}, | ||
}, | ||
"dns": score.ResourceSpec{ | ||
Type: "dns", | ||
Properties: map[string]score.ResourcePropertySpec{ | ||
"domain": {}, | ||
}, | ||
}, | ||
} | ||
|
||
var values = map[string]interface{}{ | ||
"db": map[string]interface{}{ | ||
"host": "localhost", | ||
"name": "test-db", | ||
}, | ||
"dns": map[string]interface{}{ | ||
"domain": "test.domain.name", | ||
}, | ||
} | ||
|
||
context, err := buildContext(meta, resources, values) | ||
assert.NoError(t, err) | ||
|
||
assert.Equal(t, templatesContext{ | ||
"metadata.name": "test-name", | ||
|
||
"resources.env": "env", | ||
"resources.env.DEBUG": "true", | ||
|
||
"resources.db": "db", | ||
"resources.db.host": "localhost", | ||
"resources.db.port": "5342", | ||
"resources.db.name": "test-db", | ||
|
||
"resources.dns": "dns", | ||
"resources.dns.domain": "test.domain.name", | ||
}, context) | ||
} | ||
|
||
func TestMapVar(t *testing.T) { | ||
var context = templatesContext{ | ||
"metadata.name": "test-name", | ||
|
||
"resources.env": "env", | ||
"resources.env.DEBUG": "true", | ||
|
||
"resources.db": "db", | ||
"resources.db.host": "localhost", | ||
"resources.db.port": "5342", | ||
"resources.db.name": "test-db", | ||
|
||
"resources.dns": "shared.dns", | ||
"resources.dns.domain": "test.domain.name", | ||
} | ||
|
||
assert.Equal(t, "", context.mapVar("")) | ||
assert.Equal(t, "$", context.mapVar("$")) | ||
|
||
assert.Equal(t, "test-name", context.mapVar("metadata.name")) | ||
assert.Equal(t, "", context.mapVar("metadata.name.nil")) | ||
assert.Equal(t, "", context.mapVar("metadata.nil")) | ||
|
||
assert.Equal(t, "true", context.mapVar("resources.env.DEBUG")) | ||
|
||
assert.Equal(t, "db", context.mapVar("resources.db")) | ||
assert.Equal(t, "localhost", context.mapVar("resources.db.host")) | ||
assert.Equal(t, "5342", context.mapVar("resources.db.port")) | ||
assert.Equal(t, "test-db", context.mapVar("resources.db.name")) | ||
assert.Equal(t, "", context.mapVar("resources.db.name.nil")) | ||
assert.Equal(t, "", context.mapVar("resources.db.nil")) | ||
assert.Equal(t, "", context.mapVar("resources.nil")) | ||
assert.Equal(t, "", context.mapVar("nil.db.name")) | ||
} | ||
|
||
func TestSubstitute(t *testing.T) { | ||
var context = templatesContext{ | ||
"metadata.name": "test-name", | ||
|
||
"resources.env": "env", | ||
"resources.env.DEBUG": "true", | ||
|
||
"resources.db": "db", | ||
"resources.db.host": "localhost", | ||
"resources.db.port": "5342", | ||
"resources.db.name": "test-db", | ||
|
||
"resources.dns": "dns", | ||
"resources.dns.domain": "test.domain.name", | ||
} | ||
|
||
assert.Equal(t, "", context.Substitute("")) | ||
assert.Equal(t, "abc", context.Substitute("abc")) | ||
assert.Equal(t, "abc $ abc", context.Substitute("abc $$ abc")) | ||
assert.Equal(t, "${abc}", context.Substitute("$${abc}")) | ||
|
||
assert.Equal(t, "The name is 'test-name'", context.Substitute("The name is '${metadata.name}'")) | ||
assert.Equal(t, "The name is ''", context.Substitute("The name is '${metadata.nil}'")) | ||
|
||
assert.Equal(t, "resources.env.DEBUG", context.Substitute("resources.env.DEBUG")) | ||
|
||
assert.Equal(t, "db", context.Substitute("${resources.db}")) | ||
assert.Equal(t, | ||
"postgresql://:@localhost:5342/test-db", | ||
context.Substitute("postgresql://${resources.db.user}:${resources.db.password}@${resources.db.host}:${resources.db.port}/${resources.db.name}")) | ||
} |