From 6ab7f4945979b4a9f3c448df2f8cc4e2246e2a90 Mon Sep 17 00:00:00 2001 From: Mahmood Ali Date: Wed, 21 Apr 2021 13:24:22 -0400 Subject: [PATCH] hcl2: handle unquoted undefined variables (#10419) This fixes a regression in #10326, to handle unquoted unknown variables. The HCL job may contain unquoted undefined variable references without ${...} wrapping, e.g. value = meta.node_class. In 1.0.4, this got parsed as value = "${meta.node_class}". This code performs a scan to find the relevant ${ and }, and only tries to find the closest ones with whitespace as the only separator. --- jobspec2/parse_test.go | 15 +++++++++++++++ jobspec2/types.config.go | 33 +++++++++++++++++++++++---------- 2 files changed, 38 insertions(+), 10 deletions(-) diff --git a/jobspec2/parse_test.go b/jobspec2/parse_test.go index 17492d6abfd..a815cd0da9e 100644 --- a/jobspec2/parse_test.go +++ b/jobspec2/parse_test.go @@ -873,6 +873,21 @@ func TestParse_UndefinedVariables(t *testing.T) { require.Equal(t, c, *job.Region) }) } + + t.Run("unquoted", func(t *testing.T) { + hcl := `job "example" { + region = meta.mytest +}` + + job, err := ParseWithConfig(&ParseConfig{ + Path: "input.hcl", + Body: []byte(hcl), + }) + require.NoError(t, err) + + require.Equal(t, "${meta.mytest}", *job.Region) + + }) } func TestParseServiceCheck(t *testing.T) { diff --git a/jobspec2/types.config.go b/jobspec2/types.config.go index 12bb58f344a..28e05917214 100644 --- a/jobspec2/types.config.go +++ b/jobspec2/types.config.go @@ -1,7 +1,6 @@ package jobspec2 import ( - "bytes" "fmt" "strings" @@ -315,18 +314,32 @@ func (c *jobConfig) EvalContext() *hcl.EvalContext { end := t.SourceRange().End.Byte v := string(body[start:end]) - if i := bytes.IndexByte(body[end:], '}'); i != -1 { - v += string(body[end : end+i+1]) - } else { - // fallback for unexpected cases - v += "}" + + // find the start of inclusing "${..}" if it's inclused in one; otherwise, wrap it in one + isBracketed := false + for i := start - 1; i >= 1; i-- { + if body[i] == '{' && body[i-1] == '$' { + isBracketed = true + v = string(body[i-1:start]) + v + break + } else if body[i] != ' ' { + break + } } - if i := bytes.LastIndex(body[:start], []byte("${")); i != 0 { - v = string(body[i:start]) + v + if isBracketed { + for i := end + 1; i < len(body); i++ { + if body[i] == '}' { + v += string(body[end:i]) + } else if body[i] != ' ' { + // unexpected! + v += "}" + break + } + } + } else { - // fallback for unexpected cases - v = "${" + v + v = "${" + v + "}" } return cty.StringVal(v), nil