Skip to content

Commit 82db1ef

Browse files
committed
fix debug panic
1 parent b9d9a64 commit 82db1ef

File tree

3 files changed

+112
-13
lines changed

3 files changed

+112
-13
lines changed

pkg/oc/cli/admin/diagnostics/diagnostics/cluster/app_create/app.go

+46-1
Original file line numberDiff line numberDiff line change
@@ -5,8 +5,12 @@ import (
55
"time"
66

77
corev1 "k8s.io/api/core/v1"
8+
"k8s.io/apimachinery/pkg/api/errors"
89
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
10+
"k8s.io/apimachinery/pkg/runtime/schema"
911
"k8s.io/apimachinery/pkg/util/intstr"
12+
"k8s.io/apimachinery/pkg/watch"
13+
api "k8s.io/kubernetes/pkg/apis/core"
1014
"k8s.io/kubernetes/pkg/kubectl"
1115

1216
appsv1 "github.com/openshift/api/apps/v1"
@@ -100,7 +104,7 @@ This may be a transient error. Check the master API logs for anomalies near this
100104
}
101105
defer stopWatcher(watcher)
102106
for event := range watcher.ResultChan() {
103-
running, err := kubectl.PodContainerRunning(d.appName)(event)
107+
running, err := podContainerRunning(d.appName)(event)
104108
if err != nil {
105109
d.out.Error("DCluAC009", err, fmt.Sprintf(`
106110
%s: Error while watching for app pod to deploy:
@@ -124,3 +128,44 @@ There are many reasons why this can occur; for example:
124128
`, now(), d.deployTimeout))
125129
return false
126130
}
131+
132+
// podContainerRunning returns false until the named container has ContainerStatus running (at least once),
133+
// and will return an error if the pod is deleted, runs to completion, or the container pod is not available.
134+
func podContainerRunning(containerName string) watch.ConditionFunc {
135+
return func(event watch.Event) (bool, error) {
136+
switch event.Type {
137+
case watch.Deleted:
138+
return false, errors.NewNotFound(schema.GroupResource{Resource: "pods"}, "")
139+
}
140+
switch t := event.Object.(type) {
141+
case *api.Pod:
142+
switch t.Status.Phase {
143+
case api.PodRunning, api.PodPending:
144+
case api.PodFailed, api.PodSucceeded:
145+
return false, kubectl.ErrPodCompleted
146+
default:
147+
return false, nil
148+
}
149+
for _, s := range t.Status.ContainerStatuses {
150+
if s.Name != containerName {
151+
continue
152+
}
153+
if s.State.Terminated != nil {
154+
return false, kubectl.ErrContainerTerminated
155+
}
156+
return s.State.Running != nil, nil
157+
}
158+
for _, s := range t.Status.InitContainerStatuses {
159+
if s.Name != containerName {
160+
continue
161+
}
162+
if s.State.Terminated != nil {
163+
return false, kubectl.ErrContainerTerminated
164+
}
165+
return s.State.Running != nil, nil
166+
}
167+
return false, nil
168+
}
169+
return false, nil
170+
}
171+
}

pkg/oc/cli/debug/debug.go

+53-6
Original file line numberDiff line numberDiff line change
@@ -16,11 +16,13 @@ import (
1616
batchv1 "k8s.io/api/batch/v1"
1717
corev1 "k8s.io/api/core/v1"
1818
extensionsv1beta1 "k8s.io/api/extensions/v1beta1"
19+
"k8s.io/apimachinery/pkg/api/errors"
1920
kapierrors "k8s.io/apimachinery/pkg/api/errors"
2021
"k8s.io/apimachinery/pkg/apis/meta/v1"
2122
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
2223
"k8s.io/apimachinery/pkg/labels"
2324
"k8s.io/apimachinery/pkg/runtime"
25+
"k8s.io/apimachinery/pkg/runtime/schema"
2426
"k8s.io/apimachinery/pkg/watch"
2527
corev1client "k8s.io/client-go/kubernetes/typed/core/v1"
2628
"k8s.io/kubernetes/pkg/api/legacyscheme"
@@ -103,8 +105,9 @@ type DebugOptions struct {
103105
AppsClient appsv1client.AppsV1Interface
104106
ImageClient imagev1client.ImageV1Interface
105107

106-
Printer printers.ResourcePrinter
107-
LogsForObject polymorphichelpers.LogsForObjectFunc
108+
Printer printers.ResourcePrinter
109+
LogsForObject polymorphichelpers.LogsForObjectFunc
110+
RESTClientGetter genericclioptions.RESTClientGetter
108111

109112
NoStdin bool
110113
ForceTTY bool
@@ -211,6 +214,7 @@ func (o *DebugOptions) Complete(cmd *cobra.Command, f kcmdutil.Factory, args []s
211214
return kcmdutil.UsageErrorf(cmd, "all resources must be specified before environment changes: %s", strings.Join(args, " "))
212215
}
213216
o.Resources = resources
217+
o.RESTClientGetter = f
214218

215219
switch {
216220
case o.ForceTTY && o.NoStdin:
@@ -432,7 +436,7 @@ func (o *DebugOptions) RunDebug() error {
432436
}
433437
fmt.Fprintf(o.ErrOut, "Waiting for pod to start ...\n")
434438

435-
switch containerRunningEvent, err := watch.Until(o.Timeout, w, kubectl.PodContainerRunning(o.Attach.ContainerName)); {
439+
switch containerRunningEvent, err := watch.Until(o.Timeout, w, podContainerRunning(o.Attach.ContainerName)); {
436440
// api didn't error right away but the pod wasn't even created
437441
case kapierrors.IsNotFound(err):
438442
msg := fmt.Sprintf("unable to create the debug pod %q", pod.Name)
@@ -444,12 +448,14 @@ func (o *DebugOptions) RunDebug() error {
444448
case err == kubectl.ErrPodCompleted, err == kubectl.ErrContainerTerminated, !o.Attach.Stdin:
445449
return kcmd.LogsOptions{
446450
Object: pod,
447-
Options: &kapi.PodLogOptions{
451+
Options: &corev1.PodLogOptions{
448452
Container: o.Attach.ContainerName,
449453
Follow: true,
450454
},
451-
IOStreams: o.IOStreams,
452-
LogsForObject: o.LogsForObject,
455+
RESTClientGetter: o.RESTClientGetter,
456+
ConsumeRequestFn: kcmd.DefaultConsumeRequestFn,
457+
IOStreams: o.IOStreams,
458+
LogsForObject: o.LogsForObject,
453459
}.RunLogs()
454460
case err != nil:
455461
return err
@@ -828,3 +834,44 @@ func (o *DebugOptions) approximatePodTemplateForObject(object runtime.Object) (*
828834

829835
return nil, fmt.Errorf("unable to extract pod template from type %v", reflect.TypeOf(object))
830836
}
837+
838+
// podContainerRunning returns false until the named container has ContainerStatus running (at least once),
839+
// and will return an error if the pod is deleted, runs to completion, or the container pod is not available.
840+
func podContainerRunning(containerName string) watch.ConditionFunc {
841+
return func(event watch.Event) (bool, error) {
842+
switch event.Type {
843+
case watch.Deleted:
844+
return false, errors.NewNotFound(schema.GroupResource{Resource: "pods"}, "")
845+
}
846+
switch t := event.Object.(type) {
847+
case *corev1.Pod:
848+
switch t.Status.Phase {
849+
case corev1.PodRunning, corev1.PodPending:
850+
case corev1.PodFailed, corev1.PodSucceeded:
851+
return false, kubectl.ErrPodCompleted
852+
default:
853+
return false, nil
854+
}
855+
for _, s := range t.Status.ContainerStatuses {
856+
if s.Name != containerName {
857+
continue
858+
}
859+
if s.State.Terminated != nil {
860+
return false, kubectl.ErrContainerTerminated
861+
}
862+
return s.State.Running != nil, nil
863+
}
864+
for _, s := range t.Status.InitContainerStatuses {
865+
if s.Name != containerName {
866+
continue
867+
}
868+
if s.State.Terminated != nil {
869+
return false, kubectl.ErrContainerTerminated
870+
}
871+
return s.State.Running != nil, nil
872+
}
873+
return false, nil
874+
}
875+
return false, nil
876+
}
877+
}

pkg/oc/cli/newapp/newapp.go

+13-6
Original file line numberDiff line numberDiff line change
@@ -161,8 +161,11 @@ type ObjectGeneratorOptions struct {
161161
}
162162

163163
type AppOptions struct {
164-
genericclioptions.IOStreams
165164
*ObjectGeneratorOptions
165+
166+
RESTClientGetter genericclioptions.RESTClientGetter
167+
168+
genericclioptions.IOStreams
166169
}
167170

168171
type versionedPrintObj struct {
@@ -367,6 +370,8 @@ func NewCmdNewApplication(name, baseName string, f kcmdutil.Factory, streams gen
367370

368371
// Complete sets any default behavior for the command
369372
func (o *AppOptions) Complete(baseName, commandName string, f kcmdutil.Factory, c *cobra.Command, args []string) error {
373+
o.RESTClientGetter = f
374+
370375
cmdutil.WarnAboutCommaSeparation(o.ErrOut, o.ObjectGeneratorOptions.Config.TemplateParameters, "--param")
371376
err := o.ObjectGeneratorOptions.Complete(baseName, commandName, f, c, args)
372377
if err != nil {
@@ -510,7 +515,7 @@ func (o *AppOptions) RunNewApp() error {
510515

511516
switch {
512517
case len(installing) == 1:
513-
return followInstallation(config, installing[0], o.LogsForObject)
518+
return followInstallation(config, o.RESTClientGetter, installing[0], o.LogsForObject)
514519
case len(installing) > 1:
515520
for i := range installing {
516521
fmt.Fprintf(out, "%sTrack installation of %s with '%s logs %s'.\n", indent, installing[i].Name, o.BaseName, installing[i].Name)
@@ -554,7 +559,7 @@ func getServices(items []runtime.Object) []*corev1.Service {
554559
return svc
555560
}
556561

557-
func followInstallation(config *newcmd.AppConfig, pod *corev1.Pod, logsForObjectFn polymorphichelpers.LogsForObjectFunc) error {
562+
func followInstallation(config *newcmd.AppConfig, clientGetter genericclioptions.RESTClientGetter, pod *corev1.Pod, logsForObjectFn polymorphichelpers.LogsForObjectFunc) error {
558563
fmt.Fprintf(config.Out, "--> Installing ...\n")
559564

560565
// we cannot retrieve logs until the pod is out of pending
@@ -567,12 +572,14 @@ func followInstallation(config *newcmd.AppConfig, pod *corev1.Pod, logsForObject
567572
opts := &kcmd.LogsOptions{
568573
Namespace: pod.Namespace,
569574
ResourceArg: pod.Name,
570-
Options: &kapi.PodLogOptions{
575+
Options: &corev1.PodLogOptions{
571576
Follow: true,
572577
Container: pod.Spec.Containers[0].Name,
573578
},
574-
LogsForObject: logsForObjectFn,
575-
IOStreams: genericclioptions.IOStreams{Out: config.Out},
579+
RESTClientGetter: clientGetter,
580+
ConsumeRequestFn: kcmd.DefaultConsumeRequestFn,
581+
LogsForObject: logsForObjectFn,
582+
IOStreams: genericclioptions.IOStreams{Out: config.Out},
576583
}
577584
logErr := opts.RunLogs()
578585

0 commit comments

Comments
 (0)