test_qwen2_vl.py 4.66 KB
Newer Older
1
# SPDX-License-Identifier: Apache-2.0
2
# SPDX-FileCopyrightText: Copyright contributors to the vLLM project
3

4
import pytest
5
6
from packaging.version import Version
from transformers import __version__ as TRANSFORMERS_VERSION
7

8
from vllm.multimodal import MULTIMODAL_REGISTRY
9

10
from ....conftest import ImageTestAssets
11
from ...utils import build_model_context
12
13


14
@pytest.mark.parametrize("model_id", ["Qwen/Qwen2-VL-2B-Instruct"])
15
@pytest.mark.parametrize(
16
17
    ("mm_processor_kwargs", "expected_toks_per_img", "expected_pixels_shape"),
    [
18
        ({}, 1426, (5704, 1176)),
19
        ({"min_pixels": 64**2, "max_pixels": 512**2}, 330, (1320, 1176)),
20
21
22
23
24
25
26
27
28
29
        (
            {
                "size": {
                    "shortest_edge": 64**2,
                    "longest_edge": 512**2,
                },
            },
            330,
            (1320, 1176),
        ),
30
31
    ],
)
32
@pytest.mark.parametrize("num_imgs", [1, 2])
33
@pytest.mark.parametrize("kwargs_on_init", [True, False])
34
def test_processor_override(
35
    image_assets: ImageTestAssets,
36
37
    model_id: str,
    mm_processor_kwargs: dict[str, object],
38
    expected_toks_per_img: int,
39
    expected_pixels_shape: tuple[int, int],
40
    num_imgs: int,
41
    kwargs_on_init: bool,
42
43
):
    """Ensure Qwen2VLMultiModalProcessor handles min/max pixels properly."""
44
45
46
47
48
49
    if (
        Version(TRANSFORMERS_VERSION) < Version("5.2.0")
        and "size" in mm_processor_kwargs
    ):
        pytest.skip("`size` ignored by `Qwen2VLProcessor.__call__`")

50
    ctx = build_model_context(
51
        model_id,
52
        mm_processor_kwargs=mm_processor_kwargs if kwargs_on_init else None,
53
        limit_mm_per_prompt={"image": num_imgs},
54
    )
55
    processor = MULTIMODAL_REGISTRY.create_processor(ctx.model_config)
56
    tokenizer = processor.info.get_tokenizer()
57
    hf_processor_mm_kwargs = {} if kwargs_on_init else mm_processor_kwargs
58

59
60
    # Build the image str / prompt based on the number of images we pass
    prompt = "<|vision_start|><|image_pad|><|vision_end|>" * num_imgs
61
    mm_data = {"image": [image_assets[0].pil_image] * num_imgs}
62

63
    processed_inputs = processor(
64
65
66
67
        prompt,
        mm_items=processor.info.parse_mm_data(mm_data),
        hf_processor_mm_kwargs=hf_processor_mm_kwargs,
    )
68
69

    # Ensure we have the right number of placeholders per num_crops size
70
    hf_processor = processor.info.get_hf_processor(**hf_processor_mm_kwargs)
71
72
    image_token_id = tokenizer.convert_tokens_to_ids(hf_processor.image_token)
    img_tok_count = processed_inputs["prompt_token_ids"].count(image_token_id)
73
    pixel_shape = processed_inputs["mm_kwargs"].get_data()["pixel_values"].shape
74
75
76
77

    assert img_tok_count == expected_toks_per_img * num_imgs
    assert pixel_shape[0] == expected_pixels_shape[0] * num_imgs
    assert pixel_shape[1] == expected_pixels_shape[1]
78
79
80


@pytest.mark.parametrize("model_id", ["Qwen/Qwen2-VL-2B-Instruct"])
81
82
83
84
85
86
87
88
89
@pytest.mark.parametrize(
    "mm_processor_kwargs",
    [
        {"min_pixels": 28 * 28, "max_pixels": 1280 * 28 * 28},
        {"min_pixels": 28 * 28, "max_pixels": 1283 * 28 * 28},
        {"size": {"shortest_edge": 28 * 28, "longest_edge": 1280 * 28 * 28}},
        {"size": {"shortest_edge": 28 * 28, "longest_edge": 1283 * 28 * 28}},
    ],
)
90
91
92
def test_get_image_size_with_most_features(
    image_assets: ImageTestAssets,
    model_id: str,
93
    mm_processor_kwargs: dict[str, object],
94
):
95
96
97
98
99
100
    if (
        Version(TRANSFORMERS_VERSION) < Version("5.2.0")
        and "size" in mm_processor_kwargs
    ):
        pytest.skip("`size` ignored by `Qwen2VLProcessor.__call__`")

101
102
    ctx = build_model_context(
        model_id,
103
        mm_processor_kwargs=mm_processor_kwargs,
104
105
106
107
        limit_mm_per_prompt={"image": 1},
    )
    processor = MULTIMODAL_REGISTRY.create_processor(ctx.model_config)

108
    hf_processor = processor.info.get_hf_processor(**mm_processor_kwargs)
109
110
111
112
113
114
115
    merge_size = processor.info.get_hf_config().vision_config.spatial_merge_size

    max_image_size = processor.info.get_image_size_with_most_features()
    max_tokens = processor.info.get_num_image_tokens(
        image_width=max_image_size.width,
        image_height=max_image_size.height,
        image_processor=hf_processor.image_processor,
116
        mm_kwargs=mm_processor_kwargs,
117
118
119
120
121
    )

    prompt = "<|vision_start|><|image_pad|><|vision_end|>"
    for asset in image_assets:
        mm_data = {"image": [asset.pil_image]}
122
        processed_inputs = processor(
123
124
            prompt,
            mm_items=processor.info.parse_mm_data(mm_data),
125
            hf_processor_mm_kwargs=mm_processor_kwargs,
126
        )
127
128
129
130
        grid_thw = processed_inputs["mm_kwargs"].get_data()["image_grid_thw"].tolist()
        t, h, w = grid_thw[0]
        tokens = (t * h * w) // (merge_size**2)
        assert tokens < max_tokens