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

4
from functools import lru_cache
5
from typing import TYPE_CHECKING, Any, Optional, Union, cast
6

7
8
9
10
11
12
from transformers import (
    AutoFeatureExtractor,
    AutoImageProcessor,
    AutoProcessor,
    AutoVideoProcessor,
)
13
14
from transformers.feature_extraction_utils import FeatureExtractionMixin
from transformers.image_processing_utils import BaseImageProcessor
15
from transformers.processing_utils import ProcessorMixin
16
from transformers.video_processing_utils import BaseVideoProcessor
17
18
from typing_extensions import TypeVar

19
20
from vllm.utils import get_allowed_kwarg_only_overrides

21
22
23
24
if TYPE_CHECKING:
    from vllm.config import ModelConfig

_P = TypeVar("_P", bound=ProcessorMixin, default=ProcessorMixin)
25
_V = TypeVar("_V", bound=BaseVideoProcessor, default=BaseVideoProcessor)
26
27
28
29
30
31
32
33
34
35
36
37
38


class HashableDict(dict):
    """
    A dictionary that can be hashed by lru_cache.
    """

    # NOTE: pythonic dict is not hashable,
    # we override on it directly for simplicity
    def __hash__(self) -> int:  # type: ignore[override]
        return hash(frozenset(self.items()))


39
40
41
42
43
44
45
46
47
class HashableList(list):
    """
    A list that can be hashed by lru_cache.
    """

    def __hash__(self) -> int:  # type: ignore[override]
        return hash(tuple(self))


48
49
50
51
52
53
54
55
def _get_processor_factory_fn(processor_cls: Union[type, tuple[type, ...]]):
    if isinstance(processor_cls, tuple) or processor_cls == ProcessorMixin:
        return AutoProcessor.from_pretrained
    if hasattr(processor_cls, "from_pretrained"):
        return processor_cls.from_pretrained

    return processor_cls

56

57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
def _merge_mm_kwargs(
    model_config: "ModelConfig",
    processor_cls: Union[type, tuple[type, ...]],
    /,
    **kwargs,
):
    mm_config = model_config.get_multimodal_config()
    merged_kwargs = mm_config.merge_mm_processor_kwargs(kwargs)

    factory = _get_processor_factory_fn(processor_cls)
    allowed_kwargs = get_allowed_kwarg_only_overrides(
        factory,
        merged_kwargs,
        requires_kw_only=False,
        allow_var_kwargs=True,
    )
73
74
75
76

    # NOTE: Pythonic dict is not hashable and will raise unhashable type
    # error when calling `cached_get_processor`, therefore we need to
    # wrap it to a hashable dict.
77
    for key, value in allowed_kwargs.items():
78
        if isinstance(value, dict):
79
            allowed_kwargs[key] = HashableDict(value)
80
        if isinstance(value, list):
81
82
83
            allowed_kwargs[key] = HashableList(value)

    return allowed_kwargs
84

85
86
87

def get_processor(
    processor_name: str,
88
    *args: Any,
89
    revision: Optional[str] = None,
90
    trust_remote_code: bool = False,
91
    processor_cls: Union[type[_P], tuple[type[_P], ...]] = ProcessorMixin,
92
    **kwargs: Any,
93
) -> _P:
94
    """Load a processor for the given model name via HuggingFace."""
95
96
    if revision is None:
        revision = "main"
97
98

    try:
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
        if isinstance(processor_cls, tuple) or processor_cls == ProcessorMixin:
            processor = AutoProcessor.from_pretrained(
                processor_name,
                *args,
                revision=revision,
                trust_remote_code=trust_remote_code,
                **kwargs,
            )
        elif issubclass(processor_cls, ProcessorMixin):
            processor = processor_cls.from_pretrained(
                processor_name,
                *args,
                revision=revision,
                trust_remote_code=trust_remote_code,
                **kwargs,
            )
        else:
            # Processors that are standalone classes unrelated to HF
            processor = processor_cls(*args, **kwargs)
118
119
120
121
122
123
124
125
126
127
    except ValueError as e:
        # If the error pertains to the processor class not existing or not
        # currently being imported, suggest using the --trust-remote-code flag.
        # Unlike AutoTokenizer, AutoProcessor does not separate such errors
        if not trust_remote_code:
            err_msg = (
                "Failed to load the processor. If the processor is "
                "a custom processor not yet available in the HuggingFace "
                "transformers library, consider setting "
                "`trust_remote_code=True` in LLM or using the "
128
129
                "`--trust-remote-code` flag in the CLI."
            )
130
131
132
133
            raise RuntimeError(err_msg) from e
        else:
            raise e

134
    if not isinstance(processor, processor_cls):
135
136
137
138
139
        raise TypeError(
            "Invalid type of HuggingFace processor. "
            f"Expected type: {processor_cls}, but "
            f"found type: {type(processor)}"
        )
140
141

    return processor
142
143


144
145
146
cached_get_processor = lru_cache(get_processor)


147
148
149
150
151
152
153
def cached_processor_from_config(
    model_config: "ModelConfig",
    processor_cls: Union[type[_P], tuple[type[_P], ...]] = ProcessorMixin,
    **kwargs: Any,
) -> _P:
    return cached_get_processor(
        model_config.model,
154
        revision=model_config.revision,
155
156
        trust_remote_code=model_config.trust_remote_code,
        processor_cls=processor_cls,  # type: ignore[arg-type]
157
        **_merge_mm_kwargs(model_config, processor_cls, **kwargs),
158
159
160
    )


