Skip to content

Commit 60ee93c

Browse files
committed
Simplify pkg/dockerfile interface by ditching pointer
This means slightly more typing in "zero-value" cases (`nil` vs `dockerfile.Metadata{}`), but the tradeoff is that it's simpler to use and reason about (and all the struct members are pointer-type map/slice values anyhow, so copying the struct is still pretty cheap). This also swaps the scanner error handling to return the partially parsed Metadata object alongside the scanner error -- the error already tells us the object isn't fully complete data, so it's fair/fine to return and will likely just be ignored by the caller instead. This also allows us to get to 100% code coverage. 👀 This also updates our "treat `oci-import` just like `FROM scratch`" code to *actually* parse `FROM scratch` so we can't accidentally cause "missing data" bugs there in the future, and I implemented that using `sync.OnceValues` which requires upgrading to Go 1.21, but IMO that's a worthwhile tradeoff (because `sync.OnceValues` makes that code so clean/simple).
1 parent 7ddf2be commit 60ee93c

File tree

9 files changed

+26
-34
lines changed

9 files changed

+26
-34
lines changed

Dockerfile

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
FROM golang:1.20-bullseye AS build
1+
FROM golang:1.21-bookworm AS build
22

33
SHELL ["bash", "-Eeuo", "pipefail", "-xc"]
44

Dockerfile.release

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
FROM golang:1.20-bullseye
1+
FROM golang:1.21-bookworm
22

33
SHELL ["bash", "-Eeuo", "pipefail", "-xc"]
44

Dockerfile.test

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
FROM golang:1.20-bullseye
1+
FROM golang:1.21-bookworm
22

33
SHELL ["bash", "-Eeuo", "pipefail", "-xc"]
44

bashbrew.sh

+3-3
Original file line numberDiff line numberDiff line change
@@ -6,11 +6,11 @@ set -Eeuo pipefail
66
dir="$(readlink -f "$BASH_SOURCE")"
77
dir="$(dirname "$dir")"
88

9-
: "${CGO_ENABLED:=0}"
10-
export GO111MODULE=on CGO_ENABLED
9+
: "${CGO_ENABLED=0}" "${GOTOOLCHAIN=local}"
10+
export CGO_ENABLED GOTOOLCHAIN
1111
(
1212
cd "$dir"
13-
go build -o bin/bashbrew ./cmd/bashbrew > /dev/null
13+
go build -trimpath -o bin/bashbrew ./cmd/bashbrew > /dev/null
1414
)
1515

1616
exec "$dir/bin/bashbrew" "$@"

cmd/bashbrew/docker.go

