Refactoring of ImageProcessorFast (#35069)
* add init and base image processing functions * add add_fast_image_processor to transformers-cli * add working fast image processor clip * add fast image processor to doc, working tests * remove "to be implemented" SigLip * fix unprotected import * fix unprotected vision import * update ViTImageProcessorFast * increase threshold slow fast ewuivalence * add fast img blip * add fast class in tests with cli * improve cli * add fast image processor convnext * add LlavaPatchingMixin and fast image processor for llava_next and llava_onevision * add device kwarg to ImagesKwargs for fast processing on cuda * cleanup * fix unprotected import * group images by sizes and add batch processing * Add batch equivalence tests, skip when center_crop is used * cleanup * update init and cli * fix-copies * refactor convnext, cleanup base * fix * remove patching mixins, add piped torchvision transforms for ViT * fix unbatched processing * fix f strings * protect imports * change llava onevision to class transforms (test) * fix convnext * improve formatting (following Pavel review) * fix handling device arg * improve cli * fix * fix inits * Add distinction between preprocess and _preprocess, and support for arbitrary kwargs through valid_extra_kwargs * uniformize qwen2_vl fast * fix docstrings * add add fast image processor llava * remove min_pixels max_pixels from accepted size * nit * nit * refactor fast image processors docstrings * cleanup and remove fast class transforms * update add fast image processor transformers cli * cleanup docstring * uniformize pixtral fast and make _process_image explicit * fix prepare image structure llava next/onevision * Use typed kwargs instead of explicit args * nit fix import Unpack * clearly separate pops and gets in base preprocess. Use explicit typed kwargs * make qwen2_vl preprocess arguments hashable
This commit is contained in:
@@ -20,7 +20,7 @@ from typing import Tuple, Union
|
||||
import numpy as np
|
||||
|
||||
from transformers.testing_utils import require_torch, require_vision
|
||||
from transformers.utils import is_vision_available
|
||||
from transformers.utils import is_torchvision_available, is_vision_available
|
||||
|
||||
from ...test_image_processing_common import ImageProcessingTestMixin, prepare_image_inputs
|
||||
|
||||
@@ -30,6 +30,11 @@ if is_vision_available():
|
||||
|
||||
from transformers import LlavaImageProcessor
|
||||
|
||||
if is_torchvision_available():
|
||||
from torchvision.transforms import functional as F
|
||||
|
||||
from transformers import LlavaImageProcessorFast
|
||||
|
||||
|
||||
class LlavaImageProcessingTester:
|
||||
def __init__(
|
||||
@@ -50,6 +55,7 @@ class LlavaImageProcessingTester:
|
||||
image_std=[0.26862954, 0.26130258, 0.27577711],
|
||||
do_convert_rgb=True,
|
||||
):
|
||||
super().__init__()
|
||||
size = size if size is not None else {"shortest_edge": 20}
|
||||
crop_size = crop_size if crop_size is not None else {"height": 18, "width": 18}
|
||||
self.parent = parent
|
||||
@@ -103,6 +109,7 @@ class LlavaImageProcessingTester:
|
||||
# Copied from tests.models.clip.test_image_processing_clip.CLIPImageProcessingTest with CLIP->Llava
|
||||
class LlavaImageProcessingTest(ImageProcessingTestMixin, unittest.TestCase):
|
||||
image_processing_class = LlavaImageProcessor if is_vision_available() else None
|
||||
fast_image_processing_class = LlavaImageProcessorFast if is_torchvision_available() else None
|
||||
|
||||
def setUp(self):
|
||||
super().setUp()
|
||||
@@ -114,25 +121,27 @@ class LlavaImageProcessingTest(ImageProcessingTestMixin, unittest.TestCase):
|
||||
|
||||
# Ignore copy
|
||||
def test_image_processor_properties(self):
|
||||
image_processing = self.image_processing_class(**self.image_processor_dict)
|
||||
self.assertTrue(hasattr(image_processing, "do_pad"))
|
||||
self.assertTrue(hasattr(image_processing, "do_resize"))
|
||||
self.assertTrue(hasattr(image_processing, "size"))
|
||||
self.assertTrue(hasattr(image_processing, "do_center_crop"))
|
||||
self.assertTrue(hasattr(image_processing, "center_crop"))
|
||||
self.assertTrue(hasattr(image_processing, "do_normalize"))
|
||||
self.assertTrue(hasattr(image_processing, "image_mean"))
|
||||
self.assertTrue(hasattr(image_processing, "image_std"))
|
||||
self.assertTrue(hasattr(image_processing, "do_convert_rgb"))
|
||||
for image_processing_class in self.image_processor_list:
|
||||
image_processing = image_processing_class(**self.image_processor_dict)
|
||||
self.assertTrue(hasattr(image_processing, "do_pad"))
|
||||
self.assertTrue(hasattr(image_processing, "do_resize"))
|
||||
self.assertTrue(hasattr(image_processing, "size"))
|
||||
self.assertTrue(hasattr(image_processing, "do_center_crop"))
|
||||
self.assertTrue(hasattr(image_processing, "center_crop"))
|
||||
self.assertTrue(hasattr(image_processing, "do_normalize"))
|
||||
self.assertTrue(hasattr(image_processing, "image_mean"))
|
||||
self.assertTrue(hasattr(image_processing, "image_std"))
|
||||
self.assertTrue(hasattr(image_processing, "do_convert_rgb"))
|
||||
|
||||
def test_image_processor_from_dict_with_kwargs(self):
|
||||
image_processor = self.image_processing_class.from_dict(self.image_processor_dict)
|
||||
self.assertEqual(image_processor.size, {"shortest_edge": 20})
|
||||
self.assertEqual(image_processor.crop_size, {"height": 18, "width": 18})
|
||||
for image_processing_class in self.image_processor_list:
|
||||
image_processor = image_processing_class.from_dict(self.image_processor_dict)
|
||||
self.assertEqual(image_processor.size, {"shortest_edge": 20})
|
||||
self.assertEqual(image_processor.crop_size, {"height": 18, "width": 18})
|
||||
|
||||
image_processor = self.image_processing_class.from_dict(self.image_processor_dict, size=42, crop_size=84)
|
||||
self.assertEqual(image_processor.size, {"shortest_edge": 42})
|
||||
self.assertEqual(image_processor.crop_size, {"height": 84, "width": 84})
|
||||
image_processor = image_processing_class.from_dict(self.image_processor_dict, size=42, crop_size=84)
|
||||
self.assertEqual(image_processor.size, {"shortest_edge": 42})
|
||||
self.assertEqual(image_processor.crop_size, {"height": 84, "width": 84})
|
||||
|
||||
# Ignore copy
|
||||
def test_padding(self):
|
||||
@@ -157,45 +166,72 @@ class LlavaImageProcessingTest(ImageProcessingTestMixin, unittest.TestCase):
|
||||
result.paste(image, ((height - width) // 2, 0))
|
||||
return result
|
||||
|
||||
image_processor = self.image_processing_class.from_dict(self.image_processor_dict)
|
||||
image_inputs = self.image_processor_tester.prepare_image_inputs(equal_resolution=False, numpify=True)
|
||||
for i, image_processing_class in enumerate(self.image_processor_list):
|
||||
image_processor = image_processing_class.from_dict(self.image_processor_dict)
|
||||
numpify = i == 0
|
||||
torchify = i == 1
|
||||
image_inputs = self.image_processor_tester.prepare_image_inputs(
|
||||
equal_resolution=False, numpify=numpify, torchify=torchify
|
||||
)
|
||||
|
||||
# test with images in channel-last and channel-first format
|
||||
for image in image_inputs:
|
||||
padded_image = image_processor.pad_to_square(image)
|
||||
padded_image_original = pad_to_square_original(Image.fromarray(image))
|
||||
padded_image_original = np.array(padded_image_original)
|
||||
# test with images in channel-last and channel-first format (only channel-first for torch)
|
||||
for image in image_inputs:
|
||||
padded_image = image_processor.pad_to_square(image)
|
||||
if i == 0:
|
||||
padded_image_original = pad_to_square_original(Image.fromarray(image))
|
||||
padded_image_original = np.array(padded_image_original)
|
||||
|
||||
np.testing.assert_allclose(padded_image, padded_image_original)
|
||||
np.testing.assert_allclose(padded_image, padded_image_original)
|
||||
|
||||
padded_image = image_processor.pad_to_square(image.transpose(2, 0, 1), input_data_format="channels_first")
|
||||
padded_image = padded_image.transpose(1, 2, 0)
|
||||
padded_image = image_processor.pad_to_square(
|
||||
image.transpose(2, 0, 1), input_data_format="channels_first"
|
||||
)
|
||||
padded_image = padded_image.transpose(1, 2, 0)
|
||||
|
||||
np.testing.assert_allclose(padded_image, padded_image_original)
|
||||
np.testing.assert_allclose(padded_image, padded_image_original)
|
||||
else:
|
||||
padded_image_original = pad_to_square_original(F.to_pil_image(image))
|
||||
padded_image = padded_image.permute(1, 2, 0)
|
||||
np.testing.assert_allclose(padded_image, padded_image_original)
|
||||
|
||||
# test background color
|
||||
background_color = (122, 116, 104)
|
||||
for image in image_inputs:
|
||||
padded_image = image_processor.pad_to_square(image, background_color=background_color)
|
||||
padded_image_original = pad_to_square_original(Image.fromarray(image), background_color=background_color)
|
||||
padded_image_original = np.array(padded_image_original)
|
||||
# test background color
|
||||
background_color = (122, 116, 104)
|
||||
for image in image_inputs:
|
||||
padded_image = image_processor.pad_to_square(image, background_color=background_color)
|
||||
if i == 0:
|
||||
padded_image_original = pad_to_square_original(
|
||||
Image.fromarray(image), background_color=background_color
|
||||
)
|
||||
else:
|
||||
padded_image_original = pad_to_square_original(
|
||||
F.to_pil_image(image), background_color=background_color
|
||||
)
|
||||
padded_image = padded_image.permute(1, 2, 0)
|
||||
padded_image_original = np.array(padded_image_original)
|
||||
|
||||
np.testing.assert_allclose(padded_image, padded_image_original)
|
||||
np.testing.assert_allclose(padded_image, padded_image_original)
|
||||
|
||||
background_color = 122
|
||||
for image in image_inputs:
|
||||
padded_image = image_processor.pad_to_square(image, background_color=background_color)
|
||||
padded_image_original = pad_to_square_original(Image.fromarray(image), background_color=background_color)
|
||||
padded_image_original = np.array(padded_image_original)
|
||||
background_color = 122
|
||||
for image in image_inputs:
|
||||
padded_image = image_processor.pad_to_square(image, background_color=background_color)
|
||||
if i == 0:
|
||||
padded_image_original = pad_to_square_original(
|
||||
Image.fromarray(image), background_color=background_color
|
||||
)
|
||||
else:
|
||||
padded_image_original = pad_to_square_original(
|
||||
F.to_pil_image(image), background_color=background_color
|
||||
)
|
||||
padded_image = padded_image.permute(1, 2, 0)
|
||||
padded_image_original = np.array(padded_image_original)
|
||||
np.testing.assert_allclose(padded_image, padded_image_original)
|
||||
|
||||
np.testing.assert_allclose(padded_image, padded_image_original)
|
||||
# background color length should match channel length
|
||||
with self.assertRaises(ValueError):
|
||||
padded_image = image_processor.pad_to_square(image_inputs[0], background_color=(122, 104))
|
||||
|
||||
# background color length should match channel length
|
||||
with self.assertRaises(ValueError):
|
||||
padded_image = image_processor.pad_to_square(image_inputs[0], background_color=(122, 104))
|
||||
|
||||
with self.assertRaises(ValueError):
|
||||
padded_image = image_processor.pad_to_square(image_inputs[0], background_color=(122, 104, 0, 0))
|
||||
with self.assertRaises(ValueError):
|
||||
padded_image = image_processor.pad_to_square(image_inputs[0], background_color=(122, 104, 0, 0))
|
||||
|
||||
@unittest.skip(reason="LLaVa does not support 4 channel images yet")
|
||||
# Ignore copy
|
||||
|
||||
Reference in New Issue
Block a user