forked from openshift/origin
-
Notifications
You must be signed in to change notification settings - Fork 1
/
buildpodutil.go
105 lines (94 loc) · 3.32 KB
/
buildpodutil.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
package admission
import (
"errors"
"fmt"
"k8s.io/apimachinery/pkg/runtime"
"k8s.io/apimachinery/pkg/runtime/schema"
kapi "k8s.io/kubernetes/pkg/api"
buildapi "github.com/openshift/origin/pkg/build/api"
)
// GetBuildFromPod returns a build object encoded in a pod's BUILD environment variable along with
// its encoding version
func GetBuildFromPod(pod *kapi.Pod) (*buildapi.Build, schema.GroupVersion, error) {
envVar, err := buildEnvVar(pod)
if err != nil {
return nil, schema.GroupVersion{}, fmt.Errorf("unable to get build from pod: %v", err)
}
obj, groupVersionKind, err := kapi.Codecs.UniversalDecoder().Decode([]byte(envVar.Value), nil, nil)
if err != nil {
return nil, schema.GroupVersion{}, fmt.Errorf("unable to get build from pod: %v", err)
}
build, ok := obj.(*buildapi.Build)
if !ok {
return nil, schema.GroupVersion{}, fmt.Errorf("unable to get build from pod: %v", errors.New("decoded object is not of type Build"))
}
return build, groupVersionKind.GroupVersion(), nil
}
// SetBuildInPod encodes a build object and sets it in a pod's BUILD environment variable
func SetBuildInPod(pod *kapi.Pod, build *buildapi.Build, groupVersion schema.GroupVersion) error {
envVar, err := buildEnvVar(pod)
if err != nil {
return fmt.Errorf("unable to set build in pod: %v", err)
}
encodedBuild, err := runtime.Encode(kapi.Codecs.LegacyCodec(groupVersion), build)
if err != nil {
return fmt.Errorf("unable to set build in pod: %v", err)
}
envVar.Value = string(encodedBuild)
return nil
}
// SetPodLogLevelFromBuild extracts BUILD_LOGLEVEL from the Build environment
// and feeds it as an argument to the Pod's entrypoint. The BUILD_LOGLEVEL
// environment variable may have been set in multiple ways: a default value,
// by a BuildConfig, or by the BuildDefaults admission plugin. In this method
// we finally act on the value by injecting it into the Pod.
func SetPodLogLevelFromBuild(pod *kapi.Pod, build *buildapi.Build) error {
var envs []kapi.EnvVar
// Check whether the build strategy supports --loglevel parameter.
switch {
case build.Spec.Strategy.DockerStrategy != nil:
envs = build.Spec.Strategy.DockerStrategy.Env
case build.Spec.Strategy.SourceStrategy != nil:
envs = build.Spec.Strategy.SourceStrategy.Env
default:
// The build strategy does not support --loglevel
return nil
}
buildLogLevel := "0" // The ultimate default for the build pod's loglevel if no actor sets BUILD_LOGLEVEL in the Build
for i := range envs {
env := envs[i]
if env.Name == "BUILD_LOGLEVEL" {
buildLogLevel = env.Value
break
}
}
c := &pod.Spec.Containers[0]
c.Args = append(c.Args, "--loglevel="+buildLogLevel)
return nil
}
func buildEnvVar(pod *kapi.Pod) (*kapi.EnvVar, error) {
if len(pod.Spec.Containers) == 0 {
return nil, errors.New("pod has no containers")
}
env := pod.Spec.Containers[0].Env
for i := range env {
if env[i].Name == "BUILD" {
if len(env[i].Value) == 0 {
return nil, errors.New("BUILD environment variable is empty")
}
return &env[i], nil
}
}
return nil, errors.New("pod does not have a BUILD environment variable")
}
func hasBuildEnvVar(pod *kapi.Pod) bool {
_, err := buildEnvVar(pod)
return err == nil
}
func hasBuildAnnotation(pod *kapi.Pod) bool {
if pod.Annotations == nil {
return false
}
_, hasAnnotation := pod.Annotations[buildapi.BuildAnnotation]
return hasAnnotation
}