+13-14
Original file line numberDiff line numberDiff line change
@@ -10,6 +10,7 @@ import (
1010
"os/exec"
1111
"path"
1212
"strings"
13+
"sync"
1314

1415
"github.com/docker-library/bashbrew/manifest"
1516
"github.com/docker-library/bashbrew/pkg/dockerfile"
@@ -37,27 +38,25 @@ func (r Repo) ArchDockerFroms(arch string, entry *manifest.Manifest2822Entry) ([
3738
return dockerfileMeta.Froms, nil
3839
}
3940

40-
func (r Repo) dockerfileMetadata(entry *manifest.Manifest2822Entry) (*dockerfile.Metadata, error) {
41+
func (r Repo) dockerfileMetadata(entry *manifest.Manifest2822Entry) (dockerfile.Metadata, error) {
4142
return r.archDockerfileMetadata(arch, entry)
4243
}
4344

44-
var dockerfileMetadataCache = map[string]*dockerfile.Metadata{}
45+
var (
46+
dockerfileMetadataCache = map[string]dockerfile.Metadata{}
47+
scratchDockerfileMetadata = sync.OnceValues(func() (dockerfile.Metadata, error) {
48+
return dockerfile.Parse(`FROM scratch`)
49+
})
50+
)
4551

46-
func (r Repo) archDockerfileMetadata(arch string, entry *manifest.Manifest2822Entry) (*dockerfile.Metadata, error) {
52+
func (r Repo) archDockerfileMetadata(arch string, entry *manifest.Manifest2822Entry) (dockerfile.Metadata, error) {
4753
if builder := entry.ArchBuilder(arch); builder == "oci-import" {
48-
return &dockerfile.Metadata{
49-
StageFroms: []string{
50-
"scratch",
51-
},
52-
Froms: []string{
53-
"scratch",
54-
},
55-
}, nil
54+
return scratchDockerfileMetadata()
5655
}
5756

5857
commit, err := r.fetchGitRepo(arch, entry)
5958
if err != nil {
60-
return nil, cli.NewMultiError(fmt.Errorf("failed fetching Git repo for arch %q from entry %q", arch, entry.String()), err)
59+
return dockerfile.Metadata{}, cli.NewMultiError(fmt.Errorf("failed fetching Git repo for arch %q from entry %q", arch, entry.String()), err)
6160
}
6261

6362
dockerfileFile := path.Join(entry.ArchDirectory(arch), entry.ArchFile(arch))
@@ -72,12 +71,12 @@ func (r Repo) archDockerfileMetadata(arch string, entry *manifest.Manifest2822En
7271

7372
df, err := gitShow(commit, dockerfileFile)
7473
if err != nil {
75-
return nil, cli.NewMultiError(fmt.Errorf(`failed "git show" for %q from commit %q`, dockerfileFile, commit), err)
74+
return dockerfile.Metadata{}, cli.NewMultiError(fmt.Errorf(`failed "git show" for %q from commit %q`, dockerfileFile, commit), err)
7675
}
7776

7877
meta, err := dockerfile.Parse(df)
7978
if err != nil {
80-
return nil, cli.NewMultiError(fmt.Errorf(`failed parsing Dockerfile metadata for %q from commit %q`, dockerfileFile, commit), err)
79+
return dockerfile.Metadata{}, cli.NewMultiError(fmt.Errorf(`failed parsing Dockerfile metadata for %q from commit %q`, dockerfileFile, commit), err)
8180
}
8281

8382
dockerfileMetadataCache[cacheKey] = meta

go.mod

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
module github.com/docker-library/bashbrew
22

3-
go 1.20
3+
go 1.21
44

55
require (
66
github.com/containerd/containerd v1.6.19

go.sum

+1
Original file line numberDiff line numberDiff line change
@@ -617,6 +617,7 @@ github.com/prometheus/procfs v0.1.3/go.mod h1:lV6e/gmhEcM9IjHGsFOCxxuZ+z1YqCvr4O
617617
github.com/prometheus/procfs v0.2.0/go.mod h1:lV6e/gmhEcM9IjHGsFOCxxuZ+z1YqCvr4OA4YeYWdaU=
618618
github.com/prometheus/procfs v0.6.0/go.mod h1:cz+aTbrPOrUb4q7XlbU9ygM+/jj0fzG6c1xBZuNvfVA=
619619
github.com/prometheus/procfs v0.7.3 h1:4jVXhlkAyzOScmCkXBTOLRLTz8EeU+eyjrwB/EPq0VU=
620+
github.com/prometheus/procfs v0.7.3/go.mod h1:cz+aTbrPOrUb4q7XlbU9ygM+/jj0fzG6c1xBZuNvfVA=
620621
github.com/prometheus/tsdb v0.7.1/go.mod h1:qhTCs0VvXwvX/y3TZrWD7rabWM+ijKTux40TwIPHuXU=
621622
github.com/rogpeppe/fastuuid v0.0.0-20150106093220-6724a57986af/go.mod h1:XWv6SoW27p1b0cqNHllgS5HIMJraePCO15w5zCzIWYg=
622623
github.com/rogpeppe/fastuuid v1.2.0/go.mod h1:jVj6XXZzXRy/MSR5jhDC/2q6DgLz+nrA6LYCDYWNEvQ=

pkg/dockerfile/parse.go

+4-7
Original file line numberDiff line numberDiff line change
@@ -16,12 +16,12 @@ type Metadata struct {
1616
Froms []string // every "FROM" or "COPY --from=xxx" value (minus named and/or numbered stages in the case of "--from=")
1717
}
1818

19-
func Parse(dockerfile string) (*Metadata, error) {
19+
func Parse(dockerfile string) (Metadata, error) {
2020
return ParseReader(strings.NewReader(dockerfile))
2121
}
2222

23-
func ParseReader(dockerfile io.Reader) (*Metadata, error) {
24-
meta := &Metadata{
23+
func ParseReader(dockerfile io.Reader) (Metadata, error) {
24+
meta := Metadata{
2525
// panic: assignment to entry in nil map
2626
StageNameFroms: map[string]string{},
2727
// (nil slices work fine)
@@ -171,10 +171,7 @@ func ParseReader(dockerfile io.Reader) (*Metadata, error) {
171171
}
172172
}
173173
}
174-
if err := scanner.Err(); err != nil {
175-
return nil, err
176-
}
177-
return meta, nil
174+
return meta, scanner.Err()
178175
}
179176

180177
func latestizeRepoTag(repoTag string) string {

pkg/dockerfile/parse_test.go

+1-6
Original file line numberDiff line numberDiff line change
@@ -175,14 +175,9 @@ func TestParse(t *testing.T) {
175175
if err != nil {
176176
t.Fatal(err)
177177
}
178-
179-
if parsed == nil {
178+
if !reflect.DeepEqual(parsed, td.metadata) {
180179
t.Fatalf("expected:\n%#v\ngot:\n%#v", td.metadata, parsed)
181180
}
182-
183-
if !reflect.DeepEqual(*parsed, td.metadata) {
184-
t.Fatalf("expected:\n%#v\ngot:\n%#v", td.metadata, *parsed)
185-
}
186181
})
187182
}
188183
}

0 commit comments

Comments
 (0)