From 25224ab6154cc94c74cfb1a1db74f17657ce43f0 Mon Sep 17 00:00:00 2001 From: Sebastiaan van Stijn Date: Thu, 3 Oct 2024 17:22:22 +0200 Subject: [PATCH 1/3] opts: remove ErrBadKey as it's not used as a sentinel error This error was originally introduced `ErrBadEnvVariable` in [moby/moby@500c8ba], but merely for convenience, and not used as a sentinel error. After the code was moved from the daemon to the cli repository, it was renamed to be more generic `ErrBadKey` in commit 2b17f4c8a8caad552025edb05a73db683fb8a5c6. A search on GitHub shows that there's no consumers using this error as sentinel error, and it's not used in our own code as such, so it should be safe to remove this error. This patch removes the `ErrBadKey` error-type; it also removes the prefix (`poorly formatted environment:`) to make the error more generic, because the same function was used both for env-files and label-files. [moby/moby@500c8ba]: https://github.com/vdemeester/moby/commit/500c8ba4b66c35cf2c29aeb81a9392cc406835a4 Signed-off-by: Sebastiaan van Stijn (cherry picked from commit 95e221ef4d64cd0e1036b3c91dabb6a505fe408c) Signed-off-by: Austin Vazquez --- opts/file.go | 13 ++----------- 1 file changed, 2 insertions(+), 11 deletions(-) diff --git a/opts/file.go b/opts/file.go index 5cdd8e1386d1..d1089ea41d14 100644 --- a/opts/file.go +++ b/opts/file.go @@ -12,15 +12,6 @@ import ( const whiteSpaces = " \t" -// ErrBadKey typed error for bad environment variable -type ErrBadKey struct { - msg string -} - -func (e ErrBadKey) Error() string { - return "poorly formatted environment: " + e.msg -} - func parseKeyValueFile(filename string, emptyFn func(string) (string, bool)) ([]string, error) { fh, err := os.Open(filename) if err != nil { @@ -51,10 +42,10 @@ func parseKeyValueFile(filename string, emptyFn func(string) (string, bool)) ([] // trim the front of a variable, but nothing else variable = strings.TrimLeft(variable, whiteSpaces) if strings.ContainsAny(variable, whiteSpaces) { - return []string{}, ErrBadKey{fmt.Sprintf("variable '%s' contains whitespaces", variable)} + return []string{}, fmt.Errorf("variable '%s' contains whitespaces", variable) } if len(variable) == 0 { - return []string{}, ErrBadKey{fmt.Sprintf("no variable name on line '%s'", line)} + return []string{}, fmt.Errorf("no variable name on line '%s'", line) } if hasValue { From 40539963c278d9f49b335cb4e17965e12c6c475c Mon Sep 17 00:00:00 2001 From: Sebastiaan van Stijn Date: Thu, 3 Oct 2024 22:16:04 +0200 Subject: [PATCH 2/3] opts: parseKeyValueFile: cleanup and remove redundant trimming - the function already trimmed leading whitespace from each line before parsing. keys with trailing whitespace would be invalidated, and values have whitespace preserved, so there's no need to trim whitespace for the key. - if a line is validated (key is valid), we don't need to reconstruct the key=value by concatenating, and we can add the line as-is. - check if the key is empty before checking if it contains whitespace - touch-up comments - rename some variables for readability - slight cleanup to use early returns / early continues to reduce nesting Signed-off-by: Sebastiaan van Stijn (cherry picked from commit 76196dbb01040d31ea96fabadde09ab7d7d9a85a) Signed-off-by: Austin Vazquez --- opts/file.go | 63 +++++++++++++++++++++++++++------------------------- 1 file changed, 33 insertions(+), 30 deletions(-) diff --git a/opts/file.go b/opts/file.go index d1089ea41d14..535839aa7af9 100644 --- a/opts/file.go +++ b/opts/file.go @@ -12,7 +12,7 @@ import ( const whiteSpaces = " \t" -func parseKeyValueFile(filename string, emptyFn func(string) (string, bool)) ([]string, error) { +func parseKeyValueFile(filename string, lookupFn func(string) (string, bool)) ([]string, error) { fh, err := os.Open(filename) if err != nil { return []string{}, err @@ -21,45 +21,48 @@ func parseKeyValueFile(filename string, emptyFn func(string) (string, bool)) ([] lines := []string{} scanner := bufio.NewScanner(fh) - currentLine := 0 utf8bom := []byte{0xEF, 0xBB, 0xBF} - for scanner.Scan() { + for currentLine := 1; scanner.Scan(); currentLine++ { scannedBytes := scanner.Bytes() if !utf8.Valid(scannedBytes) { - return []string{}, fmt.Errorf("env file %s contains invalid utf8 bytes at line %d: %v", filename, currentLine+1, scannedBytes) + return []string{}, fmt.Errorf("env file %s contains invalid utf8 bytes at line %d: %v", filename, currentLine, scannedBytes) } // We trim UTF8 BOM - if currentLine == 0 { + if currentLine == 1 { scannedBytes = bytes.TrimPrefix(scannedBytes, utf8bom) } - // trim the line from all leading whitespace first + // trim the line from all leading whitespace first. trailing whitespace + // is part of the value, and is kept unmodified. line := strings.TrimLeftFunc(string(scannedBytes), unicode.IsSpace) - currentLine++ - // line is not empty, and not starting with '#' - if len(line) > 0 && !strings.HasPrefix(line, "#") { - variable, value, hasValue := strings.Cut(line, "=") - // trim the front of a variable, but nothing else - variable = strings.TrimLeft(variable, whiteSpaces) - if strings.ContainsAny(variable, whiteSpaces) { - return []string{}, fmt.Errorf("variable '%s' contains whitespaces", variable) - } - if len(variable) == 0 { - return []string{}, fmt.Errorf("no variable name on line '%s'", line) - } + if len(line) == 0 || line[0] == '#' { + // skip empty lines and comments (lines starting with '#') + continue + } + + key, _, hasValue := strings.Cut(line, "=") + if len(key) == 0 { + return []string{}, fmt.Errorf("no variable name on line '%s'", line) + } + + // leading whitespace was already removed from the line, but + // variables are not allowed to contain whitespace or have + // trailing whitespace. + if strings.ContainsAny(key, whiteSpaces) { + return []string{}, fmt.Errorf("variable '%s' contains whitespaces", key) + } + + if hasValue { + // key/value pair is valid and has a value; add the line as-is. + lines = append(lines, line) + continue + } - if hasValue { - // pass the value through, no trimming - lines = append(lines, variable+"="+value) - } else { - var present bool - if emptyFn != nil { - value, present = emptyFn(line) - } - if present { - // if only a pass-through variable is given, clean it up. - lines = append(lines, strings.TrimSpace(variable)+"="+value) - } + if lookupFn != nil { + // No value given; try to look up the value. The value may be + // empty but if no value is found, the key is omitted. + if value, found := lookupFn(line); found { + lines = append(lines, key+"="+value) } } } From 9ab3d3a983c8189bcff403c00b724b760398dc1d Mon Sep 17 00:00:00 2001 From: Nicolas De Loof Date: Tue, 1 Oct 2024 21:48:40 +0200 Subject: [PATCH 3/3] move parsing key-value files to a separate package Move the code for parsing key-value files, such as used for env-files and label-files to a separate package. This allows other projects (such as compose) to use the same parsing logic, but provide custom lookup functions for their situation (which is slightly different). The new package provides utilities for parsing key-value files for either a file or an io.Reader. Most tests for EnvFile were now testing functionality that's already tested in the new package, so were (re)moved. Co-authored-by: Nicolas De Loof Signed-off-by: Nicolas De Loof Signed-off-by: Sebastiaan van Stijn (cherry picked from commit 9ecfe4f5a7634d9c375ffce1a273c421f2df716c) Signed-off-by: Austin Vazquez --- cli/command/container/opts_test.go | 2 +- opts/envfile.go | 4 +- opts/envfile_test.go | 88 ----------------- opts/file.go | 70 -------------- opts/opts.go | 2 + opts/parse.go | 3 +- pkg/kvfile/kvfile.go | 130 ++++++++++++++++++++++++++ pkg/kvfile/kvfile_test.go | 145 +++++++++++++++++++++++++++++ 8 files changed, 283 insertions(+), 161 deletions(-) delete mode 100644 opts/file.go create mode 100644 pkg/kvfile/kvfile.go create mode 100644 pkg/kvfile/kvfile_test.go diff --git a/cli/command/container/opts_test.go b/cli/command/container/opts_test.go index e03c26530e2f..0f835ee44770 100644 --- a/cli/command/container/opts_test.go +++ b/cli/command/container/opts_test.go @@ -921,7 +921,7 @@ func TestParseEnvfileVariablesWithBOMUnicode(t *testing.T) { } // UTF16 with BOM - e := "contains invalid utf8 bytes at line" + e := "invalid utf8 bytes at line" if _, _, _, err := parseRun([]string{"--env-file=testdata/utf16.env", "img", "cmd"}); err == nil || !strings.Contains(err.Error(), e) { t.Fatalf("Expected an error with message '%s', got %v", e, err) } diff --git a/opts/envfile.go b/opts/envfile.go index 26aa3c3a9094..3a16e6c189bc 100644 --- a/opts/envfile.go +++ b/opts/envfile.go @@ -2,6 +2,8 @@ package opts import ( "os" + + "github.com/docker/cli/pkg/kvfile" ) // ParseEnvFile reads a file with environment variables enumerated by lines @@ -18,5 +20,5 @@ import ( // environment variables, that's why we just strip leading whitespace and // nothing more. func ParseEnvFile(filename string) ([]string, error) { - return parseKeyValueFile(filename, os.LookupEnv) + return kvfile.Parse(filename, os.LookupEnv) } diff --git a/opts/envfile_test.go b/opts/envfile_test.go index b066d46cf646..2c01de4fd520 100644 --- a/opts/envfile_test.go +++ b/opts/envfile_test.go @@ -1,10 +1,8 @@ package opts import ( - "bufio" "os" "path/filepath" - "strings" "testing" "gotest.tools/v3/assert" @@ -19,86 +17,12 @@ func tmpFileWithContent(t *testing.T, content string) string { return fileName } -// Test ParseEnvFile for a file with a few well formatted lines -func TestParseEnvFileGoodFile(t *testing.T) { - content := `foo=bar - baz=quux -# comment - -_foobar=foobaz -with.dots=working -and_underscore=working too -` - // Adding a newline + a line with pure whitespace. - // This is being done like this instead of the block above - // because it's common for editors to trim trailing whitespace - // from lines, which becomes annoying since that's the - // exact thing we need to test. - content += "\n \t " - tmpFile := tmpFileWithContent(t, content) - - lines, err := ParseEnvFile(tmpFile) - assert.NilError(t, err) - - expectedLines := []string{ - "foo=bar", - "baz=quux", - "_foobar=foobaz", - "with.dots=working", - "and_underscore=working too", - } - - assert.Check(t, is.DeepEqual(lines, expectedLines)) -} - -// Test ParseEnvFile for an empty file -func TestParseEnvFileEmptyFile(t *testing.T) { - tmpFile := tmpFileWithContent(t, "") - - lines, err := ParseEnvFile(tmpFile) - assert.NilError(t, err) - assert.Check(t, is.Len(lines, 0)) -} - // Test ParseEnvFile for a non existent file func TestParseEnvFileNonExistentFile(t *testing.T) { _, err := ParseEnvFile("no_such_file") assert.Check(t, is.ErrorType(err, os.IsNotExist)) } -// Test ParseEnvFile for a badly formatted file -func TestParseEnvFileBadlyFormattedFile(t *testing.T) { - content := `foo=bar - f =quux -` - tmpFile := tmpFileWithContent(t, content) - - _, err := ParseEnvFile(tmpFile) - const expectedMessage = "variable 'f ' contains whitespaces" - assert.Check(t, is.ErrorContains(err, expectedMessage)) -} - -// Test ParseEnvFile for a file with a line exceeding bufio.MaxScanTokenSize -func TestParseEnvFileLineTooLongFile(t *testing.T) { - content := "foo=" + strings.Repeat("a", bufio.MaxScanTokenSize+42) - tmpFile := tmpFileWithContent(t, content) - - _, err := ParseEnvFile(tmpFile) - const expectedMessage = "bufio.Scanner: token too long" - assert.Check(t, is.ErrorContains(err, expectedMessage)) -} - -// ParseEnvFile with a random file, pass through -func TestParseEnvFileRandomFile(t *testing.T) { - content := `first line -another invalid line` - tmpFile := tmpFileWithContent(t, content) - - _, err := ParseEnvFile(tmpFile) - const expectedMessage = "variable 'first line' contains whitespaces" - assert.Check(t, is.ErrorContains(err, expectedMessage)) -} - // ParseEnvFile with environment variable import definitions func TestParseEnvVariableDefinitionsFile(t *testing.T) { content := `# comment= @@ -114,15 +38,3 @@ DEFINED_VAR expectedLines := []string{"DEFINED_VAR=defined-value"} assert.Check(t, is.DeepEqual(variables, expectedLines)) } - -// ParseEnvFile with empty variable name -func TestParseEnvVariableWithNoNameFile(t *testing.T) { - content := `# comment= -=blank variable names are an error case -` - tmpFile := tmpFileWithContent(t, content) - - _, err := ParseEnvFile(tmpFile) - const expectedMessage = "no variable name on line '=blank variable names are an error case'" - assert.Check(t, is.ErrorContains(err, expectedMessage)) -} diff --git a/opts/file.go b/opts/file.go deleted file mode 100644 index 535839aa7af9..000000000000 --- a/opts/file.go +++ /dev/null @@ -1,70 +0,0 @@ -package opts - -import ( - "bufio" - "bytes" - "fmt" - "os" - "strings" - "unicode" - "unicode/utf8" -) - -const whiteSpaces = " \t" - -func parseKeyValueFile(filename string, lookupFn func(string) (string, bool)) ([]string, error) { - fh, err := os.Open(filename) - if err != nil { - return []string{}, err - } - defer fh.Close() - - lines := []string{} - scanner := bufio.NewScanner(fh) - utf8bom := []byte{0xEF, 0xBB, 0xBF} - for currentLine := 1; scanner.Scan(); currentLine++ { - scannedBytes := scanner.Bytes() - if !utf8.Valid(scannedBytes) { - return []string{}, fmt.Errorf("env file %s contains invalid utf8 bytes at line %d: %v", filename, currentLine, scannedBytes) - } - // We trim UTF8 BOM - if currentLine == 1 { - scannedBytes = bytes.TrimPrefix(scannedBytes, utf8bom) - } - // trim the line from all leading whitespace first. trailing whitespace - // is part of the value, and is kept unmodified. - line := strings.TrimLeftFunc(string(scannedBytes), unicode.IsSpace) - - if len(line) == 0 || line[0] == '#' { - // skip empty lines and comments (lines starting with '#') - continue - } - - key, _, hasValue := strings.Cut(line, "=") - if len(key) == 0 { - return []string{}, fmt.Errorf("no variable name on line '%s'", line) - } - - // leading whitespace was already removed from the line, but - // variables are not allowed to contain whitespace or have - // trailing whitespace. - if strings.ContainsAny(key, whiteSpaces) { - return []string{}, fmt.Errorf("variable '%s' contains whitespaces", key) - } - - if hasValue { - // key/value pair is valid and has a value; add the line as-is. - lines = append(lines, line) - continue - } - - if lookupFn != nil { - // No value given; try to look up the value. The value may be - // empty but if no value is found, the key is omitted. - if value, found := lookupFn(line); found { - lines = append(lines, key+"="+value) - } - } - } - return lines, scanner.Err() -} diff --git a/opts/opts.go b/opts/opts.go index 254d7eb12853..157b30f34b3c 100644 --- a/opts/opts.go +++ b/opts/opts.go @@ -266,6 +266,8 @@ func validateDomain(val string) (string, error) { return "", fmt.Errorf("%s is not a valid domain", val) } +const whiteSpaces = " \t" + // ValidateLabel validates that the specified string is a valid label, and returns it. // // Labels are in the form of key=value; key must be a non-empty string, and not diff --git a/opts/parse.go b/opts/parse.go index 584b55ef61f4..996d4d7e7a2d 100644 --- a/opts/parse.go +++ b/opts/parse.go @@ -6,6 +6,7 @@ import ( "strconv" "strings" + "github.com/docker/cli/pkg/kvfile" "github.com/docker/docker/api/types/container" ) @@ -25,7 +26,7 @@ func ReadKVEnvStrings(files []string, override []string) ([]string, error) { func readKVStrings(files []string, override []string, emptyFn func(string) (string, bool)) ([]string, error) { var variables []string for _, ef := range files { - parsedVars, err := parseKeyValueFile(ef, emptyFn) + parsedVars, err := kvfile.Parse(ef, emptyFn) if err != nil { return nil, err } diff --git a/pkg/kvfile/kvfile.go b/pkg/kvfile/kvfile.go new file mode 100644 index 000000000000..f6ac8ef4e047 --- /dev/null +++ b/pkg/kvfile/kvfile.go @@ -0,0 +1,130 @@ +// Package kvfile provides utilities to parse line-delimited key/value files +// such as used for label-files and env-files. +// +// # File format +// +// key/value files use the following syntax: +// +// - File must be valid UTF-8. +// - BOM headers are removed. +// - Leading whitespace is removed for each line. +// - Lines starting with "#" are ignored. +// - Empty lines are ignored. +// - Key/Value pairs are provided as "KEY[=]". +// - Maximum line-length is limited to [bufio.MaxScanTokenSize]. +// +// # Interpolation, substitution, and escaping +// +// Both keys and values are used as-is; no interpolation, substitution or +// escaping is supported, and quotes are considered part of the key or value. +// Whitespace in values (including leading and trailing) is preserved. Given +// that the file format is line-delimited, neither key, nor value, can contain +// newlines. +// +// # Key/Value pairs +// +// Key/Value pairs take the following format: +// +// KEY[=] +// +// KEY is required and may not contain whitespaces or NUL characters. Any +// other character (except for the "=" delimiter) are accepted, but it is +// recommended to use a subset of the POSIX portable character set, as +// outlined in [Environment Variables]. +// +// VALUE is optional, but may be empty. If no value is provided (i.e., no +// equal sign ("=") is present), the KEY is omitted in the result, but some +// functions accept a lookup-function to provide a default value for the +// given key. +// +// [Environment Variables]: https://pubs.opengroup.org/onlinepubs/7908799/xbd/envvar.html +package kvfile + +import ( + "bufio" + "bytes" + "fmt" + "io" + "os" + "strings" + "unicode" + "unicode/utf8" +) + +// Parse parses a line-delimited key/value pairs separated by equal sign. +// It accepts a lookupFn to lookup default values for keys that do not define +// a value. An error is produced if parsing failed, the content contains invalid +// UTF-8 characters, or a key contains whitespaces. +func Parse(filename string, lookupFn func(key string) (value string, found bool)) ([]string, error) { + fh, err := os.Open(filename) + if err != nil { + return []string{}, err + } + out, err := parseKeyValueFile(fh, lookupFn) + _ = fh.Close() + if err != nil { + return []string{}, fmt.Errorf("invalid env file (%s): %v", filename, err) + } + return out, nil +} + +// ParseFromReader parses a line-delimited key/value pairs separated by equal sign. +// It accepts a lookupFn to lookup default values for keys that do not define +// a value. An error is produced if parsing failed, the content contains invalid +// UTF-8 characters, or a key contains whitespaces. +func ParseFromReader(r io.Reader, lookupFn func(key string) (value string, found bool)) ([]string, error) { + return parseKeyValueFile(r, lookupFn) +} + +const whiteSpaces = " \t" + +func parseKeyValueFile(r io.Reader, lookupFn func(string) (string, bool)) ([]string, error) { + lines := []string{} + scanner := bufio.NewScanner(r) + utf8bom := []byte{0xEF, 0xBB, 0xBF} + for currentLine := 1; scanner.Scan(); currentLine++ { + scannedBytes := scanner.Bytes() + if !utf8.Valid(scannedBytes) { + return []string{}, fmt.Errorf("invalid utf8 bytes at line %d: %v", currentLine, scannedBytes) + } + // We trim UTF8 BOM + if currentLine == 1 { + scannedBytes = bytes.TrimPrefix(scannedBytes, utf8bom) + } + // trim the line from all leading whitespace first. trailing whitespace + // is part of the value, and is kept unmodified. + line := strings.TrimLeftFunc(string(scannedBytes), unicode.IsSpace) + + if len(line) == 0 || line[0] == '#' { + // skip empty lines and comments (lines starting with '#') + continue + } + + key, _, hasValue := strings.Cut(line, "=") + if len(key) == 0 { + return []string{}, fmt.Errorf("no variable name on line '%s'", line) + } + + // leading whitespace was already removed from the line, but + // variables are not allowed to contain whitespace or have + // trailing whitespace. + if strings.ContainsAny(key, whiteSpaces) { + return []string{}, fmt.Errorf("variable '%s' contains whitespaces", key) + } + + if hasValue { + // key/value pair is valid and has a value; add the line as-is. + lines = append(lines, line) + continue + } + + if lookupFn != nil { + // No value given; try to look up the value. The value may be + // empty but if no value is found, the key is omitted. + if value, found := lookupFn(line); found { + lines = append(lines, key+"="+value) + } + } + } + return lines, scanner.Err() +} diff --git a/pkg/kvfile/kvfile_test.go b/pkg/kvfile/kvfile_test.go new file mode 100644 index 000000000000..f1a0e88e6bf5 --- /dev/null +++ b/pkg/kvfile/kvfile_test.go @@ -0,0 +1,145 @@ +package kvfile + +import ( + "bufio" + "os" + "path/filepath" + "strings" + "testing" + + "gotest.tools/v3/assert" + is "gotest.tools/v3/assert/cmp" +) + +// Test Parse for a non existent file. +func TestParseNonExistentFile(t *testing.T) { + _, err := Parse("no_such_file", nil) + assert.Check(t, is.ErrorType(err, os.IsNotExist)) +} + +// Test Parse from a file with a lookup function. +func TestParseWithLookup(t *testing.T) { + content := `# comment= +VAR=VAR_VALUE +EMPTY_VAR= +UNDEFINED_VAR +DEFINED_VAR +` + vars := map[string]string{ + "DEFINED_VAR": "defined-value", + } + lookupFn := func(name string) (string, bool) { + v, ok := vars[name] + return v, ok + } + + fileName := filepath.Join(t.TempDir(), "envfile") + err := os.WriteFile(fileName, []byte(content), 0o644) + assert.NilError(t, err) + + variables, err := Parse(fileName, lookupFn) + assert.NilError(t, err) + + expectedLines := []string{"VAR=VAR_VALUE", "EMPTY_VAR=", "DEFINED_VAR=defined-value"} + assert.Check(t, is.DeepEqual(variables, expectedLines)) +} + +// Test ParseEnvFile for a file with a few well formatted lines +func TestParseFromReaderGoodFile(t *testing.T) { + content := `foo=bar + baz=quux +# comment + +_foobar=foobaz +with.dots=working +and_underscore=working too +` + // Adding a newline + a line with pure whitespace. + // This is being done like this instead of the block above + // because it's common for editors to trim trailing whitespace + // from lines, which becomes annoying since that's the + // exact thing we need to test. + content += "\n \t " + + lines, err := ParseFromReader(strings.NewReader(content), nil) + assert.NilError(t, err) + + expectedLines := []string{ + "foo=bar", + "baz=quux", + "_foobar=foobaz", + "with.dots=working", + "and_underscore=working too", + } + + assert.Check(t, is.DeepEqual(lines, expectedLines)) +} + +// Test ParseFromReader for an empty file +func TestParseFromReaderEmptyFile(t *testing.T) { + lines, err := ParseFromReader(strings.NewReader(""), nil) + assert.NilError(t, err) + assert.Check(t, is.Len(lines, 0)) +} + +// Test ParseFromReader for a badly formatted file +func TestParseFromReaderBadlyFormattedFile(t *testing.T) { + content := `foo=bar + f =quux +` + _, err := ParseFromReader(strings.NewReader(content), nil) + const expectedMessage = "variable 'f ' contains whitespaces" + assert.Check(t, is.ErrorContains(err, expectedMessage)) +} + +// Test ParseFromReader for a file with a line exceeding bufio.MaxScanTokenSize +func TestParseFromReaderLineTooLongFile(t *testing.T) { + content := "foo=" + strings.Repeat("a", bufio.MaxScanTokenSize+42) + + _, err := ParseFromReader(strings.NewReader(content), nil) + const expectedMessage = "bufio.Scanner: token too long" + assert.Check(t, is.ErrorContains(err, expectedMessage)) +} + +// ParseEnvFile with a random file, pass through +func TestParseFromReaderRandomFile(t *testing.T) { + content := `first line +another invalid line` + + _, err := ParseFromReader(strings.NewReader(content), nil) + const expectedMessage = "variable 'first line' contains whitespaces" + assert.Check(t, is.ErrorContains(err, expectedMessage)) +} + +// Test ParseFromReader with a lookup function. +func TestParseFromReaderWithLookup(t *testing.T) { + content := `# comment= +VAR=VAR_VALUE +EMPTY_VAR= +UNDEFINED_VAR +DEFINED_VAR +` + vars := map[string]string{ + "DEFINED_VAR": "defined-value", + } + lookupFn := func(name string) (string, bool) { + v, ok := vars[name] + return v, ok + } + + variables, err := ParseFromReader(strings.NewReader(content), lookupFn) + assert.NilError(t, err) + + expectedLines := []string{"VAR=VAR_VALUE", "EMPTY_VAR=", "DEFINED_VAR=defined-value"} + assert.Check(t, is.DeepEqual(variables, expectedLines)) +} + +// Test ParseFromReader with empty variable name +func TestParseFromReaderWithNoName(t *testing.T) { + content := `# comment= +=blank variable names are an error case +` + _, err := ParseFromReader(strings.NewReader(content), nil) + const expectedMessage = "no variable name on line '=blank variable names are an error case'" + assert.Check(t, is.ErrorContains(err, expectedMessage)) +}