package bake import ( "context" "os" "testing" "github.com/stretchr/testify/require" ) func TestReadTargets(t *testing.T) { t.Parallel() fp := File{ Name: "config.hcl", Data: []byte(` target "webDEP" { args = { VAR_INHERITED = "webDEP" VAR_BOTH = "webDEP" } no-cache = true } target "webapp" { dockerfile = "Dockerfile.webapp" args = { VAR_BOTH = "webapp" } inherits = ["webDEP"] }`), } ctx := context.TODO() t.Run("NoOverrides", func(t *testing.T) { m, _, err := ReadTargets(ctx, []File{fp}, []string{"webapp"}, nil, nil) require.NoError(t, err) require.Equal(t, 1, len(m)) require.Equal(t, "Dockerfile.webapp", *m["webapp"].Dockerfile) require.Equal(t, ".", *m["webapp"].Context) require.Equal(t, "webDEP", m["webapp"].Args["VAR_INHERITED"]) require.Equal(t, true, *m["webapp"].NoCache) require.Nil(t, m["webapp"].Pull) }) t.Run("InvalidTargetOverrides", func(t *testing.T) { _, _, err := ReadTargets(ctx, []File{fp}, []string{"webapp"}, []string{"nosuchtarget.context=foo"}, nil) require.NotNil(t, err) require.Equal(t, err.Error(), "could not find any target matching 'nosuchtarget'") }) t.Run("ArgsOverrides", func(t *testing.T) { t.Run("leaf", func(t *testing.T) { os.Setenv("VAR_FROMENV"+t.Name(), "fromEnv") defer os.Unsetenv("VAR_FROM_ENV" + t.Name()) m, _, err := ReadTargets(ctx, []File{fp}, []string{"webapp"}, []string{ "webapp.args.VAR_UNSET", "webapp.args.VAR_EMPTY=", "webapp.args.VAR_SET=bananas", "webapp.args.VAR_FROMENV" + t.Name(), "webapp.args.VAR_INHERITED=override", // not overriding VAR_BOTH on purpose }, nil) require.NoError(t, err) require.Equal(t, "Dockerfile.webapp", *m["webapp"].Dockerfile) require.Equal(t, ".", *m["webapp"].Context) _, isSet := m["webapp"].Args["VAR_UNSET"] require.False(t, isSet, m["webapp"].Args["VAR_UNSET"]) _, isSet = m["webapp"].Args["VAR_EMPTY"] require.True(t, isSet, m["webapp"].Args["VAR_EMPTY"]) require.Equal(t, m["webapp"].Args["VAR_SET"], "bananas") require.Equal(t, m["webapp"].Args["VAR_FROMENV"+t.Name()], "fromEnv") require.Equal(t, m["webapp"].Args["VAR_BOTH"], "webapp") require.Equal(t, m["webapp"].Args["VAR_INHERITED"], "override") }) // building leaf but overriding parent fields t.Run("parent", func(t *testing.T) { m, _, err := ReadTargets(ctx, []File{fp}, []string{"webapp"}, []string{ "webDEP.args.VAR_INHERITED=override", "webDEP.args.VAR_BOTH=override", }, nil) require.NoError(t, err) require.Equal(t, m["webapp"].Args["VAR_INHERITED"], "override") require.Equal(t, m["webapp"].Args["VAR_BOTH"], "webapp") }) }) t.Run("ContextOverride", func(t *testing.T) { _, _, err := ReadTargets(ctx, []File{fp}, []string{"webapp"}, []string{"webapp.context"}, nil) require.NotNil(t, err) m, _, err := ReadTargets(ctx, []File{fp}, []string{"webapp"}, []string{"webapp.context=foo"}, nil) require.NoError(t, err) require.Equal(t, "foo", *m["webapp"].Context) }) t.Run("NoCacheOverride", func(t *testing.T) { m, _, err := ReadTargets(ctx, []File{fp}, []string{"webapp"}, []string{"webapp.no-cache=false"}, nil) require.NoError(t, err) require.Equal(t, false, *m["webapp"].NoCache) }) t.Run("PullOverride", func(t *testing.T) { m, _, err := ReadTargets(ctx, []File{fp}, []string{"webapp"}, []string{"webapp.pull=false"}, nil) require.NoError(t, err) require.Equal(t, false, *m["webapp"].Pull) }) t.Run("PatternOverride", func(t *testing.T) { // same check for two cases multiTargetCheck := func(t *testing.T, m map[string]*Target, err error) { require.NoError(t, err) require.Equal(t, 2, len(m)) require.Equal(t, "foo", *m["webapp"].Dockerfile) require.Equal(t, "webDEP", m["webapp"].Args["VAR_INHERITED"]) require.Equal(t, "foo", *m["webDEP"].Dockerfile) require.Equal(t, "webDEP", m["webDEP"].Args["VAR_INHERITED"]) } cases := []struct { name string targets []string overrides []string check func(*testing.T, map[string]*Target, error) }{ { name: "multi target single pattern", targets: []string{"webapp", "webDEP"}, overrides: []string{"web*.dockerfile=foo"}, check: multiTargetCheck, }, { name: "multi target multi pattern", targets: []string{"webapp", "webDEP"}, overrides: []string{"web*.dockerfile=foo", "*.args.VAR_BOTH=bar"}, check: multiTargetCheck, }, { name: "single target", targets: []string{"webapp"}, overrides: []string{"web*.dockerfile=foo"}, check: func(t *testing.T, m map[string]*Target, err error) { require.NoError(t, err) require.Equal(t, 1, len(m)) require.Equal(t, "foo", *m["webapp"].Dockerfile) require.Equal(t, "webDEP", m["webapp"].Args["VAR_INHERITED"]) }, }, { name: "nomatch", targets: []string{"webapp"}, overrides: []string{"nomatch*.dockerfile=foo"}, check: func(t *testing.T, m map[string]*Target, err error) { // NOTE: I am unsure whether failing to match should always error out // instead of simply skipping that override. // Let's enforce the error and we can relax it later if users complain. require.NotNil(t, err) require.Equal(t, err.Error(), "could not find any target matching 'nomatch*'") }, }, } for _, test := range cases { t.Run(test.name, func(t *testing.T) { m, _, err := ReadTargets(ctx, []File{fp}, test.targets, test.overrides, nil) test.check(t, m, err) }) } }) } func TestPushOverride(t *testing.T) { t.Parallel() fp := File{ Name: "docker-bake.hc", Data: []byte( `target "app" { output = ["type=image,compression=zstd"] }`), } ctx := context.TODO() m, _, err := ReadTargets(ctx, []File{fp}, []string{"app"}, []string{"*.push=true"}, nil) require.NoError(t, err) require.Equal(t, 1, len(m["app"].Outputs)) require.Equal(t, "type=image,compression=zstd,push=true", m["app"].Outputs[0]) fp = File{ Name: "docker-bake.hc", Data: []byte( `target "app" { output = ["type=image,compression=zstd"] }`), } ctx = context.TODO() m, _, err = ReadTargets(ctx, []File{fp}, []string{"app"}, []string{"*.push=false"}, nil) require.NoError(t, err) require.Equal(t, 1, len(m["app"].Outputs)) require.Equal(t, "type=image,compression=zstd,push=false", m["app"].Outputs[0]) fp = File{ Name: "docker-bake.hc", Data: []byte( `target "app" { }`), } ctx = context.TODO() m, _, err = ReadTargets(ctx, []File{fp}, []string{"app"}, []string{"*.push=true"}, nil) require.NoError(t, err) require.Equal(t, 1, len(m["app"].Outputs)) require.Equal(t, "type=image,push=true", m["app"].Outputs[0]) } func TestReadTargetsCompose(t *testing.T) { t.Parallel() fp := File{ Name: "docker-compose.yml", Data: []byte( `version: "3" services: db: build: . command: ./entrypoint.sh image: docker.io/tonistiigi/db webapp: build: dockerfile: Dockerfile.webapp args: buildno: 1 `), } fp2 := File{ Name: "docker-compose2.yml", Data: []byte( `version: "3" services: newservice: build: . webapp: build: args: buildno2: 12 `), } ctx := context.TODO() m, _, err := ReadTargets(ctx, []File{fp, fp2}, []string{"default"}, nil, nil) require.NoError(t, err) require.Equal(t, 3, len(m)) _, ok := m["newservice"] require.True(t, ok) require.Equal(t, "Dockerfile.webapp", *m["webapp"].Dockerfile) require.Equal(t, ".", *m["webapp"].Context) require.Equal(t, "1", m["webapp"].Args["buildno"]) require.Equal(t, "12", m["webapp"].Args["buildno2"]) } func TestHCLCwdPrefix(t *testing.T) { fp := File{ Name: "docker-bake.hcl", Data: []byte( `target "app" { context = "cwd://foo" dockerfile = "test" }`), } ctx := context.TODO() m, _, err := ReadTargets(ctx, []File{fp}, []string{"app"}, nil, nil) require.NoError(t, err) require.Equal(t, 1, len(m)) _, ok := m["app"] require.True(t, ok) _, err = TargetsToBuildOpt(m, &Input{}) require.NoError(t, err) require.Equal(t, "test", *m["app"].Dockerfile) require.Equal(t, "foo", *m["app"].Context) } func TestOverrideMerge(t *testing.T) { fp := File{ Name: "docker-bake.hcl", Data: []byte( `target "app" { platforms = ["linux/amd64"] output = ["foo"] }`), } ctx := context.TODO() m, _, err := ReadTargets(ctx, []File{fp}, []string{"app"}, []string{ "app.platform=linux/arm", "app.platform=linux/ppc64le", "app.output=type=registry", }, nil) require.NoError(t, err) require.Equal(t, 1, len(m)) _, ok := m["app"] require.True(t, ok) _, err = TargetsToBuildOpt(m, &Input{}) require.NoError(t, err) require.Equal(t, []string{"linux/arm", "linux/ppc64le"}, m["app"].Platforms) require.Equal(t, 1, len(m["app"].Outputs)) require.Equal(t, "type=registry", m["app"].Outputs[0]) }