161
162
163
def get_feature_extractor(
    processor_name: str,
    *args: Any,
164
    revision: Optional[str] = None,
165
166
167
    trust_remote_code: bool = False,
    **kwargs: Any,
):
168
    """Load an audio feature extractor for the given model name
169
170
171
172
173
    via HuggingFace."""
    try:
        feature_extractor = AutoFeatureExtractor.from_pretrained(
            processor_name,
            *args,
174
            revision=revision,
175
            trust_remote_code=trust_remote_code,
176
177
            **kwargs,
        )
178
179
180
181
182
183
184
185
186
187
    except ValueError as e:
        # If the error pertains to the processor class not existing or not
        # currently being imported, suggest using the --trust-remote-code flag.
        # Unlike AutoTokenizer, AutoImageProcessor does not separate such errors
        if not trust_remote_code:
            err_msg = (
                "Failed to load the feature extractor. If the feature "
                "extractor is a custom extractor not yet available in the "
                "HuggingFace transformers library, consider setting "
                "`trust_remote_code=True` in LLM or using the "
188
189
                "`--trust-remote-code` flag in the CLI."
            )
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
            raise RuntimeError(err_msg) from e
        else:
            raise e
    return cast(FeatureExtractionMixin, feature_extractor)


cached_get_feature_extractor = lru_cache(get_feature_extractor)


def cached_feature_extractor_from_config(
    model_config: "ModelConfig",
    **kwargs: Any,
):
    return cached_get_feature_extractor(
        model_config.model,
205
        revision=model_config.revision,
206
        trust_remote_code=model_config.trust_remote_code,
207
        **_merge_mm_kwargs(model_config, AutoFeatureExtractor, **kwargs),
208
209
210
    )


211
212
213
def get_image_processor(
    processor_name: str,
    *args: Any,
214
    revision: Optional[str] = None,
215
216
217
218
219
220
221
222
    trust_remote_code: bool = False,
    **kwargs: Any,
):
    """Load an image processor for the given model name via HuggingFace."""
    try:
        processor = AutoImageProcessor.from_pretrained(
            processor_name,
            *args,
223
            revision=revision,
224
            trust_remote_code=trust_remote_code,
225
226
            **kwargs,
        )
227
228
229
230
231
232
233
234
235
236
    except ValueError as e:
        # If the error pertains to the processor class not existing or not
        # currently being imported, suggest using the --trust-remote-code flag.
        # Unlike AutoTokenizer, AutoImageProcessor does not separate such errors
        if not trust_remote_code:
            err_msg = (
                "Failed to load the image processor. If the image processor is "
                "a custom processor not yet available in the HuggingFace "
                "transformers library, consider setting "
                "`trust_remote_code=True` in LLM or using the "
237
238
                "`--trust-remote-code` flag in the CLI."
            )
239
240
241
242
243
244
245
            raise RuntimeError(err_msg) from e
        else:
            raise e

    return cast(BaseImageProcessor, processor)


246
247
248
249
250
251
252
253
254
cached_get_image_processor = lru_cache(get_image_processor)


def cached_image_processor_from_config(
    model_config: "ModelConfig",
    **kwargs: Any,
):
    return cached_get_image_processor(
        model_config.model,
255
        revision=model_config.revision,
256
        trust_remote_code=model_config.trust_remote_code,
257
        **_merge_mm_kwargs(model_config, AutoImageProcessor, **kwargs),
258
    )
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276


def get_video_processor(
    processor_name: str,
    *args: Any,
    revision: Optional[str] = None,
    trust_remote_code: bool = False,
    processor_cls_overrides: Optional[type[_V]] = None,
    **kwargs: Any,
):
    """Load a video processor for the given model name via HuggingFace."""
    try:
        processor_cls = processor_cls_overrides or AutoVideoProcessor
        processor = processor_cls.from_pretrained(
            processor_name,
            *args,
            revision=revision,
            trust_remote_code=trust_remote_code,
277
278
            **kwargs,
        )
279
280
281
282
283
284
285
286
287
288
    except ValueError as e:
        # If the error pertains to the processor class not existing or not
        # currently being imported, suggest using the --trust-remote-code flag.
        # Unlike AutoTokenizer, AutoVideoProcessor does not separate such errors
        if not trust_remote_code:
            err_msg = (
                "Failed to load the video processor. If the video processor is "
                "a custom processor not yet available in the HuggingFace "
                "transformers library, consider setting "
                "`trust_remote_code=True` in LLM or using the "
289
290
                "`--trust-remote-code` flag in the CLI."
            )
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
            raise RuntimeError(err_msg) from e
        else:
            raise e

    return cast(BaseVideoProcessor, processor)


cached_get_video_processor = lru_cache(get_video_processor)


def cached_video_processor_from_config(
    model_config: "ModelConfig",
    processor_cls: Optional[type[_V]] = None,
    **kwargs: Any,
):
    return cached_get_video_processor(
        model_config.model,
        revision=model_config.revision,
        trust_remote_code=model_config.trust_remote_code,
        processor_cls_overrides=processor_cls,  # type: ignore[arg-type]
        **_merge_mm_kwargs(model_config, AutoVideoProcessor, **kwargs),
    )