mattermost-community-enterp.../channels/app/imaging/orientation.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

160 lines
4.2 KiB
Go

// Copyright (c) 2015-present Mattermost, Inc. All Rights Reserved.
// See LICENSE.txt for license information.
package imaging
import (
"errors"
"fmt"
"image"
"io"
"strings"
"github.com/anthonynsimon/bild/transform"
"github.com/bep/imagemeta"
)
const (
/*
EXIF Image Orientations
1 2 3 4 5 6 7 8
888888 888888 88 88 8888888888 88 88 8888888888
88 88 88 88 88 88 88 88 88 88 88 88
8888 8888 8888 8888 88 8888888888 8888888888 88
88 88 88 88
88 88 888888 888888
*/
Upright = iota + 1
UprightMirrored
UpsideDown
UpsideDownMirrored
RotatedCWMirrored
RotatedCCW
RotatedCCWMirrored
RotatedCW
)
var errStopDecoding = fmt.Errorf("stop decoding")
// MakeImageUpright changes the orientation of the given image.
func MakeImageUpright(img image.Image, orientation int) image.Image {
switch orientation {
case UprightMirrored:
return transform.FlipH(img)
case UpsideDown:
return transform.Rotate(img, 180, &transform.RotationOptions{ResizeBounds: true})
case UpsideDownMirrored:
return transform.FlipV(img)
case RotatedCWMirrored:
return transform.Rotate(transform.FlipH(img), -90, &transform.RotationOptions{ResizeBounds: true})
case RotatedCCW:
return transform.Rotate(img, 90, &transform.RotationOptions{ResizeBounds: true})
case RotatedCCWMirrored:
return transform.Rotate(transform.FlipV(img), -90, &transform.RotationOptions{ResizeBounds: true})
case RotatedCW:
return transform.Rotate(img, 270, &transform.RotationOptions{ResizeBounds: true})
default:
return img
}
}
type fwSeeker struct {
r io.Reader
pos int64
}
func (f *fwSeeker) Read(p []byte) (int, error) {
n, err := f.r.Read(p)
if err != nil {
return n, err
}
f.pos += int64(n)
return n, nil
}
func (f *fwSeeker) Seek(offset int64, whence int) (int64, error) {
isForwardSeek := (whence == io.SeekStart && offset >= f.pos) ||
(whence == io.SeekCurrent && offset >= 0)
// We only support seeking forward.
if !isForwardSeek {
return 0, fmt.Errorf("seeking backwards is not supported")
}
toRead := offset
if whence == io.SeekStart {
toRead -= f.pos
}
// Seeking forward means we can simply discard the data.
n, err := io.CopyN(io.Discard, f.r, toRead)
if err != nil {
return n, fmt.Errorf("failed to seek: %w", err)
}
f.pos += n
return f.pos, nil
}
// GetImageOrientation reads the input data and returns the EXIF encoded
// image orientation. Supported formats are JPEG, PNG, TIFF, and WebP.
// Passing an io.ReadSeeker is preferable as we can't guarantee a plain
// io.Reader will work for all formats (e.g. TIFF requires backwards seeking).
func GetImageOrientation(input io.Reader, format string) (int, error) {
orientation := Upright
// Strip the "image/" prefix from the format in case it's a MIME type.
format, _ = strings.CutPrefix(format, "image/")
var imgFormat imagemeta.ImageFormat
switch format {
case "jpeg":
imgFormat = imagemeta.JPEG
case "png":
imgFormat = imagemeta.PNG
case "tiff":
imgFormat = imagemeta.TIFF
case "webp":
imgFormat = imagemeta.WebP
default:
// We don't support EXIF on any other format.
return orientation, fmt.Errorf("unsupported image format: %s", format)
}
var rs io.ReadSeeker
if r, ok := input.(io.ReadSeeker); ok {
rs = r
} else {
rs = &fwSeeker{r: input}
}
opts := imagemeta.Options{
R: rs,
HandleTag: func(tag imagemeta.TagInfo) error {
if tag.Tag == "Orientation" {
if o, ok := tag.Value.(uint16); ok {
orientation = int(o)
// Stop decoding after we've found the orientation tag]
// since it's the only one we care about.
return errStopDecoding
}
}
return nil
},
ShouldHandleTag: func(tag imagemeta.TagInfo) bool {
// We only care about the orientation tag.
return tag.Tag == "Orientation"
},
Sources: imagemeta.EXIF, // We only care about EXIF data.
ImageFormat: imgFormat,
}
if err := imagemeta.Decode(opts); err != nil && !errors.Is(err, errStopDecoding) {
return Upright, fmt.Errorf("failed to decode exif data: %w", err)
}
return orientation, nil
}