processor.py 17.3 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

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


47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
def _transformers_v4_compatibility_init() -> Any:
    """Some remote code processors may define `optional_attributes` in their
    `ProcessorMixin` subclass, and then pass these arbitrary attributes directly to
    `ProcessorMixin.__init__`, which is no longer allowed in Transformers v5. For
    backward compatibility, we intercept these optional attributes and set them on the
    processor instance before calling the original `ProcessorMixin.__init__`.

    This can be removed if `Molmo2ForConditionalGeneration` is upstreamed to
    Transformers."""
    # Transformers v4
    if hasattr(ProcessorMixin, "optional_attributes"):
        return
    # Transformers v5
    if hasattr(ProcessorMixin.__init__, "_vllm_patched"):
        return

    original_init = ProcessorMixin.__init__

    def __init__(self, *args, **kwargs):
        for optional_attribute in getattr(self, "optional_attributes", []):
            if optional_attribute in kwargs:
                setattr(self, optional_attribute, kwargs.pop(optional_attribute))

        original_init(self, *args, **kwargs)

    # Only patch if ProcessorMixin is not mocked (for docs builds)
    if not hasattr(ProcessorMixin, "_mock_name"):
        __init__._vllm_patched = True  # type: ignore[attr-defined]
        ProcessorMixin.__init__ = __init__


78
_transformers_v4_compatibility_import()
79
_transformers_v4_compatibility_init()
80

81
_P = TypeVar("_P", bound=ProcessorMixin, default=ProcessorMixin)
82
_V = TypeVar("_V", bound=BaseVideoProcessor, default=BaseVideoProcessor)
83
84
85
86
87
88
89
90
91
92
93
94
95


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


96
97
98
99
100
101
102
103
104
class HashableList(list):
    """
    A list that can be hashed by lru_cache.
    """

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


105
def _get_processor_factory_fn(processor_cls: type | tuple[type, ...]):
106
107
108
109
110
111
112
    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

113

114
115
116
117
118
119
120
121
122
123
@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:
124
125
126
127
128
129
130
            # 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__", {}))
131
132
133
134
135
136
            for kw_name in kw_annotations:
                dynamic_kwargs.add(kw_name)
    dynamic_kwargs |= {"text_kwargs", "images_kwargs", "videos_kwargs", "audio_kwargs"}
    return dynamic_kwargs


137
138
def _merge_mm_kwargs(
    model_config: "ModelConfig",
139
    processor_cls: type | tuple[type, ...],
140
141
142
143
144
145
146
147
148
149
150
151
152
    /,
    **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,
    )
153
154
155
    # 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.
156
    for key, value in allowed_kwargs.items():
157
        if isinstance(value, dict):
158
            allowed_kwargs[key] = HashableDict(value)
159
        if isinstance(value, list):
160
161
162
            allowed_kwargs[key] = HashableList(value)

    return allowed_kwargs
163

164
165
166

def get_processor(
    processor_name: str,
167
    *args: Any,
168
    revision: str | None = None,
169
    trust_remote_code: bool = False,
170
    processor_cls: type[_P] | tuple[type[_P], ...] = ProcessorMixin,
171
    **kwargs: Any,
172
) -> _P:
173
    """Load a processor for the given model name via HuggingFace."""
174
175
    if revision is None:
        revision = "main"
176
    try:
177
        processor_name = convert_model_repo_to_path(processor_name)
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
        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)
197
198
199
200
201
202
203
204
205
206
    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 "
207
208
                "`--trust-remote-code` flag in the CLI."
            )
209
210
211
212
            raise RuntimeError(err_msg) from e
        else:
            raise e

213
    if not isinstance(processor, processor_cls):
214
215
216
217
218
        raise TypeError(
            "Invalid type of HuggingFace processor. "
            f"Expected type: {processor_cls}, but "
            f"found type: {type(processor)}"
        )
219
220

    return processor
221
222


223
224
225
cached_get_processor = lru_cache(get_processor)


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
@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:
257
        logger.exception("Failed to collect processor kwargs")
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
        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


295
def cached_processor_from_config(
296
    model_config: "ModelConfig",
297
    processor_cls: type[_P] | tuple[type[_P], ...] = ProcessorMixin,
298
299
    **kwargs: Any,
) -> _P:
300
    if is_gguf(model_config.model):
301
        assert not is_gguf(model_config.tokenizer), (
302
303
304
            "For multimodal GGUF models, the original tokenizer "
            "should be used to correctly load processor."
        )
