2016-07-24 05:04:09 +00:00
|
|
|
/*
|
|
|
|
Copyright 2016 The Kubernetes Authors.
|
|
|
|
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
you may not use this file except in compliance with the License.
|
|
|
|
You may obtain a copy of the License at
|
|
|
|
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
See the License for the specific language governing permissions and
|
|
|
|
limitations under the License.
|
|
|
|
*/
|
|
|
|
|
|
|
|
package framework
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"io"
|
|
|
|
"net/url"
|
|
|
|
"strings"
|
|
|
|
|
2017-01-11 14:09:48 +00:00
|
|
|
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
|
2017-02-15 10:34:49 +00:00
|
|
|
remocommandconsts "k8s.io/apimachinery/pkg/util/remotecommand"
|
2017-01-19 18:27:59 +00:00
|
|
|
restclient "k8s.io/client-go/rest"
|
2017-04-14 09:33:57 +00:00
|
|
|
"k8s.io/client-go/tools/remotecommand"
|
2016-07-24 05:04:09 +00:00
|
|
|
"k8s.io/kubernetes/pkg/api"
|
2016-11-18 20:55:17 +00:00
|
|
|
"k8s.io/kubernetes/pkg/api/v1"
|
2016-07-24 05:04:09 +00:00
|
|
|
|
|
|
|
. "github.com/onsi/gomega"
|
|
|
|
)
|
|
|
|
|
2016-11-11 19:55:37 +00:00
|
|
|
// ExecOptions passed to ExecWithOptions
|
|
|
|
type ExecOptions struct {
|
|
|
|
Command []string
|
|
|
|
|
|
|
|
Namespace string
|
|
|
|
PodName string
|
|
|
|
ContainerName string
|
|
|
|
|
|
|
|
Stdin io.Reader
|
|
|
|
CaptureStdout bool
|
|
|
|
CaptureStderr bool
|
|
|
|
// If false, whitespace in std{err,out} will be removed.
|
|
|
|
PreserveWhitespace bool
|
2016-10-04 21:04:13 +00:00
|
|
|
}
|
|
|
|
|
2016-11-11 19:55:37 +00:00
|
|
|
// ExecWithOptions executes a command in the specified container,
|
|
|
|
// returning stdout, stderr and error. `options` allowed for
|
|
|
|
// additional parameters to be passed.
|
|
|
|
func (f *Framework) ExecWithOptions(options ExecOptions) (string, string, error) {
|
|
|
|
Logf("ExecWithOptions %+v", options)
|
|
|
|
|
2016-07-24 05:04:09 +00:00
|
|
|
config, err := LoadConfig()
|
|
|
|
Expect(err).NotTo(HaveOccurred(), "failed to load restclient config")
|
2016-11-11 19:55:37 +00:00
|
|
|
|
|
|
|
const tty = false
|
|
|
|
|
2016-10-18 13:00:38 +00:00
|
|
|
req := f.ClientSet.Core().RESTClient().Post().
|
2016-07-24 05:04:09 +00:00
|
|
|
Resource("pods").
|
2016-11-11 19:55:37 +00:00
|
|
|
Name(options.PodName).
|
|
|
|
Namespace(options.Namespace).
|
2016-07-24 05:04:09 +00:00
|
|
|
SubResource("exec").
|
2016-11-11 19:55:37 +00:00
|
|
|
Param("container", options.ContainerName)
|
2016-11-18 20:55:17 +00:00
|
|
|
req.VersionedParams(&v1.PodExecOptions{
|
2016-11-11 19:55:37 +00:00
|
|
|
Container: options.ContainerName,
|
|
|
|
Command: options.Command,
|
|
|
|
Stdin: options.Stdin != nil,
|
|
|
|
Stdout: options.CaptureStdout,
|
|
|
|
Stderr: options.CaptureStderr,
|
2016-07-24 05:04:09 +00:00
|
|
|
TTY: tty,
|
|
|
|
}, api.ParameterCodec)
|
|
|
|
|
2016-11-11 19:55:37 +00:00
|
|
|
var stdout, stderr bytes.Buffer
|
|
|
|
err = execute("POST", req.URL(), config, options.Stdin, &stdout, &stderr, tty)
|
|
|
|
|
|
|
|
if options.PreserveWhitespace {
|
|
|
|
return stdout.String(), stderr.String(), err
|
|
|
|
}
|
2016-10-04 21:04:13 +00:00
|
|
|
return strings.TrimSpace(stdout.String()), strings.TrimSpace(stderr.String()), err
|
2016-07-24 05:04:09 +00:00
|
|
|
}
|
|
|
|
|
2016-11-11 19:55:37 +00:00
|
|
|
// ExecCommandInContainerWithFullOutput executes a command in the
|
|
|
|
// specified container and return stdout, stderr and error
|
|
|
|
func (f *Framework) ExecCommandInContainerWithFullOutput(podName, containerName string, cmd ...string) (string, string, error) {
|
|
|
|
return f.ExecWithOptions(ExecOptions{
|
|
|
|
Command: cmd,
|
|
|
|
Namespace: f.Namespace.Name,
|
|
|
|
PodName: podName,
|
|
|
|
ContainerName: containerName,
|
|
|
|
|
|
|
|
Stdin: nil,
|
|
|
|
CaptureStdout: true,
|
|
|
|
CaptureStderr: true,
|
|
|
|
PreserveWhitespace: false,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
// ExecCommandInContainer executes a command in the specified container.
|
|
|
|
func (f *Framework) ExecCommandInContainer(podName, containerName string, cmd ...string) string {
|
|
|
|
stdout, stderr, err := f.ExecCommandInContainerWithFullOutput(podName, containerName, cmd...)
|
|
|
|
Logf("Exec stderr: %q", stderr)
|
|
|
|
Expect(err).NotTo(HaveOccurred(),
|
|
|
|
"failed to execute command in pod %v, container %v: %v",
|
|
|
|
podName, containerName, err)
|
|
|
|
return stdout
|
|
|
|
}
|
|
|
|
|
2016-07-24 05:04:09 +00:00
|
|
|
func (f *Framework) ExecShellInContainer(podName, containerName string, cmd string) string {
|
|
|
|
return f.ExecCommandInContainer(podName, containerName, "/bin/sh", "-c", cmd)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (f *Framework) ExecCommandInPod(podName string, cmd ...string) string {
|
2016-12-07 14:40:26 +00:00
|
|
|
pod, err := f.PodClient().Get(podName, metav1.GetOptions{})
|
2016-07-24 05:04:09 +00:00
|
|
|
Expect(err).NotTo(HaveOccurred(), "failed to get pod")
|
|
|
|
Expect(pod.Spec.Containers).NotTo(BeEmpty())
|
|
|
|
return f.ExecCommandInContainer(podName, pod.Spec.Containers[0].Name, cmd...)
|
|
|
|
}
|
|
|
|
|
2016-10-04 21:04:13 +00:00
|
|
|
func (f *Framework) ExecCommandInPodWithFullOutput(podName string, cmd ...string) (string, string, error) {
|
2016-12-07 14:40:26 +00:00
|
|
|
pod, err := f.PodClient().Get(podName, metav1.GetOptions{})
|
2016-10-04 21:04:13 +00:00
|
|
|
Expect(err).NotTo(HaveOccurred(), "failed to get pod")
|
|
|
|
Expect(pod.Spec.Containers).NotTo(BeEmpty())
|
|
|
|
return f.ExecCommandInContainerWithFullOutput(podName, pod.Spec.Containers[0].Name, cmd...)
|
|
|
|
}
|
|
|
|
|
2016-07-24 05:04:09 +00:00
|
|
|
func (f *Framework) ExecShellInPod(podName string, cmd string) string {
|
|
|
|
return f.ExecCommandInPod(podName, "/bin/sh", "-c", cmd)
|
|
|
|
}
|
|
|
|
|
2016-10-04 21:04:13 +00:00
|
|
|
func (f *Framework) ExecShellInPodWithFullOutput(podName string, cmd string) (string, string, error) {
|
|
|
|
return f.ExecCommandInPodWithFullOutput(podName, "/bin/sh", "-c", cmd)
|
|
|
|
}
|
|
|
|
|
2016-07-24 05:04:09 +00:00
|
|
|
func execute(method string, url *url.URL, config *restclient.Config, stdin io.Reader, stdout, stderr io.Writer, tty bool) error {
|
|
|
|
exec, err := remotecommand.NewExecutor(config, method, url)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return exec.Stream(remotecommand.StreamOptions{
|
2017-02-15 10:34:49 +00:00
|
|
|
SupportedProtocols: remocommandconsts.SupportedStreamingProtocols,
|
2016-07-24 05:04:09 +00:00
|
|
|
Stdin: stdin,
|
|
|
|
Stdout: stdout,
|
|
|
|
Stderr: stderr,
|
|
|
|
Tty: tty,
|
|
|
|
})
|
|
|
|
}
|