Files
nomad/client/allocdir/alloc_dir_test.go
Tim Gross b25f1b66ce resources: allow job authors to configure size of secrets tmpfs (#23696)
On supported platforms, the secrets directory is a 1MiB tmpfs. But some tasks
need larger space for downloading large secrets. This is especially the case for
tasks using `templates`, which need extra room to write a temporary file to the
secrets directory that gets renamed to the old file atomically.

This changeset allows increasing the size of the tmpfs in the `resources`
block. Because this is a memory resource, we need to include it in the memory we
allocate for scheduling purposes. The task is already prevented from using more
memory in the tmpfs than the `resources.memory` field allows, but can bypass
that limit by writing to the tmpfs via `template` or `artifact` blocks.

Therefore, we need to account for the size of the tmpfs in the allocation
resources. Simply adding it to the memory needed when we create the allocation
allows it to be accounted for in all downstream consumers, and then we'll
subtract that amount from the memory resources just before configuring the task
driver.

For backwards compatibility, the default value of 1MiB is "free" and ignored by
the scheduler. Otherwise we'd be increasing the allocated resources for every
existing alloc, which could cause problems across upgrades. If a user explicitly
sets `resources.secrets = 1` it will no longer be free.

Fixes: https://github.com/hashicorp/nomad/issues/2481
Ref: https://hashicorp.atlassian.net/browse/NET-10070
2024-08-05 16:06:58 -04:00

465 lines
12 KiB
Go

// Copyright (c) HashiCorp, Inc.
// SPDX-License-Identifier: BUSL-1.1
package allocdir
import (
"archive/tar"
"bytes"
"context"
"io"
"io/fs"
"os"
"path/filepath"
"runtime"
"strings"
"syscall"
"testing"
"github.com/hashicorp/nomad/ci"
"github.com/hashicorp/nomad/helper/testlog"
"github.com/hashicorp/nomad/nomad/structs"
"github.com/hashicorp/nomad/plugins/drivers/fsisolation"
"github.com/shoenig/test/must"
"golang.org/x/sys/unix"
)
// copy from testutil to avoid import cycle
func requireNonRoot(t *testing.T) {
if unix.Geteuid() == 0 {
t.Skip("must run as non-root")
}
}
// copy from testutil to avoid import cycle
func requireRoot(t *testing.T) {
if unix.Geteuid() != 0 {
t.Skip("must run as root")
}
}
var (
t1 = &structs.Task{
Name: "web",
Driver: "exec",
Config: map[string]interface{}{
"command": "/bin/date",
"args": "+%s",
},
Resources: &structs.Resources{
DiskMB: 1,
},
}
t2 = &structs.Task{
Name: "web2",
Driver: "exec",
Config: map[string]interface{}{
"command": "/bin/date",
"args": "+%s",
},
Resources: &structs.Resources{
DiskMB: 1,
},
}
)
// Test that AllocDir.Build builds just the alloc directory.
func TestAllocDir_BuildAlloc(t *testing.T) {
ci.Parallel(t)
tmp := t.TempDir()
d := NewAllocDir(testlog.HCLogger(t), tmp, tmp, "test")
defer d.Destroy()
d.NewTaskDir(t1)
d.NewTaskDir(t2)
must.NoError(t, d.Build())
// Check that the AllocDir and each of the task directories exist.
if _, err := os.Stat(d.AllocDir); os.IsNotExist(err) {
t.Fatalf("Build() didn't create AllocDir %v", d.AllocDir)
}
for _, task := range []*structs.Task{t1, t2} {
tDir, ok := d.TaskDirs[task.Name]
must.True(t, ok)
stat, _ := os.Stat(tDir.Dir)
must.Nil(t, stat)
stat, _ = os.Stat(tDir.SecretsDir)
must.Nil(t, stat)
}
}
// HACK: This function is copy/pasted from client.testutil to prevent a test
//
// import cycle, due to testutil transitively importing allocdir. This
// should be fixed after DriverManager is implemented.
func MountCompatible(t *testing.T) {
if runtime.GOOS == "windows" {
t.Skip("Windows does not support mount")
}
if syscall.Geteuid() != 0 {
t.Skip("Must be root to run test")
}
}
func TestAllocDir_MountSharedAlloc(t *testing.T) {
ci.Parallel(t)
MountCompatible(t)
tmp := t.TempDir()
d := NewAllocDir(testlog.HCLogger(t), tmp, tmp, "test")
defer d.Destroy()
must.NoError(t, d.Build())
// Build 2 task dirs
td1 := d.NewTaskDir(t1)
must.NoError(t, td1.Build(fsisolation.Chroot, nil, "nobody"))
td2 := d.NewTaskDir(t2)
must.NoError(t, td2.Build(fsisolation.Chroot, nil, "nobody"))
// Write a file to the shared dir.
contents := []byte("foo")
const filename = "bar"
must.NoError(t, os.WriteFile(filepath.Join(d.SharedDir, filename), contents, 0o666))
// Check that the file exists in the task directories
for _, td := range []*TaskDir{td1, td2} {
taskFile := filepath.Join(td.SharedTaskDir, filename)
act, err := os.ReadFile(taskFile)
if err != nil {
t.Errorf("Failed to read shared alloc file from task dir: %v", err)
continue
}
if !bytes.Equal(act, contents) {
t.Errorf("Incorrect data read from task dir: want %v; got %v", contents, act)
}
}
}
func TestAllocDir_Snapshot(t *testing.T) {
ci.Parallel(t)
tmp := t.TempDir()
d := NewAllocDir(testlog.HCLogger(t), tmp, tmp, "test")
defer d.Destroy()
must.NoError(t, d.Build())
// Build 2 task dirs
td1 := d.NewTaskDir(t1)
must.NoError(t, td1.Build(fsisolation.None, nil, "nobody"))
td2 := d.NewTaskDir(t2)
must.NoError(t, td2.Build(fsisolation.None, nil, "nobody"))
// Write a file to the shared dir.
exp := []byte{'f', 'o', 'o'}
file := "bar"
must.NoError(t, os.WriteFile(filepath.Join(d.SharedDir, "data", file), exp, 0o666))
// Write a symlink to the shared dir
link := "qux"
must.NoError(t, os.Symlink("foo", filepath.Join(d.SharedDir, "data", link)))
// Write a file to the task local
exp = []byte{'b', 'a', 'r'}
file1 := "lol"
must.NoError(t, os.WriteFile(filepath.Join(td1.LocalDir, file1), exp, 0o666))
// Write a symlink to the task local
link1 := "baz"
must.NoError(t, os.Symlink("bar", filepath.Join(td1.LocalDir, link1)))
var b bytes.Buffer
must.NoError(t, d.Snapshot(&b))
tr := tar.NewReader(&b)
var files []string
var links []string
for {
hdr, err := tr.Next()
if err != nil && err != io.EOF {
t.Fatalf("err: %v", err)
}
if err == io.EOF {
break
}
if hdr.Typeflag == tar.TypeReg {
files = append(files, hdr.FileInfo().Name())
} else if hdr.Typeflag == tar.TypeSymlink {
links = append(links, hdr.FileInfo().Name())
}
}
must.SliceLen(t, 2, files)
must.SliceLen(t, 2, links)
}
func TestAllocDir_Move(t *testing.T) {
ci.Parallel(t)
tmp1 := t.TempDir()
tmp2 := t.TempDir()
// Create two alloc dirs
d1 := NewAllocDir(testlog.HCLogger(t), tmp1, tmp1, "test")
must.NoError(t, d1.Build())
defer d1.Destroy()
d2 := NewAllocDir(testlog.HCLogger(t), tmp2, tmp2, "test")
must.NoError(t, d2.Build())
defer d2.Destroy()
td1 := d1.NewTaskDir(t1)
must.NoError(t, td1.Build(fsisolation.None, nil, "nobody"))
// Create but don't build second task dir to mimic alloc/task runner
// behavior (AllocDir.Move() is called pre-TaskDir.Build).
d2.NewTaskDir(t1)
dataDir := filepath.Join(d1.SharedDir, SharedDataDir)
// Write a file to the shared dir.
exp1 := []byte("foo")
file1 := "bar"
must.NoError(t, os.WriteFile(filepath.Join(dataDir, file1), exp1, 0o666))
// Write a file to the task local
exp2 := []byte("bar")
file2 := "lol"
must.NoError(t, os.WriteFile(filepath.Join(td1.LocalDir, file2), exp2, 0o666))
// Move the d1 allocdir to d2
must.NoError(t, d2.Move(d1, []*structs.Task{t1}))
// Ensure the files in d1 are present in d2
fi, err := os.Stat(filepath.Join(d2.SharedDir, SharedDataDir, file1))
must.NoError(t, err)
must.NotNil(t, fi)
fi, err = os.Stat(filepath.Join(d2.TaskDirs[t1.Name].LocalDir, file2))
must.NoError(t, err)
must.NotNil(t, fi)
}
func TestAllocDir_EscapeChecking(t *testing.T) {
ci.Parallel(t)
tmp := t.TempDir()
d := NewAllocDir(testlog.HCLogger(t), tmp, tmp, "test")
must.NoError(t, d.Build())
defer d.Destroy()
// Check that issuing calls that escape the alloc dir returns errors
// List
if _, err := d.List(".."); err == nil || !strings.Contains(err.Error(), "escapes") {
t.Fatalf("List of escaping path didn't error: %v", err)
}
// Stat
if _, err := d.Stat("../foo"); err == nil || !strings.Contains(err.Error(), "escapes") {
t.Fatalf("Stat of escaping path didn't error: %v", err)
}
// ReadAt
if _, err := d.ReadAt("../foo", 0); err == nil || !strings.Contains(err.Error(), "escapes") {
t.Fatalf("ReadAt of escaping path didn't error: %v", err)
}
// BlockUntilExists
if _, err := d.BlockUntilExists(context.Background(), "../foo"); err == nil || !strings.Contains(err.Error(), "escapes") {
t.Fatalf("BlockUntilExists of escaping path didn't error: %v", err)
}
// ChangeEvents
if _, err := d.ChangeEvents(context.Background(), "../foo", 0); err == nil || !strings.Contains(err.Error(), "escapes") {
t.Fatalf("ChangeEvents of escaping path didn't error: %v", err)
}
}
// Test that `nomad fs` can't read secrets
func TestAllocDir_ReadAt_SecretDir(t *testing.T) {
ci.Parallel(t)
tmp := t.TempDir()
d := NewAllocDir(testlog.HCLogger(t), tmp, tmp, "test")
must.NoError(t, d.Build())
defer func() { _ = d.Destroy() }()
td := d.NewTaskDir(t1)
must.NoError(t, td.Build(fsisolation.None, nil, "nobody"))
// something to write and test reading
target := filepath.Join(t1.Name, TaskSecrets, "test_file")
// create target file in the task secrets dir
full := filepath.Join(d.AllocDir, target)
must.NoError(t, os.WriteFile(full, []byte("hi"), 0o600))
// ReadAt of a file in the task secrets dir should fail
_, err := d.ReadAt(target, 0)
must.EqError(t, err, "Reading secret file prohibited: web/secrets/test_file")
}
func TestAllocDir_SplitPath(t *testing.T) {
ci.Parallel(t)
dir := t.TempDir()
dest := filepath.Join(dir, "/foo/bar/baz")
must.NoError(t, os.MkdirAll(dest, os.ModePerm))
info, err := splitPath(dest)
must.NoError(t, err)
// Testing that is 6 or more rather than 6 because on osx, the temp dir is
// randomized.
must.GreaterEq(t, 6, len(info))
}
func TestAllocDir_CreateDir(t *testing.T) {
requireRoot(t)
ci.Parallel(t)
dir := t.TempDir()
// create a subdir and a file
subdir := filepath.Join(dir, "subdir")
must.NoError(t, os.MkdirAll(subdir, 0o760))
subdirMode, err := os.Stat(subdir)
must.NoError(t, err)
// Create the above hierarchy under another destination
dir1 := t.TempDir()
must.NoError(t, createDir(dir1, subdir))
// Ensure that the subdir had the right perm
fi, err := os.Stat(filepath.Join(dir1, dir, "subdir"))
must.NoError(t, err)
must.Eq(t, fi.Mode(), subdirMode.Mode())
}
func TestPathFuncs(t *testing.T) {
ci.Parallel(t)
dir := t.TempDir()
missingDir := filepath.Join(dir, "does-not-exist")
must.True(t, pathExists(dir))
must.False(t, pathExists(missingDir))
if empty, err := pathEmpty(dir); err != nil || !empty {
t.Errorf("%q is empty and exists. empty=%v error=%v", dir, empty, err)
}
if empty, err := pathEmpty(missingDir); err == nil || empty {
t.Errorf("%q is missing. empty=%v error=%v", missingDir, empty, err)
}
filename := filepath.Join(dir, "just-some-file")
f, err := os.Create(filename)
must.NoError(t, err)
f.Close()
if empty, err := pathEmpty(dir); err != nil || empty {
t.Errorf("%q is not empty. empty=%v error=%v", dir, empty, err)
}
}
func TestAllocDir_DetectContentType(t *testing.T) {
ci.Parallel(t)
inputPath := "input/"
var testFiles []string
err := filepath.Walk(inputPath, func(path string, info os.FileInfo, err error) error {
if !info.IsDir() {
testFiles = append(testFiles, path)
}
return err
})
must.NoError(t, err)
expectedEncodings := map[string]string{
"input/happy.gif": "image/gif",
"input/image.png": "image/png",
"input/nomad.jpg": "image/jpeg",
"input/test.bin": "application/octet-stream",
"input/test.json": "application/json",
"input/test.txt": "text/plain; charset=utf-8",
"input/test.go": "text/plain; charset=utf-8",
"input/test.hcl": "text/plain; charset=utf-8",
}
for _, file := range testFiles {
fileInfo, err := os.Stat(file)
must.NoError(t, err)
res := detectContentType(fileInfo, file)
must.Eq(t, expectedEncodings[file], res)
}
}
// TestAllocDir_SkipAllocDir asserts that building a chroot which contains
// itself will *not* infinitely recurse. AllocDirs should always skip embedding
// themselves into chroots.
//
// Warning: If this test fails it may fill your disk before failing, so be
// careful and/or confident.
func TestAllocDir_SkipAllocDir(t *testing.T) {
ci.Parallel(t)
MountCompatible(t)
// Create root, alloc, and other dirs
rootDir := t.TempDir()
clientAllocDir := filepath.Join(rootDir, "nomad")
mountAllocDir := filepath.Join(rootDir, "mounts")
must.NoError(t, os.Mkdir(clientAllocDir, fs.ModeDir|0o777))
otherDir := filepath.Join(rootDir, "etc")
must.NoError(t, os.Mkdir(otherDir, fs.ModeDir|0o777))
// chroot contains client.alloc_dir! This could cause infinite
// recursion.
chroot := map[string]string{
rootDir: "/",
}
allocDir := NewAllocDir(testlog.HCLogger(t), clientAllocDir, mountAllocDir, "test")
taskDir := allocDir.NewTaskDir(t1)
must.NoError(t, allocDir.Build())
defer allocDir.Destroy()
// Build chroot
err := taskDir.Build(fsisolation.Chroot, chroot, "nobody")
must.NoError(t, err)
// Assert other directory *was* embedded
embeddedOtherDir := filepath.Join(clientAllocDir, "test", t1.Name, "etc")
if _, err := os.Stat(embeddedOtherDir); os.IsNotExist(err) {
t.Fatalf("expected other directory to exist at: %q", embeddedOtherDir)
}
// Assert client.alloc_dir was *not* embedded
embeddedChroot := filepath.Join(clientAllocDir, "test", t1.Name, "nomad")
s, err := os.Stat(embeddedChroot)
if s != nil {
t.Logf("somehow you managed to embed the chroot without causing infinite recursion!")
t.Fatalf("expected chroot to not exist at: %q", embeddedChroot)
}
if !os.IsNotExist(err) {
t.Fatalf("expected chroot to not exist but error is: %v", err)
}
}