Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[receiver/k8scluster] add support for observing resources for a specific namespace #35727

Merged
Merged
Show file tree
Hide file tree
Changes from 19 commits
Commits
Show all changes
50 commits
Select commit Hold shift + click to select a range
f259d1d
replace deprecated fake client initialisation
bacherfl Oct 9, 2024
981f1c2
add e2e test scenario for namespaced k8s cluster receiver
bacherfl Oct 10, 2024
bfc23d8
remove non-fitting resources from namespaced e2e test
bacherfl Oct 10, 2024
5b9c6ac
add namespace to role binding
bacherfl Oct 10, 2024
46d11fd
adapt test expectations
bacherfl Oct 10, 2024
dc3b2f3
revert accidental change
bacherfl Oct 10, 2024
1b53fa0
omit cluster scoped observers if namespace filter is set
bacherfl Oct 10, 2024
e18f4e5
[revert when done] - adapt test setup to run locally
bacherfl Oct 16, 2024
873b489
Merge branch 'main' into feat/9401/namespaced-cluster-receiver
bacherfl Oct 17, 2024
fd62f43
Merge branch 'feat/9401/namespaced-cluster-receiver' of https://githu…
bacherfl Oct 17, 2024
a2990f3
fix informer setup
bacherfl Oct 17, 2024
c8b1f29
update expected metrics
bacherfl Oct 17, 2024
f94b075
do not try to observe cluster resource quotas when namespace limit ha…
bacherfl Oct 17, 2024
b0710fb
extend readme and add changelog entry
bacherfl Oct 17, 2024
3fa57f5
add resourcequota access to role
bacherfl Oct 17, 2024
6f751bc
add check for mutually exclusive options in config validation
bacherfl Oct 21, 2024
680aaad
Merge branch 'main' into feat/9401/namespaced-cluster-receiver
bacherfl Oct 22, 2024
aebbbc4
remove namespace check from validation and log hint about non-observa…
bacherfl Oct 22, 2024
90ac34b
Merge branch 'main' into feat/9401/namespaced-cluster-receiver
bacherfl Oct 22, 2024
7fecd06
remove sidecar deployment recommendation
bacherfl Oct 29, 2024
e2c43ac
Merge branch 'main' into feat/9401/namespaced-cluster-receiver
bacherfl Nov 5, 2024
2070e2a
Merge branch 'main' into feat/9401/namespaced-cluster-receiver
bacherfl Nov 6, 2024
4ef2cad
fix merge conflicts
bacherfl Nov 6, 2024
28862e6
fix path to testobjects directoy
bacherfl Nov 6, 2024
582f06c
fix test cleanup
bacherfl Nov 6, 2024
9916d6f
increase number of expected metrics before continuing with assertions
bacherfl Nov 6, 2024
4251eec
adapt expected.yaml to account for newly added k8s objects
bacherfl Nov 6, 2024
db9746f
adapt assertions
bacherfl Nov 6, 2024
1968099
use separate namespace for namespace-scoped test
bacherfl Nov 6, 2024
fcbebe3
fix path to testobjects directory
bacherfl Nov 6, 2024
a4f9130
ensure namespace is created before applying other tests
bacherfl Nov 6, 2024
19f947f
fix namespace in confmap
bacherfl Nov 6, 2024
dca10d1
adapt check for required number of metrics
bacherfl Nov 6, 2024
742c663
fix test expectations
bacherfl Nov 7, 2024
75c8b0e
trigger CI
bacherfl Nov 7, 2024
bc00b77
increase timeout and ignore k8s.hpa.current_replicas metric value
bacherfl Nov 7, 2024
1defed9
add debug logs
bacherfl Nov 7, 2024
4bb2346
reduce job sleep time
bacherfl Nov 7, 2024
6315edd
adapt expections
bacherfl Nov 7, 2024
14aa5c6
adapt expections
bacherfl Nov 7, 2024
10e9267
remove debug logs that are hopefully not needed anymore
bacherfl Nov 7, 2024
ca738c1
Merge branch 'main' into feat/9401/namespaced-cluster-receiver
bacherfl Nov 8, 2024
c1810ec
revert changes to test logic
bacherfl Nov 8, 2024
4a673dc
extract common helper functions
bacherfl Nov 8, 2024
86eeb7e
re-add comment for generating golden file
bacherfl Nov 8, 2024
d5cb4d2
Merge branch 'main' into feat/9401/namespaced-cluster-receiver
bacherfl Nov 8, 2024
f75b5c5
Merge branch 'main' into feat/9401/namespaced-cluster-receiver
bacherfl Nov 11, 2024
e7ed86b
Merge branch 'main' into feat/9401/namespaced-cluster-receiver
bacherfl Nov 18, 2024
3a9d281
Merge branch 'main' into feat/9401/namespaced-cluster-receiver
TylerHelmuth Nov 18, 2024
e2d8720
make generate
evan-bradley Nov 18, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
27 changes: 27 additions & 0 deletions .chloggen/k8sclusterreceiver-namespaced.yaml
Original file line number Diff line number Diff line change
@@ -0,0 +1,27 @@
# Use this changelog template to create an entry for release notes.

