feat: cache dockerfile images through warmer (#2499)
* feat: cache dockerfile images through warmer * Fix logical error in conditional statement * Addressed review feedback 1. Updated help text for the --build-arg flag to indicate it should be used with the dockerfile flag. 2. Updated the documentation to include the optional --build-arg flag. 3. Added unit tests for `ParseDockerfile`, covering scenarios for missing Dockerfile, invalid Dockerfile, single stage Dockerfile, multi-stage Dockerfile and Args Dockerfile --------- Co-authored-by: 连奔驰 <benchi.lian@thoughtworks.com>
This commit is contained in:
parent
7cd39d14e3
commit
0743c19176
|
|
@ -505,9 +505,12 @@ provide a kaniko cache warming image at `gcr.io/kaniko-project/warmer`:
|
||||||
|
|
||||||
```shell
|
```shell
|
||||||
docker run -v $(pwd):/workspace gcr.io/kaniko-project/warmer:latest --cache-dir=/workspace/cache --image=<image to cache> --image=<another image to cache>
|
docker run -v $(pwd):/workspace gcr.io/kaniko-project/warmer:latest --cache-dir=/workspace/cache --image=<image to cache> --image=<another image to cache>
|
||||||
|
docker run -v $(pwd):/workspace gcr.io/kaniko-project/warmer:latest --cache-dir=/workspace/cache --dockerfile=<path to dockerfile>
|
||||||
|
docker run -v $(pwd):/workspace gcr.io/kaniko-project/warmer:latest --cache-dir=/workspace/cache --dockerfile=<path to dockerfile> --build-args version=1.19
|
||||||
```
|
```
|
||||||
|
|
||||||
`--image` can be specified for any number of desired images. This command will
|
`--image` can be specified for any number of desired images. `--dockerfile` can
|
||||||
|
be specified for the path of dockerfile for cache.These command will combined to
|
||||||
cache those images by digest in a local directory named `cache`. Once the cache
|
cache those images by digest in a local directory named `cache`. Once the cache
|
||||||
is populated, caching is opted into with the same `--cache=true` flag as above.
|
is populated, caching is opted into with the same `--cache=true` flag as above.
|
||||||
The location of the local cache is provided via the `--cache-dir` flag,
|
The location of the local cache is provided via the `--cache-dir` flag,
|
||||||
|
|
|
||||||
|
|
@ -19,11 +19,14 @@ package cmd
|
||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
||||||
"os"
|
"os"
|
||||||
|
"path/filepath"
|
||||||
|
"regexp"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/GoogleContainerTools/kaniko/pkg/cache"
|
"github.com/GoogleContainerTools/kaniko/pkg/cache"
|
||||||
"github.com/GoogleContainerTools/kaniko/pkg/config"
|
"github.com/GoogleContainerTools/kaniko/pkg/config"
|
||||||
"github.com/GoogleContainerTools/kaniko/pkg/logging"
|
"github.com/GoogleContainerTools/kaniko/pkg/logging"
|
||||||
|
"github.com/GoogleContainerTools/kaniko/pkg/util"
|
||||||
"github.com/containerd/containerd/platforms"
|
"github.com/containerd/containerd/platforms"
|
||||||
v1 "github.com/google/go-containerregistry/pkg/v1"
|
v1 "github.com/google/go-containerregistry/pkg/v1"
|
||||||
"github.com/pkg/errors"
|
"github.com/pkg/errors"
|
||||||
|
|
@ -54,9 +57,14 @@ var RootCmd = &cobra.Command{
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
if len(opts.Images) == 0 {
|
if len(opts.Images) == 0 && opts.DockerfilePath == "" {
|
||||||
return errors.New("You must select at least one image to cache")
|
return errors.New("You must select at least one image to cache or a dockerfilepath to parse")
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if err := validateDockerfilePath(); err != nil {
|
||||||
|
return errors.Wrap(err, "error validating dockerfile path")
|
||||||
|
}
|
||||||
|
|
||||||
return nil
|
return nil
|
||||||
},
|
},
|
||||||
Run: func(cmd *cobra.Command, args []string) {
|
Run: func(cmd *cobra.Command, args []string) {
|
||||||
|
|
@ -89,6 +97,8 @@ func addKanikoOptionsFlags() {
|
||||||
RootCmd.PersistentFlags().VarP(&opts.RegistriesClientCertificates, "registry-client-cert", "", "Use the provided client certificate for mutual TLS (mTLS) communication with the given registry. Expected format is 'my.registry.url=/path/to/client/cert,/path/to/client/key'.")
|
RootCmd.PersistentFlags().VarP(&opts.RegistriesClientCertificates, "registry-client-cert", "", "Use the provided client certificate for mutual TLS (mTLS) communication with the given registry. Expected format is 'my.registry.url=/path/to/client/cert,/path/to/client/key'.")
|
||||||
RootCmd.PersistentFlags().VarP(&opts.RegistryMirrors, "registry-mirror", "", "Registry mirror to use as pull-through cache instead of docker.io. Set it repeatedly for multiple mirrors.")
|
RootCmd.PersistentFlags().VarP(&opts.RegistryMirrors, "registry-mirror", "", "Registry mirror to use as pull-through cache instead of docker.io. Set it repeatedly for multiple mirrors.")
|
||||||
RootCmd.PersistentFlags().StringVarP(&opts.CustomPlatform, "customPlatform", "", "", "Specify the build platform if different from the current host")
|
RootCmd.PersistentFlags().StringVarP(&opts.CustomPlatform, "customPlatform", "", "", "Specify the build platform if different from the current host")
|
||||||
|
RootCmd.PersistentFlags().StringVarP(&opts.DockerfilePath, "dockerfile", "d", "Dockerfile", "Path to the dockerfile to be cached. The kaniko warmer will parse and write out each stage's base image layers to the cache-dir. Using the same dockerfile path as what you plan to build in the kaniko executor is the expected usage.")
|
||||||
|
RootCmd.PersistentFlags().VarP(&opts.BuildArgs, "build-arg", "", "This flag should be used in conjunction with the dockerfile flag for scenarios where dynamic replacement of the base image is required.")
|
||||||
|
|
||||||
// Default the custom platform flag to our current platform, and validate it.
|
// Default the custom platform flag to our current platform, and validate it.
|
||||||
if opts.CustomPlatform == "" {
|
if opts.CustomPlatform == "" {
|
||||||
|
|
@ -104,6 +114,29 @@ func addHiddenFlags() {
|
||||||
RootCmd.PersistentFlags().MarkHidden("azure-container-registry-config")
|
RootCmd.PersistentFlags().MarkHidden("azure-container-registry-config")
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func validateDockerfilePath() error {
|
||||||
|
if isURL(opts.DockerfilePath) {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
if util.FilepathExists(opts.DockerfilePath) {
|
||||||
|
abs, err := filepath.Abs(opts.DockerfilePath)
|
||||||
|
if err != nil {
|
||||||
|
return errors.Wrap(err, "getting absolute path for dockerfile")
|
||||||
|
}
|
||||||
|
opts.DockerfilePath = abs
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
return errors.New("please provide a valid path to a Dockerfile within the build context with --dockerfile")
|
||||||
|
}
|
||||||
|
|
||||||
|
func isURL(path string) bool {
|
||||||
|
if match, _ := regexp.MatchString("^https?://", path); match {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
func exit(err error) {
|
func exit(err error) {
|
||||||
fmt.Println(err)
|
fmt.Println(err)
|
||||||
os.Exit(1)
|
os.Exit(1)
|
||||||
|
|
|
||||||
|
|
@ -18,13 +18,18 @@ package cache
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"bytes"
|
"bytes"
|
||||||
|
"fmt"
|
||||||
"io"
|
"io"
|
||||||
"io/ioutil"
|
"io/ioutil"
|
||||||
|
"net/http"
|
||||||
"os"
|
"os"
|
||||||
"path"
|
"path"
|
||||||
|
"regexp"
|
||||||
|
|
||||||
"github.com/GoogleContainerTools/kaniko/pkg/config"
|
"github.com/GoogleContainerTools/kaniko/pkg/config"
|
||||||
|
"github.com/GoogleContainerTools/kaniko/pkg/dockerfile"
|
||||||
"github.com/GoogleContainerTools/kaniko/pkg/image/remote"
|
"github.com/GoogleContainerTools/kaniko/pkg/image/remote"
|
||||||
|
"github.com/GoogleContainerTools/kaniko/pkg/util"
|
||||||
"github.com/google/go-containerregistry/pkg/name"
|
"github.com/google/go-containerregistry/pkg/name"
|
||||||
v1 "github.com/google/go-containerregistry/pkg/v1"
|
v1 "github.com/google/go-containerregistry/pkg/v1"
|
||||||
"github.com/google/go-containerregistry/pkg/v1/tarball"
|
"github.com/google/go-containerregistry/pkg/v1/tarball"
|
||||||
|
|
@ -34,8 +39,21 @@ import (
|
||||||
|
|
||||||
// WarmCache populates the cache
|
// WarmCache populates the cache
|
||||||
func WarmCache(opts *config.WarmerOptions) error {
|
func WarmCache(opts *config.WarmerOptions) error {
|
||||||
|
var dockerfileImages []string
|
||||||
cacheDir := opts.CacheDir
|
cacheDir := opts.CacheDir
|
||||||
images := opts.Images
|
images := opts.Images
|
||||||
|
|
||||||
|
// if opts.image is empty,we need to parse dockerfilepath to get images list
|
||||||
|
if opts.DockerfilePath != "" {
|
||||||
|
var err error
|
||||||
|
if dockerfileImages, err = ParseDockerfile(opts); err != nil {
|
||||||
|
return errors.Wrap(err, "failed to parse Dockerfile")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// TODO: Implement deduplication logic later.
|
||||||
|
images = append(images, dockerfileImages...)
|
||||||
|
|
||||||
logrus.Debugf("%s\n", cacheDir)
|
logrus.Debugf("%s\n", cacheDir)
|
||||||
logrus.Debugf("%s\n", images)
|
logrus.Debugf("%s\n", images)
|
||||||
|
|
||||||
|
|
@ -157,3 +175,41 @@ func (w *Warmer) Warm(image string, opts *config.WarmerOptions) (v1.Hash, error)
|
||||||
|
|
||||||
return digest, nil
|
return digest, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func ParseDockerfile(opts *config.WarmerOptions) ([]string, error) {
|
||||||
|
var err error
|
||||||
|
var d []uint8
|
||||||
|
var baseNames []string
|
||||||
|
match, _ := regexp.MatchString("^https?://", opts.DockerfilePath)
|
||||||
|
if match {
|
||||||
|
response, e := http.Get(opts.DockerfilePath) //nolint:noctx
|
||||||
|
if e != nil {
|
||||||
|
return nil, e
|
||||||
|
}
|
||||||
|
d, err = ioutil.ReadAll(response.Body)
|
||||||
|
} else {
|
||||||
|
d, err = ioutil.ReadFile(opts.DockerfilePath)
|
||||||
|
}
|
||||||
|
|
||||||
|
if err != nil {
|
||||||
|
return nil, errors.Wrap(err, fmt.Sprintf("reading dockerfile at path %s", opts.DockerfilePath))
|
||||||
|
}
|
||||||
|
|
||||||
|
stages, _, err := dockerfile.Parse(d)
|
||||||
|
if err != nil {
|
||||||
|
return nil, errors.Wrap(err, "parsing dockerfile")
|
||||||
|
}
|
||||||
|
|
||||||
|
for i, s := range stages {
|
||||||
|
resolvedBaseName, err := util.ResolveEnvironmentReplacement(s.BaseName, opts.BuildArgs, false)
|
||||||
|
if err != nil {
|
||||||
|
return nil, errors.Wrap(err, fmt.Sprintf("resolving base name %s", s.BaseName))
|
||||||
|
}
|
||||||
|
if s.BaseName != resolvedBaseName {
|
||||||
|
stages[i].BaseName = resolvedBaseName
|
||||||
|
}
|
||||||
|
baseNames = append(baseNames, resolvedBaseName)
|
||||||
|
}
|
||||||
|
return baseNames, nil
|
||||||
|
|
||||||
|
}
|
||||||
|
|
|
||||||
|
|
@ -18,6 +18,8 @@ package cache
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"bytes"
|
"bytes"
|
||||||
|
"io/ioutil"
|
||||||
|
"os"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
"github.com/GoogleContainerTools/kaniko/pkg/config"
|
"github.com/GoogleContainerTools/kaniko/pkg/config"
|
||||||
|
|
@ -112,3 +114,136 @@ func Test_Warmer_Warm_in_cache_expired(t *testing.T) {
|
||||||
t.Errorf("expected nothing to be written")
|
t.Errorf("expected nothing to be written")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestParseDockerfile_SingleStageDockerfile(t *testing.T) {
|
||||||
|
dockerfile := `FROM alpine:latest
|
||||||
|
LABEL maintainer="alexezio"
|
||||||
|
`
|
||||||
|
tmpfile, err := ioutil.TempFile("", "example")
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
defer os.Remove(tmpfile.Name())
|
||||||
|
|
||||||
|
if _, err := tmpfile.Write([]byte(dockerfile)); err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
if err := tmpfile.Close(); err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
opts := &config.WarmerOptions{DockerfilePath: tmpfile.Name()}
|
||||||
|
baseNames, err := ParseDockerfile(opts)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("expected no error, got %v", err)
|
||||||
|
}
|
||||||
|
if len(baseNames) != 1 {
|
||||||
|
t.Fatalf("expected 1 base name, got %d", len(baseNames))
|
||||||
|
}
|
||||||
|
if baseNames[0] != "alpine:latest" {
|
||||||
|
t.Fatalf("expected 'alpine:latest', got '%s'", baseNames[0])
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestParseDockerfile_MultiStageDockerfile(t *testing.T) {
|
||||||
|
dockerfile := `FROM golang:1.20 as BUILDER
|
||||||
|
LABEL maintainer="alexezio"
|
||||||
|
|
||||||
|
FROM alpine:latest as RUNNER
|
||||||
|
LABEL maintainer="alexezio"
|
||||||
|
`
|
||||||
|
tmpfile, err := ioutil.TempFile("", "example")
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
defer os.Remove(tmpfile.Name())
|
||||||
|
|
||||||
|
if _, err := tmpfile.Write([]byte(dockerfile)); err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
if err := tmpfile.Close(); err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
opts := &config.WarmerOptions{DockerfilePath: tmpfile.Name()}
|
||||||
|
baseNames, err := ParseDockerfile(opts)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("expected no error, got %v", err)
|
||||||
|
}
|
||||||
|
if len(baseNames) != 2 {
|
||||||
|
t.Fatalf("expected 2 base name, got %d", len(baseNames))
|
||||||
|
}
|
||||||
|
if baseNames[0] != "golang:1.20" {
|
||||||
|
t.Fatalf("expected 'golang:1.20', got '%s'", baseNames[0])
|
||||||
|
}
|
||||||
|
|
||||||
|
if baseNames[1] != "alpine:latest" {
|
||||||
|
t.Fatalf("expected 'alpine:latest', got '%s'", baseNames[0])
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestParseDockerfile_ArgsDockerfile(t *testing.T) {
|
||||||
|
dockerfile := `ARG version=latest
|
||||||
|
FROM golang:${version}
|
||||||
|
`
|
||||||
|
tmpfile, err := ioutil.TempFile("", "example")
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
defer os.Remove(tmpfile.Name())
|
||||||
|
|
||||||
|
if _, err := tmpfile.Write([]byte(dockerfile)); err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
if err := tmpfile.Close(); err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
opts := &config.WarmerOptions{DockerfilePath: tmpfile.Name(), BuildArgs: []string{"version=1.20"}}
|
||||||
|
baseNames, err := ParseDockerfile(opts)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("expected no error, got %v", err)
|
||||||
|
}
|
||||||
|
if len(baseNames) != 1 {
|
||||||
|
t.Fatalf("expected 1 base name, got %d", len(baseNames))
|
||||||
|
}
|
||||||
|
if baseNames[0] != "golang:1.20" {
|
||||||
|
t.Fatalf("expected 'golang:1.20', got '%s'", baseNames[0])
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestParseDockerfile_MissingsDockerfile(t *testing.T) {
|
||||||
|
opts := &config.WarmerOptions{DockerfilePath: "dummy-nowhere"}
|
||||||
|
baseNames, err := ParseDockerfile(opts)
|
||||||
|
if err == nil {
|
||||||
|
t.Fatal("expected an error, got nil")
|
||||||
|
}
|
||||||
|
if len(baseNames) != 0 {
|
||||||
|
t.Fatalf("expected no base names, got %d", len(baseNames))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestParseDockerfile_InvalidsDockerfile(t *testing.T) {
|
||||||
|
dockerfile := "This is a invalid dockerfile"
|
||||||
|
tmpfile, err := ioutil.TempFile("", "example")
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
defer os.Remove(tmpfile.Name())
|
||||||
|
|
||||||
|
if _, err := tmpfile.Write([]byte(dockerfile)); err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
if err := tmpfile.Close(); err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
opts := &config.WarmerOptions{DockerfilePath: tmpfile.Name()}
|
||||||
|
baseNames, err := ParseDockerfile(opts)
|
||||||
|
if err == nil {
|
||||||
|
t.Fatal("expected an error, got nil")
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(baseNames) != 0 {
|
||||||
|
t.Fatalf("expected no base names, got %d", len(baseNames))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
||||||
|
|
@ -162,4 +162,6 @@ type WarmerOptions struct {
|
||||||
CustomPlatform string
|
CustomPlatform string
|
||||||
Images multiArg
|
Images multiArg
|
||||||
Force bool
|
Force bool
|
||||||
|
DockerfilePath string
|
||||||
|
BuildArgs multiArg
|
||||||
}
|
}
|
||||||
|
|
|
||||||
Loading…
Reference in New Issue