summaryrefslogtreecommitdiff
path: root/vendor/github.com/dsoprea/go-png-image-structure/v2/media_parser.go
blob: c0e2873654e432a736d93d29229a34ef77b2de62 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
package pngstructure

import (
	"bufio"
	"bytes"
	"image"
	"io"
	"os"

	"image/png"

	"github.com/dsoprea/go-logging"
	"github.com/dsoprea/go-utility/v2/image"
)

// PngMediaParser knows how to parse a PNG stream.
type PngMediaParser struct {
}

// NewPngMediaParser returns a new `PngMediaParser` struct.
func NewPngMediaParser() *PngMediaParser {

	// TODO(dustin): Add test

	return new(PngMediaParser)
}

// Parse parses a PNG stream given a `io.ReadSeeker`.
func (pmp *PngMediaParser) Parse(rs io.ReadSeeker, size int) (mc riimage.MediaContext, err error) {
	defer func() {
		if state := recover(); state != nil {
			err = log.Wrap(state.(error))
		}
	}()

	// TODO(dustin): Add test

	ps := NewPngSplitter()

	err = ps.readHeader(rs)
	log.PanicIf(err)

	s := bufio.NewScanner(rs)

	// Since each segment can be any size, our buffer must be allowed to grow
	// as large as the file.
	buffer := []byte{}
	s.Buffer(buffer, size)
	s.Split(ps.Split)

	for s.Scan() != false {
	}

	log.PanicIf(s.Err())

	return ps.Chunks(), nil
}

// ParseFile parses a PNG stream given a file-path.
func (pmp *PngMediaParser) ParseFile(filepath string) (mc riimage.MediaContext, err error) {
	defer func() {
		if state := recover(); state != nil {
			err = log.Wrap(state.(error))
		}
	}()

	f, err := os.Open(filepath)
	log.PanicIf(err)

	defer f.Close()

	stat, err := f.Stat()
	log.PanicIf(err)

	size := stat.Size()

	chunks, err := pmp.Parse(f, int(size))
	log.PanicIf(err)

	return chunks, nil
}

// ParseBytes parses a PNG stream given a byte-slice.
func (pmp *PngMediaParser) ParseBytes(data []byte) (mc riimage.MediaContext, err error) {
	defer func() {
		if state := recover(); state != nil {
			err = log.Wrap(state.(error))
		}
	}()

	// TODO(dustin): Add test

	br := bytes.NewReader(data)

	chunks, err := pmp.Parse(br, len(data))
	log.PanicIf(err)

	return chunks, nil
}

// LooksLikeFormat returns a boolean indicating whether the stream looks like a
// PNG image.
func (pmp *PngMediaParser) LooksLikeFormat(data []byte) bool {
	return bytes.Compare(data[:len(PngSignature)], PngSignature[:]) == 0
}

// GetImage returns an image.Image-compatible struct.
func (pmp *PngMediaParser) GetImage(r io.Reader) (img image.Image, err error) {
	img, err = png.Decode(r)
	log.PanicIf(err)

	return img, nil
}

var (
	// Enforce interface conformance.
	_ riimage.MediaParser = new(PngMediaParser)
)