191 lines
5.3 KiB
Go
191 lines
5.3 KiB
Go
package dockerfile2llb
|
|
|
|
import (
|
|
"testing"
|
|
|
|
"github.com/moby/buildkit/frontend/dockerfile/instructions"
|
|
"github.com/moby/buildkit/frontend/dockerfile/shell"
|
|
"github.com/moby/buildkit/util/appcontext"
|
|
"github.com/stretchr/testify/assert"
|
|
)
|
|
|
|
func toEnvMap(args []instructions.KeyValuePairOptional, env []string) map[string]string {
|
|
m := shell.BuildEnvs(env)
|
|
|
|
for _, arg := range args {
|
|
// If key already exists, keep previous value.
|
|
if _, ok := m[arg.Key]; ok {
|
|
continue
|
|
}
|
|
if arg.Value != nil {
|
|
m[arg.Key] = arg.ValueString()
|
|
}
|
|
}
|
|
return m
|
|
}
|
|
|
|
func TestDockerfileParsing(t *testing.T) {
|
|
t.Parallel()
|
|
df := `FROM busybox
|
|
ENV FOO bar
|
|
COPY f1 f2 /sub/
|
|
RUN ls -l
|
|
`
|
|
_, _, err := Dockerfile2LLB(appcontext.Context(), []byte(df), ConvertOpt{})
|
|
assert.NoError(t, err)
|
|
|
|
df = `FROM busybox AS foo
|
|
ENV FOO bar
|
|
FROM foo
|
|
COPY --from=foo f1 /
|
|
COPY --from=0 f2 /
|
|
`
|
|
_, _, err = Dockerfile2LLB(appcontext.Context(), []byte(df), ConvertOpt{})
|
|
assert.NoError(t, err)
|
|
|
|
df = `FROM busybox AS foo
|
|
ENV FOO bar
|
|
FROM foo
|
|
COPY --from=foo f1 /
|
|
COPY --from=0 f2 /
|
|
`
|
|
_, _, err = Dockerfile2LLB(appcontext.Context(), []byte(df), ConvertOpt{
|
|
Target: "Foo",
|
|
})
|
|
assert.NoError(t, err)
|
|
|
|
_, _, err = Dockerfile2LLB(appcontext.Context(), []byte(df), ConvertOpt{
|
|
Target: "nosuch",
|
|
})
|
|
assert.Error(t, err)
|
|
|
|
df = `FROM busybox
|
|
ADD http://github.com/moby/buildkit/blob/master/README.md /
|
|
`
|
|
_, _, err = Dockerfile2LLB(appcontext.Context(), []byte(df), ConvertOpt{})
|
|
assert.NoError(t, err)
|
|
|
|
df = `FROM busybox
|
|
COPY http://github.com/moby/buildkit/blob/master/README.md /
|
|
`
|
|
_, _, err = Dockerfile2LLB(appcontext.Context(), []byte(df), ConvertOpt{})
|
|
assert.EqualError(t, err, "source can't be a URL for COPY")
|
|
|
|
df = `FROM "" AS foo`
|
|
_, _, err = Dockerfile2LLB(appcontext.Context(), []byte(df), ConvertOpt{})
|
|
assert.Error(t, err)
|
|
|
|
df = `FROM ${BLANK} AS foo`
|
|
_, _, err = Dockerfile2LLB(appcontext.Context(), []byte(df), ConvertOpt{})
|
|
assert.Error(t, err)
|
|
}
|
|
|
|
func TestAddEnv(t *testing.T) {
|
|
// k exists in env as key
|
|
// override = true
|
|
env := []string{"key1=val1", "key2=val2"}
|
|
result := addEnv(env, "key1", "value1")
|
|
assert.Equal(t, []string{"key1=value1", "key2=val2"}, result)
|
|
|
|
// k does not exist in env as key
|
|
// override = true
|
|
env = []string{"key1=val1", "key2=val2"}
|
|
result = addEnv(env, "key3", "val3")
|
|
assert.Equal(t, []string{"key1=val1", "key2=val2", "key3=val3"}, result)
|
|
|
|
// env has same keys
|
|
// override = true
|
|
env = []string{"key1=val1", "key1=val2"}
|
|
result = addEnv(env, "key1", "value1")
|
|
assert.Equal(t, []string{"key1=value1", "key1=val2"}, result)
|
|
|
|
// k matches with key only string in env
|
|
// override = true
|
|
env = []string{"key1=val1", "key2=val2", "key3"}
|
|
result = addEnv(env, "key3", "val3")
|
|
assert.Equal(t, []string{"key1=val1", "key2=val2", "key3=val3"}, result)
|
|
}
|
|
|
|
func TestParseKeyValue(t *testing.T) {
|
|
k, v := parseKeyValue("key=val")
|
|
assert.Equal(t, "key", k)
|
|
assert.Equal(t, "val", v)
|
|
|
|
k, v = parseKeyValue("key=")
|
|
assert.Equal(t, "key", k)
|
|
assert.Equal(t, "", v)
|
|
|
|
k, v = parseKeyValue("key")
|
|
assert.Equal(t, "key", k)
|
|
assert.Equal(t, "", v)
|
|
}
|
|
|
|
func TestToEnvList(t *testing.T) {
|
|
// args has no duplicated key with env
|
|
v := "val2"
|
|
args := []instructions.KeyValuePairOptional{{Key: "key2", Value: &v}}
|
|
env := []string{"key1=val1"}
|
|
resutl := toEnvMap(args, env)
|
|
assert.Equal(t, map[string]string{"key1": "val1", "key2": "val2"}, resutl)
|
|
|
|
// value of args is nil
|
|
args = []instructions.KeyValuePairOptional{{Key: "key2", Value: nil}}
|
|
env = []string{"key1=val1"}
|
|
resutl = toEnvMap(args, env)
|
|
assert.Equal(t, map[string]string{"key1": "val1"}, resutl)
|
|
|
|
// args has duplicated key with env
|
|
v = "val2"
|
|
args = []instructions.KeyValuePairOptional{{Key: "key1", Value: &v}}
|
|
env = []string{"key1=val1"}
|
|
resutl = toEnvMap(args, env)
|
|
assert.Equal(t, map[string]string{"key1": "val1"}, resutl)
|
|
|
|
v = "val2"
|
|
args = []instructions.KeyValuePairOptional{{Key: "key1", Value: &v}}
|
|
env = []string{"key1="}
|
|
resutl = toEnvMap(args, env)
|
|
assert.Equal(t, map[string]string{"key1": ""}, resutl)
|
|
|
|
v = "val2"
|
|
args = []instructions.KeyValuePairOptional{{Key: "key1", Value: &v}}
|
|
env = []string{"key1"}
|
|
resutl = toEnvMap(args, env)
|
|
assert.Equal(t, map[string]string{"key1": ""}, resutl)
|
|
|
|
// env has duplicated keys
|
|
v = "val2"
|
|
args = []instructions.KeyValuePairOptional{{Key: "key2", Value: &v}}
|
|
env = []string{"key1=val1", "key1=val1_2"}
|
|
resutl = toEnvMap(args, env)
|
|
assert.Equal(t, map[string]string{"key1": "val1_2", "key2": "val2"}, resutl)
|
|
|
|
// args has duplicated keys
|
|
v1 := "v1"
|
|
v2 := "v2"
|
|
args = []instructions.KeyValuePairOptional{{Key: "key2", Value: &v1}, {Key: "key2", Value: &v2}}
|
|
env = []string{"key1=val1"}
|
|
resutl = toEnvMap(args, env)
|
|
assert.Equal(t, map[string]string{"key1": "val1", "key2": "v1"}, resutl)
|
|
}
|
|
|
|
func TestDockerfileCircularDependencies(t *testing.T) {
|
|
// single stage depends on itself
|
|
df := `FROM busybox AS stage0
|
|
COPY --from=stage0 f1 /sub/
|
|
`
|
|
_, _, err := Dockerfile2LLB(appcontext.Context(), []byte(df), ConvertOpt{})
|
|
assert.EqualError(t, err, "circular dependency detected on stage: stage0")
|
|
|
|
// multiple stages with circular dependency
|
|
df = `FROM busybox AS stage0
|
|
COPY --from=stage2 f1 /sub/
|
|
FROM busybox AS stage1
|
|
COPY --from=stage0 f2 /sub/
|
|
FROM busybox AS stage2
|
|
COPY --from=stage1 f2 /sub/
|
|
`
|
|
_, _, err = Dockerfile2LLB(appcontext.Context(), []byte(df), ConvertOpt{})
|
|
assert.EqualError(t, err, "circular dependency detected on stage: stage0")
|
|
}
|