image_meta.go 1.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100
  1. package imagemeta
  2. import (
  3. "bufio"
  4. "errors"
  5. "io"
  6. "sync"
  7. "sync/atomic"
  8. "github.com/imgproxy/imgproxy/v3/imagetype"
  9. )
  10. type Meta interface {
  11. Format() imagetype.Type
  12. Width() int
  13. Height() int
  14. }
  15. type DecodeMetaFunc func(io.Reader) (Meta, error)
  16. type meta struct {
  17. format imagetype.Type
  18. width, height int
  19. }
  20. func (m *meta) Format() imagetype.Type {
  21. return m.format
  22. }
  23. func (m *meta) Width() int {
  24. return m.width
  25. }
  26. func (m *meta) Height() int {
  27. return m.height
  28. }
  29. type format struct {
  30. magic string
  31. decodeMeta DecodeMetaFunc
  32. }
  33. type reader interface {
  34. io.Reader
  35. Peek(int) ([]byte, error)
  36. }
  37. var (
  38. formatsMu sync.Mutex
  39. atomicFormats atomic.Value
  40. ErrFormat = errors.New("unknown image format")
  41. )
  42. func asReader(r io.Reader) reader {
  43. if rr, ok := r.(reader); ok {
  44. return rr
  45. }
  46. return bufio.NewReader(r)
  47. }
  48. func matchMagic(magic string, b []byte) bool {
  49. if len(magic) != len(b) {
  50. return false
  51. }
  52. for i, c := range b {
  53. if magic[i] != c && magic[i] != '?' {
  54. return false
  55. }
  56. }
  57. return true
  58. }
  59. func RegisterFormat(magic string, decodeMeta DecodeMetaFunc) {
  60. formatsMu.Lock()
  61. defer formatsMu.Unlock()
  62. formats, _ := atomicFormats.Load().([]format)
  63. atomicFormats.Store(append(formats, format{magic, decodeMeta}))
  64. }
  65. func DecodeMeta(r io.Reader) (Meta, error) {
  66. rr := asReader(r)
  67. formats, _ := atomicFormats.Load().([]format)
  68. for _, f := range formats {
  69. b, err := rr.Peek(len(f.magic))
  70. if err == nil && matchMagic(f.magic, b) {
  71. return f.decodeMeta(rr)
  72. }
  73. }
  74. if ok, err := IsSVG(rr); err != nil {
  75. return nil, err
  76. } else if ok {
  77. return &meta{format: imagetype.SVG, width: 1, height: 1}, nil
  78. }
  79. return nil, ErrFormat
  80. }