image_data.go 4.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157
  1. package imagedata
  2. import (
  3. "bytes"
  4. "context"
  5. "fmt"
  6. "io"
  7. "net/http"
  8. "sync"
  9. "github.com/imgproxy/imgproxy/v3/config"
  10. "github.com/imgproxy/imgproxy/v3/ierrors"
  11. "github.com/imgproxy/imgproxy/v3/imagetype"
  12. "github.com/imgproxy/imgproxy/v3/security"
  13. )
  14. var (
  15. Watermark ImageData
  16. FallbackImage ImageData
  17. FallbackImageHeaders http.Header // Headers for the fallback image
  18. )
  19. type ImageData interface {
  20. io.Closer // Close closes the image data and releases any resources held by it
  21. Reader() io.ReadSeeker // Reader returns a new ReadSeeker for the image data
  22. Format() imagetype.Type // Format returns the image format from the metadata (shortcut)
  23. Size() (int, error) // Size returns the size of the image data in bytes
  24. AddCancel(context.CancelFunc) // AddCancel attaches a cancel function to the image data
  25. }
  26. // imageDataBytes represents image data stored in a byte slice in memory
  27. type imageDataBytes struct {
  28. format imagetype.Type
  29. data []byte
  30. cancel []context.CancelFunc
  31. cancelOnce sync.Once
  32. }
  33. func (d *imageDataBytes) Close() error {
  34. d.cancelOnce.Do(func() {
  35. for _, cancel := range d.cancel {
  36. cancel()
  37. }
  38. })
  39. return nil
  40. }
  41. // Format returns the image format based on the metadata
  42. func (d *imageDataBytes) Format() imagetype.Type {
  43. return d.format
  44. }
  45. // Reader returns an io.ReadSeeker for the image data
  46. func (d *imageDataBytes) Reader() io.ReadSeeker {
  47. return bytes.NewReader(d.data)
  48. }
  49. // Size returns the size of the image data in bytes.
  50. // NOTE: asyncbuffer implementation will .Wait() for the data to be fully read
  51. func (d *imageDataBytes) Size() (int, error) {
  52. return len(d.data), nil
  53. }
  54. func (d *imageDataBytes) AddCancel(cancel context.CancelFunc) {
  55. d.cancel = append(d.cancel, cancel)
  56. }
  57. func Init() error {
  58. initRead()
  59. if err := initDownloading(); err != nil {
  60. return err
  61. }
  62. if err := loadWatermark(); err != nil {
  63. return err
  64. }
  65. if err := loadFallbackImage(); err != nil {
  66. return err
  67. }
  68. return nil
  69. }
  70. func loadWatermark() error {
  71. var err error
  72. switch {
  73. case len(config.WatermarkData) > 0:
  74. Watermark, err = NewFromBase64(config.WatermarkData, security.DefaultOptions())
  75. // NOTE: this should be something like err = ierrors.Wrap(err).WithStackDeep(0).WithPrefix("watermark")
  76. // In the NewFromBase64 all errors should be wrapped to something like
  77. // .WithPrefix("load from base64")
  78. if err != nil {
  79. return ierrors.Wrap(err, 0, ierrors.WithPrefix("can't load watermark from Base64"))
  80. }
  81. case len(config.WatermarkPath) > 0:
  82. Watermark, err = NewFromPath(config.WatermarkPath, security.DefaultOptions())
  83. if err != nil {
  84. return ierrors.Wrap(err, 0, ierrors.WithPrefix("can't read watermark from file"))
  85. }
  86. case len(config.WatermarkURL) > 0:
  87. Watermark, _, err = Download(context.Background(), config.WatermarkURL, "watermark", DownloadOptions{Header: nil, CookieJar: nil}, security.DefaultOptions())
  88. if err != nil {
  89. return ierrors.Wrap(err, 0, ierrors.WithPrefix("can't download from URL"))
  90. }
  91. default:
  92. Watermark = nil
  93. }
  94. return nil
  95. }
  96. func loadFallbackImage() (err error) {
  97. switch {
  98. case len(config.FallbackImageData) > 0:
  99. FallbackImage, err = NewFromBase64(config.FallbackImageData, security.DefaultOptions())
  100. if err != nil {
  101. return ierrors.Wrap(err, 0, ierrors.WithPrefix("can't load fallback image from Base64"))
  102. }
  103. case len(config.FallbackImagePath) > 0:
  104. FallbackImage, err = NewFromPath(config.FallbackImagePath, security.DefaultOptions())
  105. if err != nil {
  106. return ierrors.Wrap(err, 0, ierrors.WithPrefix("can't read fallback image from file"))
  107. }
  108. case len(config.FallbackImageURL) > 0:
  109. FallbackImage, FallbackImageHeaders, err = Download(context.Background(), config.FallbackImageURL, "fallback image", DownloadOptions{Header: nil, CookieJar: nil}, security.DefaultOptions())
  110. if err != nil {
  111. return ierrors.Wrap(err, 0, ierrors.WithPrefix("can't download from URL"))
  112. }
  113. default:
  114. FallbackImage = nil
  115. }
  116. return err
  117. }
  118. func Download(ctx context.Context, imageURL, desc string, opts DownloadOptions, secopts security.Options) (ImageData, http.Header, error) {
  119. imgdata, h, err := download(ctx, imageURL, opts, secopts)
  120. if err != nil {
  121. return nil, h, ierrors.Wrap(
  122. err, 0,
  123. ierrors.WithPrefix(fmt.Sprintf("Can't download %s", desc)),
  124. )
  125. }
  126. return imgdata, h, nil
  127. }