mattermost-community-enterp.../vendor/github.com/bep/imagemeta/imagedecoder_jpg.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

135 lines
2.5 KiB
Go

// Copyright 2024 Bjørn Erik Pedersen
// SPDX-License-Identifier: MIT
package imagemeta
import (
"bytes"
"io"
)
type imageDecoderJPEG struct {
*baseStreamingDecoder
}
func (e *imageDecoderJPEG) decode() error {
// JPEG SOI marker.
soi, err := e.read2E()
if err != nil {
return nil
}
if soi != markerSOI {
return nil
}
// These are the sources we support.
sourceSet := EXIF | IPTC | XMP
// Remove sources that are not requested.
sourceSet = sourceSet & e.opts.Sources
for {
if sourceSet.IsZero() {
// Done.
return nil
}
marker := e.read2()
if e.isEOF {
return nil
}
if marker == 0 {
continue
}
if marker == markerSOS {
// Start of scan. We're done.
return nil
}
// Read the 16-bit length of the segment. The value includes the 2 bytes for the
// length itself, so we subtract 2 to get the number of remaining bytes.
length := e.read2()
if length < 2 {
return errInvalidFormat
}
length -= 2
if marker == markerApp1EXIF && sourceSet.Has(EXIF) {
sourceSet = sourceSet.Remove(EXIF)
if err := e.handleEXIF(int64(length)); err != nil {
return err
}
continue
}
if marker == markerApp13 && sourceSet.Has(IPTC) {
sourceSet = sourceSet.Remove(IPTC)
if err := e.handleIPTC(int(length)); err != nil {
return err
}
continue
}
if marker == markerrApp1XMP && sourceSet.Has(XMP) {
const xmpMarkerLen = 29
oldPos := e.pos()
b, err := e.readBytesVolatileE(xmpMarkerLen)
if err != nil && err != io.ErrUnexpectedEOF {
return err
}
if err == nil && bytes.Equal(b, markerXMP) {
length -= xmpMarkerLen
sourceSet = sourceSet.Remove(XMP)
r := io.LimitReader(e.r, int64(length))
if err := decodeXMP(r, e.opts); err != nil {
return err
}
continue
} else {
// Not XMP, rewind.
e.seek(oldPos)
}
}
e.skip(int64(length))
}
}
func (e *imageDecoderJPEG) handleEXIF(length int64) error {
thumbnailOffset := e.pos()
r, err := e.bufferedReader(length)
if err != nil {
return err
}
defer r.Close()
exifr := newMetaDecoderEXIF(r, e.byteOrder, thumbnailOffset, e.opts)
header := exifr.read4()
if header != exifHeader {
return err
}
exifr.skip(2)
if err := exifr.decode(); err != nil {
return err
}
return nil
}
func (e *imageDecoderJPEG) handleIPTC(length int) error {
const headerLength = 14
// Skip the IPTC header.
e.skip(headerLength)
r, err := e.bufferedReader(int64(length - headerLength))
if err != nil {
return err
}
defer r.Close()
dec := newMetaDecoderIPTC(r, e.opts)
return dec.decodeBlocks()
}