# One of 'breaking', 'deprecation', 'new_component', 'enhancement', 'bug_fix'
change_type: enhancement

# The name of the component, or a single word describing the area of concern, (e.g. filelogreceiver)
component: k8sclusterreceiver

# A brief description of the change. Surround your text with quotes ("") if it needs to start with a backtick (`).
note: Add support for limiting observed resources to a specific namespace.

# Mandatory: One or more tracking issues related to the change. You can use the PR number here if no issue exists.
issues: [9401]

# (Optional) One or more lines of additional information to render under the primary note.
# These lines will be padded with 2 spaces and then inserted directly into the document.
# Use pipe (|) for multiline entries.
subtext: This change allows to make use of this receiver with `Roles`/`RoleBindings`, as opposed to giving the collector cluster-wide read access.

# If your change doesn't affect end users or the exported elements of any package,
# you should instead start your pull request title with [chore] or use the "Skip Changelog" label.
# Optional: The change log or logs in which this entry should be included.
# e.g. '[user]' or '[user, api]'
# Include 'user' if the change is relevant to end users.
# Include 'api' if there is a change to a library API.
# Default: '[user]'
change_logs: []
92 changes: 92 additions & 0 deletions receiver/k8sclusterreceiver/README.md
Original file line number Diff line number Diff line change
Expand Up @@ -64,6 +64,7 @@ The following allocatable resource types are available.
- storage
- `metrics`: Allows to enable/disable metrics.
- `resource_attributes`: Allows to enable/disable resource attributes.
- `namespace`: Allows to observe resources for a particular namespace only. If this option is set to a non-empty string, `Nodes`, `Namespaces` and `ClusterResourceQuotas` will not be observed.

Example:

