Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Substituting variables into the devfile from the CLI #5749

Merged
Show file tree
Hide file tree
Changes from 8 commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
Expand Up @@ -64,3 +64,10 @@ components:
- name: main
image: {{CONTAINER_IMAGE}}
```

## Sustituting variables

The Devfile can define variables to make the Devfile parameterizable. The Devfile can define values for these variables, and you
can override the values for variables from the command line when running `odo deploy`, using the `--var` and `--var-file` options.

See [Sustituting variables in odo dev](dev.md#sustituting-variables) for more information.
11 changes: 11 additions & 0 deletions docs/website/versioned_docs/version-3.0.0/command-reference/dev.md
Original file line number Diff line number Diff line change
Expand Up @@ -43,6 +43,17 @@ In the above example, three things have happened:
You can press Ctrl-c at any time to terminate the development session. The command can take a few moment to terminate, as it
will first delete all resources deployed into the cluster for this session before terminating.

### Sustituting variables

The Devfile can define variables to make the Devfile parameterizable. The Devfile can define values for these variables, and you
can override the values for variables from the command line when running `odo dev`, using the `--var` and `--var-file` options.

The `--var` option takes a comma-separated list of `variable=value` pairs, where `=value` is optional. If the `=value` is omitted, the value is extracted from the environment variable named `variable`. In this case, if the environment variable with this name is not defined, the value defined into the Devfile will be used.
rm3l marked this conversation as resolved.
Show resolved Hide resolved

The `--var-file` option takes a filename as argument. The file contains a line-separated list of `variable=value` pairs, with the same behaviour as before.

Note that the values passed with the `--var` option overrides the values obtained with the `--var-file` option.

## Devfile (Advanced Usage)


Expand Down
4 changes: 2 additions & 2 deletions go.mod
Original file line number Diff line number Diff line change
Expand Up @@ -7,8 +7,8 @@ require (
github.com/Netflix/go-expect v0.0.0-20201125194554-85d881c3777e
github.com/Xuanwo/go-locale v1.0.0
github.com/blang/semver v3.5.1+incompatible
github.com/devfile/api/v2 v2.0.0-20220117162434-6e6e6a8bc14c
github.com/devfile/library v1.2.1-0.20220217161036-0f5995513e92
github.com/devfile/api/v2 v2.0.0-20220309195345-48ebbf1e51cf
github.com/devfile/library v1.2.1-0.20220602130922-85a4805bd59c
github.com/devfile/registry-support/index/generator v0.0.0-20220222194908-7a90a4214f3e
github.com/devfile/registry-support/registry-library v0.0.0-20220504150710-21de53798172
github.com/fatih/color v1.13.0
Expand Down
4 changes: 4 additions & 0 deletions go.sum
Original file line number Diff line number Diff line change
Expand Up @@ -353,10 +353,14 @@ github.com/devfile/api/v2 v2.0.0-20210910153124-da620cd1a7a1/go.mod h1:kLX/nW93g
github.com/devfile/api/v2 v2.0.0-20211021164004-dabee4e633ed/go.mod h1:d99eTN6QxgzihOOFyOZA+VpUyD4Q1pYRYHZ/ci9J96Q=
github.com/devfile/api/v2 v2.0.0-20220117162434-6e6e6a8bc14c h1:sjghKUov/WT71dBreHYQcTgQctxHT/p4uAhUFdGQfSU=
github.com/devfile/api/v2 v2.0.0-20220117162434-6e6e6a8bc14c/go.mod h1:d99eTN6QxgzihOOFyOZA+VpUyD4Q1pYRYHZ/ci9J96Q=
github.com/devfile/api/v2 v2.0.0-20220309195345-48ebbf1e51cf h1:FkwAOQtepscB5B0j++9S/eoicXj707MaP5HPIScz0sA=
github.com/devfile/api/v2 v2.0.0-20220309195345-48ebbf1e51cf/go.mod h1:kLX/nW93gigOHXK3NLeJL2fSS/sgEe+OHu8bo3aoOi4=
github.com/devfile/library v1.1.1-0.20210910214722-7c5ff63711ec/go.mod h1:svPWwWb+BP15SXCHl0dyOeE4Sohrjl5a2BaOzc/riLc=
github.com/devfile/library v1.2.1-0.20211104222135-49d635cb492f/go.mod h1:uFZZdTuRqA68FVe/JoJHP92CgINyQkyWnM2Qyiim+50=
github.com/devfile/library v1.2.1-0.20220217161036-0f5995513e92 h1:RIrd0pBAET9vLHEZGyaG1PSjp/lJXa+CZfuWiih2p6g=
github.com/devfile/library v1.2.1-0.20220217161036-0f5995513e92/go.mod h1:GSPfJaBg0+bBjBHbwBE5aerJLH6tWGQu2q2rHYd9czM=
github.com/devfile/library v1.2.1-0.20220602130922-85a4805bd59c h1:Puqg/NJ6pP1LPFxPYBX9c0Oh0Ttu5PVdNJL5WR/CRTk=
github.com/devfile/library v1.2.1-0.20220602130922-85a4805bd59c/go.mod h1:kpkIA9YI7gkfglRDsKJ3n5QgVyyBmnasr2U2djYwg6s=
github.com/devfile/registry-support/index/generator v0.0.0-20211012185733-0a73f866043f h1:fKNUmoOPh7yAs69uMRZWHvev+m3e7T4jBL/hOXZB9ys=
github.com/devfile/registry-support/index/generator v0.0.0-20211012185733-0a73f866043f/go.mod h1:bLGagbW2SFn7jo5+kUPlCMehIGqWkRtLKc5O0OyJMJM=
github.com/devfile/registry-support/index/generator v0.0.0-20220222194908-7a90a4214f3e h1:3WAjUoyAmCBzUpx+sO0dSgkH74uSW1y886wGbojD2D8=
Expand Down
3 changes: 2 additions & 1 deletion pkg/dev/dev.go
Original file line number Diff line number Diff line change
Expand Up @@ -56,7 +56,7 @@ func (o *DevClient) Start(devfileObj parser.DevfileObj, platformContext kubernet
return nil
}

func (o *DevClient) Watch(devfileObj parser.DevfileObj, path string, ignorePaths []string, out io.Writer, h Handler, ctx context.Context, debug bool) error {
func (o *DevClient) Watch(devfileObj parser.DevfileObj, path string, ignorePaths []string, out io.Writer, h Handler, ctx context.Context, debug bool, variables map[string]string) error {
envSpecificInfo, err := envinfo.NewEnvSpecificInfo(path)
if err != nil {
return err
Expand All @@ -72,6 +72,7 @@ func (o *DevClient) Watch(devfileObj parser.DevfileObj, path string, ignorePaths
InitialDevfileObj: devfileObj,
Debug: debug,
DebugPort: envSpecificInfo.GetDebugPort(),
Variables: variables,
}

return o.watchClient.WatchAndPush(out, watchParameters, ctx)
Expand Down
2 changes: 1 addition & 1 deletion pkg/dev/interface.go
Original file line number Diff line number Diff line change
Expand Up @@ -20,7 +20,7 @@ type Client interface {
// It logs messages to out and uses the Handler h to perform push operation when anything changes in path.
// It uses devfileObj to notify user to restart odo dev if they change endpoint information in the devfile.
// If debug is true, the debug command will be started after a sync, or the run command by default
Watch(devfileObj parser.DevfileObj, path string, ignorePaths []string, out io.Writer, h Handler, ctx context.Context, debug bool) error
Watch(devfileObj parser.DevfileObj, path string, ignorePaths []string, out io.Writer, h Handler, ctx context.Context, debug bool, variables map[string]string) error
}

type Handler interface {
Expand Down
8 changes: 4 additions & 4 deletions pkg/dev/mock.go

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

9 changes: 9 additions & 0 deletions pkg/devfile/devfile.go
Original file line number Diff line number Diff line change
Expand Up @@ -47,6 +47,15 @@ func ParseAndValidateFromFile(devfilePath string) (parser.DevfileObj, error) {
return parseDevfile(parser.ParserArgs{Path: devfilePath})
}

// ParseAndValidateFromFile reads, parses and validates devfile from a file
rm3l marked this conversation as resolved.
Show resolved Hide resolved
// if there are warning it logs them on stdout
func ParseAndValidateFromFileWithVariables(devfilePath string, variables map[string]string) (parser.DevfileObj, error) {
return parseDevfile(parser.ParserArgs{
Path: devfilePath,
ExternalVariables: variables,
})
}

func variableWarning(section string, variable string, messages []string) string {
quotedVars := []string{}
for _, v := range messages {
Expand Down
19 changes: 17 additions & 2 deletions pkg/odo/cli/deploy/deploy.go
Original file line number Diff line number Diff line change
Expand Up @@ -16,6 +16,7 @@ import (
"github.com/redhat-developer/odo/pkg/odo/genericclioptions/clientset"
odoutil "github.com/redhat-developer/odo/pkg/odo/util"
scontext "github.com/redhat-developer/odo/pkg/segment/context"
"github.com/redhat-developer/odo/pkg/vars"
"github.com/redhat-developer/odo/pkg/version"

"os"
Expand All @@ -36,6 +37,13 @@ type DeployOptions struct {
// Clients
clientset *clientset.Clientset

// Flags
varFileFlag string
varsFlag []string

// Variables to override Devfile variables
variables map[string]string

// working directory
contextDir string
}
Expand Down Expand Up @@ -85,7 +93,12 @@ func (o *DeployOptions) Complete(cmdline cmdline.Cmdline, args []string) (err er
return err
}

o.Context, err = genericclioptions.New(genericclioptions.NewCreateParameters(cmdline).NeedDevfile(o.contextDir).CreateAppIfNeeded())
o.variables, err = vars.GetVariables(o.clientset.FS, o.varFileFlag, o.varsFlag, os.LookupEnv)
if err != nil {
return err
}

o.Context, err = genericclioptions.New(genericclioptions.NewCreateParameters(cmdline).NeedDevfile(o.contextDir).WithVariables(o.variables).CreateAppIfNeeded())
if err != nil {
return err
}
Expand Down Expand Up @@ -161,7 +174,9 @@ func NewCmdDeploy(name, fullName string) *cobra.Command {
genericclioptions.GenericRun(o, cmd, args)
},
}
clientset.Add(deployCmd, clientset.INIT, clientset.DEPLOY)
deployCmd.Flags().StringArrayVar(&o.varsFlag, "var", []string{}, "Variable to override Devfile variable and variables in var-file")
deployCmd.Flags().StringVar(&o.varFileFlag, "var-file", "", "File containing variables to override Devfile variables")
clientset.Add(deployCmd, clientset.INIT, clientset.DEPLOY, clientset.FILESYSTEM)

// Add a defined annotation in order to appear in the help menu
deployCmd.Annotations["command"] = "main"
Expand Down
22 changes: 17 additions & 5 deletions pkg/odo/cli/dev/dev.go
Original file line number Diff line number Diff line change
Expand Up @@ -32,6 +32,7 @@ import (
odoutil "github.com/redhat-developer/odo/pkg/odo/util"
scontext "github.com/redhat-developer/odo/pkg/segment/context"
"github.com/redhat-developer/odo/pkg/util"
"github.com/redhat-developer/odo/pkg/vars"
"github.com/redhat-developer/odo/pkg/version"
"github.com/redhat-developer/odo/pkg/watch"
)
Expand Down Expand Up @@ -65,6 +66,11 @@ type DevOptions struct {
noWatchFlag bool
randomPortsFlag bool
debugFlag bool
varFileFlag string
varsFlag []string

// Variables to override Devfile variables
variables map[string]string
}

type Handler struct{}
Expand Down Expand Up @@ -122,7 +128,12 @@ func (o *DevOptions) Complete(cmdline cmdline.Cmdline, args []string) error {
return err
}

o.Context, err = genericclioptions.New(genericclioptions.NewCreateParameters(cmdline).NeedDevfile(""))
o.variables, err = vars.GetVariables(o.clientset.FS, o.varFileFlag, o.varsFlag, os.LookupEnv)
if err != nil {
return err
}

o.Context, err = genericclioptions.New(genericclioptions.NewCreateParameters(cmdline).NeedDevfile("").WithVariables(o.variables))
if err != nil {
return fmt.Errorf("unable to create context: %v", err)
}
Expand Down Expand Up @@ -257,7 +268,7 @@ func (o *DevOptions) Run(ctx context.Context) (err error) {
err = o.clientset.WatchClient.CleanupDevResources(devFileObj, log.GetStdout())
} else {
d := Handler{}
err = o.clientset.DevClient.Watch(devFileObj, path, o.ignorePaths, o.out, &d, o.ctx, o.debugFlag)
err = o.clientset.DevClient.Watch(devFileObj, path, o.ignorePaths, o.out, &d, o.ctx, o.debugFlag, o.variables)
}
return err
}
Expand All @@ -280,7 +291,7 @@ func (o *Handler) RegenerateAdapterAndPush(pushParams common.PushParameters, wat
}

func regenerateComponentAdapterFromWatchParams(parameters watch.WatchParameters) (common.ComponentAdapter, error) {
devObj, err := ododevfile.ParseAndValidateFromFile(location.DevfileLocation(""))
devObj, err := ododevfile.ParseAndValidateFromFileWithVariables(location.DevfileLocation(""), parameters.Variables)
if err != nil {
return nil, err
}
Expand Down Expand Up @@ -321,8 +332,9 @@ It forwards endpoints with exposure values 'public' or 'internal' to a port on l
devCmd.Flags().BoolVar(&o.noWatchFlag, "no-watch", false, "Do not watch for file changes")
devCmd.Flags().BoolVar(&o.randomPortsFlag, "random-ports", false, "Assign random ports to redirected ports")
devCmd.Flags().BoolVar(&o.debugFlag, "debug", false, "Execute the debug command within the component")

clientset.Add(devCmd, clientset.DEV, clientset.INIT, clientset.KUBERNETES, clientset.STATE)
devCmd.Flags().StringArrayVar(&o.varsFlag, "var", []string{}, "Variable to override Devfile variable and variables in var-file")
devCmd.Flags().StringVar(&o.varFileFlag, "var-file", "", "File containing variables to override Devfile variables")
clientset.Add(devCmd, clientset.DEV, clientset.INIT, clientset.KUBERNETES, clientset.STATE, clientset.FILESYSTEM)
// Add a defined annotation in order to appear in the help menu
devCmd.Annotations["command"] = "main"
devCmd.SetUsageTemplate(odoutil.CmdUsageTemplate)
Expand Down
9 changes: 8 additions & 1 deletion pkg/odo/genericclioptions/context.go
Original file line number Diff line number Diff line change
Expand Up @@ -60,6 +60,8 @@ type CreateParameters struct {
devfile bool
offline bool
appIfNeeded bool
// variables override devfile variables
variables map[string]string
}

func NewCreateParameters(cmdline cmdline.Cmdline) CreateParameters {
Expand All @@ -82,6 +84,11 @@ func (o CreateParameters) CreateAppIfNeeded() CreateParameters {
return o
}

func (o CreateParameters) WithVariables(variables map[string]string) CreateParameters {
o.variables = variables
return o
}

// New creates a context based on the given parameters
func New(parameters CreateParameters) (*Context, error) {
ctx := internalCxt{}
Expand Down Expand Up @@ -124,7 +131,7 @@ func New(parameters CreateParameters) (*Context, error) {
isDevfile := odoutil.CheckPathExists(ctx.devfilePath)
if isDevfile {
// Parse devfile and validate
devObj, err := devfile.ParseAndValidateFromFile(ctx.devfilePath)
devObj, err := devfile.ParseAndValidateFromFileWithVariables(ctx.devfilePath, parameters.variables)
if err != nil {
return nil, fmt.Errorf("failed to parse the devfile %s, with error: %s", ctx.devfilePath, err)
}
Expand Down
2 changes: 2 additions & 0 deletions pkg/vars/doc.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,2 @@
// The var package analyzes variables defined in env file and as flags
package vars
15 changes: 15 additions & 0 deletions pkg/vars/errors.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,15 @@
package vars

import "fmt"

type ErrBadKey struct {
msg string
}

func NewErrBadKey(msg string) ErrBadKey {
return ErrBadKey{msg: msg}
}

func (e ErrBadKey) Error() string {
return fmt.Sprintf("poorly formatted environment: %s", e.msg)
}
112 changes: 112 additions & 0 deletions pkg/vars/vars.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,112 @@
package vars

import (
"bufio"
"fmt"
"strings"
"unicode"

"github.com/redhat-developer/odo/pkg/testingutil/filesystem"
)

// GetVariables returns a map of key/value from pairs defined in the file and in the list of strings in the format KEY=VALUE or KEY
// For a KEY entry, the value wbe obtained from the environment, and if the value is not defined in the environment, the entry KEY will be ignored
feloy marked this conversation as resolved.
Show resolved Hide resolved
// An empty filename will skip the extraction of pairs from file
func GetVariables(fs filesystem.Filesystem, filename string, override []string, lookupEnv func(string) (string, bool)) (map[string]string, error) {

result := map[string]string{}
var err error
if len(filename) > 0 {
result, err = parseKeyValueFile(fs, filename, lookupEnv)
if err != nil {
return nil, err
}
}
overrideVars, err := parseKeyValueStrings(override, lookupEnv)
if err != nil {
return nil, err
}

for k, v := range overrideVars {
result[k] = v
}

return result, nil
}

// parseKeyValueFile parses a file for "KEY=VALUE" lines and returns a map of keys/values
// If a key is defined without value "KEY", the value is searched into the environment with lookupEnv function
feloy marked this conversation as resolved.
Show resolved Hide resolved
// Note that "KEY=" defines an empty value for KEY, but "KEY" indicates to search for value in environment
// If the KEY environment variable is not defined, this entry will be skipped
func parseKeyValueFile(fs filesystem.Filesystem, filename string, lookupEnv func(string) (string, bool)) (map[string]string, error) {
f, err := fs.Open(filename)
if err != nil {
return nil, err
}
defer f.Close()

result := map[string]string{}

scanner := bufio.NewScanner(f)
for scanner.Scan() {
scannedText := scanner.Text()

key, value, err := parseKeyValueString(scannedText, lookupEnv)
if err != nil {
return nil, err
}
if len(key) == 0 {
continue
}
result[key] = value
}

return result, nil
}

func parseKeyValueStrings(strs []string, lookupEnv func(string) (string, bool)) (map[string]string, error) {
result := map[string]string{}

for _, str := range strs {
key, value, err := parseKeyValueString(str, lookupEnv)
if err != nil {
return nil, err
}
if len(key) == 0 {
continue
}
result[key] = value
}
return result, nil
}

// parseKeyValueString parses a string to extract a key and its associated value
// if a line is empty or a comment, a nil error and an empty key are returned
// if a key does not define a value, the value will be obtained from the environment
// in this case, if the environment does not define the valriable, the entrey will be ignored
feloy marked this conversation as resolved.
Show resolved Hide resolved
func parseKeyValueString(s string, lookupEnv func(string) (string, bool)) (string, string, error) {
line := strings.TrimLeftFunc(s, unicode.IsSpace)
if len(line) == 0 || strings.HasPrefix(line, "#") {
return "", "", nil
}
parts := strings.SplitN(line, "=", 2)
key := parts[0]

var value string
if len(parts) > 1 {
value = parts[1]
} else {
var found bool
value, found = lookupEnv(key)
if !found {
return "", "", nil
}
}

// TODO validate key format

if len(key) == 0 {
return "", "", NewErrBadKey(fmt.Sprintf("no key defined in line %q", s))
}
rm3l marked this conversation as resolved.
Show resolved Hide resolved
return key, value, nil
}
Loading