-
Notifications
You must be signed in to change notification settings - Fork 1.8k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Load builtin entrypoint while redirecting steps
Currently, any steps in a Pipeline that rely on the built in entrypoint of a container will not have the expected behaviour while running due to the entrypoint being overridden at runtime. This fixes #175. A major side effect of this work is that the override step will now possibly make HTTP calls every time a step is overridden. There is very rudimentary caching in place, however this could likely be improved if performance becomes an issue. Fixes #175
- Loading branch information
Showing
9 changed files
with
411 additions
and
164 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
162 changes: 162 additions & 0 deletions
162
pkg/reconciler/v1alpha1/taskrun/entrypoint/entrypoint.go
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,162 @@ | ||
/* | ||
Copyright 2018 The Knative Authors | ||
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 entrypoint | ||
|
||
import ( | ||
"encoding/json" | ||
"fmt" | ||
"sync" | ||
|
||
"github.com/google/go-containerregistry/pkg/authn" | ||
"github.com/google/go-containerregistry/pkg/name" | ||
"github.com/google/go-containerregistry/pkg/v1/remote" | ||
"github.com/knative/build/pkg/apis/build/v1alpha1" | ||
|
||
corev1 "k8s.io/api/core/v1" | ||
) | ||
|
||
const ( | ||
// MountName is the name of the pvc being mounted (which | ||
// will contain the entrypoint binary and eventually the logs) | ||
MountName = "tools" | ||
MountPoint = "/tools" | ||
BinaryLocation = MountPoint + "/entrypoint" | ||
JSONConfigEnvVar = "ENTRYPOINT_OPTIONS" | ||
Image = "gcr.io/k8s-prow/entrypoint@sha256:7c7cd8906ce4982ffee326218e9fc75da2d4896d53cabc9833b9cc8d2d6b2b8f" | ||
InitContainerName = "place-tools" | ||
ProcessLogFile = "/tools/process-log.txt" | ||
MarkerFile = "/tools/marker-file.txt" | ||
) | ||
|
||
var toolsMount = corev1.VolumeMount{ | ||
Name: MountName, | ||
MountPath: MountPoint, | ||
} | ||
|
||
// Cache is a simple caching mechanism allowing for caching the results of | ||
// getting the Entrypoint of a container image from a remote registry. It | ||
// is synchronized via a mutex so that we can share a single Cache across | ||
// each worker thread that the reconciler is running. The mutex is necessary | ||
// due to the possibility of a panic if two workers were to attempt to read and | ||
// write to the internal map at the same time. | ||
type Cache struct { | ||
mtx sync.RWMutex | ||
cache map[string][]string | ||
} | ||
|
||
// NewCache is a simple helper function that returns a pointer to a Cache that | ||
// has had the internal cache map initialized. | ||
func NewCache() *Cache { | ||
return &Cache{ | ||
cache: make(map[string][]string), | ||
} | ||
} | ||
|
||
func (c *Cache) get(sha string) ([]string, bool) { | ||
c.mtx.RLock() | ||
ep, ok := c.cache[sha] | ||
c.mtx.RUnlock() | ||
return ep, ok | ||
} | ||
|
||
func (c *Cache) set(sha string, ep []string) { | ||
c.mtx.Lock() | ||
c.cache[sha] = ep | ||
c.mtx.Unlock() | ||
} | ||
|
||
// AddCopyStep will prepend a BuildStep (Container) that will | ||
// copy the entrypoint binary from the entrypoint image into the | ||
// volume mounted at MountPoint, so that it can be mounted by | ||
// subsequent steps and used to capture logs. | ||
func AddCopyStep(b *v1alpha1.BuildSpec) { | ||
cp := corev1.Container{ | ||
Name: InitContainerName, | ||
Image: Image, | ||
Command: []string{"/bin/cp"}, | ||
Args: []string{"/entrypoint", BinaryLocation}, | ||
VolumeMounts: []corev1.VolumeMount{toolsMount}, | ||
} | ||
b.Steps = append([]corev1.Container{cp}, b.Steps...) | ||
|
||
} | ||
|
||
type entrypointArgs struct { | ||
Args []string `json:"args"` | ||
ProcessLog string `json:"process_log"` | ||
MarkerFile string `json:"marker_file"` | ||
} | ||
|
||
func getEnvVar(cmd, args []string) (string, error) { | ||
entrypointArgs := entrypointArgs{ | ||
Args: append(cmd, args...), | ||
ProcessLog: ProcessLogFile, | ||
MarkerFile: MarkerFile, | ||
} | ||
j, err := json.Marshal(entrypointArgs) | ||
if err != nil { | ||
return "", fmt.Errorf("couldn't marshal arguments %q for entrypoint env var: %s", entrypointArgs, err) | ||
} | ||
return string(j), nil | ||
} | ||
|
||
// GetRemoteEntrypoint accepts a cache of image lookups, as well as the image | ||
// to look for. If the cache does not contain the image, it will lookup the | ||
// metadata from the images registry, and then commit that to the cache | ||
func GetRemoteEntrypoint(cache *Cache, image string) ([]string, error) { | ||
if ep, ok := cache.get(image); ok { | ||
return ep, nil | ||
} | ||
// verify the image name, then download the remote config file | ||
ref, err := name.ParseReference(image, name.WeakValidation) | ||
if err != nil { | ||
return nil, fmt.Errorf("couldn't parse image %s: %v", image, err) | ||
} | ||
img, err := remote.Image(ref, remote.WithAuthFromKeychain(authn.DefaultKeychain)) | ||
if err != nil { | ||
return nil, fmt.Errorf("couldn't get container image info from registry %s: %v", image, err) | ||
} | ||
cfg, err := img.ConfigFile() | ||
if err != nil { | ||
return nil, fmt.Errorf("couldn't get config for image %s: %v", image, err) | ||
} | ||
cache.set(image, cfg.ContainerConfig.Entrypoint) | ||
return cfg.ContainerConfig.Entrypoint, nil | ||
} | ||
|
||
// RedirectSteps will modify each of the steps/containers such that | ||
// the binary being run is no longer the one specified by the Command | ||
// and the Args, but is instead the entrypoint binary, which will | ||
// itself invoke the Command and Args, but also capture logs. | ||
func RedirectSteps(steps []corev1.Container) error { | ||
for i := range steps { | ||
step := &steps[i] | ||
e, err := getEnvVar(step.Command, step.Args) | ||
if err != nil { | ||
return fmt.Errorf("couldn't get env var for entrypoint: %s", err) | ||
} | ||
step.Command = []string{BinaryLocation} | ||
step.Args = []string{} | ||
|
||
step.Env = append(step.Env, corev1.EnvVar{ | ||
Name: JSONConfigEnvVar, | ||
Value: e, | ||
}) | ||
step.VolumeMounts = append(step.VolumeMounts, toolsMount) | ||
} | ||
return nil | ||
} |
103 changes: 103 additions & 0 deletions
103
pkg/reconciler/v1alpha1/taskrun/entrypoint/entrypoint_test.go
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,103 @@ | ||
package entrypoint_test | ||
|
||
import ( | ||
"github.com/knative/build-pipeline/pkg/reconciler/v1alpha1/taskrun/entrypoint" | ||
"github.com/knative/build/pkg/apis/build/v1alpha1" | ||
"k8s.io/api/core/v1" | ||
"testing" | ||
) | ||
const ( | ||
kanikoImage = "gcr.io/kaniko-project/executor" | ||
kanikoEntrypoint = "/kaniko/executor" | ||
) | ||
|
||
func TestAddEntrypoint(t *testing.T) { | ||
inputs := []v1.Container{ | ||
{ | ||
Image: kanikoImage, | ||
}, | ||
{ | ||
Image: kanikoImage, | ||
Args: []string{"abcd"}, | ||
}, | ||
{ | ||
Image: kanikoImage, | ||
Command: []string{"abcd"}, | ||
Args: []string{"efgh"}, | ||
}, | ||
} | ||
// The first test case showcases the downloading of the entrypoint for the | ||
// image. The second test shows downloading the image as well as the args | ||
// being passed in. The third command shows a set Command overriding the | ||
// remote one. | ||
envVarStrings := []string{ | ||
`{"args":null,"process_log":"/tools/process-log.txt","marker_file":"/tools/marker-file.txt"}`, | ||
`{"args":["abcd"],"process_log":"/tools/process-log.txt","marker_file":"/tools/marker-file.txt"}`, | ||
`{"args":["abcd","efgh"],"process_log":"/tools/process-log.txt","marker_file":"/tools/marker-file.txt"}`, | ||
|
||
} | ||
err := entrypoint.RedirectSteps(inputs) | ||
if err != nil { | ||
t.Errorf("failed to get resources: %v", err) | ||
} | ||
for i, input := range inputs { | ||
if len(input.Command) == 0 || input.Command[0] != entrypoint.BinaryLocation { | ||
t.Errorf("command incorrectly set: %q", input.Command) | ||
} | ||
if len(input.Args) > 0 { | ||
t.Errorf("containers should have no args") | ||
} | ||
if len(input.Env) == 0 { | ||
t.Error("there should be atleast one envvar") | ||
} | ||
for _, e := range input.Env { | ||
if e.Name == entrypoint.JSONConfigEnvVar && e.Value != envVarStrings[i] { | ||
t.Errorf("envvar \n%s\n does not match \n%s", e.Value, envVarStrings[i]) | ||
} | ||
} | ||
found := false | ||
for _, vm := range input.VolumeMounts { | ||
if vm.Name == entrypoint.MountName { | ||
found = true | ||
break | ||
} | ||
} | ||
if !found { | ||
t.Error("could not find tools volume mount") | ||
} | ||
} | ||
} | ||
|
||
func TestGetRemoteEntrypoint(t *testing.T) { | ||
ep, err := entrypoint.GetRemoteEntrypoint(entrypoint.NewCache(), kanikoImage) | ||
if err != nil { | ||
t.Errorf("couldn't get entrypoint remote: %v", err) | ||
} | ||
if len(ep) != 1 { | ||
t.Errorf("remote entrypoint should only have one item") | ||
} | ||
if ep[0] != kanikoEntrypoint { | ||
t.Errorf("entrypoints do not match: %s should be %s", ep[0], kanikoEntrypoint) | ||
} | ||
} | ||
|
||
func TestAddCopyStep(t *testing.T) { | ||
bs := &v1alpha1.BuildSpec{ | ||
Steps: []v1.Container{ | ||
{ | ||
Name: "test", | ||
}, | ||
{ | ||
Name: "test", | ||
}, | ||
}, | ||
} | ||
expectedSteps := len(bs.Steps) + 1 | ||
entrypoint.AddCopyStep(bs) | ||
if len(bs.Steps) != 3 { | ||
t.Errorf("BuildSpec has the wrong step count: %d should be %d", len(bs.Steps), expectedSteps) | ||
} | ||
if bs.Steps[0].Name != entrypoint.InitContainerName { | ||
t.Errorf("entrypoint is incorrect: %s should be %s", bs.Steps[0].Name, entrypoint.InitContainerName) | ||
} | ||
} |
Oops, something went wrong.