Expand Down Expand Up @@ -275,6 +276,97 @@ subjects:
EOF
```

As an alternative to setting up a `ClusterRole`/`ClusterRoleBinding`, it is also possible to limit the observed resources to a
particular namespace by setting the `namespace` option of the receiver. This allows the collector to only rely on `Roles`/`RoleBindings`,
instead of granting the collector cluster-wide read access to resources.
Note however, that in this case the following resources will not be observed by the `k8sclusterreceiver`:

- `Nodes`
- `Namespaces`
- `ClusterResourceQuotas`

To use this approach, use the commands below to create the required `Role` and `RoleBinding`:

```bash
<<EOF | kubectl apply -f -
metadata:
name: otelcontribcol
labels:
app: otelcontribcol
namespace: default
rules:
- apiGroups:
- ""
resources:
- events
- pods
- pods/status
- replicationcontrollers
- replicationcontrollers/status
- services
verbs:
- get
- list
- watch
- apiGroups:
- apps
resources:
- daemonsets
- deployments
- replicasets
- statefulsets
verbs:
- get
- list
- watch
- apiGroups:
- extensions
resources:
- daemonsets
- deployments
- replicasets
verbs:
- get
- list
- watch
- apiGroups:
- batch
resources:
- jobs
- cronjobs
verbs:
- get
- list
- watch
- apiGroups:
- autoscaling
resources:
- horizontalpodautoscalers
verbs:
- get
- list
- watch
EOF
```

```bash
<<EOF | kubectl apply -f -
apiVersion: rbac.authorization.k8s.io/v1
kind: RoleBinding
metadata:
name: otelcontribcol
namespace: default
roleRef:
apiGroup: rbac.authorization.k8s.io
kind: Role
name: otelcontribcol
subjects:
- kind: ServiceAccount
name: otelcontribcol
namespace: default
EOF
```

### Deployment

Create a [Deployment](https://kubernetes.io/docs/concepts/workloads/controllers/deployment/) to deploy the collector.
Expand Down
6 changes: 6 additions & 0 deletions receiver/k8sclusterreceiver/config.go
Original file line number Diff line number Diff line change
Expand Up @@ -39,6 +39,11 @@ type Config struct {

// MetricsBuilderConfig allows customizing scraped metrics/attributes representation.
metadata.MetricsBuilderConfig `mapstructure:",squash"`

// Namespace to fetch resources from. If this is set, certain cluster-wide resources such as Nodes or Namespaces
// will not be able to be observed. Setting this option is recommended for sidecar deployment patterns where the collector runs
// as a sidecar for a pod, or in environments where due to security restrictions the collector can not be granted cluster-wide permissions.
ChrsMark marked this conversation as resolved.
Show resolved Hide resolved
Namespace string `mapstructure:"namespace"`
ChrsMark marked this conversation as resolved.
Show resolved Hide resolved
}

func (cfg *Config) Validate() error {
Expand All @@ -48,5 +53,6 @@ func (cfg *Config) Validate() error {
default:
return fmt.Errorf("\"%s\" is not a supported distribution. Must be one of: \"openshift\", \"kubernetes\"", cfg.Distribution)
}

return nil
}
76 changes: 72 additions & 4 deletions receiver/k8sclusterreceiver/e2e_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -28,17 +28,17 @@ import (

const testKubeConfig = "/tmp/kube-config-otelcol-e2e-testing"

// TestE2E tests the k8s cluster receiver with a real k8s cluster.
// TestE2EClusterScoped tests the k8s cluster receiver with a real k8s cluster.
// The test requires a prebuilt otelcontribcol image uploaded to a kind k8s cluster defined in
// `/tmp/kube-config-otelcol-e2e-testing`. Run the following command prior to running the test locally:
//
// kind create cluster --kubeconfig=/tmp/kube-config-otelcol-e2e-testing
// make docker-otelcontribcol
// KUBECONFIG=/tmp/kube-config-otelcol-e2e-testing kind load docker-image otelcontribcol:latest
func TestE2E(t *testing.T) {
func TestE2EClusterScoped(t *testing.T) {

var expected pmetric.Metrics
expectedFile := filepath.Join("testdata", "e2e", "expected.yaml")
expectedFile := filepath.Join("testdata", "e2e", "cluster-scoped", "expected.yaml")
expected, err := golden.ReadMetrics(expectedFile)
require.NoError(t, err)

Expand All @@ -50,7 +50,7 @@ func TestE2E(t *testing.T) {
defer shutdownSink()

testID := uuid.NewString()[:8]
collectorObjs := k8stest.CreateCollectorObjects(t, k8sClient, testID, "")
collectorObjs := k8stest.CreateCollectorObjects(t, k8sClient, testID, filepath.Join(".", "testdata", "e2e", "cluster-scoped", "collector"))

defer func() {
for _, obj := range append(collectorObjs) {
Expand Down Expand Up @@ -108,6 +108,74 @@ func TestE2E(t *testing.T) {
)
}

// TestE2ENamespaceScoped tests the k8s cluster receiver with a real k8s cluster.
// The test requires a prebuilt otelcontribcol image uploaded to a kind k8s cluster defined in
// `/tmp/kube-config-otelcol-e2e-testing`. Run the following command prior to running the test locally:
//
// kind create cluster --kubeconfig=/tmp/kube-config-otelcol-e2e-testing
// make docker-otelcontribcol
// KUBECONFIG=/tmp/kube-config-otelcol-e2e-testing kind load docker-image otelcontribcol:latest
func TestE2ENamespaceScoped(t *testing.T) {

var expected pmetric.Metrics
expectedFile := filepath.Join("testdata", "e2e", "namespace-scoped", "expected.yaml")
expected, err := golden.ReadMetrics(expectedFile)
require.NoError(t, err)

k8sClient, err := k8stest.NewK8sClient(testKubeConfig)
require.NoError(t, err)

metricsConsumer := new(consumertest.MetricsSink)
shutdownSink := startUpSink(t, metricsConsumer)
defer shutdownSink()

testID := uuid.NewString()[:8]
collectorObjs := k8stest.CreateCollectorObjects(t, k8sClient, testID, filepath.Join(".", "testdata", "e2e", "namespace-scoped", "collector"))

defer func() {
for _, obj := range append(collectorObjs) {
require.NoErrorf(t, k8stest.DeleteObject(k8sClient, obj), "failed to delete object %s", obj.GetName())
}
}()

wantEntries := 4 // Minimal number of metrics to wait for.
waitForData(t, wantEntries, metricsConsumer)

replaceWithStar := func(string) string { return "*" }
shortenNames := func(value string) string {
if strings.HasPrefix(value, "otelcol") {
return "otelcol"
}
return value
}
containerImageShorten := func(value string) string {
return value[(strings.LastIndex(value, "/") + 1):]
}
require.NoError(t, pmetrictest.CompareMetrics(expected, metricsConsumer.AllMetrics()[len(metricsConsumer.AllMetrics())-1],
pmetrictest.IgnoreTimestamp(),
pmetrictest.IgnoreStartTimestamp(),
pmetrictest.IgnoreMetricValues("k8s.deployment.desired", "k8s.deployment.available", "k8s.container.restarts", "k8s.container.cpu_request", "k8s.container.memory_request", "k8s.container.memory_limit"),
pmetrictest.ChangeResourceAttributeValue("k8s.deployment.name", shortenNames),
pmetrictest.ChangeResourceAttributeValue("k8s.pod.name", shortenNames),
pmetrictest.ChangeResourceAttributeValue("k8s.replicaset.name", shortenNames),
pmetrictest.ChangeResourceAttributeValue("k8s.deployment.uid", replaceWithStar),
pmetrictest.ChangeResourceAttributeValue("k8s.pod.uid", replaceWithStar),
pmetrictest.ChangeResourceAttributeValue("k8s.replicaset.uid", replaceWithStar),
pmetrictest.ChangeResourceAttributeValue("container.id", replaceWithStar),
pmetrictest.ChangeResourceAttributeValue("container.image.tag", replaceWithStar),
pmetrictest.ChangeResourceAttributeValue("k8s.node.uid", replaceWithStar),
pmetrictest.ChangeResourceAttributeValue("k8s.namespace.uid", replaceWithStar),
pmetrictest.ChangeResourceAttributeValue("k8s.daemonset.uid", replaceWithStar),
pmetrictest.ChangeResourceAttributeValue("container.image.name", containerImageShorten),
pmetrictest.IgnoreScopeVersion(),
pmetrictest.IgnoreResourceMetricsOrder(),
pmetrictest.IgnoreMetricsOrder(),
pmetrictest.IgnoreScopeMetricsOrder(),
pmetrictest.IgnoreMetricDataPointsOrder(),
),
)
}

func startUpSink(t *testing.T, mc *consumertest.MetricsSink) func() {
f := otlpreceiver.NewFactory()
cfg := f.CreateDefaultConfig().(*otlpreceiver.Config)
Expand Down
65 changes: 59 additions & 6 deletions receiver/k8sclusterreceiver/receiver_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -54,7 +54,7 @@ func TestReceiver(t *testing.T) {
osQuotaClient := fakeQuota.NewSimpleClientset()
sink := new(consumertest.MetricsSink)

r := setupReceiver(client, osQuotaClient, sink, nil, 10*time.Second, tt)
r := setupReceiver(client, osQuotaClient, sink, nil, 10*time.Second, tt, "")

// Setup k8s resources.
numPods := 2
Expand Down Expand Up @@ -93,6 +93,57 @@ func TestReceiver(t *testing.T) {
require.NoError(t, r.Shutdown(ctx))
}

func TestNamespacedReceiver(t *testing.T) {
tt, err := componenttest.SetupTelemetry(component.NewID(metadata.Type))
require.NoError(t, err)
defer func() {
require.NoError(t, tt.Shutdown(context.Background()))
}()

client := newFakeClientWithAllResources()
osQuotaClient := fakeQuota.NewSimpleClientset()
sink := new(consumertest.MetricsSink)

r := setupReceiver(client, osQuotaClient, sink, nil, 10*time.Second, tt, "test")

// Setup k8s resources.
numPods := 2
numNodes := 1
numQuotas := 2

createPods(t, client, numPods)
createNodes(t, client, numNodes)
createClusterQuota(t, osQuotaClient, numQuotas)

ctx := context.Background()
require.NoError(t, r.Start(ctx, newNopHost()))

// Expects metric data from pods only, where each metric data
// struct corresponds to one resource.
// Nodes and ClusterResourceQuotas should not be observed as these are non-namespaced resources
expectedNumMetrics := numPods
var initialDataPointCount int
require.Eventually(t, func() bool {
initialDataPointCount = sink.DataPointCount()
return initialDataPointCount == expectedNumMetrics
}, 10*time.Second, 100*time.Millisecond,
"metrics not collected")

numPodsToDelete := 1
deletePods(t, client, numPodsToDelete)

// Expects metric data from a node, since other resources were deleted.
expectedNumMetrics = numPods - numPodsToDelete
var metricsCountDelta int
require.Eventually(t, func() bool {
metricsCountDelta = sink.DataPointCount() - initialDataPointCount
return metricsCountDelta == expectedNumMetrics
}, 10*time.Second, 100*time.Millisecond,
"updated metrics not collected")

require.NoError(t, r.Shutdown(ctx))
}

func TestReceiverTimesOutAfterStartup(t *testing.T) {
tt, err := componenttest.SetupTelemetry(component.NewID(metadata.Type))
require.NoError(t, err)
Expand All @@ -102,7 +153,7 @@ func TestReceiverTimesOutAfterStartup(t *testing.T) {
client := newFakeClientWithAllResources()

// Mock initial cache sync timing out, using a small timeout.
r := setupReceiver(client, nil, consumertest.NewNop(), nil, 1*time.Millisecond, tt)
r := setupReceiver(client, nil, consumertest.NewNop(), nil, 1*time.Millisecond, tt, "")

createPods(t, client, 1)

Expand All @@ -125,7 +176,7 @@ func TestReceiverWithManyResources(t *testing.T) {
osQuotaClient := fakeQuota.NewSimpleClientset()
sink := new(consumertest.MetricsSink)

r := setupReceiver(client, osQuotaClient, sink, nil, 10*time.Second, tt)
r := setupReceiver(client, osQuotaClient, sink, nil, 10*time.Second, tt, "")

numPods := 1000
numQuotas := 2
Expand Down Expand Up @@ -165,7 +216,7 @@ func TestReceiverWithMetadata(t *testing.T) {

logsConsumer := new(consumertest.LogsSink)

r := setupReceiver(client, nil, metricsConsumer, logsConsumer, 10*time.Second, tt)
r := setupReceiver(client, nil, metricsConsumer, logsConsumer, 10*time.Second, tt, "")
r.config.MetadataExporters = []string{"nop/withmetadata"}

// Setup k8s resources.
Expand Down Expand Up @@ -225,7 +276,8 @@ func setupReceiver(
metricsConsumer consumer.Metrics,
logsConsumer consumer.Logs,
initialSyncTimeout time.Duration,
tt componenttest.TestTelemetry) *kubernetesReceiver {
tt componenttest.TestTelemetry,
namespace string) *kubernetesReceiver {

distribution := distributionKubernetes
if osQuotaClient != nil {
Expand All @@ -238,6 +290,7 @@ func setupReceiver(
AllocatableTypesToReport: []string{"cpu", "memory"},
Distribution: distribution,
MetricsBuilderConfig: metadata.DefaultMetricsBuilderConfig(),
Namespace: namespace,
}

r, _ := newReceiver(context.Background(), receiver.Settings{ID: component.NewID(metadata.Type), TelemetrySettings: tt.TelemetrySettings(), BuildInfo: component.NewDefaultBuildInfo()}, config)
Expand All @@ -255,7 +308,7 @@ func setupReceiver(
}

func newFakeClientWithAllResources() *fake.Clientset {
client := fake.NewSimpleClientset()
client := fake.NewClientset()
client.Resources = []*v1.APIResourceList{
{
GroupVersion: "v1",
Expand Down
Loading