1
0
mirror of https://github.com/coalaura/ffwebp.git synced 2025-09-07 21:45:31 +00:00

better sniff and dng

This commit is contained in:
Laura
2025-08-13 21:32:29 +02:00
parent 580e76f192
commit 113b6641ea
11 changed files with 317 additions and 68 deletions

View File

@@ -6,7 +6,7 @@ FFWebP is a small, single-binary CLI for converting images between formats, thin
- Single binary: no external tools required - Single binary: no external tools required
- Auto-detects input codec and infers output from the file extension - Auto-detects input codec and infers output from the file extension
- Supports AVIF, BMP, Farbfeld, GIF, HEIF/HEIC (decode-only), ICO/CUR, JPEG, JPEG XL, PCX, PNG, PNM (PBM/PGM/PPM/PAM), PSD (decode-only), QOI, SVG (decode-only), TGA, TIFF, WebP, XBM, XCF (decode-only) and XPM - Supports AVIF, BMP, DNG (decode-only), Farbfeld, GIF, HEIF/HEIC (decode-only), ICO/CUR, JPEG, JPEG XL, PCX, PNG, PNM (PBM/PGM/PPM/PAM), PSD (decode-only), QOI, SVG (decode-only), TGA, TIFF, WebP, XBM, XCF (decode-only) and XPM
- Lossy or lossless output with configurable quality - Lossy or lossless output with configurable quality
- Thumbnail generation via Lanczos3 resampling - Thumbnail generation via Lanczos3 resampling
- Per-codec flags for fine-grained control (see `ffwebp --help`) - Per-codec flags for fine-grained control (see `ffwebp --help`)

1
go.mod
View File

