test_common.py 8.76 KB
Newer Older
1
2
# SPDX-License-Identifier: Apache-2.0

3
import copy
4
from functools import partial
5
from typing import Optional
6
7
8
9
10
11
12
13
14

import numpy as np
import pytest
from PIL import Image

from vllm.config import ModelConfig
from vllm.inputs import InputProcessingContext
from vllm.multimodal import MULTIMODAL_REGISTRY
from vllm.multimodal.processing import ProcessingCache
15
from vllm.transformers_utils.tokenizer import cached_tokenizer_from_config
16
17

from ....multimodal.utils import random_audio, random_image, random_video
18
from ...registry import HF_EXAMPLE_MODELS
19
20
21
22
23
24
25


def _test_processing_correctness(
    model_id: str,
    hit_rate: float,
    num_batches: int,
    simplify_rate: float,
26
    ignore_mm_keys: Optional[list[str]] = None,
27
):
28
29
30
    model_info = HF_EXAMPLE_MODELS.find_hf_info(model_id)
    model_info.check_available_online(on_fail="skip")
    model_info.check_transformers_version(on_fail="skip")
31
32
33
34

    model_config = ModelConfig(
        model_id,
        task="auto",
35
36
        tokenizer=model_info.tokenizer or model_id,
        tokenizer_mode=model_info.tokenizer_mode,
37
        trust_remote_code=model_info.trust_remote_code,
38
39
40
        seed=0,
        dtype="float16",
        revision=None,
41
        hf_overrides=model_info.hf_overrides,
42
43
44
45
46
47
    )

    model_cls = MULTIMODAL_REGISTRY._get_model_cls(model_config)
    factories = MULTIMODAL_REGISTRY._processor_factories[model_cls]
    ctx = InputProcessingContext(
        model_config,
48
        tokenizer=cached_tokenizer_from_config(model_config),
49
50
51
52
    )
    # Ensure that it can fit all of the data
    cache = ProcessingCache(capacity=1 << 30)

53
54
55
56
57
58
59
60
61
    processing_info = factories.info(ctx)
    supported_mm_limits = processing_info.get_supported_mm_limits()
    limit_mm_per_prompt = {
        modality: 3 if limit is None else limit
        for modality, limit in supported_mm_limits.items()
    }

    model_config.get_multimodal_config().limit_per_prompt = limit_mm_per_prompt

62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
    baseline_processor = factories.build_processor(ctx, cache=None)
    cached_processor = factories.build_processor(ctx, cache=cache)
    dummy_inputs = baseline_processor.dummy_inputs
    tokenizer = baseline_processor.info.get_tokenizer()

    rng = np.random.RandomState(0)

    input_to_hit = {
        "image": Image.new("RGB", size=(128, 128)),
        "video": np.zeros((4, 128, 128, 3), dtype=np.uint8),
        "audio": (np.zeros((512, )), 16000),
    }
    input_factory = {
        "image":
        partial(random_image, rng, min_wh=128, max_wh=256),
        "video":
        partial(random_video,
                rng,
                min_frames=2,
                max_frames=8,
                min_wh=128,
                max_wh=256),
        "audio":
        partial(random_audio, rng, min_len=512, max_len=1024, sr=16000),
    }

88
    tokenizer_encode_kwargs = {}
89
90
    if model_config.hf_config.model_type in ("mllama", "whisper", "ultravox"):
        # For some multimodal models, tokenizer will always add bos_token
91
92
93
        # at the beginning of prompt by default, causing hf_processor outputs
        # incorrect token ids. So we need use `add_special_tokens=False` here
        # to leave bos_token to be added by the processor.
94
95
        tokenizer_encode_kwargs = {"add_special_tokens": False}

96
97
98
99
    for batch_idx in range(num_batches):
        mm_data = {
            k:
            [(input_to_hit[k] if rng.rand() < hit_rate else input_factory[k]())
100
             for _ in range(rng.randint(limit + 1))]
101
            for k, limit in limit_mm_per_prompt.items()
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
        }

        mm_counts = {k: len(vs) for k, vs in mm_data.items()}
        prompt = dummy_inputs.get_dummy_processor_inputs(
            model_config.max_model_len,
            mm_counts,
        ).prompt_text

        # Drop unnecessary keys and test single -> multi conversion
        if rng.rand() < simplify_rate:
            for k in list(mm_data.keys()):
                if not mm_data[k]:
                    del mm_data[k]
                elif len(mm_data[k]) == 1:
                    mm_data[k] = mm_data[k][0]

        baseline_result = baseline_processor.apply(
            prompt,
            mm_data=mm_data,
            hf_processor_mm_kwargs={},
        )
        cached_result = cached_processor.apply(
            prompt,
            mm_data=mm_data,
            hf_processor_mm_kwargs={},
        )

129
130
131
132
        assert _drop_mm_kwargs_keys(
            baseline_result, ignore_mm_keys) == _drop_mm_kwargs_keys(
                cached_result, ignore_mm_keys), (
                    f"Failed ({batch_idx=}, {prompt=}, {mm_data=})")
133
134

        baseline_tokenized_result = baseline_processor.apply(
135
            tokenizer.encode(prompt, **tokenizer_encode_kwargs),
136
137
138
139
            mm_data=mm_data,
            hf_processor_mm_kwargs={},
        )

