/* Copyright The containerd 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 v2 import ( "fmt" "os" "os/exec" "testing" "time" "github.com/opencontainers/runtime-spec/specs-go" "github.com/stretchr/testify/assert" "go.uber.org/goleak" ) func TestEventChanCleanupOnCgroupRemoval(t *testing.T) { checkCgroupMode(t) cmd := exec.Command("cat") stdin, err := cmd.StdinPipe() if err != nil { t.Fatalf("Failed to create cat process: %v", err) } if err := cmd.Start(); err != nil { t.Fatalf("Failed to start cat process: %v", err) } proc := cmd.Process if proc == nil { t.Fatal("Process is nil") } group := fmt.Sprintf("testing-watcher-%d.scope", proc.Pid) c, err := NewSystemd("", group, proc.Pid, &Resources{}) if err != nil { t.Fatalf("Failed to init new cgroup manager: %v", err) } evCh, errCh := c.EventChan() // give event goroutine a chance to start time.Sleep(500 * time.Millisecond) if err := stdin.Close(); err != nil { t.Fatalf("Failed closing stdin: %v", err) } if err := cmd.Wait(); err != nil { t.Fatalf("Failed waiting for cmd: %v", err) } done := false for !done { select { case <-evCh: case err := <-errCh: if err != nil { t.Fatalf("Unexpected error on error channel: %v", err) } done = true case <-time.After(5 * time.Second): t.Fatal("Timed out") } } goleak.VerifyNone(t) } func TestSystemdFullPath(t *testing.T) { tests := []struct { inputSlice string inputGroup string expectedOut string }{ { inputSlice: "user.slice", inputGroup: "myGroup.slice", expectedOut: "/sys/fs/cgroup/user.slice/myGroup.slice", }, { inputSlice: "/", inputGroup: "myGroup.slice", expectedOut: "/sys/fs/cgroup/myGroup.slice", }, { inputSlice: "system.slice", inputGroup: "myGroup.slice", expectedOut: "/sys/fs/cgroup/system.slice/myGroup.slice", }, { inputSlice: "user.slice", inputGroup: "my-group.slice", expectedOut: "/sys/fs/cgroup/user.slice/my.slice/my-group.slice", }, { inputSlice: "user.slice", inputGroup: "my-group-more-dashes.slice", expectedOut: "/sys/fs/cgroup/user.slice/my.slice/my-group.slice/my-group-more.slice/my-group-more-dashes.slice", }, { inputSlice: "user.slice", inputGroup: "my-group-dashes.slice", expectedOut: "/sys/fs/cgroup/user.slice/my.slice/my-group.slice/my-group-dashes.slice", }, { inputSlice: "user.slice", inputGroup: "myGroup.scope", expectedOut: "/sys/fs/cgroup/user.slice/myGroup.scope", }, { inputSlice: "user.slice", inputGroup: "my-group-dashes.scope", expectedOut: "/sys/fs/cgroup/user.slice/my-group-dashes.scope", }, { inputSlice: "test-waldo.slice", inputGroup: "my-group.slice", expectedOut: "/sys/fs/cgroup/test.slice/test-waldo.slice/my.slice/my-group.slice", }, { inputSlice: "test-waldo.slice", inputGroup: "my.service", expectedOut: "/sys/fs/cgroup/test.slice/test-waldo.slice/my.service", }, } for _, test := range tests { actual := getSystemdFullPath(test.inputSlice, test.inputGroup) assert.Equal(t, test.expectedOut, actual) } } func TestMoveTo(t *testing.T) { checkCgroupMode(t) manager, err := NewManager(defaultCgroup2Path, "/test1", ToResources(&specs.LinuxResources{})) if err != nil { t.Error(err) return } proc := os.Getpid() if err := manager.AddProc(uint64(proc)); err != nil { t.Error(err) return } destination, err := NewManager(defaultCgroup2Path, "/test2", ToResources(&specs.LinuxResources{})) if err != nil { t.Error(err) return } if err := manager.MoveTo(destination); err != nil { t.Error(err) return } desProcs, err := destination.Procs(true) desMap := make(map[int]bool) for _, p := range desProcs { desMap[int(p)] = true } if !desMap[proc] { t.Errorf("process %v not in destination cgroup", proc) return } }