@@ -14,6 +14,7 @@ require (
github.com/gonutz/xcf v0.0.0-20180404091035-c002b9533d97 github.com/gonutz/xcf v0.0.0-20180404091035-c002b9533d97
github.com/hullerob/go.farbfeld v0.0.0-20181222022525-3661193c725f github.com/hullerob/go.farbfeld v0.0.0-20181222022525-3661193c725f
github.com/kriticalflare/qoi v0.0.0-20240815192827-34f66f23bcef github.com/kriticalflare/qoi v0.0.0-20240815192827-34f66f23bcef
github.com/mdouchement/dng v0.0.0-20230730131840-4066c9106942
github.com/nfnt/resize v0.0.0-20180221191011-83c6a9932646 github.com/nfnt/resize v0.0.0-20180221191011-83c6a9932646
github.com/oov/psd v0.0.0-20220121172623-5db5eafcecbb github.com/oov/psd v0.0.0-20220121172623-5db5eafcecbb
github.com/samuel/go-pcx v0.0.0-20210515040514-6a5ce4d132f7 github.com/samuel/go-pcx v0.0.0-20210515040514-6a5ce4d132f7

2
go.sum
View File

@@ -28,6 +28,8 @@ github.com/hullerob/go.farbfeld v0.0.0-20181222022525-3661193c725f h1:1LkiAnH6Rh
github.com/hullerob/go.farbfeld v0.0.0-20181222022525-3661193c725f/go.mod h1:mQEoc766DxPTAwQ54neWTK/lFqIeSO7OU6bqZsceglw= github.com/hullerob/go.farbfeld v0.0.0-20181222022525-3661193c725f/go.mod h1:mQEoc766DxPTAwQ54neWTK/lFqIeSO7OU6bqZsceglw=
github.com/kriticalflare/qoi v0.0.0-20240815192827-34f66f23bcef h1:XHb/eK43B8XuqAO5jHILCXzZP3pBamGmn5PcGjTZTuE= github.com/kriticalflare/qoi v0.0.0-20240815192827-34f66f23bcef h1:XHb/eK43B8XuqAO5jHILCXzZP3pBamGmn5PcGjTZTuE=
github.com/kriticalflare/qoi v0.0.0-20240815192827-34f66f23bcef/go.mod h1:skc5Zgfi3XE//1zgGGPC1abynJwsZhFxOiwkCrwL4Z8= github.com/kriticalflare/qoi v0.0.0-20240815192827-34f66f23bcef/go.mod h1:skc5Zgfi3XE//1zgGGPC1abynJwsZhFxOiwkCrwL4Z8=
github.com/mdouchement/dng v0.0.0-20230730131840-4066c9106942 h1:UA97jLO4tz9u69BhytirCXKwQhce4FoaUEYj0Sgp/HQ=
github.com/mdouchement/dng v0.0.0-20230730131840-4066c9106942/go.mod h1:pui8xMvtvG4x7qr6cUrto3et8w2np6n9cTLZFpN/ELY=
github.com/nfnt/resize v0.0.0-20180221191011-83c6a9932646 h1:zYyBkD/k9seD2A7fsi6Oo2LfFZAehjjQMERAvZLEDnQ= github.com/nfnt/resize v0.0.0-20180221191011-83c6a9932646 h1:zYyBkD/k9seD2A7fsi6Oo2LfFZAehjjQMERAvZLEDnQ=
github.com/nfnt/resize v0.0.0-20180221191011-83c6a9932646/go.mod h1:jpp1/29i3P1S/RLdc7JQKbRpFeM1dOBd8T9ki5s+AY8= github.com/nfnt/resize v0.0.0-20180221191011-83c6a9932646/go.mod h1:jpp1/29i3P1S/RLdc7JQKbRpFeM1dOBd8T9ki5s+AY8=
github.com/oov/psd v0.0.0-20220121172623-5db5eafcecbb h1:JF9kOhBBk4WPF7luXFu5yR+WgaFm9L/KiHJHhU9vDwA= github.com/oov/psd v0.0.0-20220121172623-5db5eafcecbb h1:JF9kOhBBk4WPF7luXFu5yR+WgaFm9L/KiHJHhU9vDwA=

View File

@@ -15,10 +15,12 @@ import (
var ( var (
executable = "ffwebp" executable = "ffwebp"
encodeOnly = map[string]bool{ encodeOnly = map[string]bool{
"dng": true,
"heic": true, "heic": true,
"heif": true, "heif": true,
"psd": true, "psd": true,
"svg": true, "svg": true,
"xcf": true,
} }
) )

View File

@@ -6,6 +6,7 @@ import (
"fmt" "fmt"
"io" "io"
"path/filepath" "path/filepath"
"sort"
"strings" "strings"
) )
@@ -30,17 +31,15 @@ func (s *Sniffed) String() string {
} }
func Sniff(reader io.Reader, input string, ignoreExtension bool) (*Sniffed, io.Reader, error) { func Sniff(reader io.Reader, input string, ignoreExtension bool) (*Sniffed, io.Reader, error) {
var (
hintedExt string
hintedCodec Codec
)
if !ignoreExtension { if !ignoreExtension {
ext := strings.ToLower(strings.TrimPrefix(filepath.Ext(input), ".")) hintedExt = strings.ToLower(strings.TrimPrefix(filepath.Ext(input), "."))
if ext != "" { if hintedExt != "" {
codec, _ := FindCodec(ext, false) hintedCodec, _ = FindCodec(hintedExt, false)
if codec != nil {
return &Sniffed{
Header: []byte("." + ext),
Confidence: 100,
Codec: codec,
}, reader, nil
}
} }
} }
@@ -51,10 +50,15 @@ func Sniff(reader io.Reader, input string, ignoreExtension bool) (*Sniffed, io.R
ra := bytes.NewReader(buf) ra := bytes.NewReader(buf)
type candidate struct {
codec Codec
confidence int
header []byte
}
var ( var (
best int best int
magic []byte list []candidate
guess Codec
) )
for _, codec := range codecs { for _, codec := range codecs {
@@ -69,21 +73,59 @@ func Sniff(reader io.Reader, input string, ignoreExtension bool) (*Sniffed, io.R
return nil, nil, err return nil, nil, err
} }
fmt.Println(codec.String(), confidence)
if confidence <= 0 {
continue
}
list = append(list, candidate{
codec: codec,
confidence: confidence,
header: header,
})
if confidence > best { if confidence > best {
best = confidence best = confidence
magic = header
guess = codec
} }
} }
if guess == nil { if len(list) == 0 || best <= 0 {
return nil, nil, errors.New("unknown input format") return nil, nil, errors.New("unknown input format")
} }
var top []candidate
for _, cand := range list {
if cand.confidence == best {
top = append(top, cand)
}
}
if hintedCodec != nil {
for _, cand := range top {
if cand.codec != hintedCodec {
continue
}
return &Sniffed{
Header: cand.header,
Confidence: cand.confidence,
Codec: cand.codec,
}, bytes.NewReader(buf), nil
}
}
sort.Slice(top, func(i, j int) bool {
return top[i].codec.String() < top[j].codec.String()
})
chosen := top[0]
return &Sniffed{ return &Sniffed{
Header: magic, Header: chosen.header,
Confidence: best, Confidence: chosen.confidence,
Codec: guess, Codec: chosen.codec,
}, bytes.NewReader(buf), nil }, bytes.NewReader(buf), nil
} }

