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

4
5
import importlib
import inspect
6
from functools import lru_cache
7
from typing import TYPE_CHECKING, Any, cast, get_args, get_type_hints
8

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

22
from vllm.logger import init_logger
23
24
from vllm.transformers_utils.gguf_utils import is_gguf
from vllm.transformers_utils.utils import convert_model_repo_to_path
25
from vllm.utils.func_utils import get_allowed_kwarg_only_overrides
26

27
28
logger = init_logger(__name__)

29
if TYPE_CHECKING:
30
    from vllm.config import ModelConfig
31

32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48

def _transformers_v4_compatibility_import():
    """Some remote code processors still import `ChatTemplateLoadKwargs` which was a
    subset of `ProcessorChatTemplateKwargs` as defined in Transformers v4.
    In Transformers v5 these were merged into `ProcessorChatTemplateKwargs` and
    `ChatTemplateLoadKwargs` was removed. For backward compatibility, we add an alias
    for `ChatTemplateLoadKwargs` if it doesn't exist.

    This can be removed if `HCXVisionForCausalLM` is upstreamed to Transformers."""
    old_import = getattr(processing_utils, "ChatTemplateLoadKwargs", None)
    new_import = getattr(processing_utils, "ProcessorChatTemplateKwargs", None)
    if old_import is None and new_import is not None:
        processing_utils.ChatTemplateLoadKwargs = new_import


_transformers_v4_compatibility_import()

49
_P = TypeVar("_P", bound=ProcessorMixin, default=ProcessorMixin)
50
_V = TypeVar("_V", bound=BaseVideoProcessor, default=BaseVideoProcessor)
51
52
53
54
55
56
57
58
59
60
61
62
63


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()))


64
65
66
67
68
69
70
71
72
class HashableList(list):
    """
    A list that can be hashed by lru_cache.
    """

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


73
def _get_processor_factory_fn(processor_cls: type | tuple[type, ...]):
74
75
76
77
78
79
80
    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

81

82
83
84
85
86
87
88
89
90
91
@lru_cache
def _collect_dynamic_keys_from_processing_kwargs(kwargs_cls: type) -> set[str]:
    dynamic_kwargs: set[str] = set()
    if kwargs_cls is None:
        return dynamic_kwargs
    # get kwargs annotations in processor
    # merge text_kwargs / images_kwargs / videos_kwargs / audio_kwargs
    kwargs_type_annotations = get_type_hints(kwargs_cls)
    for kw_type in ("text_kwargs", "images_kwargs", "videos_kwargs", "audio_kwargs"):
        if kw_type in kwargs_type_annotations:
92
93
94
95
96
97
98
            # Use __annotations__ instead of get_type_hints() to avoid
            # NameError from unresolved forward references (e.g.
            # PILImageResampling). We only need key names, not types.
            kw_cls = kwargs_type_annotations[kw_type]
            kw_annotations: dict[str, Any] = {}
            for base in reversed(kw_cls.__mro__):
                kw_annotations.update(getattr(base, "__annotations__", {}))
99
100
101
102
103
104
            for kw_name in kw_annotations:
                dynamic_kwargs.add(kw_name)
    dynamic_kwargs |= {"text_kwargs", "images_kwargs", "videos_kwargs", "audio_kwargs"}
    return dynamic_kwargs


105
106
def _merge_mm_kwargs(
    model_config: "ModelConfig",
107
    processor_cls: type | tuple[type, ...],
108
109
110
111
112
113
114
115
116
117
118
119
120
    /,
    **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,
    )
121
122
123
    # 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.
124
    for key, value in allowed_kwargs.items():
125
        if isinstance(value, dict):
126
            allowed_kwargs[key] = HashableDict(value)
127
        if isinstance(value, list):
128
129
130
            allowed_kwargs[key] = HashableList(value)

    return allowed_kwargs
131

132
133
134

def get_processor(
    processor_name: str,
135
    *args: Any,
136
    revision: str | None = None,
137
    trust_remote_code: bool = False,
138
    processor_cls: type[_P] | tuple[type[_P], ...] = ProcessorMixin,
139
    **kwargs: Any,
140
) -> _P:
141
    """Load a processor for the given model name via HuggingFace."""
