249 lines
6.7 KiB
Go
249 lines
6.7 KiB
Go
package main
|
|
|
|
import (
|
|
"encoding/base64"
|
|
"fmt"
|
|
"io/ioutil"
|
|
"os"
|
|
"strings"
|
|
|
|
"github.com/joho/godotenv"
|
|
"github.com/pkg/errors"
|
|
"github.com/sirupsen/logrus"
|
|
"github.com/urfave/cli"
|
|
|
|
kaniko "github.com/drone/drone-kaniko"
|
|
"github.com/drone/drone-kaniko/pkg/artifact"
|
|
)
|
|
|
|
const (
|
|
// Docker file path
|
|
dockerPath string = "/kaniko/.docker"
|
|
dockerConfigPath string = "/kaniko/.docker/config.json"
|
|
|
|
v1RegistryURL string = "https://index.docker.io/v1/" // Default registry
|
|
v2RegistryURL string = "https://index.docker.io/v2/" // v2 registry is not supported
|
|
v2HubRegistryURL string = "https://registry.hub.docker.com/v2/"
|
|
|
|
defaultDigestFile string = "/kaniko/digest-file"
|
|
)
|
|
|
|
var (
|
|
version = "unknown"
|
|
)
|
|
|
|
func main() {
|
|
// Load env-file if it exists first
|
|
if env := os.Getenv("PLUGIN_ENV_FILE"); env != "" {
|
|
if err := godotenv.Load(env); err != nil {
|
|
logrus.Fatal(err)
|
|
}
|
|
}
|
|
|
|
app := cli.NewApp()
|
|
app.Name = "kaniko docker plugin"
|
|
app.Usage = "kaniko docker plugin"
|
|
app.Action = run
|
|
app.Version = version
|
|
app.Flags = []cli.Flag{
|
|
cli.StringFlag{
|
|
Name: "dockerfile",
|
|
Usage: "build dockerfile",
|
|
Value: "Dockerfile",
|
|
EnvVar: "PLUGIN_DOCKERFILE",
|
|
},
|
|
cli.StringFlag{
|
|
Name: "context",
|
|
Usage: "build context",
|
|
Value: ".",
|
|
EnvVar: "PLUGIN_CONTEXT",
|
|
},
|
|
cli.StringSliceFlag{
|
|
Name: "tags",
|
|
Usage: "build tags",
|
|
Value: &cli.StringSlice{"latest"},
|
|
EnvVar: "PLUGIN_TAGS",
|
|
FilePath: ".tags",
|
|
},
|
|
cli.BoolFlag{
|
|
Name: "auto_tag",
|
|
Usage: "enable for semver tagging",
|
|
EnvVar: "PLUGIN_AUTO_TAG",
|
|
},
|
|
cli.StringSliceFlag{
|
|
Name: "args",
|
|
Usage: "build args",
|
|
EnvVar: "PLUGIN_BUILD_ARGS",
|
|
},
|
|
cli.StringFlag{
|
|
Name: "target",
|
|
Usage: "build target",
|
|
EnvVar: "PLUGIN_TARGET",
|
|
},
|
|
cli.StringFlag{
|
|
Name: "repo",
|
|
Usage: "docker repository",
|
|
EnvVar: "PLUGIN_REPO",
|
|
},
|
|
cli.StringSliceFlag{
|
|
Name: "custom-labels",
|
|
Usage: "additional k=v labels",
|
|
EnvVar: "PLUGIN_CUSTOM_LABELS",
|
|
},
|
|
cli.StringFlag{
|
|
Name: "registry",
|
|
Usage: "docker registry",
|
|
Value: v1RegistryURL,
|
|
EnvVar: "PLUGIN_REGISTRY",
|
|
},
|
|
cli.StringFlag{
|
|
Name: "username",
|
|
Usage: "docker username",
|
|
EnvVar: "PLUGIN_USERNAME",
|
|
},
|
|
cli.StringFlag{
|
|
Name: "password",
|
|
Usage: "docker password",
|
|
EnvVar: "PLUGIN_PASSWORD",
|
|
},
|
|
cli.BoolFlag{
|
|
Name: "skip-tls-verify",
|
|
Usage: "Skip registry tls verify",
|
|
EnvVar: "PLUGIN_SKIP_TLS_VERIFY",
|
|
},
|
|
cli.StringFlag{
|
|
Name: "snapshot-mode",
|
|
Usage: "Specify one of full, redo or time as snapshot mode",
|
|
EnvVar: "PLUGIN_SNAPSHOT_MODE",
|
|
},
|
|
cli.BoolFlag{
|
|
Name: "enable-cache",
|
|
Usage: "Set this flag to opt into caching with kaniko",
|
|
EnvVar: "PLUGIN_ENABLE_CACHE",
|
|
},
|
|
cli.StringFlag{
|
|
Name: "cache-repo",
|
|
Usage: "Remote repository that will be used to store cached layers. enable-cache needs to be set to use this flag",
|
|
EnvVar: "PLUGIN_CACHE_REPO",
|
|
},
|
|
cli.IntFlag{
|
|
Name: "cache-ttl",
|
|
Usage: "Cache timeout in hours. Defaults to two weeks.",
|
|
EnvVar: "PLUGIN_CACHE_TTL",
|
|
},
|
|
cli.StringFlag{
|
|
Name: "artifact-file",
|
|
Usage: "Artifact file location that will be generated by the plugin. This file will include information of docker images that are uploaded by the plugin.",
|
|
EnvVar: "PLUGIN_ARTIFACT_FILE",
|
|
},
|
|
cli.BoolFlag{
|
|
Name: "no-push",
|
|
Usage: "Set this flag if you only want to build the image, without pushing to a registry",
|
|
EnvVar: "PLUGIN_NO_PUSH",
|
|
},
|
|
cli.StringFlag{
|
|
Name: "verbosity",
|
|
Usage: "Set this flag with value as oneof <panic|fatal|error|warn|info|debug|trace> to set the logging level for kaniko. Defaults to info.",
|
|
EnvVar: "PLUGIN_VERBOSITY",
|
|
},
|
|
cli.StringFlag{
|
|
Name: "platform",
|
|
Usage: "Allows to build with another default platform than the host, similarly to docker build --platform",
|
|
EnvVar: "PLUGIN_PLATFORM",
|
|
},
|
|
}
|
|
|
|
if err := app.Run(os.Args); err != nil {
|
|
logrus.Fatal(err)
|
|
}
|
|
}
|
|
|
|
func run(c *cli.Context) error {
|
|
username := c.String("username")
|
|
noPush := c.Bool("no-push")
|
|
|
|
// only setup auth when pushing or credentials are defined
|
|
if !noPush || username != "" {
|
|
if err := createDockerCfgFile(username, c.String("password"), c.String("registry")); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
plugin := kaniko.Plugin{
|
|
Build: kaniko.Build{
|
|
Dockerfile: c.String("dockerfile"),
|
|
Context: c.String("context"),
|
|
Tags: c.StringSlice("tags"),
|
|
AutoTag: c.Bool("auto_tag"),
|
|
Args: c.StringSlice("args"),
|
|
Target: c.String("target"),
|
|
Repo: c.String("repo"),
|
|
Labels: c.StringSlice("custom-labels"),
|
|
SkipTlsVerify: c.Bool("skip-tls-verify"),
|
|
SnapshotMode: c.String("snapshot-mode"),
|
|
EnableCache: c.Bool("enable-cache"),
|
|
CacheRepo: c.String("cache-repo"),
|
|
CacheTTL: c.Int("cache-ttl"),
|
|
DigestFile: defaultDigestFile,
|
|
NoPush: noPush,
|
|
Verbosity: c.String("verbosity"),
|
|
Platform: c.String("platform"),
|
|
},
|
|
Artifact: kaniko.Artifact{
|
|
Tags: c.StringSlice("tags"),
|
|
Repo: buildRepo(c.String("registry"), c.String("repo")),
|
|
Registry: c.String("registry"),
|
|
ArtifactFile: c.String("artifact-file"),
|
|
RegistryType: artifact.Docker,
|
|
},
|
|
}
|
|
return plugin.Exec()
|
|
}
|
|
|
|
// Create the docker config file for authentication
|
|
func createDockerCfgFile(username, password, registry string) error {
|
|
if username == "" {
|
|
return fmt.Errorf("Username must be specified")
|
|
}
|
|
if password == "" {
|
|
return fmt.Errorf("Password must be specified")
|
|
}
|
|
if registry == "" {
|
|
return fmt.Errorf("Registry must be specified")
|
|
}
|
|
|
|
if registry == v2RegistryURL || registry == v2HubRegistryURL {
|
|
fmt.Println("Docker v2 registry is not supported in kaniko. Refer issue: https://github.com/GoogleContainerTools/kaniko/issues/1209")
|
|
fmt.Printf("Using v1 registry instead: %s\n", v1RegistryURL)
|
|
registry = v1RegistryURL
|
|
}
|
|
|
|
err := os.MkdirAll(dockerPath, 0600)
|
|
if err != nil {
|
|
return errors.Wrap(err, fmt.Sprintf("failed to create %s directory", dockerPath))
|
|
}
|
|
|
|
authBytes := []byte(fmt.Sprintf("%s:%s", username, password))
|
|
encodedString := base64.StdEncoding.EncodeToString(authBytes)
|
|
jsonBytes := []byte(fmt.Sprintf(`{"auths": {"%s": {"auth": "%s"}}}`, registry, encodedString))
|
|
err = ioutil.WriteFile(dockerConfigPath, jsonBytes, 0644)
|
|
if err != nil {
|
|
return errors.Wrap(err, "failed to create docker config file")
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func buildRepo(registry, repo string) string {
|
|
if registry == "" {
|
|
// No custom registry, just return the repo name
|
|
return repo
|
|
}
|
|
if strings.HasPrefix(repo, registry + "/") {
|
|
// Repo already includes the registry prefix
|
|
// For backward compatibility, we won't add the prefix again.
|
|
return repo
|
|
}
|
|
// Prefix the repo with the registry
|
|
return registry + "/" + repo
|
|
}
|