/
pod.go
188 lines (168 loc) · 5.51 KB
/
pod.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
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
package runner
import (
"context"
"errors"
"fmt"
"path/filepath"
"time"
"github.com/openshift/osde2e/pkg/common/util"
kubev1 "k8s.io/api/core/v1"
kerror "k8s.io/apimachinery/pkg/api/errors"
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
"k8s.io/apimachinery/pkg/util/intstr"
"k8s.io/apimachinery/pkg/util/wait"
"k8s.io/utils/pointer"
)
const (
configMapCreateTimeout = 30 * time.Second
podCreateTimeout = 90 * time.Second
podPendingTimeout = 20 // 20 Iterations
fastPoll = 5 * time.Second
slowPoll = 15 * time.Second
resultsPort = 8000
resultsPortName = "results"
osde2ePayload = "osde2e-payload"
osde2ePayloadMountPath = "/osde2e-payload"
osde2ePayloadScript = "payload.sh"
)
var fullPayloadScriptPath string
func init() {
fullPayloadScriptPath = filepath.Join(osde2ePayloadMountPath, osde2ePayloadScript)
}
// DefaultContainer is used by the DefaultRunner to run workloads
var DefaultContainer = kubev1.Container{
Ports: []kubev1.ContainerPort{
{
Name: resultsPortName,
ContainerPort: resultsPort,
Protocol: kubev1.ProtocolTCP,
},
},
ImagePullPolicy: kubev1.PullAlways,
ReadinessProbe: &kubev1.Probe{
ProbeHandler: kubev1.ProbeHandler{
HTTPGet: &kubev1.HTTPGetAction{
Path: "/",
Port: intstr.FromInt(resultsPort),
},
},
PeriodSeconds: 7,
},
SecurityContext: &kubev1.SecurityContext{
RunAsUser: pointer.Int64(0),
},
}
// volumeMounts returns a v1.VolumeMount given a specific name and the static payloadMountPath
func volumeMounts(name string) []kubev1.VolumeMount {
return []kubev1.VolumeMount{
{
Name: name,
MountPath: osde2ePayloadMountPath,
},
}
}
// volumes returns a v1.Volume given a specific name
func volumes(name string) []kubev1.Volume {
return []kubev1.Volume{
{
Name: name,
VolumeSource: kubev1.VolumeSource{
ConfigMap: &kubev1.ConfigMapVolumeSource{
LocalObjectReference: kubev1.LocalObjectReference{
Name: name,
},
DefaultMode: pointer.Int32(0o755),
},
},
},
}
}
// createPod for creates a runner-based pod and typically collects generated artifacts from it.
func (r *Runner) createPod(ctx context.Context) (pod *kubev1.Pod, err error) {
// configure pod to run workload
cmName := fmt.Sprintf("%s-%s", osde2ePayload, util.RandomStr(5))
pod = &kubev1.Pod{
ObjectMeta: r.meta(),
Spec: r.PodSpec,
}
if len(r.Cmd) != 0 {
cmd, err := r.Command()
if err != nil {
return nil, fmt.Errorf("couldn't template cmd: %v", err)
}
configMap, err := r.Kube.CoreV1().ConfigMaps(r.Namespace).Create(ctx, &kubev1.ConfigMap{
ObjectMeta: metav1.ObjectMeta{
Name: cmName,
},
BinaryData: map[string][]byte{
osde2ePayloadScript: cmd,
},
}, metav1.CreateOptions{})
if err != nil {
return nil, fmt.Errorf("error creating ConfigMap: %v", err)
}
// Verify the configMap has been created before proceeding
err = wait.PollUntilContextTimeout(ctx, fastPoll, configMapCreateTimeout, false, func(ctx context.Context) (done bool, err error) {
if configMap, err = r.Kube.CoreV1().ConfigMaps(r.Namespace).Get(ctx, configMap.Name, metav1.GetOptions{}); err != nil {
r.Error(err, fmt.Sprintf("error creating %s config map", configMap.Name))
}
return err == nil, nil
})
if err != nil {
return nil, err
}
}
// A pod can have multiple containers. Create all the necessary mounts per-container.
for i, container := range pod.Spec.Containers {
if container.Name == "" || container.Name == r.Name {
pod.Spec.Containers[i].Name = r.Name
pod.Spec.Containers[i].Image = r.ImageName
pod.Spec.Containers[i].Env = append(pod.Spec.Containers[i].Env, kubev1.EnvVar{
Name: "OSDE2E",
Value: "true",
})
// run command in pod if, present
if len(r.Cmd) != 0 {
pod.Spec.Containers[i].Command = []string{
fullPayloadScriptPath,
}
pod.Spec.Containers[i].VolumeMounts = volumeMounts(cmName)
}
pod.Spec.Volumes = volumes(cmName)
}
}
// setup git repos to be cloned in init containers
r.Repos.ConfigurePod(&pod.Spec)
// retry until Pod can be created or timeout occurs
var createdPod *kubev1.Pod
err = wait.PollUntilContextTimeout(ctx, fastPoll, podCreateTimeout, false, func(ctx context.Context) (done bool, err error) {
if createdPod, err = r.Kube.CoreV1().Pods(r.Namespace).Create(ctx, pod, metav1.CreateOptions{}); err != nil {
r.Error(err, fmt.Sprintf("error creating %s/%s runner Pod", r.Namespace, r.Name))
}
return err == nil, nil
})
return createdPod, err
}
// waitForRunningPod, given a v1.Pod, will wait for 3 minutes for a pod to enter the running phase or return an error.
func (r *Runner) waitForPodRunning(ctx context.Context, pod *kubev1.Pod) error {
pendingCount := 0
return wait.PollUntilContextTimeout(ctx, fastPoll, 3*time.Minute, false, func(ctx context.Context) (done bool, err error) {
pod, err = r.Kube.CoreV1().Pods(pod.Namespace).Get(ctx, pod.Name, metav1.GetOptions{})
if err != nil && !kerror.IsNotFound(err) {
return
} else if pod == nil {
err = errors.New("pod can't be nil")
} else if pod.Status.Phase == kubev1.PodFailed || pod.Status.Phase == kubev1.PodUnknown {
err = fmt.Errorf("failed waiting for Pod: the Pod has a phase of %s", pod.Status.Phase)
} else if pod.Status.Phase == kubev1.PodRunning {
done = true
} else {
pendingCount++
if pendingCount > podPendingTimeout {
err = errors.New("timed out waiting for pod to start")
}
r.Info(fmt.Sprintf("Waiting for Pod '%s/%s' to start Running...", pod.Namespace, pod.Name))
}
return
})
}