diff --git a/frontend/dockerfile/dockerfile2llb/convert.go b/frontend/dockerfile/dockerfile2llb/convert.go index 8bff5aac45162..93df59ac90cd7 100644 --- a/frontend/dockerfile/dockerfile2llb/convert.go +++ b/frontend/dockerfile/dockerfile2llb/convert.go @@ -227,7 +227,7 @@ func toDispatchState(ctx context.Context, dt []byte, opt ConvertOpt) (*dispatchS if v, ok := opt.BuildArgs[metaArg.Key]; !ok { if metaArg.Value != nil { result, _ := shlex.ProcessWordWithMatches(*metaArg.Value, metaArgsToMap(optMetaArgs)) - *metaArg.Value = result.Word + *metaArg.Value = result.Result info.deps = result.Matched } } else { @@ -250,22 +250,17 @@ func toDispatchState(ctx context.Context, dt []byte, opt ConvertOpt) (*dispatchS // set base state for every image for i, st := range stages { - result, err := shlex.ProcessWordWithMatches(st.BaseName, metaArgsToMap(optMetaArgs)) - name := result.Word - used := result.Matched - if len(result.Unmatched) > 0 { - for unmatched := range result.Unmatched { - msg := linter.RuleUndeclaredArgInFrom.Format(unmatched) - linter.RuleUndeclaredArgInFrom.Run(opt.Warn, st.Location, msg) - } - } + nameMatch, err := shlex.ProcessWordWithMatches(st.BaseName, metaArgsToMap(optMetaArgs)) + reportUnusedFromArgs(nameMatch.Unmatched, st.Location, opt.Warn) + used := nameMatch.Matched + if err != nil { return nil, parser.WithLocation(err, st.Location) } - if name == "" { + if nameMatch.Result == "" { return nil, parser.WithLocation(errors.Errorf("base name (%s) should not be blank", st.BaseName), st.Location) } - st.BaseName = name + st.BaseName = nameMatch.Result ds := &dispatchState{ stage: st, @@ -279,27 +274,22 @@ func toDispatchState(ctx context.Context, dt []byte, opt ConvertOpt) (*dispatchS } if v := st.Platform; v != "" { - result, err := shlex.ProcessWordWithMatches(v, metaArgsToMap(optMetaArgs)) - v := result.Word - u := result.Matched - - if len(result.Unmatched) > 0 { - for unmatched := range result.Unmatched { - msg := linter.RuleUndeclaredArgInFrom.Format(unmatched) - linter.RuleUndeclaredArgInFrom.Run(opt.Warn, st.Location, msg) - } - } + platMatch, err := shlex.ProcessWordWithMatches(v, metaArgsToMap(optMetaArgs)) + reportUnusedFromArgs(platMatch.Unmatched, st.Location, opt.Warn) + if err != nil { - return nil, parser.WithLocation(errors.Wrapf(err, "failed to process arguments for platform %s", v), st.Location) + return nil, parser.WithLocation(errors.Wrapf(err, "failed to process arguments for platform %s", platMatch.Result), st.Location) } - p, err := platforms.Parse(v) + p, err := platforms.Parse(platMatch.Result) if err != nil { - return nil, parser.WithLocation(errors.Wrapf(err, "failed to parse platform %s", v), st.Location) + return nil, parser.WithLocation(errors.Wrapf(err, "failed to parse platform %s", platMatch.Result), st.Location) } - for k := range u { + + for k := range platMatch.Matched { used[k] = struct{}{} } + ds.platform = &p } @@ -2010,3 +2000,10 @@ func validateCommandCasing(dockerfile *parser.Result, warn linter.LintWarnFunc) } } } + +func reportUnusedFromArgs(unmatched map[string]struct{}, location []parser.Range, warn linter.LintWarnFunc) { + for arg := range unmatched { + msg := linter.RuleUndeclaredArgInFrom.Format(arg) + linter.RuleUndeclaredArgInFrom.Run(warn, location, msg) + } +} diff --git a/frontend/dockerfile/dockerfile_lint_test.go b/frontend/dockerfile/dockerfile_lint_test.go index ee7ef8e1603af..8a216163c58ea 100644 --- a/frontend/dockerfile/dockerfile_lint_test.go +++ b/frontend/dockerfile/dockerfile_lint_test.go @@ -40,20 +40,23 @@ FROM scratch as base2 FROM scratch AS base3 `) - checkLinterWarnings(t, sb, dockerfile, []expectedLintWarning{ - { - RuleName: "StageNameCasing", - Description: "Stage names should be lowercase", - Detail: "Stage name 'BadStageName' should be lowercase", - Line: 3, - Level: 1, - }, - { - RuleName: "FromAsCasing", - Description: "The 'as' keyword should match the case of the 'from' keyword", - Detail: "'as' and 'FROM' keywords' casing do not match", - Line: 6, - Level: 1, + checkLinterWarnings(t, sb, &lintTestParams{ + Dockerfile: dockerfile, + Warnings: []expectedLintWarning{ + { + RuleName: "StageNameCasing", + Description: "Stage names should be lowercase", + Detail: "Stage name 'BadStageName' should be lowercase", + Line: 3, + Level: 1, + }, + { + RuleName: "FromAsCasing", + Description: "The 'as' keyword should match the case of the 'from' keyword", + Detail: "'as' and 'FROM' keywords' casing do not match", + Line: 6, + Level: 1, + }, }, }) @@ -63,13 +66,17 @@ from scratch AS base from scratch as base2 `) - checkLinterWarnings(t, sb, dockerfile, []expectedLintWarning{ - { - RuleName: "FromAsCasing", - Description: "The 'as' keyword should match the case of the 'from' keyword", - Detail: "'AS' and 'from' keywords' casing do not match", - Level: 1, - Line: 3, + + checkLinterWarnings(t, sb, &lintTestParams{ + Dockerfile: dockerfile, + Warnings: []expectedLintWarning{ + { + RuleName: "FromAsCasing", + Description: "The 'as' keyword should match the case of the 'from' keyword", + Detail: "'AS' and 'from' keywords' casing do not match", + Line: 3, + Level: 1, + }, }, }) } @@ -85,14 +92,17 @@ COPY Dockerfile \ . `) - checkLinterWarnings(t, sb, dockerfile, []expectedLintWarning{ - { - RuleName: "NoEmptyContinuations", - Description: "Empty continuation lines will become errors in a future release", - Detail: "Empty continuation line", - Level: 1, - Line: 6, - URL: "https://github.com/moby/moby/pull/33719", + checkLinterWarnings(t, sb, &lintTestParams{ + Dockerfile: dockerfile, + Warnings: []expectedLintWarning{ + { + RuleName: "NoEmptyContinuations", + Description: "Empty continuation lines will become errors in a future release", + Detail: "Empty continuation line", + Level: 1, + Line: 6, + URL: "https://github.com/moby/moby/pull/33719", + }, }, }) } @@ -103,27 +113,34 @@ func testSelfConsistentCommandCasing(t *testing.T, sb integration.Sandbox) { From scratch as base FROM scratch AS base2 `) - checkLinterWarnings(t, sb, dockerfile, []expectedLintWarning{ - { - RuleName: "SelfConsistentCommandCasing", - Description: "Commands should be in consistent casing (all lower or all upper)", - Detail: "Command 'From' should be consistently cased", - Level: 1, - Line: 3, + checkLinterWarnings(t, sb, &lintTestParams{ + Dockerfile: dockerfile, + Warnings: []expectedLintWarning{ + { + RuleName: "SelfConsistentCommandCasing", + Description: "Commands should be in consistent casing (all lower or all upper)", + Detail: "Command 'From' should be consistently cased", + Level: 1, + Line: 3, + }, }, }) + dockerfile = []byte(` # warning: 'FROM' should be either lowercased or uppercased frOM scratch as base from scratch as base2 `) - checkLinterWarnings(t, sb, dockerfile, []expectedLintWarning{ - { - RuleName: "SelfConsistentCommandCasing", - Description: "Commands should be in consistent casing (all lower or all upper)", - Detail: "Command 'frOM' should be consistently cased", - Line: 3, - Level: 1, + checkLinterWarnings(t, sb, &lintTestParams{ + Dockerfile: dockerfile, + Warnings: []expectedLintWarning{ + { + RuleName: "SelfConsistentCommandCasing", + Description: "Commands should be in consistent casing (all lower or all upper)", + Detail: "Command 'frOM' should be consistently cased", + Line: 3, + Level: 1, + }, }, }) } @@ -135,13 +152,17 @@ FROM scratch copy Dockerfile /foo COPY Dockerfile /bar `) - checkLinterWarnings(t, sb, dockerfile, []expectedLintWarning{ - { - RuleName: "FileConsistentCommandCasing", - Description: "All commands within the Dockerfile should use the same casing (either upper or lower)", - Detail: "Command 'copy' should match the case of the command majority (uppercase)", - Line: 4, - Level: 1, + + checkLinterWarnings(t, sb, &lintTestParams{ + Dockerfile: dockerfile, + Warnings: []expectedLintWarning{ + { + RuleName: "FileConsistentCommandCasing", + Description: "All commands within the Dockerfile should use the same casing (either upper or lower)", + Detail: "Command 'copy' should match the case of the command majority (uppercase)", + Line: 4, + Level: 1, + }, }, }) @@ -151,13 +172,16 @@ from scratch COPY Dockerfile /foo copy Dockerfile /bar `) - checkLinterWarnings(t, sb, dockerfile, []expectedLintWarning{ - { - RuleName: "FileConsistentCommandCasing", - Description: "All commands within the Dockerfile should use the same casing (either upper or lower)", - Detail: "Command 'COPY' should match the case of the command majority (lowercase)", - Line: 4, - Level: 1, + checkLinterWarnings(t, sb, &lintTestParams{ + Dockerfile: dockerfile, + Warnings: []expectedLintWarning{ + { + RuleName: "FileConsistentCommandCasing", + Description: "All commands within the Dockerfile should use the same casing (either upper or lower)", + Detail: "Command 'COPY' should match the case of the command majority (lowercase)", + Line: 4, + Level: 1, + }, }, }) @@ -168,13 +192,16 @@ COPY Dockerfile /foo COPY Dockerfile /bar COPY Dockerfile /baz `) - checkLinterWarnings(t, sb, dockerfile, []expectedLintWarning{ - { - RuleName: "FileConsistentCommandCasing", - Description: "All commands within the Dockerfile should use the same casing (either upper or lower)", - Detail: "Command 'from' should match the case of the command majority (uppercase)", - Line: 3, - Level: 1, + checkLinterWarnings(t, sb, &lintTestParams{ + Dockerfile: dockerfile, + Warnings: []expectedLintWarning{ + { + RuleName: "FileConsistentCommandCasing", + Description: "All commands within the Dockerfile should use the same casing (either upper or lower)", + Detail: "Command 'from' should match the case of the command majority (uppercase)", + Line: 3, + Level: 1, + }, }, }) @@ -185,13 +212,16 @@ copy Dockerfile /foo copy Dockerfile /bar copy Dockerfile /baz `) - checkLinterWarnings(t, sb, dockerfile, []expectedLintWarning{ - { - RuleName: "FileConsistentCommandCasing", - Description: "All commands within the Dockerfile should use the same casing (either upper or lower)", - Detail: "Command 'FROM' should match the case of the command majority (lowercase)", - Line: 3, - Level: 1, + checkLinterWarnings(t, sb, &lintTestParams{ + Dockerfile: dockerfile, + Warnings: []expectedLintWarning{ + { + RuleName: "FileConsistentCommandCasing", + Description: "All commands within the Dockerfile should use the same casing (either upper or lower)", + Detail: "Command 'FROM' should match the case of the command majority (lowercase)", + Line: 3, + Level: 1, + }, }, }) @@ -200,14 +230,14 @@ from scratch copy Dockerfile /foo copy Dockerfile /bar `) - checkLinterWarnings(t, sb, dockerfile, []expectedLintWarning{}) + checkLinterWarnings(t, sb, &lintTestParams{Dockerfile: dockerfile}) dockerfile = []byte(` FROM scratch COPY Dockerfile /foo COPY Dockerfile /bar `) - checkLinterWarnings(t, sb, dockerfile, []expectedLintWarning{}) + checkLinterWarnings(t, sb, &lintTestParams{Dockerfile: dockerfile}) } func testMaintainerDeprecated(t *testing.T, sb integration.Sandbox) { @@ -239,32 +269,60 @@ ARG base=scratch FROM $base COPY Dockerfile . `) - checkLinterWarnings(t, sb, dockerfile, []expectedLintWarning{}) + checkLinterWarnings(t, sb, &lintTestParams{Dockerfile: dockerfile}) dockerfile = []byte(` -ARG platform=linux/amd64 -FROM --platform=$platform scratch +ARG BUILDPLATFORM=linux/amd64 +FROM --platform=$BUILDPLATFORM scratch COPY Dockerfile . `) - checkLinterWarnings(t, sb, dockerfile, []expectedLintWarning{}) + checkLinterWarnings(t, sb, &lintTestParams{Dockerfile: dockerfile}) + + dockerfile = []byte(` +FROM --platform=$BUILDPLATFORM scratch +COPY Dockerfile . +`) + checkLinterWarnings(t, sb, &lintTestParams{Dockerfile: dockerfile}) + + dockerfile = []byte(` +FROM --platform=$BULIDPLATFORM scratch +COPY Dockerfile . +`) + checkLinterWarnings(t, sb, &lintTestParams{ + Dockerfile: dockerfile, + Warnings: []expectedLintWarning{ + { + RuleName: "UndeclaredArgInFrom", + Description: "FROM command must use declared ARGs", + Detail: "FROM argument 'BULIDPLATFORM' is not declared", + Level: 1, + Line: 2, + }, + }, + StreamSolveErr: `failed to solve: failed to parse platform : "" is an invalid component of "": platform specifier component must match "^[A-Za-z0-9_-]+$": invalid argument`, + UnmarshalSolveErr: `failed to solve: failed to parse platform : "" is an invalid component of "": platform specifier component must match "^[A-Za-z0-9_-]+$": invalid argument`, + }) dockerfile = []byte(` ARG tag=latest FROM busybox:${tag}${version} AS b COPY Dockerfile . `) - checkLinterWarnings(t, sb, dockerfile, []expectedLintWarning{ - { - RuleName: "UndeclaredArgInFrom", - Description: "FROM command must use declared ARGs", - Detail: "FROM argument 'version' is not declared", - Level: 1, - Line: 3, + checkLinterWarnings(t, sb, &lintTestParams{ + Dockerfile: dockerfile, + Warnings: []expectedLintWarning{ + { + RuleName: "UndeclaredArgInFrom", + Description: "FROM command must use declared ARGs", + Detail: "FROM argument 'version' is not declared", + Level: 1, + Line: 3, + }, }, }) } -func checkUnmarshal(t *testing.T, sb integration.Sandbox, c *client.Client, dir *integration.TmpDirWithName, expected []expectedLintWarning) { +func checkUnmarshal(t *testing.T, sb integration.Sandbox, lintTest *lintTestParams) { destDir, err := os.MkdirTemp("", "buildkit") require.NoError(t, err) defer os.RemoveAll(destDir) @@ -279,12 +337,14 @@ func checkUnmarshal(t *testing.T, sb integration.Sandbox, c *client.Client, dir }, Frontend: "dockerfile.v0", }) - require.NoError(t, err) + if err != nil { + return nil, err + } lintResults, err := unmarshalLintResults(res) require.NoError(t, err) - require.Equal(t, len(expected), len(lintResults.Warnings)) + require.Equal(t, len(lintTest.Warnings), len(lintResults.Warnings)) sort.Slice(lintResults.Warnings, func(i, j int) bool { // sort by line number in ascending order firstRange := lintResults.Warnings[i].Location.Ranges[0] @@ -293,23 +353,26 @@ func checkUnmarshal(t *testing.T, sb integration.Sandbox, c *client.Client, dir }) // Compare expectedLintWarning with actual lint results for i, w := range lintResults.Warnings { - checkLintWarning(t, w, expected[i]) + checkLintWarning(t, w, lintTest.Warnings[i]) } called = true return nil, nil } - _, err = c.Build(sb.Context(), client.SolveOpt{ + _, err = lintTest.Client.Build(sb.Context(), client.SolveOpt{ LocalMounts: map[string]fsutil.FS{ - dockerui.DefaultLocalNameDockerfile: dir, + dockerui.DefaultLocalNameDockerfile: lintTest.TmpDir, }, }, "", frontend, nil) - require.NoError(t, err) - - require.True(t, called) + if lintTest.UnmarshalSolveErr == "" { + require.NoError(t, err) + require.True(t, called) + } else { + require.EqualError(t, err, lintTest.UnmarshalSolveErr) + } } -func checkProgressStream(t *testing.T, sb integration.Sandbox, c *client.Client, dir *integration.TmpDirWithName, expected []expectedLintWarning) { +func checkProgressStream(t *testing.T, sb integration.Sandbox, lintTest *lintTestParams) { t.Helper() status := make(chan *client.SolveStatus) @@ -335,16 +398,20 @@ func checkProgressStream(t *testing.T, sb integration.Sandbox, c *client.Client, f := getFrontend(t, sb) - _, err := f.Solve(sb.Context(), c, client.SolveOpt{ + _, err := f.Solve(sb.Context(), lintTest.Client, client.SolveOpt{ FrontendAttrs: map[string]string{ "platform": "linux/amd64,linux/arm64", }, LocalMounts: map[string]fsutil.FS{ - dockerui.DefaultLocalNameDockerfile: dir, - dockerui.DefaultLocalNameContext: dir, + dockerui.DefaultLocalNameDockerfile: lintTest.TmpDir, + dockerui.DefaultLocalNameContext: lintTest.TmpDir, }, }, status) - require.NoError(t, err) + if lintTest.StreamSolveErr == "" { + require.NoError(t, err) + } else { + require.EqualError(t, err, lintTest.StreamSolveErr) + } select { case <-statusDone: @@ -353,7 +420,7 @@ func checkProgressStream(t *testing.T, sb integration.Sandbox, c *client.Client, } // two platforms only show one warning - require.Equal(t, len(expected), len(warnings)) + require.Equal(t, len(lintTest.Warnings), len(warnings)) sort.Slice(warnings, func(i, j int) bool { w1 := warnings[i] w2 := warnings[j] @@ -365,30 +432,39 @@ func checkProgressStream(t *testing.T, sb integration.Sandbox, c *client.Client, return w1.Range[0].Start.Line < w2.Range[0].Start.Line }) for i, w := range warnings { - checkVertexWarning(t, w, expected[i]) + checkVertexWarning(t, w, lintTest.Warnings[i]) } } -func checkLinterWarnings(t *testing.T, sb integration.Sandbox, dockerfile []byte, expected []expectedLintWarning) { - // As a note, this test depends on the `expected` lint - // warnings to be in order of appearance in the Dockerfile. +func checkLinterWarnings(t *testing.T, sb integration.Sandbox, lintTest *lintTestParams) { + if lintTest.Warnings != nil { + sort.Slice(lintTest.Warnings, func(i, j int) bool { + return lintTest.Warnings[i].Line < lintTest.Warnings[j].Line + }) + } integration.SkipOnPlatform(t, "windows") - dir := integration.Tmpdir( - t, - fstest.CreateFile("Dockerfile", dockerfile, 0600), - ) + if lintTest.TmpDir == nil { + lintTest.TmpDir = integration.Tmpdir( + t, + fstest.CreateFile("Dockerfile", lintTest.Dockerfile, 0600), + ) + } - c, err := client.New(sb.Context(), sb.Address()) - require.NoError(t, err) - defer c.Close() + if lintTest.Client == nil { + var err error + lintTest.Client, err = client.New(sb.Context(), sb.Address()) + require.NoError(t, err) + defer lintTest.Client.Close() + } t.Run("warntype=progress", func(t *testing.T) { - checkProgressStream(t, sb, c, dir, expected) + checkProgressStream(t, sb, lintTest) }) + t.Run("warntype=unmarshal", func(t *testing.T) { - checkUnmarshal(t, sb, c, dir, expected) + checkUnmarshal(t, sb, lintTest) }) } @@ -429,3 +505,12 @@ type expectedLintWarning struct { URL string Level int } + +type lintTestParams struct { + Client *client.Client + TmpDir *integration.TmpDirWithName + Dockerfile []byte + Warnings []expectedLintWarning + StreamSolveErr string + UnmarshalSolveErr string +} diff --git a/frontend/dockerfile/shell/lex.go b/frontend/dockerfile/shell/lex.go index c0a7bbfb98538..3ae23f5ea668c 100644 --- a/frontend/dockerfile/shell/lex.go +++ b/frontend/dockerfile/shell/lex.go @@ -58,7 +58,7 @@ func (s *Lex) ProcessWordWithMap(word string, env map[string]string) (string, er } type ProcessWordMatchesResult struct { - Word string + Result string Matched map[string]struct{} Unmatched map[string]struct{} } @@ -69,7 +69,7 @@ func (s *Lex) ProcessWordWithMatches(word string, env map[string]string) (Proces sw := s.init(word, env) word, _, err := sw.process(word) return ProcessWordMatchesResult{ - Word: word, + Result: word, Matched: sw.matches, Unmatched: sw.nonmatches, }, err diff --git a/frontend/dockerfile/shell/lex_test.go b/frontend/dockerfile/shell/lex_test.go index 288fcf2be697b..43d65afa5acb2 100644 --- a/frontend/dockerfile/shell/lex_test.go +++ b/frontend/dockerfile/shell/lex_test.go @@ -271,6 +271,7 @@ func TestProcessWithMatches(t *testing.T) { expected string expectedErr bool matches map[string]struct{} + unmatched map[string]struct{} }{ { input: "x", @@ -481,7 +482,7 @@ func TestProcessWithMatches(t *testing.T) { c := c t.Run(c.input, func(t *testing.T) { result, err := shlex.ProcessWordWithMatches(c.input, c.envs) - w := result.Word + w := result.Result matches := result.Matched if c.expectedErr { require.Error(t, err) @@ -514,7 +515,7 @@ func TestProcessWithMatchesPlatform(t *testing.T) { "TARGETVARIANT": "v7", }) require.NoError(t, err) - require.Equal(t, "something-v1.2.3.linux-arm-v7.tar.gz", results.Word) + require.Equal(t, "something-v1.2.3.linux-arm-v7.tar.gz", results.Result) results, err = shlex.ProcessWordWithMatches(release, map[string]string{ "VERSION": version, @@ -523,7 +524,7 @@ func TestProcessWithMatchesPlatform(t *testing.T) { "TARGETVARIANT": "", }) require.NoError(t, err) - require.Equal(t, "something-v1.2.3.linux-arm64.tar.gz", results.Word) + require.Equal(t, "something-v1.2.3.linux-arm64.tar.gz", results.Result) results, err = shlex.ProcessWordWithMatches(release, map[string]string{ "VERSION": version, @@ -532,5 +533,5 @@ func TestProcessWithMatchesPlatform(t *testing.T) { // No "TARGETVARIANT": "", }) require.NoError(t, err) - require.Equal(t, "something-v1.2.3.linux-arm64.tar.gz", results.Word) + require.Equal(t, "something-v1.2.3.linux-arm64.tar.gz", results.Result) }