169
internal/codec/dng/dng.go Normal file
View File

@@ -0,0 +1,169 @@
package dng
import (
"bytes"
"encoding/binary"
"errors"
"image"
"io"
"github.com/coalaura/ffwebp/internal/codec"
"github.com/coalaura/ffwebp/internal/opts"
"github.com/urfave/cli/v3"
// pure-go DNG preview extractor
"github.com/mdouchement/dng"
)
func init() {
codec.Register(impl{})
}
type impl struct{}
func (impl) String() string {
return "dng"
}
func (impl) Extensions() []string {
return []string{"dng"}
}
func (impl) CanEncode() bool {
return false
}
func (impl) Flags(flags []cli.Flag) []cli.Flag {
return flags
}
func (impl) Sniff(reader io.ReaderAt) (int, []byte, error) {
header := make([]byte, 16)
if _, err := reader.ReadAt(header, 0); err != nil && err != io.EOF {
return 0, nil, err
}
if len(header) < 8 {
return 0, nil, nil
}
isLE := bytes.Equal(header[0:2], []byte{'I', 'I'})
isBE := bytes.Equal(header[0:2], []byte{'M', 'M'})
if !isLE && !isBE {
return 0, nil, nil
}
var ord binary.ByteOrder
if isLE {
ord = binary.LittleEndian
} else {
ord = binary.BigEndian
}
sig := ord.Uint16(header[2:4])
const (
tiffClassic = 42
tiffBig = 43
tagDNGVersion = 0xC612
)
switch sig {
case tiffClassic:
ifd0, err := readU32(reader, ord, 4)
if err != nil || ifd0 == 0 {
return 0, nil, nil
}
n, err := readU16(reader, ord, int64(ifd0))
if err != nil {
return 0, nil, nil
}
for i := 0; i < int(n); i++ {
off := int64(ifd0) + 2 + int64(i)*12
tag, err := readU16(reader, ord, off)
if err != nil {
return 0, nil, nil
}
if uint16(tag) == uint16(tagDNGVersion) {
return 110, header[:8], nil
}
}
case tiffBig:
ifd0, err := readU64(reader, ord, 8)
if err != nil || ifd0 == 0 {
return 0, nil, nil
}
var bcnt [8]byte
if _, err := reader.ReadAt(bcnt[:], int64(ifd0)); err != nil {
return 0, nil, nil
}
n := ord.Uint64(bcnt[:])
max := n
if max > 1024 {
max = 1024
}
for i := uint64(0); i < max; i++ {
off := int64(ifd0) + 8 + int64(i)*20
tag, err := readU16(reader, ord, off)
if err != nil {
return 0, nil, nil
}
if uint16(tag) == uint16(tagDNGVersion) {
return 110, header[:8], nil
}
}
}
return 0, nil, nil
}
func (impl) Decode(r io.Reader) (image.Image, error) {
return dng.Decode(r)
}
func (impl) Encode(w io.Writer, img image.Image, _ opts.Common) error {
return errors.New("dng: encode not supported")
}
func readU16(reader io.ReaderAt, ord binary.ByteOrder, off int64) (uint16, error) {
var b [2]byte
if _, err := reader.ReadAt(b[:], off); err != nil {
return 0, err
}
return ord.Uint16(b[:]), nil
}
func readU32(reader io.ReaderAt, ord binary.ByteOrder, off int64) (uint32, error) {
var b [4]byte
if _, err := reader.ReadAt(b[:], off); err != nil {
return 0, err
}
return ord.Uint32(b[:]), nil
}
func readU64(reader io.ReaderAt, ord binary.ByteOrder, off int64) (uint64, error) {
var b [8]byte
if _, err := reader.ReadAt(b[:], off); err != nil {
return 0, err
}
return ord.Uint64(b[:]), nil
}

