2026-05-29 23:44:05 +03:00
|
|
|
package jobs
|
|
|
|
|
|
|
|
|
|
import (
|
2026-06-08 03:43:43 +03:00
|
|
|
"archive/zip"
|
2026-05-29 23:44:05 +03:00
|
|
|
"bytes"
|
2026-06-08 03:43:43 +03:00
|
|
|
"encoding/json"
|
2026-05-29 23:44:05 +03:00
|
|
|
"image"
|
|
|
|
|
"image/color"
|
2026-06-03 14:55:19 +03:00
|
|
|
"image/jpeg"
|
2026-05-29 23:44:05 +03:00
|
|
|
"image/png"
|
|
|
|
|
"io"
|
|
|
|
|
"log/slog"
|
|
|
|
|
"mime/multipart"
|
|
|
|
|
"net/http/httptest"
|
|
|
|
|
"net/textproto"
|
2026-06-03 14:55:19 +03:00
|
|
|
"strings"
|
2026-05-29 23:44:05 +03:00
|
|
|
"testing"
|
|
|
|
|
|
|
|
|
|
"warpbox.dev/backend/libs/services"
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
func TestGenerateMissingThumbnailsForBox(t *testing.T) {
|
|
|
|
|
service := newThumbnailTestUploadService(t)
|
|
|
|
|
result := createThumbnailTestBox(t, service)
|
|
|
|
|
box, err := service.GetBox(result.BoxID)
|
|
|
|
|
if err != nil {
|
|
|
|
|
t.Fatalf("GetBox returned error: %v", err)
|
|
|
|
|
}
|
|
|
|
|
if box.Files[0].Thumbnail != "" {
|
|
|
|
|
t.Fatalf("thumbnail should start empty")
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
jobResult, err := generateMissingThumbnailsForBox(service, slog.New(slog.NewTextHandler(io.Discard, nil)), box)
|
|
|
|
|
if err != nil {
|
|
|
|
|
t.Fatalf("generateMissingThumbnailsForBox returned error: %v", err)
|
|
|
|
|
}
|
|
|
|
|
if jobResult.Generated != 1 || jobResult.Failed != 0 {
|
|
|
|
|
t.Fatalf("job result = %+v, want 1 generated and 0 failed", jobResult)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
updated, err := service.GetBox(result.BoxID)
|
|
|
|
|
if err != nil {
|
|
|
|
|
t.Fatalf("GetBox after thumbnail returned error: %v", err)
|
|
|
|
|
}
|
|
|
|
|
if updated.Files[0].Thumbnail == "" {
|
|
|
|
|
t.Fatalf("thumbnail was not saved to box metadata")
|
|
|
|
|
}
|
|
|
|
|
if service.ThumbnailPath(updated, updated.Files[0]) == "" {
|
|
|
|
|
t.Fatalf("thumbnail path was empty")
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2026-06-03 14:55:19 +03:00
|
|
|
func TestCreateTextThumbnailRendersMarkdownAsJPEG(t *testing.T) {
|
|
|
|
|
data, err := createTextThumbnail(services.File{
|
|
|
|
|
Name: "notes.md",
|
|
|
|
|
ContentType: "text/markdown",
|
|
|
|
|
}, strings.NewReader("# Meeting notes\n\n```go\nfunc main() {}\n```\n\nA rendered Markdown preview."))
|
|
|
|
|
if err != nil {
|
|
|
|
|
t.Fatalf("createTextThumbnail returned error: %v", err)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
img, err := jpeg.Decode(bytes.NewReader(data))
|
|
|
|
|
if err != nil {
|
|
|
|
|
t.Fatalf("jpeg.Decode returned error: %v", err)
|
|
|
|
|
}
|
|
|
|
|
if img.Bounds().Dx() != 360 || img.Bounds().Dy() != 240 {
|
|
|
|
|
t.Fatalf("thumbnail size = %dx%d, want 360x240", img.Bounds().Dx(), img.Bounds().Dy())
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func TestNeedsThumbnailIncludesCodeTextFiles(t *testing.T) {
|
|
|
|
|
if !needsThumbnail(services.File{Name: "main.go", ContentType: "text/plain"}) {
|
|
|
|
|
t.Fatalf("Go source file should get a text thumbnail")
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2026-06-05 10:42:30 +03:00
|
|
|
func TestUsableVideoFrameRejectsBlackFrame(t *testing.T) {
|
|
|
|
|
var dark bytes.Buffer
|
|
|
|
|
if err := jpeg.Encode(&dark, solidTestImage(color.RGBA{A: 255}), nil); err != nil {
|
|
|
|
|
t.Fatalf("jpeg.Encode dark returned error: %v", err)
|
|
|
|
|
}
|
|
|
|
|
if usableVideoFrame(dark.Bytes()) {
|
|
|
|
|
t.Fatalf("black video frame should not be usable")
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var bright bytes.Buffer
|
|
|
|
|
if err := jpeg.Encode(&bright, solidTestImage(color.RGBA{R: 180, G: 80, B: 40, A: 255}), nil); err != nil {
|
|
|
|
|
t.Fatalf("jpeg.Encode bright returned error: %v", err)
|
|
|
|
|
}
|
|
|
|
|
if !usableVideoFrame(bright.Bytes()) {
|
|
|
|
|
t.Fatalf("bright video frame should be usable")
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func TestRenderVideoScenesThumbnailReturnsLargeJPEG(t *testing.T) {
|
|
|
|
|
data := renderVideoScenesThumbnail(
|
|
|
|
|
services.File{Name: "clip.mp4", ContentType: "video/mp4"},
|
|
|
|
|
videoInfo{Codec: "h264", Width: 1920, Height: 1080, Duration: 125, FrameRate: "24.00 fps"},
|
|
|
|
|
[]videoSceneFrame{
|
|
|
|
|
{Timestamp: "00:00:10", Image: solidTestImage(color.RGBA{R: 140, G: 40, B: 80, A: 255})},
|
|
|
|
|
{Timestamp: "00:00:35", Image: solidTestImage(color.RGBA{R: 40, G: 120, B: 150, A: 255})},
|
|
|
|
|
},
|
|
|
|
|
)
|
|
|
|
|
img, err := jpeg.Decode(bytes.NewReader(data))
|
|
|
|
|
if err != nil {
|
|
|
|
|
t.Fatalf("jpeg.Decode returned error: %v", err)
|
|
|
|
|
}
|
|
|
|
|
if img.Bounds().Dx() != 1200 || img.Bounds().Dy() != 630 {
|
|
|
|
|
t.Fatalf("scene preview size = %dx%d, want 1200x630", img.Bounds().Dx(), img.Bounds().Dy())
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2026-06-08 03:43:43 +03:00
|
|
|
func TestCreateArchiveListingRendersZipTree(t *testing.T) {
|
|
|
|
|
var archive bytes.Buffer
|
|
|
|
|
writer := zip.NewWriter(&archive)
|
|
|
|
|
addZipTestFile(t, writer, "docs/readme.md", "hello")
|
|
|
|
|
addZipTestFile(t, writer, "src/main.go", "package main\n")
|
|
|
|
|
if err := writer.Close(); err != nil {
|
|
|
|
|
t.Fatalf("zip.Close returned error: %v", err)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
data, err := createArchiveListing(services.File{Name: "bundle.zip", ContentType: "application/zip"}, bytes.NewReader(archive.Bytes()))
|
|
|
|
|
if err != nil {
|
|
|
|
|
t.Fatalf("createArchiveListing returned error: %v", err)
|
|
|
|
|
}
|
|
|
|
|
var listing archiveListingData
|
|
|
|
|
if err := json.Unmarshal(data, &listing); err != nil {
|
|
|
|
|
t.Fatalf("json.Unmarshal returned error: %v\n%s", err, string(data))
|
|
|
|
|
}
|
|
|
|
|
if listing.Name != "bundle.zip" || listing.FileCount != 2 || listing.FolderCount != 2 {
|
|
|
|
|
t.Fatalf("archive listing metadata = %+v", listing)
|
|
|
|
|
}
|
|
|
|
|
if listing.Root == nil || len(listing.Root.Items) != 2 {
|
|
|
|
|
t.Fatalf("archive listing root = %+v", listing.Root)
|
|
|
|
|
}
|
|
|
|
|
if listing.Root.Items[0].Name != "docs" || listing.Root.Items[0].Icon != "folder" {
|
|
|
|
|
t.Fatalf("first archive folder = %+v", listing.Root.Items[0])
|
|
|
|
|
}
|
|
|
|
|
if listing.Root.Items[0].Items[0].Name != "readme.md" || listing.Root.Items[0].Items[0].Icon != "txt" {
|
|
|
|
|
t.Fatalf("markdown archive file = %+v", listing.Root.Items[0].Items[0])
|
|
|
|
|
}
|
|
|
|
|
if listing.Root.Items[1].Items[0].Name != "main.go" || listing.Root.Items[1].Items[0].Icon != "code" {
|
|
|
|
|
t.Fatalf("go archive file = %+v", listing.Root.Items[1].Items[0])
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func addZipTestFile(t *testing.T, writer *zip.Writer, name, body string) {
|
|
|
|
|
t.Helper()
|
|
|
|
|
file, err := writer.Create(name)
|
|
|
|
|
if err != nil {
|
|
|
|
|
t.Fatalf("zip.Create returned error: %v", err)
|
|
|
|
|
}
|
|
|
|
|
if _, err := file.Write([]byte(body)); err != nil {
|
|
|
|
|
t.Fatalf("zip file write returned error: %v", err)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2026-06-05 10:42:30 +03:00
|
|
|
func solidTestImage(c color.Color) image.Image {
|
|
|
|
|
img := image.NewRGBA(image.Rect(0, 0, 32, 24))
|
|
|
|
|
for y := 0; y < img.Bounds().Dy(); y++ {
|
|
|
|
|
for x := 0; x < img.Bounds().Dx(); x++ {
|
|
|
|
|
img.Set(x, y, c)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return img
|
|
|
|
|
}
|
|
|
|
|
|
2026-05-29 23:44:05 +03:00
|
|
|
func newThumbnailTestUploadService(t *testing.T) *services.UploadService {
|
|
|
|
|
t.Helper()
|
|
|
|
|
service, err := services.NewUploadService(1024*1024, t.TempDir(), "http://example.test", slog.New(slog.NewTextHandler(io.Discard, nil)))
|
|
|
|
|
if err != nil {
|
|
|
|
|
t.Fatalf("NewUploadService returned error: %v", err)
|
|
|
|
|
}
|
|
|
|
|
t.Cleanup(func() {
|
|
|
|
|
if err := service.Close(); err != nil {
|
|
|
|
|
t.Fatalf("Close returned error: %v", err)
|
|
|
|
|
}
|
|
|
|
|
})
|
|
|
|
|
return service
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func createThumbnailTestBox(t *testing.T, service *services.UploadService) services.UploadResult {
|
|
|
|
|
t.Helper()
|
|
|
|
|
result, err := service.CreateBox(thumbnailTestFileHeaders(t), services.UploadOptions{MaxDays: 1})
|
|
|
|
|
if err != nil {
|
|
|
|
|
t.Fatalf("CreateBox returned error: %v", err)
|
|
|
|
|
}
|
|
|
|
|
return result
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func thumbnailTestFileHeaders(t *testing.T) []*multipart.FileHeader {
|
|
|
|
|
t.Helper()
|
|
|
|
|
var imageData bytes.Buffer
|
|
|
|
|
img := image.NewRGBA(image.Rect(0, 0, 2, 2))
|
|
|
|
|
img.Set(0, 0, color.RGBA{R: 255, A: 255})
|
|
|
|
|
if err := png.Encode(&imageData, img); err != nil {
|
|
|
|
|
t.Fatalf("png.Encode returned error: %v", err)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var payload bytes.Buffer
|
|
|
|
|
writer := multipart.NewWriter(&payload)
|
|
|
|
|
header := make(textproto.MIMEHeader)
|
|
|
|
|
header.Set("Content-Disposition", `form-data; name="file"; filename="thumb.png"`)
|
|
|
|
|
header.Set("Content-Type", "image/png")
|
|
|
|
|
part, err := writer.CreatePart(header)
|
|
|
|
|
if err != nil {
|
|
|
|
|
t.Fatalf("CreateFormFile returned error: %v", err)
|
|
|
|
|
}
|
|
|
|
|
if _, err := part.Write(imageData.Bytes()); err != nil {
|
|
|
|
|
t.Fatalf("part.Write returned error: %v", err)
|
|
|
|
|
}
|
|
|
|
|
if err := writer.Close(); err != nil {
|
|
|
|
|
t.Fatalf("writer.Close returned error: %v", err)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
request := httptest.NewRequest("POST", "/upload", &payload)
|
|
|
|
|
request.Header.Set("Content-Type", writer.FormDataContentType())
|
|
|
|
|
if err := request.ParseMultipartForm(1024 * 1024); err != nil {
|
|
|
|
|
t.Fatalf("ParseMultipartForm returned error: %v", err)
|
|
|
|
|
}
|
|
|
|
|
return request.MultipartForm.File["file"]
|
|
|
|
|
}
|