Skip to content

Commit 2ec4b40

Browse files
committed
Make exec more consistent with the rest of the kubectl commands.
1 parent b7caede commit 2ec4b40

File tree

7 files changed

+103
-28
lines changed

7 files changed

+103
-28
lines changed

contrib/completions/bash/kubectl

-4
Original file line numberDiff line numberDiff line change
@@ -485,10 +485,6 @@ _kubectl_exec()
485485
flags+=("-t")
486486

487487
must_have_one_flag=()
488-
must_have_one_flag+=("--container=")
489-
must_have_one_flag+=("-c")
490-
must_have_one_flag+=("--pod=")
491-
must_have_one_flag+=("-p")
492488
must_have_one_noun=()
493489
}
494490

docs/kubectl_exec.md

+7-4
Original file line numberDiff line numberDiff line change
@@ -8,17 +8,20 @@ Execute a command in a container.
88
Execute a command in a container.
99

1010
```
11-
kubectl exec -p POD -c CONTAINER -- COMMAND [args...]
11+
kubectl exec POD -c CONTAINER -- COMMAND [args...]
1212
```
1313

1414
### Examples
1515

1616
```
17+
// get output from running 'date' from pod 123456-7890, using the first container by default
18+
$ kubectl exec 123456-7890 date
19+
1720
// get output from running 'date' in ruby-container from pod 123456-7890
18-
$ kubectl exec -p 123456-7890 -c ruby-container date
21+
$ kubectl exec 123456-7890 -c ruby-container date
1922
2023
//switch to raw terminal mode, sends stdin to 'bash' in ruby-container from pod 123456-780 and sends stdout/stderr from 'bash' back to the client
21-
$ kubectl exec -p 123456-7890 -c ruby-container -i -t -- bash -il
24+
$ kubectl exec 123456-7890 -c ruby-container -i -t -- bash -il
2225
```
2326

2427
### Options
@@ -63,6 +66,6 @@ $ kubectl exec -p 123456-7890 -c ruby-container -i -t -- bash -il
6366
### SEE ALSO
6467
* [kubectl](kubectl.md) - kubectl controls the Kubernetes cluster manager
6568

66-
###### Auto generated by spf13/cobra at 2015-05-21 10:33:11.186469192 +0000 UTC
69+
###### Auto generated by spf13/cobra at 2015-05-27 22:47:02.898315735 +0000 UTC
6770