View File

@@ -64,7 +64,7 @@ func (impl) Flags(flags []cli.Flag) []cli.Flag {
} }
func (impl) Sniff(reader io.ReaderAt) (int, []byte, error) { func (impl) Sniff(reader io.ReaderAt) (int, []byte, error) {
buf := make([]byte, 128) buf := make([]byte, 256)
n, err := reader.ReadAt(buf, 0) n, err := reader.ReadAt(buf, 0)
if err != nil && err != io.EOF { if err != nil && err != io.EOF {

View File

@@ -1,14 +1,15 @@
package tga package tga
import ( import (
"image" "encoding/binary"
"io" "image"
"io"
"github.com/ftrvxmtrx/tga" "github.com/ftrvxmtrx/tga"
"github.com/coalaura/ffwebp/internal/codec" "github.com/coalaura/ffwebp/internal/codec"
"github.com/coalaura/ffwebp/internal/opts" "github.com/coalaura/ffwebp/internal/opts"
"github.com/urfave/cli/v3" "github.com/urfave/cli/v3"
) )
func init() { func init() {
@@ -34,38 +35,62 @@ func (impl) Flags(flags []cli.Flag) []cli.Flag {
} }
func (impl) Sniff(reader io.ReaderAt) (int, []byte, error) { func (impl) Sniff(reader io.ReaderAt) (int, []byte, error) {
buf := make([]byte, 3) // Validate full 18-byte TGA header to reduce false positives.
// Ref: https://www.fileformat.info/format/tga/egff.htm
hdr := make([]byte, 18)
if _, err := reader.ReadAt(hdr, 0); err != nil && err != io.EOF {
return 0, nil, err
}
if len(hdr) < 18 {
return 0, nil, nil
}
if _, err := reader.ReadAt(buf, 0); err != nil { idLength := hdr[0]
return 0, nil, err colorMapType := hdr[1]
} imageType := hdr[2]
colorMapType := buf[1] if colorMapType > 1 {
return 0, nil, nil
}
if colorMapType > 1 { switch imageType {
return 0, nil, nil case 1, 2, 3, 9, 10, 11:
} // valid image types
default:
// Exclude type 0 (no image data) to avoid matching random files like ISO BMFF.
return 0, nil, nil
}
validImageTypes := map[byte]bool{ // Width/height must be > 0
0: true, // no image data width := binary.LittleEndian.Uint16(hdr[12:14])
1: true, // colormapped, uncompressed height := binary.LittleEndian.Uint16(hdr[14:16])
2: true, // truecolor, uncompressed if width == 0 || height == 0 {
3: true, // grayscale, uncompressed return 0, nil, nil
9: true, // colormapped, RLE }
10: true, // truecolor, RLE
11: true, // grayscale, RLE
}
imageType := buf[2] // Pixel depth must be one of common values
bpp := hdr[16]
switch bpp {
case 8, 15, 16, 24, 32:
// ok
default:
return 0, nil, nil
}
if !validImageTypes[imageType] { // If color map is present, validate that the length is non-zero
return 0, nil, nil if colorMapType == 1 {
} colorMapLength := binary.LittleEndian.Uint16(hdr[5:7])
if colorMapLength == 0 {
return 0, nil, nil
}
}
header := make([]byte, 3) // Basic sanity: idLength must not push us past file start (not strictly necessary for sniff)
copy(header, buf) _ = idLength
return 100, header, nil header := make([]byte, 18)
copy(header, hdr)
return 100, header, nil
} }
func (impl) Decode(reader io.Reader) (image.Image, error) { func (impl) Decode(reader io.Reader) (image.Image, error) {

View File

@@ -62,24 +62,32 @@ func (impl) Flags(flags []cli.Flag) []cli.Flag {
} }
func (impl) Sniff(reader io.ReaderAt) (int, []byte, error) { func (impl) Sniff(reader io.ReaderAt) (int, []byte, error) {
magicLE := []byte{0x49, 0x49, 0x2A, 0x00} // Recognize classic TIFF (II*\0 or MM\0*) and BigTIFF (signature 43)
magicBE := []byte{0x4D, 0x4D, 0x00, 0x2A} buf := make([]byte, 16)
if _, err := reader.ReadAt(buf, 0); err != nil && err != io.EOF {
return 0, nil, err
}
buf := make([]byte, 4) if len(buf) < 8 {
return 0, nil, nil
}
if _, err := reader.ReadAt(buf, 0); err != nil { if bytes.Equal(buf[0:4], []byte{0x49, 0x49, 0x2A, 0x00}) ||
return 0, nil, err bytes.Equal(buf[0:4], []byte{0x4D, 0x4D, 0x00, 0x2A}) {
} return 100, buf[:8], nil
}
if bytes.Equal(buf, magicLE) { // BigTIFF: byte order + 43 marker, bytesize=8
return 100, magicLE, nil isLE := bytes.Equal(buf[0:2], []byte{'I', 'I'})
} isBE := bytes.Equal(buf[0:2], []byte{'M', 'M'})
if isLE || isBE {
if (isLE && buf[2] == 0x2B && buf[3] == 0x00 && buf[4] == 0x08 && buf[5] == 0x00) ||
(isBE && buf[2] == 0x00 && buf[3] == 0x2B && buf[4] == 0x00 && buf[5] == 0x08) {
return 100, buf[:8], nil
}
}
if bytes.Equal(buf, magicBE) { return 0, nil, nil
return 100, magicBE, nil
}
return 0, nil, nil
} }
func (impl) Decode(reader io.Reader) (image.Image, error) { func (impl) Decode(reader io.Reader) (image.Image, error) {

View File

@@ -54,7 +54,7 @@ func (impl) Sniff(reader io.ReaderAt) (int, []byte, error) {
buf = buf[:n] buf = buf[:n]
if bytes.Contains(buf, []byte("#define")) && bytes.Contains(buf, []byte("bits[]")) { if bytes.Contains(buf, []byte("#define")) && bytes.Contains(buf, []byte("bits[]")) {
return 90, buf, nil return 80, buf, nil
} }
return 0, nil, nil return 0, nil, nil

View File

@@ -53,8 +53,8 @@ func (impl) Sniff(reader io.ReaderAt) (int, []byte, error) {
buf = buf[:n] buf = buf[:n]
if bytes.Contains(buf, []byte("/* XPM */")) && bytes.Contains(buf, []byte("bits[]")) { if bytes.Contains(buf, []byte("/* XPM */")) && bytes.Contains(buf, []byte("[] = {")) {
return 90, buf, nil return 80, buf, nil
} }
return 0, nil, nil return 0, nil, nil