s3manager-web/internal/app/s3manager/bucket_view_test.go
2023-06-13 18:34:15 +02:00

182 lines
5.2 KiB
Go

package s3manager_test
import (
"context"
"fmt"
"io"
"net/http"
"net/http/httptest"
"os"
"path/filepath"
"strings"
"testing"
"github.com/cloudlena/s3manager/internal/app/s3manager"
"github.com/cloudlena/s3manager/internal/app/s3manager/mocks"
"github.com/gorilla/mux"
"github.com/matryer/is"
"github.com/minio/minio-go/v7"
)
func TestHandleBucketView(t *testing.T) {
t.Parallel()
cases := []struct {
it string
listObjectsFunc func(context.Context, string, minio.ListObjectsOptions) <-chan minio.ObjectInfo
bucketName string
path string
expectedStatusCode int
expectedBodyContains string
}{
{
it: "renders a bucket containing a file",
listObjectsFunc: func(context.Context, string, minio.ListObjectsOptions) <-chan minio.ObjectInfo {
objCh := make(chan minio.ObjectInfo)
go func() {
objCh <- minio.ObjectInfo{Key: "FILE-NAME"}
close(objCh)
}()
return objCh
},
bucketName: "BUCKET-NAME",
expectedStatusCode: http.StatusOK,
expectedBodyContains: "FILE-NAME",
},
{
it: "renders placeholder for an empty bucket",
listObjectsFunc: func(context.Context, string, minio.ListObjectsOptions) <-chan minio.ObjectInfo {
objCh := make(chan minio.ObjectInfo)
close(objCh)
return objCh
},
bucketName: "BUCKET-NAME",
expectedStatusCode: http.StatusOK,
expectedBodyContains: "No objects in",
},
{
it: "renders a bucket containing an archive",
listObjectsFunc: func(context.Context, string, minio.ListObjectsOptions) <-chan minio.ObjectInfo {
objCh := make(chan minio.ObjectInfo)
go func() {
objCh <- minio.ObjectInfo{Key: "archive.tar.gz"}
close(objCh)
}()
return objCh
},
bucketName: "BUCKET-NAME",
expectedStatusCode: http.StatusOK,
expectedBodyContains: "archive",
},
{
it: "renders a bucket containing an image",
listObjectsFunc: func(context.Context, string, minio.ListObjectsOptions) <-chan minio.ObjectInfo {
objCh := make(chan minio.ObjectInfo)
go func() {
objCh <- minio.ObjectInfo{Key: "FILE-NAME.png"}
close(objCh)
}()
return objCh
},
bucketName: "BUCKET-NAME",
expectedStatusCode: http.StatusOK,
expectedBodyContains: "photo",
},
{
it: "renders a bucket containing a sound file",
listObjectsFunc: func(context.Context, string, minio.ListObjectsOptions) <-chan minio.ObjectInfo {
objCh := make(chan minio.ObjectInfo)
go func() {
objCh <- minio.ObjectInfo{Key: "FILE-NAME.mp3"}
close(objCh)
}()
return objCh
},
bucketName: "BUCKET-NAME",
expectedStatusCode: http.StatusOK,
expectedBodyContains: "music_note",
},
{
it: "returns error if the bucket doesn't exist",
listObjectsFunc: func(context.Context, string, minio.ListObjectsOptions) <-chan minio.ObjectInfo {
objCh := make(chan minio.ObjectInfo)
go func() {
objCh <- minio.ObjectInfo{Err: errBucketDoesNotExist}
close(objCh)
}()
return objCh
},
bucketName: "BUCKET-NAME",
expectedStatusCode: http.StatusNotFound,
expectedBodyContains: http.StatusText(http.StatusNotFound),
},
{
it: "returns error if there is an S3 error",
listObjectsFunc: func(context.Context, string, minio.ListObjectsOptions) <-chan minio.ObjectInfo {
objCh := make(chan minio.ObjectInfo)
go func() {
objCh <- minio.ObjectInfo{Err: errS3}
close(objCh)
}()
return objCh
},
bucketName: "BUCKET-NAME",
expectedStatusCode: http.StatusInternalServerError,
expectedBodyContains: http.StatusText(http.StatusInternalServerError),
},
{
it: "renders a bucket with folder",
listObjectsFunc: func(context.Context, string, minio.ListObjectsOptions) <-chan minio.ObjectInfo {
objCh := make(chan minio.ObjectInfo)
go func() {
objCh <- minio.ObjectInfo{Key: "AFolder/"}
close(objCh)
}()
return objCh
},
bucketName: "BUCKET-NAME",
expectedStatusCode: http.StatusOK,
expectedBodyContains: "folder",
},
{
it: "renders a bucket with path",
listObjectsFunc: func(context.Context, string, minio.ListObjectsOptions) <-chan minio.ObjectInfo {
objCh := make(chan minio.ObjectInfo)
close(objCh)
return objCh
},
bucketName: "BUCKET-NAME",
path: "abc/def",
expectedStatusCode: http.StatusOK,
expectedBodyContains: "def",
},
}
for _, tc := range cases {
tc := tc
t.Run(tc.it, func(t *testing.T) {
t.Parallel()
is := is.New(t)
s3 := &mocks.S3Mock{
ListObjectsFunc: tc.listObjectsFunc,
}
templates := os.DirFS(filepath.Join("..", "..", "..", "web", "template"))
r := mux.NewRouter()
r.PathPrefix("/buckets/").Handler(s3manager.HandleBucketView(s3, templates, true, true)).Methods(http.MethodGet)
ts := httptest.NewServer(r)
defer ts.Close()
resp, err := http.Get(fmt.Sprintf("%s/buckets/%s/%s", ts.URL, tc.bucketName, tc.path))
is.NoErr(err)
defer resp.Body.Close()
body, err := io.ReadAll(resp.Body)
is.NoErr(err)
is.Equal(tc.expectedStatusCode, resp.StatusCode) // status code
is.True(strings.Contains(string(body), tc.expectedBodyContains)) // body
})
}
}