@@ -16,11 +16,13 @@ import (
16
16
batchv1 "k8s.io/api/batch/v1"
17
17
corev1 "k8s.io/api/core/v1"
18
18
extensionsv1beta1 "k8s.io/api/extensions/v1beta1"
19
+ "k8s.io/apimachinery/pkg/api/errors"
19
20
kapierrors "k8s.io/apimachinery/pkg/api/errors"
20
21
"k8s.io/apimachinery/pkg/apis/meta/v1"
21
22
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
22
23
"k8s.io/apimachinery/pkg/labels"
23
24
"k8s.io/apimachinery/pkg/runtime"
25
+ "k8s.io/apimachinery/pkg/runtime/schema"
24
26
"k8s.io/apimachinery/pkg/watch"
25
27
corev1client "k8s.io/client-go/kubernetes/typed/core/v1"
26
28
"k8s.io/kubernetes/pkg/api/legacyscheme"
@@ -103,8 +105,9 @@ type DebugOptions struct {
103
105
AppsClient appsv1client.AppsV1Interface
104
106
ImageClient imagev1client.ImageV1Interface
105
107
106
- Printer printers.ResourcePrinter
107
- LogsForObject polymorphichelpers.LogsForObjectFunc
108
+ Printer printers.ResourcePrinter
109
+ LogsForObject polymorphichelpers.LogsForObjectFunc
110
+ RESTClientGetter genericclioptions.RESTClientGetter
108
111
109
112
NoStdin bool
110
113
ForceTTY bool
@@ -211,6 +214,7 @@ func (o *DebugOptions) Complete(cmd *cobra.Command, f kcmdutil.Factory, args []s
211
214
return kcmdutil .UsageErrorf (cmd , "all resources must be specified before environment changes: %s" , strings .Join (args , " " ))
212
215
}
213
216
o .Resources = resources
217
+ o .RESTClientGetter = f
214
218
215
219
switch {
216
220
case o .ForceTTY && o .NoStdin :
@@ -432,7 +436,7 @@ func (o *DebugOptions) RunDebug() error {
432
436
}
433
437
fmt .Fprintf (o .ErrOut , "Waiting for pod to start ...\n " )
434
438
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 )); {
436
440
// api didn't error right away but the pod wasn't even created
437
441
case kapierrors .IsNotFound (err ):
438
442
msg := fmt .Sprintf ("unable to create the debug pod %q" , pod .Name )
@@ -444,12 +448,14 @@ func (o *DebugOptions) RunDebug() error {
444
448
case err == kubectl .ErrPodCompleted , err == kubectl .ErrContainerTerminated , ! o .Attach .Stdin :
445
449
return kcmd.LogsOptions {
446
450
Object : pod ,
447
- Options : & kapi .PodLogOptions {
451
+ Options : & corev1 .PodLogOptions {
448
452
Container : o .Attach .ContainerName ,
449
453
Follow : true ,
450
454
},
451
- IOStreams : o .IOStreams ,
452
- LogsForObject : o .LogsForObject ,
455
+ RESTClientGetter : o .RESTClientGetter ,
456
+ ConsumeRequestFn : kcmd .DefaultConsumeRequestFn ,
457
+ IOStreams : o .IOStreams ,
458
+ LogsForObject : o .LogsForObject ,
453
459
}.RunLogs ()
454
460
case err != nil :
455
461
return err
@@ -828,3 +834,44 @@ func (o *DebugOptions) approximatePodTemplateForObject(object runtime.Object) (*
828
834
829
835
return nil , fmt .Errorf ("unable to extract pod template from type %v" , reflect .TypeOf (object ))
830
836
}
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
+ }
0 commit comments