image_meta.go 1.4 KB

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