package filesystem_test import ( "bytes" "image" "image/png" "mime/multipart" "net/http" "net/http/httptest" "os" "path/filepath" "strings" "testing" "github.com/pocketbase/pocketbase/tools/filesystem" ) func TestFileSystemExists(t *testing.T) { dir := createTestDir(t) defer os.RemoveAll(dir) fs, err := filesystem.NewLocal(dir) if err != nil { t.Fatal(err) } defer fs.Close() scenarios := []struct { file string exists bool }{ {"sub1.txt", false}, {"test/sub1.txt", true}, {"test/sub2.txt", true}, {"image.png", true}, } for i, scenario := range scenarios { exists, _ := fs.Exists(scenario.file) if exists != scenario.exists { t.Errorf("(%d) Expected %v, got %v", i, scenario.exists, exists) } } } func TestFileSystemAttributes(t *testing.T) { dir := createTestDir(t) defer os.RemoveAll(dir) fs, err := filesystem.NewLocal(dir) if err != nil { t.Fatal(err) } defer fs.Close() scenarios := []struct { file string expectError bool expectContentType string }{ {"sub1.txt", true, ""}, {"test/sub1.txt", false, "application/octet-stream"}, {"test/sub2.txt", false, "application/octet-stream"}, {"image.png", false, "image/png"}, } for i, scenario := range scenarios { attr, err := fs.Attributes(scenario.file) if err == nil && scenario.expectError { t.Errorf("(%d) Expected error, got nil", i) } if err != nil && !scenario.expectError { t.Errorf("(%d) Expected nil, got error, %v", i, err) } if err == nil && attr.ContentType != scenario.expectContentType { t.Errorf("(%d) Expected attr.ContentType to be %q, got %q", i, scenario.expectContentType, attr.ContentType) } } } func TestFileSystemDelete(t *testing.T) { dir := createTestDir(t) defer os.RemoveAll(dir) fs, err := filesystem.NewLocal(dir) if err != nil { t.Fatal(err) } defer fs.Close() if err := fs.Delete("missing.txt"); err == nil { t.Fatal("Expected error, got nil") } if err := fs.Delete("image.png"); err != nil { t.Fatalf("Expected nil, got error %v", err) } } func TestFileSystemDeletePrefix(t *testing.T) { dir := createTestDir(t) defer os.RemoveAll(dir) fs, err := filesystem.NewLocal(dir) if err != nil { t.Fatal(err) } defer fs.Close() if errs := fs.DeletePrefix(""); len(errs) == 0 { t.Fatal("Expected error, got nil", errs) } if errs := fs.DeletePrefix("missing/"); len(errs) != 0 { t.Fatalf("Not existing prefix shouldn't error, got %v", errs) } if errs := fs.DeletePrefix("test"); len(errs) != 0 { t.Fatalf("Expected nil, got errors %v", errs) } // ensure that the test/ files are deleted if exists, _ := fs.Exists("test/sub1.txt"); exists { t.Fatalf("Expected test/sub1.txt to be deleted") } if exists, _ := fs.Exists("test/sub2.txt"); exists { t.Fatalf("Expected test/sub2.txt to be deleted") } } func TestFileSystemUploadMultipart(t *testing.T) { dir := createTestDir(t) defer os.RemoveAll(dir) // create multipart form file body := new(bytes.Buffer) mp := multipart.NewWriter(body) w, err := mp.CreateFormFile("test", "test") if err != nil { t.Fatalf("Failed creating form file: %v", err) } w.Write([]byte("demo")) mp.Close() req := httptest.NewRequest(http.MethodPost, "/", body) req.Header.Add("Content-Type", mp.FormDataContentType()) file, fh, err := req.FormFile("test") if err != nil { t.Fatalf("Failed to fetch form file: %v", err) } defer file.Close() // --- fs, err := filesystem.NewLocal(dir) if err != nil { t.Fatal(err) } defer fs.Close() fileKey := "newdir/newkey.txt" uploadErr := fs.UploadMultipart(fh, fileKey) if uploadErr != nil { t.Fatal(uploadErr) } if exists, _ := fs.Exists(fileKey); !exists { t.Fatalf("Expected %q to exist", fileKey) } attrs, err := fs.Attributes(fileKey) if err != nil { t.Fatalf("Failed to fetch file attributes: %v", err) } if name, ok := attrs.Metadata["original-filename"]; !ok || name != "test" { t.Fatalf("Expected original-filename to be %q, got %q", "test", name) } } func TestFileSystemUploadFile(t *testing.T) { dir := createTestDir(t) defer os.RemoveAll(dir) fs, err := filesystem.NewLocal(dir) if err != nil { t.Fatal(err) } defer fs.Close() fileKey := "newdir/newkey.txt" file, err := filesystem.NewFileFromPath(filepath.Join(dir, "image.svg")) if err != nil { t.Fatalf("Failed to load test file: %v", err) } file.OriginalName = "test.txt" uploadErr := fs.UploadFile(file, fileKey) if uploadErr != nil { t.Fatal(uploadErr) } if exists, _ := fs.Exists(fileKey); !exists { t.Fatalf("Expected %q to exist", fileKey) } attrs, err := fs.Attributes(fileKey) if err != nil { t.Fatalf("Failed to fetch file attributes: %v", err) } if name, ok := attrs.Metadata["original-filename"]; !ok || name != file.OriginalName { t.Fatalf("Expected original-filename to be %q, got %q", file.OriginalName, name) } } func TestFileSystemUpload(t *testing.T) { dir := createTestDir(t) defer os.RemoveAll(dir) fs, err := filesystem.NewLocal(dir) if err != nil { t.Fatal(err) } defer fs.Close() fileKey := "newdir/newkey.txt" uploadErr := fs.Upload([]byte("demo"), fileKey) if uploadErr != nil { t.Fatal(uploadErr) } if exists, _ := fs.Exists(fileKey); !exists { t.Fatalf("Expected %s to exist", fileKey) } } func TestFileSystemServe(t *testing.T) { dir := createTestDir(t) defer os.RemoveAll(dir) fs, err := filesystem.NewLocal(dir) if err != nil { t.Fatal(err) } defer fs.Close() csp := "default-src 'none'; media-src 'self'; style-src 'unsafe-inline'; sandbox" cacheControl := "max-age=2592000, stale-while-revalidate=86400" scenarios := []struct { path string name string customHeaders map[string]string expectError bool expectHeaders map[string]string }{ { // missing "missing.txt", "test_name.txt", nil, true, nil, }, { // existing regular file "test/sub1.txt", "test_name.txt", nil, false, map[string]string{ "Content-Disposition": "attachment; filename=test_name.txt", "Content-Type": "application/octet-stream", "Content-Length": "0", "Content-Security-Policy": csp, "Cache-Control": cacheControl, }, }, { // png inline "image.png", "test_name.png", nil, false, map[string]string{ "Content-Disposition": "inline; filename=test_name.png", "Content-Type": "image/png", "Content-Length": "73", "Content-Security-Policy": csp, "Cache-Control": cacheControl, }, }, { // svg exception "image.svg", "test_name.svg", nil, false, map[string]string{ "Content-Disposition": "attachment; filename=test_name.svg", "Content-Type": "image/svg+xml", "Content-Length": "0", "Content-Security-Policy": csp, "Cache-Control": cacheControl, }, }, { // css exception "style.css", "test_name.css", nil, false, map[string]string{ "Content-Disposition": "attachment; filename=test_name.css", "Content-Type": "text/css", "Content-Length": "0", "Content-Security-Policy": csp, "Cache-Control": cacheControl, }, }, { // custom header "test/sub2.txt", "test_name.txt", map[string]string{ "Content-Disposition": "1", "Content-Type": "2", "Content-Length": "3", "Content-Security-Policy": "4", "Cache-Control": "5", "X-Custom": "6", }, false, map[string]string{ "Content-Disposition": "1", "Content-Type": "2", "Content-Length": "0", // overwriten by http.ServeContent "Content-Security-Policy": "4", "Cache-Control": "5", "X-Custom": "6", }, }, } for _, s := range scenarios { res := httptest.NewRecorder() req := httptest.NewRequest("GET", "/", nil) for k, v := range s.customHeaders { res.Header().Set(k, v) } err := fs.Serve(res, req, s.path, s.name) hasErr := err != nil if hasErr != s.expectError { t.Errorf("(%s) Expected hasError %v, got %v (%v)", s.path, s.expectError, hasErr, err) continue } if s.expectError { continue } result := res.Result() for hName, hValue := range s.expectHeaders { v := result.Header.Get(hName) if v != hValue { t.Errorf("(%s) Expected value %q for header %q, got %q", s.path, hValue, hName, v) } } } } func TestFileSystemGetFile(t *testing.T) { dir := createTestDir(t) defer os.RemoveAll(dir) fs, err := filesystem.NewLocal(dir) if err != nil { t.Fatal(err) } defer fs.Close() f, err := fs.GetFile("image.png") if err != nil { t.Fatal(err) } defer f.Close() if f == nil { t.Fatal("File is supposed to be found") } } func TestFileSystemServeSingleRange(t *testing.T) { dir := createTestDir(t) defer os.RemoveAll(dir) fs, err := filesystem.NewLocal(dir) if err != nil { t.Fatal(err) } defer fs.Close() res := httptest.NewRecorder() req := httptest.NewRequest("GET", "/", nil) req.Header.Add("Range", "bytes=0-20") if err := fs.Serve(res, req, "image.png", "image.png"); err != nil { t.Fatal(err) } result := res.Result() if result.StatusCode != http.StatusPartialContent { t.Fatalf("Expected StatusCode %d, got %d", http.StatusPartialContent, result.StatusCode) } expectedRange := "bytes 0-20/73" if cr := result.Header.Get("Content-Range"); cr != expectedRange { t.Fatalf("Expected Content-Range %q, got %q", expectedRange, cr) } if l := result.Header.Get("Content-Length"); l != "21" { t.Fatalf("Expected Content-Length %v, got %v", 21, l) } } func TestFileSystemServeMultiRange(t *testing.T) { dir := createTestDir(t) defer os.RemoveAll(dir) fs, err := filesystem.NewLocal(dir) if err != nil { t.Fatal(err) } defer fs.Close() res := httptest.NewRecorder() req := httptest.NewRequest("GET", "/", nil) req.Header.Add("Range", "bytes=0-20, 25-30") if err := fs.Serve(res, req, "image.png", "image.png"); err != nil { t.Fatal(err) } result := res.Result() if result.StatusCode != http.StatusPartialContent { t.Fatalf("Expected StatusCode %d, got %d", http.StatusPartialContent, result.StatusCode) } if ct := result.Header.Get("Content-Type"); !strings.HasPrefix(ct, "multipart/byteranges; boundary=") { t.Fatalf("Expected Content-Type to be multipart/byteranges, got %v", ct) } } func TestFileSystemCreateThumb(t *testing.T) { dir := createTestDir(t) defer os.RemoveAll(dir) fs, err := filesystem.NewLocal(dir) if err != nil { t.Fatal(err) } defer fs.Close() scenarios := []struct { file string thumb string cropCenter bool expectError bool }{ // missing {"missing.txt", "thumb_test_missing", true, true}, // non-image existing file {"test/sub1.txt", "thumb_test_sub1", true, true}, // existing image file - crop center {"image.png", "thumb_file_center", true, false}, // existing image file - crop top {"image.png", "thumb_file_top", false, false}, // existing image file with existing thumb path = should fail {"image.png", "test", true, true}, } for i, scenario := range scenarios { err := fs.CreateThumb(scenario.file, scenario.thumb, "100x100") hasErr := err != nil if hasErr != scenario.expectError { t.Errorf("(%d) Expected hasErr to be %v, got %v (%v)", i, scenario.expectError, hasErr, err) continue } if scenario.expectError { continue } if exists, _ := fs.Exists(scenario.thumb); !exists { t.Errorf("(%d) Couldn't find %q thumb", i, scenario.thumb) } } } // --- func createTestDir(t *testing.T) string { dir, err := os.MkdirTemp(os.TempDir(), "pb_test") if err != nil { t.Fatal(err) } if err := os.MkdirAll(filepath.Join(dir, "test"), os.ModePerm); err != nil { t.Fatal(err) } file1, err := os.OpenFile(filepath.Join(dir, "test/sub1.txt"), os.O_WRONLY|os.O_CREATE, 0644) if err != nil { t.Fatal(err) } file1.Close() file2, err := os.OpenFile(filepath.Join(dir, "test/sub2.txt"), os.O_WRONLY|os.O_CREATE, 0644) if err != nil { t.Fatal(err) } file2.Close() file3, err := os.OpenFile(filepath.Join(dir, "image.png"), os.O_WRONLY|os.O_CREATE, 0644) if err != nil { t.Fatal(err) } imgRect := image.Rect(0, 0, 1, 1) // tiny 1x1 png png.Encode(file3, imgRect) file3.Close() err2 := os.WriteFile(filepath.Join(dir, "image.png.attrs"), []byte(`{"user.cache_control":"","user.content_disposition":"","user.content_encoding":"","user.content_language":"","user.content_type":"image/png","user.metadata":null}`), 0644) if err2 != nil { t.Fatal(err2) } file4, err := os.OpenFile(filepath.Join(dir, "image.svg"), os.O_WRONLY|os.O_CREATE, 0644) if err != nil { t.Fatal(err) } file4.Close() file5, err := os.OpenFile(filepath.Join(dir, "style.css"), os.O_WRONLY|os.O_CREATE, 0644) if err != nil { t.Fatal(err) } file5.Close() file6, err := os.OpenFile(filepath.Join(dir, "image_! noext"), os.O_WRONLY|os.O_CREATE, 0644) if err != nil { t.Fatal(err) } png.Encode(file6, image.Rect(0, 0, 1, 1)) // tiny 1x1 png file6.Close() return dir }