mattermost-community-enterp.../vendor/github.com/bep/imagemeta/imagedecoder_png.go
Claude ec1f89217a Merge: Complete Mattermost Server with Community Enterprise
Full Mattermost server source with integrated Community Enterprise features.
Includes vendor directory for offline/air-gapped builds.

Structure:
- enterprise-impl/: Enterprise feature implementations
- enterprise-community/: Init files that register implementations
- enterprise/: Bridge imports (community_imports.go)
- vendor/: All dependencies for offline builds

Build (online):
  go build ./cmd/mattermost

Build (offline/air-gapped):
  go build -mod=vendor ./cmd/mattermost

🤖 Generated with [Claude Code](https://claude.com/claude-code)

Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
2025-12-17 23:59:07 +09:00

123 lines
3.4 KiB
Go

// Copyright 2024 Bjørn Erik Pedersen
// SPDX-License-Identifier: MIT
package imagemeta
import (
"bytes"
"compress/zlib"
"encoding/hex"
"fmt"
"io"
)
type imageDecoderPNG struct {
*baseStreamingDecoder
}
// See https://exiftool.org/TagNames/PNG.html
var (
pngTagIDExif = []byte("eXIf")
pngCompressedText = []byte("zTXt") // See https://exiftool.org/forum/index.php?topic=7988.msg40759#msg40759
pngRawProfileTypeIPTC = []byte("Raw profile type iptc")
pngRawProfileTypeEXIF = []byte("Raw profile type exif")
)
func (e *imageDecoderPNG) decode() error {
// Skip header.
e.skip(8)
sources := e.opts.Sources
skipTag := func(chunkLength uint32) {
e.skip(int64(chunkLength))
e.skip(4) // skip CRC
}
for {
if sources.IsZero() {
return nil
}
chunkLength := e.read4()
tagID := e.readBytesVolatile(4)
if sources.Has(EXIF) && bytes.Equal(tagID, pngTagIDExif) {
sources = sources.Remove(EXIF)
if err := func() error {
r, err := e.bufferedReader(int64(chunkLength))
if err != nil {
return err
}
defer r.Close()
exifr := newMetaDecoderEXIF(r, e.byteOrder, 0, e.opts)
return exifr.decode()
}(); err != nil {
return err
}
e.skip(4) // skip CRC
} else if bytes.Equal(tagID, pngCompressedText) {
// Profile Name is 1-79 bytes, followed by the null character.
// Note that profileNameLength includes the null character.
profileName, profileNameLength := e.readNullTerminatedBytes(79 + 1)
// See https://exiftool.org/forum/index.php?topic=7988.msg40759#msg40759
if bytes.Equal(profileName, pngRawProfileTypeIPTC) {
if sources.Has(IPTC) {
sources = sources.Remove(IPTC)
dataLen := int(chunkLength) - int(profileNameLength)
if dataLen < 0 {
return newInvalidFormatErrorf("invalid data length %d", dataLen)
}
// TODO(bep) According to the spec, this should always return Latin-1 encoded text.
// The image editors out there does not seem to care much about this.
// See https://github.com/bep/imagemeta/issues/19
data, err := decompressZTXt(e.readBytesVolatile(dataLen))
if err != nil {
return newInvalidFormatError(fmt.Errorf("decompressing zTXt: %w", err))
}
data = data[profileNameLength:] // Skip the header bytes.
data = bytes.ReplaceAll(data, []byte("\n"), []byte(""))
d := make([]byte, hex.DecodedLen(len(data)))
_, err = hex.Decode(d, data)
if err != nil {
return fmt.Errorf("decoding hex: %w", err)
}
r := bytes.NewReader(d)
iptcDec := newMetaDecoderIPTC(r, e.opts)
if err := iptcDec.decodeBlocks(); err != nil {
return err
}
} else {
e.skip(int64(chunkLength) - profileNameLength)
}
} else if bytes.Equal(profileName, pngRawProfileTypeEXIF) {
e.skip(int64(chunkLength) - profileNameLength)
} else {
e.skip(int64(chunkLength) - profileNameLength)
}
e.skip(4) // skip CRC
} else {
skipTag(chunkLength)
}
}
}
func decompressZTXt(data []byte) ([]byte, error) {
// The first byte indicates the compression method, for which only deflate is currently defined (method zero).
compressionMethod := data[0]
if compressionMethod != 0 {
return nil, fmt.Errorf("unknown PNG compression method %v", compressionMethod)
}
b := bytes.NewReader(data[1:])
z, err := zlib.NewReader(b)
if err != nil {
return nil, err
}
defer z.Close()
p, err := io.ReadAll(z)
return p, err
}