140
141
142
143
        assert _drop_mm_kwargs_keys(
            baseline_result, ignore_mm_keys) == _drop_mm_kwargs_keys(
                baseline_tokenized_result, ignore_mm_keys), (
                    f"Failed ({batch_idx=}, {prompt=}, {mm_data=})")
144
145

        cached_tokenized_result = cached_processor.apply(
146
            tokenizer.encode(prompt, **tokenizer_encode_kwargs),
147
148
149
150
            mm_data=mm_data,
            hf_processor_mm_kwargs={},
        )

151
152
153
154
        assert _drop_mm_kwargs_keys(
            cached_result, ignore_mm_keys) == _drop_mm_kwargs_keys(
                cached_tokenized_result, ignore_mm_keys), (
                    f"Failed ({batch_idx=}, {prompt=}, {mm_data=})")
155
156
157


# yapf: disable
158
159
160
161
162
@pytest.mark.parametrize("model_id", [
    "rhymes-ai/Aria",
    "Salesforce/blip2-opt-2.7b",
    "facebook/chameleon-7b",
    "deepseek-ai/deepseek-vl2-tiny",
163
    "microsoft/Florence-2-base",
164
    "adept/fuyu-8b",
165
    "THUDM/glm-4v-9b",
166
167
    "h2oai/h2ovl-mississippi-800m",
    "OpenGVLab/InternVL2-1B",
168
    "HuggingFaceM4/Idefics3-8B-Llama3",
169
170
171
172
    "llava-hf/llava-1.5-7b-hf",
    "llava-hf/llava-v1.6-mistral-7b-hf",
    "llava-hf/LLaVA-NeXT-Video-7B-hf",
    "llava-hf/llava-onevision-qwen2-0.5b-ov-hf",
173
    "meta-llama/Llama-3.2-11B-Vision-Instruct",
174
175
    "TIGER-Lab/Mantis-8B-siglip-llama3",
    "mistral-community/pixtral-12b",
176
177
    "openbmb/MiniCPM-o-2_6",
    "openbmb/MiniCPM-V-2_6",
178
179
    "allenai/Molmo-7B-D-0924",
    "allenai/Molmo-7B-O-0924",
180
    "nvidia/NVLM-D-72B",
181
182
    "Qwen/Qwen-VL-Chat",
    "Qwen/Qwen2-VL-2B-Instruct",
Roger Wang's avatar
Roger Wang committed
183
    "Qwen/Qwen2.5-VL-3B-Instruct",
184
    "Qwen/Qwen2-Audio-7B-Instruct",
185
    "fixie-ai/ultravox-v0_5-llama-3_2-1b",
186
    "openai/whisper-large-v3",
187
188
    "google/paligemma-3b-mix-224",
    "google/paligemma2-3b-ft-docci-448",
189
190
191
192
193
194
195
196
197
198
199
])
@pytest.mark.parametrize("hit_rate", [0.3, 0.5, 1.0])
@pytest.mark.parametrize("num_batches", [32])
@pytest.mark.parametrize("simplify_rate", [1.0])
# yapf: enable
def test_processing_correctness(
    model_id: str,
    hit_rate: float,
    num_batches: int,
    simplify_rate: float,
):
200
201
202
203
204
205
206
    ignore_mm_keys = None
    if 'ultravox' in model_id:
        # In Ultravox, the audio_features can be different depending on padding
        # The slight difference should not be a problem though, since
        # attention_mask lets us ignore the difference.
        ignore_mm_keys = ['audio_features']

207
208
209
210
211
    _test_processing_correctness(
        model_id,
        hit_rate=hit_rate,
        num_batches=num_batches,
        simplify_rate=simplify_rate,
212
        ignore_mm_keys=ignore_mm_keys,
213
214
215
216
    )


# yapf: disable
217
@pytest.mark.parametrize("model_id", ["microsoft/Phi-3-vision-128k-instruct"])
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
@pytest.mark.parametrize("hit_rate", [0.3, 0.5, 1.0])
@pytest.mark.parametrize("num_batches", [32])
@pytest.mark.parametrize("simplify_rate", [1.0])
# yapf: enable
def test_processing_correctness_phi3v(
    model_id: str,
    hit_rate: float,
    num_batches: int,
    simplify_rate: float,
):
    # HACK - this is an attempted workaround for the following bug
    # https://github.com/huggingface/transformers/issues/34307
    from transformers import AutoImageProcessor  # noqa: F401
    from transformers import AutoProcessor  # noqa: F401

    AutoImageProcessor.from_pretrained(model_id, trust_remote_code=True)

    _test_processing_correctness(
        model_id,
        hit_rate=hit_rate,
        num_batches=num_batches,
        simplify_rate=simplify_rate,
    )
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266


def _drop_mm_kwargs_keys(result: dict,
                         ignore_mm_keys: Optional[list[str]] = None) -> dict:
    """Drop specified keys from result['mm_kwargs'].

    This is mainly to avoid doing exact match of audio_features in ultravox.

    Args:
        result: Result to drop keys from
        ignore_mm_keys: List of keys to ignore, e.g. ['audio_features']
    """
    if not ignore_mm_keys:
        return result

    if 'mm_kwargs' in result:
        result = copy.deepcopy(result)
        mm_kwargs = result['mm_kwargs']
        for key in ignore_mm_keys:
            mm_kwargs.pop(key, None)
        for items in mm_kwargs._items_by_modality.values():
            for item in items:
                for key in ignore_mm_keys:
                    item.pop(key, None)

    return result