// Copyright The OpenTelemetry Authors // SPDX-License-Identifier: Apache-2.0 // Keep the original Uber license. // Copyright (c) 2017 Uber Technologies, Inc. // // Permission is hereby granted, free of charge, to any person obtaining a copy // of this software and associated documentation files (the "Software"), to deal // in the Software without restriction, including without limitation the rights // to use, copy, modify, merge, publish, distribute, sublicense, and/or sell // copies of the Software, and to permit persons to whom the Software is // furnished to do so, subject to the following conditions: // // The above copyright notice and this permission notice shall be included in // all copies or substantial portions of the Software. // // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN // THE SOFTWARE. //go:build linux package cgroups import ( "path/filepath" "testing" "github.com/stretchr/testify/assert" "github.com/stretchr/testify/require" ) func TestNewCGroups(t *testing.T) { cgroupsProcCGroupPath := filepath.Join(testDataProcPath, "cgroups", "cgroup") cgroupsProcMountInfoPath := filepath.Join(testDataProcPath, "cgroups", "mountinfo") testTable := []struct { subsys string path string }{ {_cgroupSubsysCPU, "/sys/fs/cgroup/cpu,cpuacct"}, {_cgroupSubsysCPUAcct, "/sys/fs/cgroup/cpu,cpuacct"}, {_cgroupSubsysCPUSet, "/sys/fs/cgroup/cpuset"}, {_cgroupSubsysMemory, "/sys/fs/cgroup/memory/large"}, } cgroups, err := NewCGroups(cgroupsProcMountInfoPath, cgroupsProcCGroupPath) assert.Len(t, cgroups, len(testTable)) require.NoError(t, err) for _, tt := range testTable { cgroup, exists := cgroups[tt.subsys] assert.True(t, exists, "%q expected to present in `cgroups`", tt.subsys) assert.Equal(t, tt.path, cgroup.path, "%q expected for `cgroups[%q].path`, got %q", tt.path, tt.subsys, cgroup.path) } } func TestNewCGroupsWithErrors(t *testing.T) { testTable := []struct { mountInfoPath string cgroupPath string }{ {"non-existing-file", "/dev/null"}, {"/dev/null", "non-existing-file"}, { "/dev/null", filepath.Join(testDataProcPath, "invalid-cgroup", "cgroup"), }, { filepath.Join(testDataProcPath, "invalid-mountinfo", "mountinfo"), "/dev/null", }, { filepath.Join(testDataProcPath, "untranslatable", "mountinfo"), filepath.Join(testDataProcPath, "untranslatable", "cgroup"), }, } for _, tt := range testTable { cgroups, err := NewCGroups(tt.mountInfoPath, tt.cgroupPath) assert.Nil(t, cgroups) assert.Error(t, err) } } func TestCGroupsMemoryQuota(t *testing.T) { testTable := []struct { name string expectedQuota int64 expectedDefined bool shouldHaveError bool }{ { name: "undefined", expectedQuota: int64(-1.0), expectedDefined: false, shouldHaveError: true, }, { name: "memory", expectedQuota: int64(8796093018112), expectedDefined: true, shouldHaveError: false, }, } cgroups := make(CGroups) quota, defined, err := cgroups.MemoryQuota() assert.Equal(t, int64(-1), quota, "nonexistent") assert.False(t, defined, "nonexistent") require.NoError(t, err, "nonexistent") for _, tt := range testTable { cgroupPath := filepath.Join(testDataCGroupsPath, tt.name) cgroups[_cgroupSubsysMemory] = NewCGroup(cgroupPath) quota, defined, err := cgroups.MemoryQuota() assert.Equal(t, tt.expectedQuota, quota, tt.name) assert.Equal(t, tt.expectedDefined, defined, tt.name) if tt.shouldHaveError { assert.Error(t, err, tt.name) } else { assert.NoError(t, err, tt.name) } } } func TestCGroupsIsCGroupV2(t *testing.T) { testTable := []struct { name string expectedIsV2 bool shouldHaveError bool }{ { name: "cgroupv1", expectedIsV2: false, shouldHaveError: false, }, { name: "cgroupv1v2", expectedIsV2: false, shouldHaveError: false, }, { name: "cgroupv2", expectedIsV2: true, shouldHaveError: false, }, { name: "nonexistent", expectedIsV2: false, shouldHaveError: true, }, } for _, tt := range testTable { mountInfoPath := filepath.Join(testDataProcPath, "v2", tt.name, "mountinfo") isV2, err := isCGroupV2(mountInfoPath) assert.Equal(t, tt.expectedIsV2, isV2, tt.name) if tt.shouldHaveError { assert.Error(t, err, tt.name) } else { assert.NoError(t, err, tt.name) } } } func TestCGroupsMemoryQuotaV2(t *testing.T) { testTable := []struct { name string expectedQuota int64 expectedDefined bool shouldHaveError bool }{ { name: "memory", expectedQuota: int64(250000000), expectedDefined: true, shouldHaveError: false, }, { name: "undefined", expectedQuota: int64(-1), expectedDefined: false, shouldHaveError: false, }, { name: "invalid", expectedQuota: int64(-1), expectedDefined: false, shouldHaveError: true, }, { name: "empty", expectedQuota: int64(-1), expectedDefined: false, shouldHaveError: true, }, } quota, defined, err := memoryQuotaV2("nonexistent", "nonexistent") assert.Equal(t, int64(-1), quota, "nonexistent") assert.False(t, defined, "nonexistent") require.NoError(t, err, "nonexistent") cgroupBasePath := filepath.Join(testDataCGroupsPath, "v2") for _, tt := range testTable { cgroupPath := filepath.Join(cgroupBasePath, tt.name) quota, defined, err := memoryQuotaV2(cgroupPath, "memory.max") assert.Equal(t, tt.expectedQuota, quota, tt.name) assert.Equal(t, tt.expectedDefined, defined, tt.name) if tt.shouldHaveError { assert.Error(t, err, tt.name) } else { assert.NoError(t, err, tt.name) } } }