159 lines
4.3 KiB
Go
159 lines
4.3 KiB
Go
/*
|
|
Copyright 2018 Google LLC
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
you may not use this file except in compliance with the License.
|
|
You may obtain a copy of the License at
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
See the License for the specific language governing permissions and
|
|
limitations under the License.
|
|
*/
|
|
|
|
package executor
|
|
|
|
import (
|
|
"fmt"
|
|
"io/ioutil"
|
|
"os"
|
|
|
|
"github.com/GoogleCloudPlatform/kaniko/pkg/commands"
|
|
"github.com/GoogleCloudPlatform/kaniko/pkg/constants"
|
|
"github.com/GoogleCloudPlatform/kaniko/pkg/dockerfile"
|
|
"github.com/GoogleCloudPlatform/kaniko/pkg/image"
|
|
"github.com/GoogleCloudPlatform/kaniko/pkg/snapshot"
|
|
"github.com/GoogleCloudPlatform/kaniko/pkg/util"
|
|
"github.com/containers/image/manifest"
|
|
"github.com/docker/docker/builder/dockerfile/instructions"
|
|
"github.com/sirupsen/logrus"
|
|
)
|
|
|
|
func DoBuild(dockerfilePath, srcContext, destination, snapshotMode string) error {
|
|
// Parse dockerfile and unpack base image to root
|
|
d, err := ioutil.ReadFile(dockerfilePath)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
stages, err := dockerfile.Parse(d)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
baseImage := stages[0].BaseName
|
|
|
|
// Unpack file system to root
|
|
logrus.Infof("Unpacking filesystem of %s...", baseImage)
|
|
if err := util.ExtractFileSystemFromImage(baseImage); err != nil {
|
|
return err
|
|
}
|
|
|
|
hasher, err := getHasher(snapshotMode)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
l := snapshot.NewLayeredMap(hasher)
|
|
snapshotter := snapshot.NewSnapshotter(l, constants.RootDir)
|
|
|
|
// Take initial snapshot
|
|
if err := snapshotter.Init(); err != nil {
|
|
return err
|
|
}
|
|
|
|
// Initialize source image
|
|
sourceImage, err := image.NewSourceImage(baseImage)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
// Set environment variables within the image
|
|
if err := image.SetEnvVariables(sourceImage); err != nil {
|
|
return err
|
|
}
|
|
|
|
imageConfig := sourceImage.Config()
|
|
// Currently only supports single stage builds
|
|
for _, stage := range stages {
|
|
if err := resolveOnBuild(&stage, imageConfig); err != nil {
|
|
return err
|
|
}
|
|
for _, cmd := range stage.Commands {
|
|
dockerCommand, err := commands.GetCommand(cmd, srcContext)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if dockerCommand == nil {
|
|
continue
|
|
}
|
|
if err := dockerCommand.ExecuteCommand(imageConfig); err != nil {
|
|
return err
|
|
}
|
|
// Now, we get the files to snapshot from this command and take the snapshot
|
|
snapshotFiles := dockerCommand.FilesToSnapshot()
|
|
contents, err := snapshotter.TakeSnapshot(snapshotFiles)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
util.MoveVolumeWhitelistToWhitelist()
|
|
if contents == nil {
|
|
logrus.Info("No files were changed, appending empty layer to config.")
|
|
sourceImage.AppendConfigHistory(constants.Author, true)
|
|
continue
|
|
}
|
|
// Append the layer to the image
|
|
if err := sourceImage.AppendLayer(contents, constants.Author); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
}
|
|
// Push the image
|
|
if err := setDefaultEnv(); err != nil {
|
|
return err
|
|
}
|
|
return image.PushImage(sourceImage, destination)
|
|
}
|
|
|
|
func getHasher(snapshotMode string) (func(string) (string, error), error) {
|
|
if snapshotMode == constants.SnapshotModeTime {
|
|
logrus.Info("Only file modification time will be considered when snapshotting")
|
|
return util.MtimeHasher(), nil
|
|
}
|
|
if snapshotMode == constants.SnapshotModeFull {
|
|
return util.Hasher(), nil
|
|
}
|
|
return nil, fmt.Errorf("%s is not a valid snapshot mode", snapshotMode)
|
|
}
|
|
|
|
func resolveOnBuild(stage *instructions.Stage, config *manifest.Schema2Config) error {
|
|
if config.OnBuild == nil {
|
|
return nil
|
|
}
|
|
// Otherwise, parse into commands
|
|
cmds, err := dockerfile.ParseCommands(config.OnBuild)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
// Append to the beginning of the commands in the stage
|
|
stage.Commands = append(cmds, stage.Commands...)
|
|
logrus.Infof("Executing %v build triggers", len(cmds))
|
|
return nil
|
|
}
|
|
|
|
// setDefaultEnv sets default values for HOME and PATH so that
|
|
// config.json and docker-credential-gcr can be accessed
|
|
func setDefaultEnv() error {
|
|
defaultEnvs := map[string]string{
|
|
"HOME": "/root",
|
|
"PATH": "/usr/local/bin/",
|
|
}
|
|
for key, val := range defaultEnvs {
|
|
if err := os.Setenv(key, val); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
return nil
|
|
}
|