processor.py 8.17 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
from transformers import (AutoFeatureExtractor, AutoImageProcessor,
                          AutoProcessor)
from transformers.feature_extraction_utils import FeatureExtractionMixin
from transformers.image_processing_utils import BaseImageProcessor
11
from transformers.processing_utils import ProcessorMixin
12
13
from typing_extensions import TypeVar

14
15
from vllm.utils import get_allowed_kwarg_only_overrides

16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
if TYPE_CHECKING:
    from vllm.config import ModelConfig

_P = TypeVar("_P", bound=ProcessorMixin, default=ProcessorMixin)


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


33
34
35
36
37
38
39
40
41
class HashableList(list):
    """
    A list that can be hashed by lru_cache.
    """

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


42
43
44
45
46
47
48
49
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

50

51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
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,
    )
67
68
69
70

    # 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.
71
    for key, value in allowed_kwargs.items():
72
        if isinstance(value, dict):
73
            allowed_kwargs[key] = HashableDict(value)
74
        if isinstance(value, list):
75
76
77
            allowed_kwargs[key] = HashableList(value)

    return allowed_kwargs
78

79
80
81

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

    try:
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
        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)
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
    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 "
                "`--trust-remote-code` flag in the CLI.")
            raise RuntimeError(err_msg) from e
        else:
            raise e

127
128
129
130
131
132
    if not isinstance(processor, processor_cls):
        raise TypeError("Invalid type of HuggingFace processor. "
                        f"Expected type: {processor_cls}, but "
                        f"found type: {type(processor)}")

    return processor
133
134


135
136
137
cached_get_processor = lru_cache(get_processor)


138
139
140
141
142
143
144
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,
145
        revision=model_config.revision,
146
147
        trust_remote_code=model_config.trust_remote_code,
        processor_cls=processor_cls,  # type: ignore[arg-type]
148
        **_merge_mm_kwargs(model_config, processor_cls, **kwargs),
149
150
151
    )


152
153
154
def get_feature_extractor(
    processor_name: str,
    *args: Any,
155
    revision: Optional[str] = None,
156
157
158
159
160
161
162
163
164
    trust_remote_code: bool = False,
    **kwargs: Any,
):
    """Load an audio feature extractor for the given model name 
    via HuggingFace."""
    try:
        feature_extractor = AutoFeatureExtractor.from_pretrained(
            processor_name,
            *args,
165
            revision=revision,
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
            trust_remote_code=trust_remote_code,
            **kwargs)
    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 "
                "`--trust-remote-code` flag in the CLI.")
            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,
194
        revision=model_config.revision,
195
        trust_remote_code=model_config.trust_remote_code,
196
        **_merge_mm_kwargs(model_config, AutoFeatureExtractor, **kwargs),
197
198
199
    )


200
201
202
def get_image_processor(
    processor_name: str,
    *args: Any,
203
    revision: Optional[str] = None,
204
205
206
207
208
209
210
211
    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,
212
            revision=revision,
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
            trust_remote_code=trust_remote_code,
            **kwargs)
    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 "
                "`--trust-remote-code` flag in the CLI.")
            raise RuntimeError(err_msg) from e
        else:
            raise e

    return cast(BaseImageProcessor, processor)


233
234
235
236
237
238
239
240
241
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,
242
        revision=model_config.revision,
243
        trust_remote_code=model_config.trust_remote_code,
244
        **_merge_mm_kwargs(model_config, AutoImageProcessor, **kwargs),
245
    )