mirror of
https://github.com/k3s-io/kubernetes.git
synced 2025-07-20 02:11:09 +00:00
This is the result of automatically editing source files like this: go install golang.org/x/tools/cmd/goimports@latest find ./test/e2e* -name "*.go" | xargs env PATH=$GOPATH/bin:$PATH ./e2e-framework-sed.sh with e2e-framework-sed.sh containing this: sed -i \ -e "s/\(f\|fr\|\w\w*\.[fF]\w*\)\.ExecCommandInContainer(/e2epod.ExecCommandInContainer(\1, /" \ -e "s/\(f\|fr\|\w\w*\.[fF]\w*\)\.ExecCommandInContainerWithFullOutput(/e2epod.ExecCommandInContainerWithFullOutput(\1, /" \ -e "s/\(f\|fr\|\w\w*\.[fF]\w*\)\.ExecShellInContainer(/e2epod.ExecShellInContainer(\1, /" \ -e "s/\(f\|fr\|\w\w*\.[fF]\w*\)\.ExecShellInPod(/e2epod.ExecShellInPod(\1, /" \ -e "s/\(f\|fr\|\w\w*\.[fF]\w*\)\.ExecShellInPodWithFullOutput(/e2epod.ExecShellInPodWithFullOutput(\1, /" \ -e "s/\(f\|fr\|\w\w*\.[fF]\w*\)\.ExecWithOptions(/e2epod.ExecWithOptions(\1, /" \ -e "s/\(f\|fr\|\w\w*\.[fF]\w*\)\.MatchContainerOutput(/e2eoutput.MatchContainerOutput(\1, /" \ -e "s/\(f\|fr\|\w\w*\.[fF]\w*\)\.PodClient(/e2epod.NewPodClient(\1, /" \ -e "s/\(f\|fr\|\w\w*\.[fF]\w*\)\.PodClientNS(/e2epod.PodClientNS(\1, /" \ -e "s/\(f\|fr\|\w\w*\.[fF]\w*\)\.TestContainerOutput(/e2eoutput.TestContainerOutput(\1, /" \ -e "s/\(f\|fr\|\w\w*\.[fF]\w*\)\.TestContainerOutputRegexp(/e2eoutput.TestContainerOutputRegexp(\1, /" \ -e "s/framework.AddOrUpdateLabelOnNode\b/e2enode.AddOrUpdateLabelOnNode/" \ -e "s/framework.AllNodes\b/e2edebug.AllNodes/" \ -e "s/framework.AllNodesReady\b/e2enode.AllNodesReady/" \ -e "s/framework.ContainerResourceGatherer\b/e2edebug.ContainerResourceGatherer/" \ -e "s/framework.ContainerResourceUsage\b/e2edebug.ContainerResourceUsage/" \ -e "s/framework.CreateEmptyFileOnPod\b/e2eoutput.CreateEmptyFileOnPod/" \ -e "s/framework.DefaultPodDeletionTimeout\b/e2epod.DefaultPodDeletionTimeout/" \ -e "s/framework.DumpAllNamespaceInfo\b/e2edebug.DumpAllNamespaceInfo/" \ -e "s/framework.DumpDebugInfo\b/e2eoutput.DumpDebugInfo/" \ -e "s/framework.DumpNodeDebugInfo\b/e2edebug.DumpNodeDebugInfo/" \ -e "s/framework.EtcdUpgrade\b/e2eproviders.EtcdUpgrade/" \ -e "s/framework.EventsLister\b/e2edebug.EventsLister/" \ -e "s/framework.ExecOptions\b/e2epod.ExecOptions/" \ -e "s/framework.ExpectNodeHasLabel\b/e2enode.ExpectNodeHasLabel/" \ -e "s/framework.ExpectNodeHasTaint\b/e2enode.ExpectNodeHasTaint/" \ -e "s/framework.GCEUpgradeScript\b/e2eproviders.GCEUpgradeScript/" \ -e "s/framework.ImagePrePullList\b/e2epod.ImagePrePullList/" \ -e "s/framework.KubectlBuilder\b/e2ekubectl.KubectlBuilder/" \ -e "s/framework.LocationParamGKE\b/e2eproviders.LocationParamGKE/" \ -e "s/framework.LogSizeDataTimeseries\b/e2edebug.LogSizeDataTimeseries/" \ -e "s/framework.LogSizeGatherer\b/e2edebug.LogSizeGatherer/" \ -e "s/framework.LogsSizeData\b/e2edebug.LogsSizeData/" \ -e "s/framework.LogsSizeDataSummary\b/e2edebug.LogsSizeDataSummary/" \ -e "s/framework.LogsSizeVerifier\b/e2edebug.LogsSizeVerifier/" \ -e "s/framework.LookForStringInLog\b/e2eoutput.LookForStringInLog/" \ -e "s/framework.LookForStringInPodExec\b/e2eoutput.LookForStringInPodExec/" \ -e "s/framework.LookForStringInPodExecToContainer\b/e2eoutput.LookForStringInPodExecToContainer/" \ -e "s/framework.MasterAndDNSNodes\b/e2edebug.MasterAndDNSNodes/" \ -e "s/framework.MasterNodes\b/e2edebug.MasterNodes/" \ -e "s/framework.MasterUpgradeGKE\b/e2eproviders.MasterUpgradeGKE/" \ -e "s/framework.NewKubectlCommand\b/e2ekubectl.NewKubectlCommand/" \ -e "s/framework.NewLogsVerifier\b/e2edebug.NewLogsVerifier/" \ -e "s/framework.NewNodeKiller\b/e2enode.NewNodeKiller/" \ -e "s/framework.NewResourceUsageGatherer\b/e2edebug.NewResourceUsageGatherer/" \ -e "s/framework.NodeHasTaint\b/e2enode.NodeHasTaint/" \ -e "s/framework.NodeKiller\b/e2enode.NodeKiller/" \ -e "s/framework.NodesSet\b/e2edebug.NodesSet/" \ -e "s/framework.PodClient\b/e2epod.PodClient/" \ -e "s/framework.RemoveLabelOffNode\b/e2enode.RemoveLabelOffNode/" \ -e "s/framework.ResourceConstraint\b/e2edebug.ResourceConstraint/" \ -e "s/framework.ResourceGathererOptions\b/e2edebug.ResourceGathererOptions/" \ -e "s/framework.ResourceUsagePerContainer\b/e2edebug.ResourceUsagePerContainer/" \ -e "s/framework.ResourceUsageSummary\b/e2edebug.ResourceUsageSummary/" \ -e "s/framework.RunHostCmd\b/e2eoutput.RunHostCmd/" \ -e "s/framework.RunHostCmdOrDie\b/e2eoutput.RunHostCmdOrDie/" \ -e "s/framework.RunHostCmdWithFullOutput\b/e2eoutput.RunHostCmdWithFullOutput/" \ -e "s/framework.RunHostCmdWithRetries\b/e2eoutput.RunHostCmdWithRetries/" \ -e "s/framework.RunKubectl\b/e2ekubectl.RunKubectl/" \ -e "s/framework.RunKubectlInput\b/e2ekubectl.RunKubectlInput/" \ -e "s/framework.RunKubectlOrDie\b/e2ekubectl.RunKubectlOrDie/" \ -e "s/framework.RunKubectlOrDieInput\b/e2ekubectl.RunKubectlOrDieInput/" \ -e "s/framework.RunKubectlWithFullOutput\b/e2ekubectl.RunKubectlWithFullOutput/" \ -e "s/framework.RunKubemciCmd\b/e2ekubectl.RunKubemciCmd/" \ -e "s/framework.RunKubemciWithKubeconfig\b/e2ekubectl.RunKubemciWithKubeconfig/" \ -e "s/framework.SingleContainerSummary\b/e2edebug.SingleContainerSummary/" \ -e "s/framework.SingleLogSummary\b/e2edebug.SingleLogSummary/" \ -e "s/framework.TimestampedSize\b/e2edebug.TimestampedSize/" \ -e "s/framework.WaitForAllNodesSchedulable\b/e2enode.WaitForAllNodesSchedulable/" \ -e "s/framework.WaitForSSHTunnels\b/e2enode.WaitForSSHTunnels/" \ -e "s/framework.WorkItem\b/e2edebug.WorkItem/" \ "$@" for i in "$@"; do # Import all sub packages and let goimports figure out which of those # are redundant (= already imported) or not needed. sed -i -e '/"k8s.io.kubernetes.test.e2e.framework"/a e2edebug "k8s.io/kubernetes/test/e2e/framework/debug"' "$i" sed -i -e '/"k8s.io.kubernetes.test.e2e.framework"/a e2ekubectl "k8s.io/kubernetes/test/e2e/framework/kubectl"' "$i" sed -i -e '/"k8s.io.kubernetes.test.e2e.framework"/a e2enode "k8s.io/kubernetes/test/e2e/framework/node"' "$i" sed -i -e '/"k8s.io.kubernetes.test.e2e.framework"/a e2eoutput "k8s.io/kubernetes/test/e2e/framework/pod/output"' "$i" sed -i -e '/"k8s.io.kubernetes.test.e2e.framework"/a e2epod "k8s.io/kubernetes/test/e2e/framework/pod"' "$i" sed -i -e '/"k8s.io.kubernetes.test.e2e.framework"/a e2eproviders "k8s.io/kubernetes/test/e2e/framework/providers"' "$i" goimports -w "$i" done
310 lines
11 KiB
Go
310 lines
11 KiB
Go
/*
|
|
Copyright 2017 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 monitoring
|
|
|
|
import (
|
|
"fmt"
|
|
"os/exec"
|
|
"strings"
|
|
|
|
appsv1 "k8s.io/api/apps/v1"
|
|
v1 "k8s.io/api/core/v1"
|
|
rbacv1 "k8s.io/api/rbac/v1"
|
|
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
|
|
"k8s.io/kubernetes/test/e2e/framework"
|
|
e2edeployment "k8s.io/kubernetes/test/e2e/framework/deployment"
|
|
e2ekubectl "k8s.io/kubernetes/test/e2e/framework/kubectl"
|
|
imageutils "k8s.io/kubernetes/test/utils/image"
|
|
|
|
gcm "google.golang.org/api/monitoring/v3"
|
|
)
|
|
|
|
var (
|
|
// CustomMetricName is the metrics name used in test cases.
|
|
CustomMetricName = "foo"
|
|
// UnusedMetricName is the unused metrics name used in test cases.
|
|
UnusedMetricName = "unused"
|
|
// CustomMetricValue is the value for CustomMetricName.
|
|
CustomMetricValue = int64(448)
|
|
// UnusedMetricValue is the value for UnusedMetricName.
|
|
UnusedMetricValue = int64(446)
|
|
// StackdriverExporter is exporter name.
|
|
StackdriverExporter = "stackdriver-exporter"
|
|
// HPAPermissions is a ClusterRoleBinding that grants unauthenticated user permissions granted for
|
|
// HPA for testing purposes, i.e. it should grant permission to read custom metrics.
|
|
HPAPermissions = &rbacv1.ClusterRoleBinding{
|
|
ObjectMeta: metav1.ObjectMeta{
|
|
Name: "custom-metrics-reader",
|
|
},
|
|
RoleRef: rbacv1.RoleRef{
|
|
APIGroup: "rbac.authorization.k8s.io",
|
|
Kind: "ClusterRole",
|
|
Name: "system:controller:horizontal-pod-autoscaler",
|
|
},
|
|
Subjects: []rbacv1.Subject{
|
|
{
|
|
APIGroup: "rbac.authorization.k8s.io",
|
|
Kind: "Group",
|
|
Name: "system:unauthenticated",
|
|
},
|
|
},
|
|
}
|
|
// StagingDeploymentsLocation is the location where the adapter deployment files are stored.
|
|
StagingDeploymentsLocation = "https://raw.githubusercontent.com/GoogleCloudPlatform/k8s-stackdriver/master/custom-metrics-stackdriver-adapter/deploy/staging/"
|
|
// AdapterForOldResourceModel is file name for the old resource model.
|
|
AdapterForOldResourceModel = "adapter_old_resource_model.yaml"
|
|
// AdapterForNewResourceModel is file name for the new resource model.
|
|
AdapterForNewResourceModel = "adapter_new_resource_model.yaml"
|
|
// AdapterDefault is the default model.
|
|
AdapterDefault = AdapterForOldResourceModel
|
|
// ClusterAdminBinding is the cluster rolebinding name for test cases.
|
|
ClusterAdminBinding = "e2e-test-cluster-admin-binding"
|
|
)
|
|
|
|
// CustomMetricContainerSpec allows to specify a config for StackdriverExporterDeployment
|
|
// with multiple containers exporting different metrics.
|
|
type CustomMetricContainerSpec struct {
|
|
Name string
|
|
MetricName string
|
|
MetricValue int64
|
|
}
|
|
|
|
// SimpleStackdriverExporterDeployment is a Deployment of simple application that exports a metric of
|
|
// fixed value to Stackdriver in a loop.
|
|
func SimpleStackdriverExporterDeployment(name, namespace string, replicas int32, metricValue int64) *appsv1.Deployment {
|
|
return StackdriverExporterDeployment(name, namespace, replicas,
|
|
[]CustomMetricContainerSpec{
|
|
{
|
|
Name: StackdriverExporter,
|
|
MetricName: CustomMetricName,
|
|
MetricValue: metricValue,
|
|
},
|
|
})
|
|
}
|
|
|
|
// StackdriverExporterDeployment is a Deployment of an application that can expose
|
|
// an arbitrary amount of metrics of fixed value to Stackdriver in a loop. Each metric
|
|
// is exposed by a different container in one pod.
|
|
// The metric names and values are configured via the containers parameter.
|
|
func StackdriverExporterDeployment(name, namespace string, replicas int32, containers []CustomMetricContainerSpec) *appsv1.Deployment {
|
|
podSpec := v1.PodSpec{Containers: []v1.Container{}}
|
|
for _, containerSpec := range containers {
|
|
podSpec.Containers = append(podSpec.Containers, stackdriverExporterContainerSpec(containerSpec.Name, namespace, containerSpec.MetricName, containerSpec.MetricValue))
|
|
}
|
|
|
|
d := e2edeployment.NewDeployment(name, replicas, map[string]string{"name": name}, "", "", appsv1.RollingUpdateDeploymentStrategyType)
|
|
d.ObjectMeta.Namespace = namespace
|
|
d.Spec.Template.Spec = podSpec
|
|
return d
|
|
}
|
|
|
|
// StackdriverExporterPod is a Pod of simple application that exports a metric of fixed value to
|
|
// Stackdriver in a loop.
|
|
func StackdriverExporterPod(podName, namespace, podLabel, metricName string, metricValue int64) *v1.Pod {
|
|
return &v1.Pod{
|
|
ObjectMeta: metav1.ObjectMeta{
|
|
Name: podName,
|
|
Namespace: namespace,
|
|
Labels: map[string]string{
|
|
"name": podLabel,
|
|
},
|
|
},
|
|
Spec: v1.PodSpec{
|
|
Containers: []v1.Container{stackdriverExporterContainerSpec(StackdriverExporter, namespace, metricName, metricValue)},
|
|
},
|
|
}
|
|
}
|
|
|
|
func stackdriverExporterContainerSpec(name string, namespace string, metricName string, metricValue int64) v1.Container {
|
|
return v1.Container{
|
|
Name: name,
|
|
Image: imageutils.GetE2EImage(imageutils.SdDummyExporter),
|
|
ImagePullPolicy: v1.PullPolicy("Always"),
|
|
Command: []string{
|
|
"/bin/sh",
|
|
"-c",
|
|
strings.Join([]string{
|
|
"./sd_dummy_exporter",
|
|
"--pod-id=$(POD_ID)",
|
|
"--pod-name=$(POD_NAME)",
|
|
"--namespace=" + namespace,
|
|
"--metric-name=" + metricName,
|
|
fmt.Sprintf("--metric-value=%v", metricValue),
|
|
"--use-old-resource-model",
|
|
"--use-new-resource-model",
|
|
}, " "),
|
|
},
|
|
Env: []v1.EnvVar{
|
|
{
|
|
Name: "POD_ID",
|
|
ValueFrom: &v1.EnvVarSource{
|
|
FieldRef: &v1.ObjectFieldSelector{
|
|
FieldPath: "metadata.uid",
|
|
},
|
|
},
|
|
},
|
|
{
|
|
Name: "POD_NAME",
|
|
ValueFrom: &v1.EnvVarSource{
|
|
FieldRef: &v1.ObjectFieldSelector{
|
|
FieldPath: "metadata.name",
|
|
},
|
|
},
|
|
},
|
|
},
|
|
Ports: []v1.ContainerPort{{ContainerPort: 80}},
|
|
}
|
|
}
|
|
|
|
// PrometheusExporterDeployment is a Deployment of simple application with two containers
|
|
// one exposing a metric in prometheus format and second a prometheus-to-sd container
|
|
// that scrapes the metric and pushes it to stackdriver.
|
|
func PrometheusExporterDeployment(name, namespace string, replicas int32, metricValue int64) *appsv1.Deployment {
|
|
d := e2edeployment.NewDeployment(name, replicas, map[string]string{"name": name}, "", "", appsv1.RollingUpdateDeploymentStrategyType)
|
|
d.ObjectMeta.Namespace = namespace
|
|
d.Spec.Template.Spec = prometheusExporterPodSpec(CustomMetricName, metricValue, 8080)
|
|
return d
|
|
}
|
|
|
|
func prometheusExporterPodSpec(metricName string, metricValue int64, port int32) v1.PodSpec {
|
|
return v1.PodSpec{
|
|
Containers: []v1.Container{
|
|
{
|
|
Name: "prometheus-exporter",
|
|
Image: imageutils.GetE2EImage(imageutils.PrometheusDummyExporter),
|
|
ImagePullPolicy: v1.PullPolicy("Always"),
|
|
Command: []string{"/prometheus_dummy_exporter", "--metric-name=" + metricName,
|
|
fmt.Sprintf("--metric-value=%v", metricValue), fmt.Sprintf("=--port=%d", port)},
|
|
Ports: []v1.ContainerPort{{ContainerPort: port}},
|
|
},
|
|
{
|
|
Name: "prometheus-to-sd",
|
|
Image: imageutils.GetE2EImage(imageutils.PrometheusToSd),
|
|
ImagePullPolicy: v1.PullPolicy("Always"),
|
|
Command: []string{"/monitor", fmt.Sprintf("--source=:http://localhost:%d", port),
|
|
"--stackdriver-prefix=custom.googleapis.com", "--pod-id=$(POD_ID)", "--namespace-id=$(POD_NAMESPACE)"},
|
|
Env: []v1.EnvVar{
|
|
{
|
|
Name: "POD_ID",
|
|
ValueFrom: &v1.EnvVarSource{
|
|
FieldRef: &v1.ObjectFieldSelector{
|
|
FieldPath: "metadata.uid",
|
|
},
|
|
},
|
|
},
|
|
{
|
|
Name: "POD_NAMESPACE",
|
|
ValueFrom: &v1.EnvVarSource{
|
|
FieldRef: &v1.ObjectFieldSelector{
|
|
FieldPath: "metadata.namespace",
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
}
|
|
}
|
|
|
|
// CreateAdapter creates Custom Metrics - Stackdriver adapter
|
|
// adapterDeploymentFile should be a filename for adapter deployment located in StagingDeploymentLocation
|
|
func CreateAdapter(adapterDeploymentFile string) error {
|
|
// A workaround to make the work on GKE. GKE doesn't normally allow to create cluster roles,
|
|
// which the adapter deployment does. The solution is to create cluster role binding for
|
|
// cluster-admin role and currently used service account.
|
|
err := createClusterAdminBinding()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
adapterURL := StagingDeploymentsLocation + adapterDeploymentFile
|
|
err = exec.Command("wget", adapterURL).Run()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
stat, err := e2ekubectl.RunKubectl("", "apply", "-f", adapterURL)
|
|
framework.Logf(stat)
|
|
return err
|
|
}
|
|
|
|
func createClusterAdminBinding() error {
|
|
stdout, stderr, err := framework.RunCmd("gcloud", "config", "get-value", "core/account")
|
|
if err != nil {
|
|
framework.Logf(stderr)
|
|
return err
|
|
}
|
|
serviceAccount := strings.TrimSpace(stdout)
|
|
framework.Logf("current service account: %q", serviceAccount)
|
|
stat, err := e2ekubectl.RunKubectl("", "create", "clusterrolebinding", ClusterAdminBinding, "--clusterrole=cluster-admin", "--user="+serviceAccount)
|
|
framework.Logf(stat)
|
|
return err
|
|
}
|
|
|
|
// CreateDescriptors creates descriptors for metrics: CustomMetricName and UnusedMetricName.
|
|
func CreateDescriptors(service *gcm.Service, projectID string) error {
|
|
_, err := service.Projects.MetricDescriptors.Create(fmt.Sprintf("projects/%s", projectID), &gcm.MetricDescriptor{
|
|
Name: CustomMetricName,
|
|
ValueType: "INT64",
|
|
Type: "custom.googleapis.com/" + CustomMetricName,
|
|
MetricKind: "GAUGE",
|
|
}).Do()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
_, err = service.Projects.MetricDescriptors.Create(fmt.Sprintf("projects/%s", projectID), &gcm.MetricDescriptor{
|
|
Name: UnusedMetricName,
|
|
ValueType: "INT64",
|
|
Type: "custom.googleapis.com/" + UnusedMetricName,
|
|
MetricKind: "GAUGE",
|
|
}).Do()
|
|
return err
|
|
}
|
|
|
|
// CleanupDescriptors deletes descriptors for metrics: CustomMetricName and UnusedMetricName.
|
|
// TODO: Cleanup time series as well
|
|
func CleanupDescriptors(service *gcm.Service, projectID string) {
|
|
_, err := service.Projects.MetricDescriptors.Delete(fmt.Sprintf("projects/%s/metricDescriptors/custom.googleapis.com/%s", projectID, CustomMetricName)).Do()
|
|
if err != nil {
|
|
framework.Logf("Failed to delete descriptor for metric '%s': %v", CustomMetricName, err)
|
|
}
|
|
_, err = service.Projects.MetricDescriptors.Delete(fmt.Sprintf("projects/%s/metricDescriptors/custom.googleapis.com/%s", projectID, UnusedMetricName)).Do()
|
|
if err != nil {
|
|
framework.Logf("Failed to delete descriptor for metric '%s': %v", CustomMetricName, err)
|
|
}
|
|
}
|
|
|
|
// CleanupAdapter deletes Custom Metrics - Stackdriver adapter deployments.
|
|
func CleanupAdapter(adapterDeploymentFile string) {
|
|
stat, err := e2ekubectl.RunKubectl("", "delete", "-f", adapterDeploymentFile)
|
|
framework.Logf(stat)
|
|
if err != nil {
|
|
framework.Logf("Failed to delete adapter deployments: %s", err)
|
|
}
|
|
err = exec.Command("rm", adapterDeploymentFile).Run()
|
|
if err != nil {
|
|
framework.Logf("Failed to delete adapter deployment file: %s", err)
|
|
}
|
|
cleanupClusterAdminBinding()
|
|
}
|
|
|
|
func cleanupClusterAdminBinding() {
|
|
stat, err := e2ekubectl.RunKubectl("", "delete", "clusterrolebinding", ClusterAdminBinding)
|
|
framework.Logf(stat)
|
|
if err != nil {
|
|
framework.Logf("Failed to delete cluster admin binding: %s", err)
|
|
}
|
|
}
|