-
Notifications
You must be signed in to change notification settings - Fork 34
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Add end-to-end tests for local cache (#292)
By submitting this pull request, I confirm that you can use, modify, copy, and redistribute this contribution, under the terms of your choice. --------- Signed-off-by: Burak Varlı <[email protected]>
- Loading branch information
Showing
3 changed files
with
254 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,237 @@ | ||
package custom_testsuites | ||
|
||
import ( | ||
"context" | ||
"fmt" | ||
"path/filepath" | ||
"slices" | ||
"time" | ||
|
||
"github.com/aws/aws-sdk-go-v2/aws" | ||
"github.com/aws/aws-sdk-go-v2/service/s3" | ||
"github.com/google/uuid" | ||
. "github.com/onsi/ginkgo/v2" | ||
v1 "k8s.io/api/core/v1" | ||
"k8s.io/apimachinery/pkg/util/errors" | ||
"k8s.io/kubernetes/test/e2e/framework" | ||
e2epod "k8s.io/kubernetes/test/e2e/framework/pod" | ||
e2eskipper "k8s.io/kubernetes/test/e2e/framework/skipper" | ||
e2evolume "k8s.io/kubernetes/test/e2e/framework/volume" | ||
storageframework "k8s.io/kubernetes/test/e2e/storage/framework" | ||
admissionapi "k8s.io/pod-security-admission/api" | ||
"k8s.io/utils/ptr" | ||
) | ||
|
||
const volumeName1 = "volume1" | ||
const root = int64(0) | ||
const defaultNonRootGroup = int64(2000) | ||
|
||
type s3CSICacheTestSuite struct { | ||
tsInfo storageframework.TestSuiteInfo | ||
} | ||
|
||
func InitS3CSICacheTestSuite() storageframework.TestSuite { | ||
return &s3CSICacheTestSuite{ | ||
tsInfo: storageframework.TestSuiteInfo{ | ||
Name: "cache", | ||
TestPatterns: []storageframework.TestPattern{ | ||
storageframework.DefaultFsPreprovisionedPV, | ||
}, | ||
}, | ||
} | ||
} | ||
|
||
func (t *s3CSICacheTestSuite) GetTestSuiteInfo() storageframework.TestSuiteInfo { | ||
return t.tsInfo | ||
} | ||
|
||
func (t *s3CSICacheTestSuite) SkipUnsupportedTests(_ storageframework.TestDriver, pattern storageframework.TestPattern) { | ||
if pattern.VolType != storageframework.PreprovisionedPV { | ||
e2eskipper.Skipf("Suite %q does not support %v", t.tsInfo.Name, pattern.VolType) | ||
} | ||
} | ||
|
||
func (t *s3CSICacheTestSuite) DefineTests(driver storageframework.TestDriver, pattern storageframework.TestPattern) { | ||
f := framework.NewFrameworkWithCustomTimeouts(NamespacePrefix+"cache", storageframework.GetDriverTimeouts(driver)) | ||
// This is required for now due to hack mentioned in `ensureCacheDirExistsInNode` function, see the comments there for more context. | ||
f.NamespacePodSecurityLevel = admissionapi.LevelPrivileged | ||
|
||
type local struct { | ||
config *storageframework.PerTestConfig | ||
|
||
// A list of cleanup functions to be called after each test to clean resources created during the test. | ||
cleanup []func(context.Context) error | ||
} | ||
|
||
var l local | ||
|
||
deferCleanup := func(f func(context.Context) error) { | ||
l.cleanup = append(l.cleanup, f) | ||
} | ||
|
||
cleanup := func(ctx context.Context) { | ||
var errs []error | ||
slices.Reverse(l.cleanup) // clean items in reverse order similar to how `defer` works | ||
for _, f := range l.cleanup { | ||
errs = append(errs, f(ctx)) | ||
} | ||
framework.ExpectNoError(errors.NewAggregate(errs), "while cleanup resource") | ||
} | ||
BeforeEach(func(ctx context.Context) { | ||
l = local{} | ||
l.config = driver.PrepareTest(ctx, f) | ||
DeferCleanup(cleanup) | ||
}) | ||
|
||
// checkBasicFileOperations verifies basic file operations works in the given `basePath` inside the `pod`. | ||
checkBasicFileOperations := func(ctx context.Context, pod *v1.Pod, bucketName, basePath string) { | ||
framework.Logf("Checking basic file operations inside pod %s at %s", pod.UID, basePath) | ||
|
||
dir := filepath.Join(basePath, "test-dir") | ||
first := filepath.Join(basePath, "first") | ||
second := filepath.Join(dir, "second") | ||
|
||
seed := time.Now().UTC().UnixNano() | ||
testWriteSize := 1024 // 1KB | ||
|
||
checkWriteToPath(f, pod, first, testWriteSize, seed) | ||
checkListingPathWithEntries(f, pod, basePath, []string{"first"}) | ||
// Test reading multiple times to ensure cached-read works | ||
for i := 0; i < 3; i++ { | ||
checkReadFromPath(f, pod, first, testWriteSize, seed) | ||
} | ||
|
||
// Now remove the file from S3 | ||
deleteObjectFromS3(ctx, bucketName, "first") | ||
|
||
// Ensure the data still read from the cache - without cache this would fail as its removed from underlying bucket | ||
checkReadFromPath(f, pod, first, testWriteSize, seed) | ||
|
||
e2evolume.VerifyExecInPodSucceed(f, pod, fmt.Sprintf("mkdir %s && cd %s && echo 'second!' > %s", dir, dir, second)) | ||
e2evolume.VerifyExecInPodSucceed(f, pod, fmt.Sprintf("cat %s | grep -q 'second!'", second)) | ||
checkListingPathWithEntries(f, pod, dir, []string{"second"}) | ||
checkListingPathWithEntries(f, pod, basePath, []string{"test-dir"}) | ||
checkDeletingPath(f, pod, first) | ||
checkDeletingPath(f, pod, second) | ||
} | ||
|
||
createPod := func(ctx context.Context, additionalMountOptions []string, podModifiers ...func(*v1.Pod)) (*v1.Pod, string) { | ||
cacheDir := randomCacheDir() | ||
|
||
vol := createVolumeResourceWithMountOptions(ctx, l.config, pattern, append(additionalMountOptions, fmt.Sprintf("cache %s", cacheDir))) | ||
deferCleanup(vol.CleanupResource) | ||
|
||
bucketName := bucketNameFromVolumeResource(vol) | ||
|
||
pod := e2epod.MakePod(f.Namespace.Name, nil, []*v1.PersistentVolumeClaim{vol.Pvc}, admissionapi.LevelBaseline, "") | ||
ensureCacheDirExistsInNode(pod, cacheDir) | ||
for _, pm := range podModifiers { | ||
pm(pod) | ||
} | ||
|
||
pod, err := createPod(ctx, f.ClientSet, f.Namespace.Name, pod) | ||
framework.ExpectNoError(err) | ||
deferCleanup(func(ctx context.Context) error { return e2epod.DeletePodWithWait(ctx, f.ClientSet, pod) }) | ||
|
||
return pod, bucketName | ||
} | ||
|
||
Describe("Cache", func() { | ||
Describe("Local", func() { | ||
It("basic file operations as root", func(ctx context.Context) { | ||
pod, bucketName := createPod(ctx, []string{"allow-delete"}, func(pod *v1.Pod) { | ||
pod.Spec.Containers[0].SecurityContext.RunAsUser = ptr.To(root) | ||
pod.Spec.Containers[0].SecurityContext.RunAsGroup = ptr.To(root) | ||
}) | ||
checkBasicFileOperations(ctx, pod, bucketName, e2epod.VolumeMountPath1) | ||
}) | ||
|
||
It("basic file operations as non-root", func(ctx context.Context) { | ||
mountOptions := []string{ | ||
"allow-delete", | ||
"allow-other", | ||
fmt.Sprintf("uid=%d", *e2epod.GetDefaultNonRootUser()), | ||
fmt.Sprintf("gid=%d", defaultNonRootGroup), | ||
} | ||
pod, bucketName := createPod(ctx, mountOptions, func(pod *v1.Pod) { | ||
pod.Spec.Containers[0].SecurityContext.RunAsUser = e2epod.GetDefaultNonRootUser() | ||
pod.Spec.Containers[0].SecurityContext.RunAsGroup = ptr.To(defaultNonRootGroup) | ||
pod.Spec.Containers[0].SecurityContext.RunAsNonRoot = ptr.To(true) | ||
}) | ||
|
||
checkBasicFileOperations(ctx, pod, bucketName, e2epod.VolumeMountPath1) | ||
}) | ||
|
||
It("two containers in the same pod using the same cache", func(ctx context.Context) { | ||
testFile := filepath.Join(e2epod.VolumeMountPath1, "helloworld.txt") | ||
|
||
pod, _ := createPod(ctx, []string{"allow-delete"}, func(pod *v1.Pod) { | ||
// Make it init container to ensure it runs before regular containers | ||
pod.Spec.InitContainers = append(pod.Spec.InitContainers, v1.Container{ | ||
Name: "populate-cache", | ||
Image: e2epod.GetDefaultTestImage(), | ||
Command: e2epod.GenerateScriptCmd( | ||
fmt.Sprintf("echo 'hello world!' > %s && cat %s | grep -q 'hello world!'", testFile, testFile)), | ||
VolumeMounts: []v1.VolumeMount{ | ||
{ | ||
Name: volumeName1, | ||
MountPath: e2epod.VolumeMountPath1, | ||
}, | ||
}, | ||
}) | ||
}) | ||
|
||
e2evolume.VerifyExecInPodSucceed(f, pod, fmt.Sprintf("cat %s | grep -q 'hello world!'", testFile)) | ||
}) | ||
}) | ||
}) | ||
} | ||
|
||
// deleteObjectFromS3 deletes an object from given bucket by using S3 SDK. | ||
// This is useful to create some side-effects by bypassing Mountpoint. | ||
func deleteObjectFromS3(ctx context.Context, bucket string, key string) { | ||
client := s3.NewFromConfig(awsConfig(ctx)) | ||
_, err := client.DeleteObject(ctx, &s3.DeleteObjectInput{ | ||
Bucket: aws.String(bucket), | ||
Key: aws.String(key), | ||
}) | ||
framework.ExpectNoError(err) | ||
} | ||
|
||
func randomCacheDir() string { | ||
return filepath.Join("/tmp/mp-cache", uuid.New().String()) | ||
} | ||
|
||
// ensureCacheDirExistsInNode adds a hostPath for given `cacheDir` with `DirectoryOrCreate` type. | ||
// This hack required because Mountpoint process is running on the underlying host and not inside the container, | ||
// so we need to ensure cache directory exists on the host. | ||
// This hack hopefully will go away with https://github.com/awslabs/mountpoint-s3-csi-driver/issues/279. | ||
func ensureCacheDirExistsInNode(pod *v1.Pod, cacheDir string) { | ||
cacheVolumeMount := v1.VolumeMount{ | ||
Name: "make-cache-dir", | ||
MountPath: "/cache", | ||
} | ||
|
||
// The directory created with `DirectoryOrCreate` will have 0755 permissions and will be owned by kubelet. | ||
// Unless we change permissions here, non-root containers won't be able to access to the cache dir. | ||
pod.Spec.InitContainers = append(pod.Spec.DeepCopy().InitContainers, v1.Container{ | ||
Name: "chmod-cache-dir", | ||
Image: e2epod.GetDefaultTestImage(), | ||
Command: e2epod.GenerateScriptCmd("chmod -R 777 /cache"), | ||
SecurityContext: &v1.SecurityContext{ | ||
RunAsUser: ptr.To(root), | ||
RunAsGroup: ptr.To(root), | ||
}, | ||
VolumeMounts: []v1.VolumeMount{cacheVolumeMount}, | ||
}) | ||
pod.Spec.Volumes = append(pod.Spec.Volumes, v1.Volume{ | ||
Name: "make-cache-dir", | ||
VolumeSource: v1.VolumeSource{ | ||
HostPath: &v1.HostPathVolumeSource{ | ||
Path: cacheDir, | ||
Type: ptr.To(v1.HostPathDirectoryOrCreate), | ||
}, | ||
}, | ||
}) | ||
pod.Spec.Containers[0].VolumeMounts = append(pod.Spec.Containers[0].VolumeMounts, cacheVolumeMount) | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters