image_data.go 5.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203
  1. package imagedata
  2. import (
  3. "bytes"
  4. "context"
  5. "io"
  6. "net/http"
  7. "sync"
  8. "github.com/imgproxy/imgproxy/v3/asyncbuffer"
  9. "github.com/imgproxy/imgproxy/v3/config"
  10. "github.com/imgproxy/imgproxy/v3/ierrors"
  11. "github.com/imgproxy/imgproxy/v3/imagetype"
  12. )
  13. var (
  14. Watermark ImageData
  15. FallbackImage ImageData
  16. FallbackImageHeaders http.Header // Headers for the fallback image
  17. )
  18. // ImageData represents the data of an image that can be read from a source.
  19. // Please note that this interface can be backed by any reader, including lazy AsyncBuffer.
  20. // There is no other way to guarantee that the data is read without errors except reading it till EOF.
  21. type ImageData interface {
  22. io.Closer // Close closes the image data and releases any resources held by it
  23. Reader() io.ReadSeeker // Reader returns a new ReadSeeker for the image data
  24. Format() imagetype.Type // Format returns the image format from the metadata (shortcut)
  25. Size() (int, error) // Size returns the size of the image data in bytes
  26. Error() error // Error returns any error that occurred during reading data from source
  27. // AddCancel attaches a cancel function to the image data.
  28. // Please note that Cancel functions must be idempotent: for instance, an implementation
  29. // could wrap cancel into sync.Once.
  30. AddCancel(context.CancelFunc)
  31. }
  32. // imageDataBytes represents image data stored in a byte slice in memory
  33. type imageDataBytes struct {
  34. format imagetype.Type
  35. data []byte
  36. cancel []context.CancelFunc
  37. cancelOnce sync.Once
  38. }
  39. // imageDataAsyncBuffer is a struct that implements the ImageData interface backed by an AsyncBuffer
  40. type imageDataAsyncBuffer struct {
  41. b *asyncbuffer.AsyncBuffer
  42. format imagetype.Type
  43. cancel []context.CancelFunc
  44. cancelOnce sync.Once
  45. }
  46. // Close closes the image data and releases any resources held by it
  47. func (d *imageDataBytes) Close() error {
  48. d.cancelOnce.Do(func() {
  49. for _, cancel := range d.cancel {
  50. cancel()
  51. }
  52. })
  53. return nil
  54. }
  55. // Format returns the image format based on the metadata
  56. func (d *imageDataBytes) Format() imagetype.Type {
  57. return d.format
  58. }
  59. // Reader returns an io.ReadSeeker for the image data
  60. func (d *imageDataBytes) Reader() io.ReadSeeker {
  61. return bytes.NewReader(d.data)
  62. }
  63. // Size returns the size of the image data in bytes.
  64. func (d *imageDataBytes) Size() (int, error) {
  65. return len(d.data), nil
  66. }
  67. // AddCancel attaches a cancel function to the image data
  68. func (d *imageDataBytes) AddCancel(cancel context.CancelFunc) {
  69. d.cancel = append(d.cancel, cancel)
  70. }
  71. func (d *imageDataBytes) Error() error {
  72. // No error handling for in-memory data, return nil
  73. return nil
  74. }
  75. // Reader returns a ReadSeeker for the image data
  76. func (d *imageDataAsyncBuffer) Reader() io.ReadSeeker {
  77. return d.b.Reader()
  78. }
  79. // Close closes the response body (hence, response) and the async buffer itself
  80. func (d *imageDataAsyncBuffer) Close() error {
  81. d.cancelOnce.Do(func() {
  82. d.b.Close()
  83. for _, cancel := range d.cancel {
  84. cancel()
  85. }
  86. })
  87. return nil
  88. }
  89. // Format returns the image format from the metadata
  90. func (d *imageDataAsyncBuffer) Format() imagetype.Type {
  91. return d.format
  92. }
  93. // Size returns the size of the image data in bytes.
  94. // It waits for the async buffer to finish reading.
  95. func (d *imageDataAsyncBuffer) Size() (int, error) {
  96. return d.b.Wait()
  97. }
  98. // AddCancel attaches a cancel function to the image data
  99. func (d *imageDataAsyncBuffer) AddCancel(cancel context.CancelFunc) {
  100. d.cancel = append(d.cancel, cancel)
  101. }
  102. // Error returns any error that occurred during reading data from
  103. // async buffer or the underlying source.
  104. func (d *imageDataAsyncBuffer) Error() error {
  105. return d.b.Error()
  106. }
  107. func Init() error {
  108. if err := initDownloading(); err != nil {
  109. return err
  110. }
  111. if err := loadWatermark(); err != nil {
  112. return err
  113. }
  114. if err := loadFallbackImage(); err != nil {
  115. return err
  116. }
  117. return nil
  118. }
  119. func loadWatermark() error {
  120. var err error
  121. switch {
  122. case len(config.WatermarkData) > 0:
  123. Watermark, err = NewFromBase64(config.WatermarkData)
  124. // NOTE: this should be something like err = ierrors.Wrap(err).WithStackDeep(0).WithPrefix("watermark")
  125. // In the NewFromBase64 all errors should be wrapped to something like
  126. // .WithPrefix("load from base64")
  127. if err != nil {
  128. return ierrors.Wrap(err, 0, ierrors.WithPrefix("can't load watermark from Base64"))
  129. }
  130. case len(config.WatermarkPath) > 0:
  131. Watermark, err = NewFromPath(config.WatermarkPath)
  132. if err != nil {
  133. return ierrors.Wrap(err, 0, ierrors.WithPrefix("can't read watermark from file"))
  134. }
  135. case len(config.WatermarkURL) > 0:
  136. Watermark, _, err = DownloadSync(context.Background(), config.WatermarkURL, "watermark", DefaultDownloadOptions())
  137. if err != nil {
  138. return ierrors.Wrap(err, 0, ierrors.WithPrefix("can't download from URL"))
  139. }
  140. default:
  141. Watermark = nil
  142. }
  143. return nil
  144. }
  145. func loadFallbackImage() (err error) {
  146. switch {
  147. case len(config.FallbackImageData) > 0:
  148. FallbackImage, err = NewFromBase64(config.FallbackImageData)
  149. if err != nil {
  150. return ierrors.Wrap(err, 0, ierrors.WithPrefix("can't load fallback image from Base64"))
  151. }
  152. case len(config.FallbackImagePath) > 0:
  153. FallbackImage, err = NewFromPath(config.FallbackImagePath)
  154. if err != nil {
  155. return ierrors.Wrap(err, 0, ierrors.WithPrefix("can't read fallback image from file"))
  156. }
  157. case len(config.FallbackImageURL) > 0:
  158. FallbackImage, FallbackImageHeaders, err = DownloadSync(context.Background(), config.FallbackImageURL, "fallback image", DefaultDownloadOptions())
  159. if err != nil {
  160. return ierrors.Wrap(err, 0, ierrors.WithPrefix("can't download from URL"))
  161. }
  162. default:
  163. FallbackImage = nil
  164. }
  165. return err
  166. }