305
306
        model = model_config.tokenizer
        revision = model_config.tokenizer_revision
307
308
309
310
    else:
        model = model_config.model
        revision = model_config.revision

311
    return cached_get_processor_without_dynamic_kwargs(
312
313
        model,
        revision=revision,
314
315
        trust_remote_code=model_config.trust_remote_code,
        processor_cls=processor_cls,  # type: ignore[arg-type]
316
        **_merge_mm_kwargs(model_config, processor_cls, **kwargs),
317
318
319
    )


320
321
322
def get_feature_extractor(
    processor_name: str,
    *args: Any,
323
    revision: str | None = None,
324
325
326
    trust_remote_code: bool = False,
    **kwargs: Any,
):
327
    """Load an audio feature extractor for the given model name
328
329
    via HuggingFace."""
    try:
330
        processor_name = convert_model_repo_to_path(processor_name)
331
332
333
        feature_extractor = AutoFeatureExtractor.from_pretrained(
            processor_name,
            *args,
334
            revision=revision,
335
            trust_remote_code=trust_remote_code,
336
337
            **kwargs,
        )
338
339
340
341
342
343
344
345
346
347
    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 "
348
349
                "`--trust-remote-code` flag in the CLI."
            )
350
351
352
353
354
355
356
357
358
359
            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(
360
    model_config: "ModelConfig",
361
362
363
364
    **kwargs: Any,
):
    return cached_get_feature_extractor(
        model_config.model,
365
        revision=model_config.revision,
366
        trust_remote_code=model_config.trust_remote_code,
367
        **_merge_mm_kwargs(model_config, AutoFeatureExtractor, **kwargs),
368
369
370
    )


371
372
373
def get_image_processor(
    processor_name: str,
    *args: Any,
374
    revision: str | None = None,
375
376
377
378
379
    trust_remote_code: bool = False,
    **kwargs: Any,
):
    """Load an image processor for the given model name via HuggingFace."""
    try:
380
        processor_name = convert_model_repo_to_path(processor_name)
381
382
383
        processor = AutoImageProcessor.from_pretrained(
            processor_name,
            *args,
384
            revision=revision,
385
            trust_remote_code=trust_remote_code,
386
387
            **kwargs,
        )
388
389
390
391
392
393
394
395
396
397
    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 "
398
399
                "`--trust-remote-code` flag in the CLI."
            )
400
401
402
403
404
405
406
            raise RuntimeError(err_msg) from e
        else:
            raise e

    return cast(BaseImageProcessor, processor)


407
408
409
410
cached_get_image_processor = lru_cache(get_image_processor)


def cached_image_processor_from_config(
411
    model_config: "ModelConfig",
412
413
    **kwargs: Any,
):
414
    if is_gguf(model_config.model):
415
        assert not is_gguf(model_config.tokenizer), (
416
417
418
            "For multimodal GGUF models, the original tokenizer "
            "should be used to correctly load image processor."
        )
419
420
        model = model_config.tokenizer
        revision = model_config.tokenizer_revision
421
422
423
    else:
        model = model_config.model
        revision = model_config.revision
424
    return cached_get_image_processor(
425
426
        model,
        revision=revision,
427
        trust_remote_code=model_config.trust_remote_code,
428
        **_merge_mm_kwargs(model_config, AutoImageProcessor, **kwargs),
429
    )
430
431
432
433
434


def get_video_processor(
    processor_name: str,
    *args: Any,
435
    revision: str | None = None,
436
    trust_remote_code: bool = False,
437
    processor_cls_overrides: type[_V] | None = None,
438
439
440
441
    **kwargs: Any,
):
    """Load a video processor for the given model name via HuggingFace."""
    try:
442
        processor_name = convert_model_repo_to_path(processor_name)
443
444
445
446
447
448
        processor_cls = processor_cls_overrides or AutoVideoProcessor
        processor = processor_cls.from_pretrained(
            processor_name,
            *args,
            revision=revision,
            trust_remote_code=trust_remote_code,
449
450
            **kwargs,
        )
451
452
453
454
455
456
457
458
459
460
    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 "
461
462
                "`--trust-remote-code` flag in the CLI."
            )
463
464
465
466
467
468
469
470
471
472
473
            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(
474
    model_config: "ModelConfig",
475
    processor_cls: type[_V] | None = None,
476
477
478
479
480
481
482
483
484
    **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),
    )