agi.image_test.go 3.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111
  1. package agi
  2. import (
  3. "bytes"
  4. "image"
  5. "image/jpeg"
  6. "os"
  7. "path/filepath"
  8. "testing"
  9. "time"
  10. "imuslab.com/arozos/mod/filesystem"
  11. "imuslab.com/arozos/mod/filesystem/abstractions/localfs"
  12. )
  13. // newImageTestFSH creates a FileSystemHandler backed by a temporary local
  14. // directory for exercising the imagelib helpers.
  15. func newImageTestFSH(t *testing.T) (*filesystem.FileSystemHandler, string) {
  16. t.Helper()
  17. dir := t.TempDir()
  18. abs := localfs.NewLocalFileSystemAbstraction("TEST", dir+"/", "public", false)
  19. fsh := &filesystem.FileSystemHandler{
  20. Name: "test",
  21. UUID: "TEST",
  22. Path: dir + "/",
  23. ReadOnly: false,
  24. Hierarchy: "public",
  25. InitiationTime: time.Now().Unix(),
  26. FileSystemAbstraction: abs,
  27. Filesystem: "ext4",
  28. }
  29. return fsh, dir
  30. }
  31. // smallJPEGBytes returns a small solid image encoded as JPEG. Written into a
  32. // RAW-extension file it stands in for the embedded preview that real RAW photos
  33. // carry, which is what convertRawToJPEG extracts.
  34. func smallJPEGBytes(t *testing.T) []byte {
  35. t.Helper()
  36. img := image.NewRGBA(image.Rect(0, 0, 32, 24))
  37. var buf bytes.Buffer
  38. if err := jpeg.Encode(&buf, img, &jpeg.Options{Quality: 90}); err != nil {
  39. t.Fatalf("jpeg encode: %v", err)
  40. }
  41. return buf.Bytes()
  42. }
  43. func TestConvertRawToJPEG(t *testing.T) {
  44. jpegBytes := smallJPEGBytes(t)
  45. t.Run("RAW with embedded JPEG is converted", func(t *testing.T) {
  46. fsh, dir := newImageTestFSH(t)
  47. src := filepath.Join(dir, "photo.arw")
  48. if err := os.WriteFile(src, jpegBytes, 0644); err != nil {
  49. t.Fatalf("write src: %v", err)
  50. }
  51. dst := filepath.Join(dir, "photo.jpg")
  52. if err := convertRawToJPEG(fsh, src, fsh, dst); err != nil {
  53. t.Fatalf("convertRawToJPEG returned error: %v", err)
  54. }
  55. out, err := os.ReadFile(dst)
  56. if err != nil {
  57. t.Fatalf("output not written: %v", err)
  58. }
  59. img, format, err := image.Decode(bytes.NewReader(out))
  60. if err != nil {
  61. t.Fatalf("output is not a decodable image: %v", err)
  62. }
  63. if format != "jpeg" {
  64. t.Errorf("expected jpeg output, got %q", format)
  65. }
  66. if b := img.Bounds(); b.Dx() != 32 || b.Dy() != 24 {
  67. t.Errorf("unexpected output dimensions: %dx%d", b.Dx(), b.Dy())
  68. }
  69. })
  70. t.Run("non-RAW source is rejected", func(t *testing.T) {
  71. fsh, dir := newImageTestFSH(t)
  72. src := filepath.Join(dir, "photo.png")
  73. if err := os.WriteFile(src, jpegBytes, 0644); err != nil {
  74. t.Fatalf("write src: %v", err)
  75. }
  76. if err := convertRawToJPEG(fsh, src, fsh, filepath.Join(dir, "out.jpg")); err == nil {
  77. t.Errorf("expected error for non-RAW source, got nil")
  78. }
  79. })
  80. t.Run("non-JPEG output extension is rejected", func(t *testing.T) {
  81. fsh, dir := newImageTestFSH(t)
  82. src := filepath.Join(dir, "photo.arw")
  83. if err := os.WriteFile(src, jpegBytes, 0644); err != nil {
  84. t.Fatalf("write src: %v", err)
  85. }
  86. if err := convertRawToJPEG(fsh, src, fsh, filepath.Join(dir, "out.png")); err == nil {
  87. t.Errorf("expected error for non-JPEG output, got nil")
  88. }
  89. })
  90. t.Run("RAW without embedded JPEG fails cleanly", func(t *testing.T) {
  91. fsh, dir := newImageTestFSH(t)
  92. src := filepath.Join(dir, "garbage.cr2")
  93. if err := os.WriteFile(src, []byte("not a real raw file"), 0644); err != nil {
  94. t.Fatalf("write src: %v", err)
  95. }
  96. if err := convertRawToJPEG(fsh, src, fsh, filepath.Join(dir, "out.jpg")); err == nil {
  97. t.Errorf("expected error for RAW without embedded JPEG, got nil")
  98. }
  99. })
  100. }