142
143
    if revision is None:
        revision = "main"
144
    try:
145
        processor_name = convert_model_repo_to_path(processor_name)
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
        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)
165
166
167
168
169
170
171
172
173
174
    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 "
175
176
                "`--trust-remote-code` flag in the CLI."
            )
177
178
179
180
            raise RuntimeError(err_msg) from e
        else:
            raise e

181
    if not isinstance(processor, processor_cls):
182
183
184
185
186
        raise TypeError(
            "Invalid type of HuggingFace processor. "
            f"Expected type: {processor_cls}, but "
            f"found type: {type(processor)}"
        )
187
188

    return processor
189
190


191
192
193
cached_get_processor = lru_cache(get_processor)


194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
@lru_cache
def get_processor_kwargs_from_processor(processor: _P) -> set[str]:
    try:
        # get kwargs annotations in processor
        call_kwargs = inspect.signature(type(processor).__call__).parameters.get(
            "kwargs"
        )
        call_kwargs_annotations = call_kwargs.annotation if call_kwargs else None
        # if the processor has explicit kwargs annotation, use it
        if call_kwargs_annotations not in (None, inspect._empty):
            # get_type_hints will parse all type annotations at runtime,
            # and if an annotation refers to a type or
            # name that hasn’t been imported or defined, it will raise an error.
            # So we use __annotations__ to get the raw annotations directly.
            return _collect_dynamic_keys_from_processing_kwargs(
                get_args(call_kwargs_annotations)[0]
            )
        # otherwise, try to get from ProcessingKwargs
        else:
            module_name = type(processor).__module__
            mod = importlib.import_module(module_name)
            # find *ProcessingKwargs in the module
            processor_kwargs: set[str] = set()
            for name, obj in vars(mod).items():
                if name.endswith("ProcessingKwargs"):
                    processor_kwargs = (
                        processor_kwargs
                        | _collect_dynamic_keys_from_processing_kwargs(obj)
                    )
            return processor_kwargs
    except Exception:
225
        logger.exception("Failed to collect processor kwargs")
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
        return set()


def cached_get_processor_without_dynamic_kwargs(
    processor_name: str,
    *args: Any,
    revision: str | None = None,
    trust_remote_code: bool = False,
    processor_cls: type[_P] | tuple[type[_P], ...] = ProcessorMixin,
    **kwargs: Any,
) -> _P:
    # Step 1: use default kwargs to get a temporary processor instance
    processor = cached_get_processor(
        processor_name,
        revision=revision,
        trust_remote_code=trust_remote_code,
        processor_cls=processor_cls,  # type: ignore[arg-type]
    )

    # Step 2: use temporary processor collect dynamic keys
    dynamic_keys = get_processor_kwargs_from_processor(processor)

    # Step 3: use dynamic_keys filter kwargs
    filtered_kwargs = {k: v for k, v in kwargs.items() if k not in dynamic_keys}

    # Step 4: use filtered kwargs to get final processor instance
    final_processor = cached_get_processor(
        processor_name,
        revision=revision,
        trust_remote_code=trust_remote_code,
        processor_cls=processor_cls,  # type: ignore[arg-type]
        **filtered_kwargs,
    )

    return final_processor


263
def cached_processor_from_config(
264
    model_config: "ModelConfig",
265
    processor_cls: type[_P] | tuple[type[_P], ...] = ProcessorMixin,
266
267
    **kwargs: Any,
) -> _P:
268
    if is_gguf(model_config.model):
269
        assert not is_gguf(model_config.tokenizer), (
270
271
272
            "For multimodal GGUF models, the original tokenizer "
            "should be used to correctly load processor."
        )
273
274
        model = model_config.tokenizer
        revision = model_config.tokenizer_revision
275
276
277
278
    else:
        model = model_config.model
        revision = model_config.revision