6871
[![Analytics](https://kubernetes-site.appspot.com/UA-36037335-10/GitHub/docs/kubectl_exec.md?pixel)]()

docs/man/man1/kubectl-exec.1

+5-2
Original file line numberDiff line numberDiff line change
@@ -141,11 +141,14 @@ Execute a command in a container.
141141
.RS
142142

143143
.nf
144+
// get output from running 'date' from pod 123456\-7890, using the first container by default
145+
$ kubectl exec 123456\-7890 date
146+
144147
// get output from running 'date' in ruby\-container from pod 123456\-7890
145-
$ kubectl exec \-p 123456\-7890 \-c ruby\-container date
148+
$ kubectl exec 123456\-7890 \-c ruby\-container date
146149

147150
//switch to raw terminal mode, sends stdin to 'bash' in ruby\-container from pod 123456\-780 and sends stdout/stderr from 'bash' back to the client
148-
$ kubectl exec \-p 123456\-7890 \-c ruby\-container \-i \-t \-\- bash \-il
151+
$ kubectl exec 123456\-7890 \-c ruby\-container \-i \-t \-\- bash \-il
149152

150153
.fi
151154
.RE

pkg/kubectl/cmd/exec.go

+29-15
Original file line numberDiff line numberDiff line change
@@ -32,17 +32,20 @@ import (
3232
)
3333

3434
const (
35-
exec_example = `// get output from running 'date' in ruby-container from pod 123456-7890
36-
$ kubectl exec -p 123456-7890 -c ruby-container date
35+
exec_example = `// get output from running 'date' from pod 123456-7890, using the first container by default
36+
$ kubectl exec 123456-7890 date
37+
38+
// get output from running 'date' in ruby-container from pod 123456-7890
39+
$ kubectl exec 123456-7890 -c ruby-container date
3740
3841
//switch to raw terminal mode, sends stdin to 'bash' in ruby-container from pod 123456-780 and sends stdout/stderr from 'bash' back to the client
39-
$ kubectl exec -p 123456-7890 -c ruby-container -i -t -- bash -il`
42+
$ kubectl exec 123456-7890 -c ruby-container -i -t -- bash -il`
4043
)
4144

4245
func NewCmdExec(f *cmdutil.Factory, cmdIn io.Reader, cmdOut, cmdErr io.Writer) *cobra.Command {
4346
params := &execParams{}
4447
cmd := &cobra.Command{
45-
Use: "exec -p POD -c CONTAINER -- COMMAND [args...]",
48+
Use: "exec POD -c CONTAINER -- COMMAND [args...]",
4649
Short: "Execute a command in a container.",
4750
Long: "Execute a command in a container.",
4851
Example: exec_example,
@@ -52,10 +55,8 @@ func NewCmdExec(f *cmdutil.Factory, cmdIn io.Reader, cmdOut, cmdErr io.Writer) *
5255
},
5356
}
5457
cmd.Flags().StringVarP(&params.podName, "pod", "p", "", "Pod name")
55-
cmd.MarkFlagRequired("pod")
5658
// TODO support UID
5759
cmd.Flags().StringVarP(&params.containerName, "container", "c", "", "Container name")
58-
cmd.MarkFlagRequired("container")
5960
cmd.Flags().BoolVarP(&params.stdin, "stdin", "i", false, "Pass stdin to the container")
6061
cmd.Flags().BoolVarP(&params.tty, "tty", "t", false, "Stdin is a TTY")
6162
return cmd
@@ -79,14 +80,27 @@ type execParams struct {
7980
tty bool
8081
}
8182

82-
func RunExec(f *cmdutil.Factory, cmd *cobra.Command, cmdIn io.Reader, cmdOut, cmdErr io.Writer, p *execParams, args []string, re remoteExecutor) error {
83-
if len(p.podName) == 0 {
84-
return cmdutil.UsageError(cmd, "POD is required for exec")
83+
func extractPodAndContainer(cmd *cobra.Command, args []string, p *execParams) (podName string, containerName string, err error) {
84+
if len(p.podName) == 0 && len(args) == 0 {
85+
return "", "", cmdutil.UsageError(cmd, "POD is required for exec")
8586
}
86-
87-
if len(args) < 1 {
88-
return cmdutil.UsageError(cmd, "COMMAND is required for exec")
87+
if len(p.podName) != 0 {
88+
printDeprecationWarning("exec POD", "-p POD")
89+
podName = p.podName
90+
if len(args) < 1 {
91+
return "", "", cmdutil.UsageError(cmd, "COMMAND is required for exec")
92+
}
93+
} else {
94+
podName = args[0]
95+
if len(args) < 2 {
96+
return "", "", cmdutil.UsageError(cmd, "COMMAND is required for exec")
97+
}
8998
}
99+
return podName, p.containerName, nil
100+
}
101+
102+
func RunExec(f *cmdutil.Factory, cmd *cobra.Command, cmdIn io.Reader, cmdOut, cmdErr io.Writer, p *execParams, args []string, re remoteExecutor) error {
103+
podName, containerName, err := extractPodAndContainer(cmd, args, p)
90104
namespace, err := f.DefaultNamespace()
91105
if err != nil {
92106
return err
@@ -97,17 +111,17 @@ func RunExec(f *cmdutil.Factory, cmd *cobra.Command, cmdIn io.Reader, cmdOut, cm
97111
return err
98112
}
99113

100-
pod, err := client.Pods(namespace).Get(p.podName)
114+
pod, err := client.Pods(namespace).Get(podName)
101115
if err != nil {
102116
return err
103117
}
104118

105119
if pod.Status.Phase != api.PodRunning {
106-
glog.Fatalf("Unable to execute command because pod is not running. Current status=%v", pod.Status.Phase)
120+
glog.Fatalf("Unable to execute command because pod %s is not running. Current status=%v", podName, pod.Status.Phase)
107121
}
108122

109-
containerName := p.containerName
110123
if len(containerName) == 0 {
124+
glog.V(4).Infof("defaulting container name to %s", pod.Spec.Containers[0].Name)
111125
containerName = pod.Spec.Containers[0].Name
112126
}
113127

pkg/kubectl/cmd/exec_test.go

+60-1
Original file line numberDiff line numberDiff line change
@@ -39,8 +39,67 @@ func (f *fakeRemoteExecutor) Execute(req *client.Request, config *client.Config,
3939
return f.execErr
4040
}
4141

42-
func TestExec(t *testing.T) {
42+
func TestPodAndContainer(t *testing.T) {
43+
tests := []struct {
44+
args []string
45+
p *execParams
46+
expectError bool
47+
expectedPod string
48+
expectedContainer string
49+
}{
50+
{
51+
p: &execParams{},
52+
expectError: true,
53+
},
54+
{
55+
p: &execParams{podName: "foo"},
56+
expectError: true,
57+
},
58+
{
59+
p: &execParams{podName: "foo", containerName: "bar"},
60+
expectError: true,
61+
},
62+
{
63+
p: &execParams{podName: "foo"},
64+
args: []string{"cmd"},
65+
expectedPod: "foo",
66+
},
67+
{
68+
p: &execParams{},
69+
args: []string{"foo"},
70+
expectError: true,
71+
},
72+
{
73+
p: &execParams{},
74+
args: []string{"foo", "cmd"},
75+
expectedPod: "foo",
76+
},
77+
{
78+
p: &execParams{containerName: "bar"},
79+
args: []string{"foo", "cmd"},
80+
expectedPod: "foo",
81+
expectedContainer: "bar",
82+
},
83+
}
84+
for _, test := range tests {
85+
cmd := &cobra.Command{}
86+
podName, containerName, err := extractPodAndContainer(cmd, test.args, test.p)
87+
if podName != test.expectedPod {
88+
t.Errorf("expected: %s, got: %s", test.expectedPod, podName)
89+
}
90+
if containerName != test.expectedContainer {
91+
t.Errorf("expected: %s, got: %s", test.expectedContainer, containerName)
92+
}
93+
if test.expectError && err == nil {
94+
t.Error("unexpected non-error")
95+
}
96+
if !test.expectError && err != nil {
97+
t.Errorf("unexpected error: %v", err)
98+
}
99+
}
100+
}
43101

102+
func TestExec(t *testing.T) {
44103
tests := []struct {
45104
name, version, podPath, execPath, container string
46105
nsInQuery bool

pkg/kubelet/dockertools/manager.go

+1-1
Original file line numberDiff line numberDiff line change
@@ -1021,7 +1021,7 @@ func (dm *DockerManager) ExecInContainer(containerId string, cmd []string, stdin
10211021
} else {
10221022
if stdin != nil {
10231023
// Use an os.Pipe here as it returns true *os.File objects.
1024-
// This way, if you run 'kubectl exec -p <pod> -i bash' (no tty) and type 'exit',
1024+
// This way, if you run 'kubectl exec <pod> -i bash' (no tty) and type 'exit',
10251025
// the call below to command.Run() can unblock because its Stdin is the read half
10261026
// of the pipe.
10271027
r, w, err := os.Pipe()

pkg/kubelet/rkt/rkt.go

+1-1
Original file line numberDiff line numberDiff line change
@@ -980,7 +980,7 @@ func (r *runtime) ExecInContainer(containerID string, cmd []string, stdin io.Rea
980980
}
981981
if stdin != nil {
982982
// Use an os.Pipe here as it returns true *os.File objects.
983-
// This way, if you run 'kubectl exec -p <pod> -i bash' (no tty) and type 'exit',
983+
// This way, if you run 'kubectl exec <pod> -i bash' (no tty) and type 'exit',
984984
// the call below to command.Run() can unblock because its Stdin is the read half
985985
// of the pipe.
986986
r, w, err := os.Pipe()

0 commit comments

Comments
 (0)