Adapt test suite for AVIF support
This commit is contained in:
parent
615d02e07e
commit
6e72b75efa
@ -9,7 +9,7 @@ addons:
|
|||||||
apt:
|
apt:
|
||||||
update: true
|
update: true
|
||||||
sources: &common_sources
|
sources: &common_sources
|
||||||
# add support for HEIF files
|
# add support for AVIF files
|
||||||
- sourceline: 'ppa:strukturag/libheif'
|
- sourceline: 'ppa:strukturag/libheif'
|
||||||
- sourceline: 'ppa:strukturag/libde265'
|
- sourceline: 'ppa:strukturag/libde265'
|
||||||
packages: &common_packages
|
packages: &common_packages
|
||||||
|
@ -23,9 +23,9 @@ statistics and others. It supports a large range of [numeric
|
|||||||
types](http://libvips.github.io/libvips/API/current/VipsImage.html#VipsBandFormat),
|
types](http://libvips.github.io/libvips/API/current/VipsImage.html#VipsBandFormat),
|
||||||
from 8-bit int to 128-bit complex. Images can have any number of bands.
|
from 8-bit int to 128-bit complex. Images can have any number of bands.
|
||||||
It supports a good range of image formats, including JPEG, TIFF, PNG,
|
It supports a good range of image formats, including JPEG, TIFF, PNG,
|
||||||
WebP, HEIC, FITS, Matlab, OpenEXR, PDF, SVG, HDR, PPM / PGM / PFM, CSV,
|
WebP, HEIC, AVIF, FITS, Matlab, OpenEXR, PDF, SVG, HDR, PPM / PGM / PFM,
|
||||||
GIF, Analyze, NIfTI, DeepZoom, and OpenSlide. It can also load images via
|
CSV, GIF, Analyze, NIfTI, DeepZoom, and OpenSlide. It can also load images
|
||||||
ImageMagick or GraphicsMagick, letting it work with formats like DICOM.
|
via ImageMagick or GraphicsMagick, letting it work with formats like DICOM.
|
||||||
|
|
||||||
It comes with bindings for
|
It comes with bindings for
|
||||||
[C](http://libvips.github.io/libvips/API/current/using-from-c.html),
|
[C](http://libvips.github.io/libvips/API/current/using-from-c.html),
|
||||||
@ -279,7 +279,7 @@ files: Aperio, Hamamatsu, Leica, MIRAX, Sakura, Trestle, and Ventana.
|
|||||||
|
|
||||||
### libheif
|
### libheif
|
||||||
|
|
||||||
If available, libvips can load and save HEIC images.
|
If available, libvips can load and save HEIC and AVIF images.
|
||||||
|
|
||||||
# Contributors
|
# Contributors
|
||||||
|
|
||||||
|
@ -41,7 +41,7 @@ DICOM_FILE = os.path.join(IMAGES, "dicom_test_image.dcm")
|
|||||||
BMP_FILE = os.path.join(IMAGES, "MARBLES.BMP")
|
BMP_FILE = os.path.join(IMAGES, "MARBLES.BMP")
|
||||||
NIFTI_FILE = os.path.join(IMAGES, "avg152T1_LR_nifti.nii.gz")
|
NIFTI_FILE = os.path.join(IMAGES, "avg152T1_LR_nifti.nii.gz")
|
||||||
ICO_FILE = os.path.join(IMAGES, "favicon.ico")
|
ICO_FILE = os.path.join(IMAGES, "favicon.ico")
|
||||||
HEIC_FILE = os.path.join(IMAGES, "heic-orientation-6.heic")
|
AVIF_FILE = os.path.join(IMAGES, "avif-orientation-6.avif")
|
||||||
MOSAIC_FILES = [os.path.join(IMAGES, "cd1.1.jpg"), os.path.join(IMAGES, "cd1.2.jpg"),
|
MOSAIC_FILES = [os.path.join(IMAGES, "cd1.1.jpg"), os.path.join(IMAGES, "cd1.2.jpg"),
|
||||||
os.path.join(IMAGES, "cd2.1.jpg"), os.path.join(IMAGES, "cd2.2.jpg"),
|
os.path.join(IMAGES, "cd2.1.jpg"), os.path.join(IMAGES, "cd2.2.jpg"),
|
||||||
os.path.join(IMAGES, "cd3.1.jpg"), os.path.join(IMAGES, "cd3.2.jpg"),
|
os.path.join(IMAGES, "cd3.1.jpg"), os.path.join(IMAGES, "cd3.2.jpg"),
|
||||||
|
Binary file not shown.
BIN
test/test-suite/images/avif-orientation-6.avif
Normal file
BIN
test/test-suite/images/avif-orientation-6.avif
Normal file
Binary file not shown.
After Width: | Height: | Size: 164 KiB |
Binary file not shown.
@ -72,6 +72,8 @@ class TestCreate:
|
|||||||
assert im.max() == 255.0
|
assert im.max() == 255.0
|
||||||
assert im.min() == 0.0
|
assert im.min() == 0.0
|
||||||
|
|
||||||
|
@pytest.mark.skipif(pyvips.type_find("VipsOperation", "fwfft") == 0,
|
||||||
|
reason="no FFTW, skipping test")
|
||||||
def test_fractsurf(self):
|
def test_fractsurf(self):
|
||||||
im = pyvips.Image.fractsurf(100, 90, 2.5)
|
im = pyvips.Image.fractsurf(100, 90, 2.5)
|
||||||
assert im.width == 100
|
assert im.width == 100
|
||||||
|
@ -11,7 +11,7 @@ from helpers import \
|
|||||||
JPEG_FILE, SRGB_FILE, MATLAB_FILE, PNG_FILE, TIF_FILE, OME_FILE, \
|
JPEG_FILE, SRGB_FILE, MATLAB_FILE, PNG_FILE, TIF_FILE, OME_FILE, \
|
||||||
ANALYZE_FILE, GIF_FILE, WEBP_FILE, EXR_FILE, FITS_FILE, OPENSLIDE_FILE, \
|
ANALYZE_FILE, GIF_FILE, WEBP_FILE, EXR_FILE, FITS_FILE, OPENSLIDE_FILE, \
|
||||||
PDF_FILE, SVG_FILE, SVGZ_FILE, SVG_GZ_FILE, GIF_ANIM_FILE, DICOM_FILE, \
|
PDF_FILE, SVG_FILE, SVGZ_FILE, SVG_GZ_FILE, GIF_ANIM_FILE, DICOM_FILE, \
|
||||||
BMP_FILE, NIFTI_FILE, ICO_FILE, HEIC_FILE, TRUNCATED_FILE, \
|
BMP_FILE, NIFTI_FILE, ICO_FILE, AVIF_FILE, TRUNCATED_FILE, \
|
||||||
GIF_ANIM_EXPECTED_PNG_FILE, GIF_ANIM_DISPOSE_BACKGROUND_FILE, \
|
GIF_ANIM_EXPECTED_PNG_FILE, GIF_ANIM_DISPOSE_BACKGROUND_FILE, \
|
||||||
GIF_ANIM_DISPOSE_BACKGROUND_EXPECTED_PNG_FILE, \
|
GIF_ANIM_DISPOSE_BACKGROUND_EXPECTED_PNG_FILE, \
|
||||||
GIF_ANIM_DISPOSE_PREVIOUS_FILE, \
|
GIF_ANIM_DISPOSE_PREVIOUS_FILE, \
|
||||||
@ -75,7 +75,7 @@ class TestForeign:
|
|||||||
max_diff = (im - x).abs().max()
|
max_diff = (im - x).abs().max()
|
||||||
assert max_diff == 0
|
assert max_diff == 0
|
||||||
|
|
||||||
def save_load_file(self, format, options, im, thresh):
|
def save_load_file(self, format, options, im, max_diff=0):
|
||||||
# yuk!
|
# yuk!
|
||||||
# but we can't set format parameters for pyvips.Image.new_temp_file()
|
# but we can't set format parameters for pyvips.Image.new_temp_file()
|
||||||
filename = temp_filename(self.tempdir, format)
|
filename = temp_filename(self.tempdir, format)
|
||||||
@ -86,8 +86,7 @@ class TestForeign:
|
|||||||
assert im.width == x.width
|
assert im.width == x.width
|
||||||
assert im.height == x.height
|
assert im.height == x.height
|
||||||
assert im.bands == x.bands
|
assert im.bands == x.bands
|
||||||
max_diff = (im - x).abs().max()
|
assert (im - x).abs().max() <= max_diff
|
||||||
assert max_diff <= thresh
|
|
||||||
x = None
|
x = None
|
||||||
|
|
||||||
def save_load_buffer(self, saver, loader, im, max_diff=0, **kwargs):
|
def save_load_buffer(self, saver, loader, im, max_diff=0, **kwargs):
|
||||||
@ -115,7 +114,7 @@ class TestForeign:
|
|||||||
assert (im - x).abs().max() <= max_diff
|
assert (im - x).abs().max() <= max_diff
|
||||||
|
|
||||||
def test_vips(self):
|
def test_vips(self):
|
||||||
self.save_load_file(".v", "", self.colour, 0)
|
self.save_load_file(".v", "", self.colour)
|
||||||
|
|
||||||
# check we can save and restore metadata
|
# check we can save and restore metadata
|
||||||
filename = temp_filename(self.tempdir, ".v")
|
filename = temp_filename(self.tempdir, ".v")
|
||||||
@ -321,8 +320,8 @@ class TestForeign:
|
|||||||
self.save_load_buffer("pngsave_buffer", "pngload_buffer", self.colour)
|
self.save_load_buffer("pngsave_buffer", "pngload_buffer", self.colour)
|
||||||
self.save_load("%s.png", self.mono)
|
self.save_load("%s.png", self.mono)
|
||||||
self.save_load("%s.png", self.colour)
|
self.save_load("%s.png", self.colour)
|
||||||
self.save_load_file(".png", "[interlace]", self.colour, 0)
|
self.save_load_file(".png", "[interlace]", self.colour)
|
||||||
self.save_load_file(".png", "[interlace]", self.mono, 0)
|
self.save_load_file(".png", "[interlace]", self.mono)
|
||||||
|
|
||||||
# size of a regular mono PNG
|
# size of a regular mono PNG
|
||||||
len_mono = len(self.mono.write_to_buffer(".png"))
|
len_mono = len(self.mono.write_to_buffer(".png"))
|
||||||
@ -391,30 +390,30 @@ class TestForeign:
|
|||||||
self.save_load("%s.tif", self.cmyk)
|
self.save_load("%s.tif", self.cmyk)
|
||||||
|
|
||||||
self.save_load("%s.tif", self.onebit)
|
self.save_load("%s.tif", self.onebit)
|
||||||
self.save_load_file(".tif", "[bitdepth=1]", self.onebit, 0)
|
self.save_load_file(".tif", "[bitdepth=1]", self.onebit)
|
||||||
self.save_load_file(".tif", "[miniswhite]", self.onebit, 0)
|
self.save_load_file(".tif", "[miniswhite]", self.onebit)
|
||||||
self.save_load_file(".tif", "[bitdepth=1,miniswhite]", self.onebit, 0)
|
self.save_load_file(".tif", "[bitdepth=1,miniswhite]", self.onebit)
|
||||||
|
|
||||||
self.save_load_file(".tif",
|
self.save_load_file(".tif",
|
||||||
"[profile={0}]".format(SRGB_FILE),
|
"[profile={0}]".format(SRGB_FILE),
|
||||||
self.colour, 0)
|
self.colour)
|
||||||
self.save_load_file(".tif", "[tile]", self.colour, 0)
|
self.save_load_file(".tif", "[tile]", self.colour)
|
||||||
self.save_load_file(".tif", "[tile,pyramid]", self.colour, 0)
|
self.save_load_file(".tif", "[tile,pyramid]", self.colour)
|
||||||
self.save_load_file(".tif", "[tile,pyramid,subifd]", self.colour, 0)
|
self.save_load_file(".tif", "[tile,pyramid,subifd]", self.colour)
|
||||||
self.save_load_file(".tif",
|
self.save_load_file(".tif",
|
||||||
"[tile,pyramid,compression=jpeg]", self.colour, 80)
|
"[tile,pyramid,compression=jpeg]", self.colour, 80)
|
||||||
self.save_load_file(".tif",
|
self.save_load_file(".tif",
|
||||||
"[tile,pyramid,subifd,compression=jpeg]",
|
"[tile,pyramid,subifd,compression=jpeg]",
|
||||||
self.colour, 80)
|
self.colour, 80)
|
||||||
self.save_load_file(".tif", "[bigtiff]", self.colour, 0)
|
self.save_load_file(".tif", "[bigtiff]", self.colour)
|
||||||
self.save_load_file(".tif", "[compression=jpeg]", self.colour, 80)
|
self.save_load_file(".tif", "[compression=jpeg]", self.colour, 80)
|
||||||
self.save_load_file(".tif",
|
self.save_load_file(".tif",
|
||||||
"[tile,tile-width=256]", self.colour, 10)
|
"[tile,tile-width=256]", self.colour, 10)
|
||||||
|
|
||||||
im = pyvips.Image.new_from_file(TIF2_FILE)
|
im = pyvips.Image.new_from_file(TIF2_FILE)
|
||||||
self.save_load_file(".tif", "[bitdepth=2]", im, 0)
|
self.save_load_file(".tif", "[bitdepth=2]", im)
|
||||||
im = pyvips.Image.new_from_file(TIF4_FILE)
|
im = pyvips.Image.new_from_file(TIF4_FILE)
|
||||||
self.save_load_file(".tif", "[bitdepth=4]", im, 0)
|
self.save_load_file(".tif", "[bitdepth=4]", im)
|
||||||
|
|
||||||
filename = temp_filename(self.tempdir, '.tif')
|
filename = temp_filename(self.tempdir, '.tif')
|
||||||
x = pyvips.Image.new_from_file(TIF_FILE)
|
x = pyvips.Image.new_from_file(TIF_FILE)
|
||||||
@ -1072,37 +1071,42 @@ class TestForeign:
|
|||||||
def test_heifload(self):
|
def test_heifload(self):
|
||||||
def heif_valid(im):
|
def heif_valid(im):
|
||||||
a = im(10, 10)
|
a = im(10, 10)
|
||||||
# different versions of HEIC decode have slightly different
|
# different versions of libheif decode have slightly different
|
||||||
# rounding
|
# rounding
|
||||||
assert_almost_equal_objects(a, [197.0, 181.0, 158.0], threshold=2)
|
assert_almost_equal_objects(a, [197.0, 181.0, 158.0], threshold=2)
|
||||||
assert im.width == 3024
|
assert im.width == 3024
|
||||||
assert im.height == 4032
|
assert im.height == 4032
|
||||||
assert im.bands == 3
|
assert im.bands == 3
|
||||||
|
|
||||||
self.file_loader("heifload", HEIC_FILE, heif_valid)
|
self.file_loader("heifload", AVIF_FILE, heif_valid)
|
||||||
self.buffer_loader("heifload_buffer", HEIC_FILE, heif_valid)
|
self.buffer_loader("heifload_buffer", AVIF_FILE, heif_valid)
|
||||||
|
|
||||||
@skip_if_no("heifsave")
|
@skip_if_no("heifsave")
|
||||||
def test_heifsave(self):
|
def test_heifsave(self):
|
||||||
self.save_load_buffer("heifsave_buffer", "heifload_buffer",
|
self.save_load_buffer("heifsave_buffer", "heifload_buffer",
|
||||||
self.colour, 80)
|
self.colour, 80, compression="av1")
|
||||||
self.save_load("%s.heic", self.colour)
|
# TODO: perhaps we should automatically set the compression to
|
||||||
|
# av1 when we save to *.avif?
|
||||||
|
#self.save_load("%s.avif", self.colour)
|
||||||
|
self.save_load_file(".avif", "[compression=av1]",
|
||||||
|
self.colour, 80)
|
||||||
|
|
||||||
# test lossless mode
|
# uncomment to test lossless mode, will take a while
|
||||||
im = pyvips.Image.new_from_file(HEIC_FILE)
|
#im = pyvips.Image.new_from_file(AVIF_FILE)
|
||||||
buf = im.heifsave_buffer(lossless=True)
|
#buf = im.heifsave_buffer(lossless=True, compression="av1")
|
||||||
im2 = pyvips.Image.new_from_buffer(buf, "")
|
#im2 = pyvips.Image.new_from_buffer(buf, "")
|
||||||
# not in fact quite lossless
|
# not in fact quite lossless
|
||||||
assert abs(im.avg() - im2.avg()) < 3
|
#assert abs(im.avg() - im2.avg()) < 3
|
||||||
|
|
||||||
# higher Q should mean a bigger buffer
|
# higher Q should mean a bigger buffer, needs libheif >= v1.8.0,
|
||||||
b1 = im.heifsave_buffer(Q=10)
|
# see: https://github.com/libvips/libvips/issues/1757
|
||||||
b2 = im.heifsave_buffer(Q=90)
|
b1 = self.mono.heifsave_buffer(Q=10, compression="av1")
|
||||||
|
b2 = self.mono.heifsave_buffer(Q=90, compression="av1")
|
||||||
assert len(b2) > len(b1)
|
assert len(b2) > len(b1)
|
||||||
|
|
||||||
# try saving an image with an ICC profile and reading it back
|
# try saving an image with an ICC profile and reading it back
|
||||||
# not all libheif have profile support, so put it in an if
|
# not all libheif have profile support, so put it in an if
|
||||||
buf = self.colour.heifsave_buffer()
|
buf = self.colour.heifsave_buffer(Q=10, compression="av1")
|
||||||
im = pyvips.Image.new_from_buffer(buf, "")
|
im = pyvips.Image.new_from_buffer(buf, "")
|
||||||
p1 = self.colour.get("icc-profile-data")
|
p1 = self.colour.get("icc-profile-data")
|
||||||
if im.get_typeof("icc-profile-data") != 0:
|
if im.get_typeof("icc-profile-data") != 0:
|
||||||
@ -1113,18 +1117,16 @@ class TestForeign:
|
|||||||
# the exif test will need us to be able to walk the header,
|
# the exif test will need us to be able to walk the header,
|
||||||
# we can't just check exif-data
|
# we can't just check exif-data
|
||||||
|
|
||||||
# libheif 1.1 (on ubuntu 18.04, current LTS) does not support exif
|
|
||||||
# write, so this test is commented out
|
|
||||||
|
|
||||||
# test that exif changes change the output of heifsave
|
# test that exif changes change the output of heifsave
|
||||||
# first make sure we have exif support
|
# first make sure we have exif support
|
||||||
#z = pyvips.Image.new_from_file(JPEG_FILE)
|
z = pyvips.Image.new_from_file(AVIF_FILE)
|
||||||
#if z.get_typeof("exif-ifd0-Orientation") != 0:
|
if z.get_typeof("exif-ifd0-Make") != 0:
|
||||||
# x = self.colour.copy()
|
x = z.copy()
|
||||||
# x.set("exif-ifd0-Make", "banana")
|
x.set("exif-ifd0-Make", "banana")
|
||||||
# buf = x.heifsave_buffer()
|
buf = x.heifsave_buffer(Q=10, compression="av1")
|
||||||
# y = pyvips.Image.new_from_buffer(buf, "")
|
y = pyvips.Image.new_from_buffer(buf, "")
|
||||||
# assert y.get("exif-ifd0-Make").split(" ")[0] == "banana"
|
assert y.get("exif-ifd0-Make").split(" ")[0] == "banana"
|
||||||
|
|
||||||
|
|
||||||
if __name__ == '__main__':
|
if __name__ == '__main__':
|
||||||
pytest.main()
|
pytest.main()
|
||||||
|
@ -2,7 +2,7 @@
|
|||||||
import pytest
|
import pytest
|
||||||
|
|
||||||
import pyvips
|
import pyvips
|
||||||
from helpers import JPEG_FILE, OME_FILE, HEIC_FILE, TIF_FILE, all_formats, have
|
from helpers import JPEG_FILE, OME_FILE, AVIF_FILE, TIF_FILE, all_formats, have
|
||||||
|
|
||||||
|
|
||||||
# Run a function expecting a complex image on a two-band image
|
# Run a function expecting a complex image on a two-band image
|
||||||
@ -195,8 +195,8 @@ class TestResample:
|
|||||||
|
|
||||||
if have("heifload"):
|
if have("heifload"):
|
||||||
# this image is orientation 6 ... thumbnail should flip it
|
# this image is orientation 6 ... thumbnail should flip it
|
||||||
im = pyvips.Image.new_from_file(HEIC_FILE)
|
im = pyvips.Image.new_from_file(AVIF_FILE)
|
||||||
thumb = pyvips.Image.thumbnail(HEIC_FILE, 100)
|
thumb = pyvips.Image.thumbnail(AVIF_FILE, 100)
|
||||||
|
|
||||||
# thumb should be portrait
|
# thumb should be portrait
|
||||||
assert thumb.width < thumb.height
|
assert thumb.width < thumb.height
|
||||||
|
Loading…
x
Reference in New Issue
Block a user