279
    return cached_get_processor_without_dynamic_kwargs(
280
281
        model,
        revision=revision,
282
283
        trust_remote_code=model_config.trust_remote_code,
        processor_cls=processor_cls,  # type: ignore[arg-type]
284
        **_merge_mm_kwargs(model_config, processor_cls, **kwargs),
285
286
287
    )


288
289
290
def get_feature_extractor(
    processor_name: str,
    *args: Any,
291
    revision: str | None = None,
292
293
294
    trust_remote_code: bool = False,
    **kwargs: Any,
):
295
    """Load an audio feature extractor for the given model name
296
297
    via HuggingFace."""
    try:
298
        processor_name = convert_model_repo_to_path(processor_name)
299
300
301
        feature_extractor = AutoFeatureExtractor.from_pretrained(
            processor_name,
            *args,
302
            revision=revision,
303
            trust_remote_code=trust_remote_code,
304
305
            **kwargs,
        )
306
307
308
309
310
311
312
313
314
315
    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 "
316
317
                "`--trust-remote-code` flag in the CLI."
            )
318
319
320
321
322
323
324
325
326
327
            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(
328
    model_config: "ModelConfig",
329
330
331
332
    **kwargs: Any,
):
    return cached_get_feature_extractor(
        model_config.model,
333
        revision=model_config.revision,
334
        trust_remote_code=model_config.trust_remote_code,
335
        **_merge_mm_kwargs(model_config, AutoFeatureExtractor, **kwargs),
336
337
338
    )


339
340
341
def get_image_processor(
    processor_name: str,
    *args: Any,
342
    revision: str | None = None,
343
344
345
346
347
    trust_remote_code: bool = False,
    **kwargs: Any,
):
    """Load an image processor for the given model name via HuggingFace."""
    try:
348
        processor_name = convert_model_repo_to_path(processor_name)
349
350
351
        processor = AutoImageProcessor.from_pretrained(
            processor_name,
            *args,
352
            revision=revision,
353
            trust_remote_code=trust_remote_code,
354
355
            **kwargs,
        )
356
357
358
359
360
361
362
363
364
365
    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 "
366
367
                "`--trust-remote-code` flag in the CLI."
            )
368
369
370
371
372
373
374
            raise RuntimeError(err_msg) from e
        else:
            raise e

    return cast(BaseImageProcessor, processor)


375
376
377
378
cached_get_image_processor = lru_cache(get_image_processor)


def cached_image_processor_from_config(
379
    model_config: "ModelConfig",
380
381
    **kwargs: Any,
):
382
    if is_gguf(model_config.model):
383
        assert not is_gguf(model_config.tokenizer), (
384
385
386
            "For multimodal GGUF models, the original tokenizer "
            "should be used to correctly load image processor."
        )
387
388
        model = model_config.tokenizer
        revision = model_config.tokenizer_revision
389
390
391
    else:
        model = model_config.model
        revision = model_config.revision
392
    return cached_get_image_processor(
393
394
        model,
        revision=revision,
395
        trust_remote_code=model_config.trust_remote_code,
396
        **_merge_mm_kwargs(model_config, AutoImageProcessor, **kwargs),
397
    )
398
399
400
401
402


def get_video_processor(
    processor_name: str,
    *args: Any,
403
    revision: str | None = None,
404
    trust_remote_code: bool = False,
405
    processor_cls_overrides: type[_V] | None = None,
406
407
408
409
    **kwargs: Any,
):
    """Load a video processor for the given model name via HuggingFace."""
    try:
410
        processor_name = convert_model_repo_to_path(processor_name)
411
412
413
414
415
416
        processor_cls = processor_cls_overrides or AutoVideoProcessor
        processor = processor_cls.from_pretrained(
            processor_name,
            *args,
            revision=revision,
            trust_remote_code=trust_remote_code,
417
418
            **kwargs,
        )
419
420
421
422
423
424
425
426
427
428
    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 "
429
430
                "`--trust-remote-code` flag in the CLI."
            )
431
432
433
434
435
436
437
438
439
440
441
            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(
442
    model_config: "ModelConfig",
443
    processor_cls: type[_V] | None = None,
444
445
446
447
448
449
450
451
452
    **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),
    )