image.py 1.91 KB
Newer Older
1
from functools import lru_cache
2
3
4
5

import torch
from PIL import Image

6
7
from vllm.config import ModelConfig
from vllm.inputs.registry import InputContext
8
from vllm.logger import init_logger
9
from vllm.transformers_utils.image_processor import get_image_processor
10
from vllm.utils import is_list_of
11

12
from .base import MultiModalData, MultiModalInputs, MultiModalPlugin
13
14
15

logger = init_logger(__name__)

16
cached_get_image_processor = lru_cache(get_image_processor)
17
18


19
class ImagePlugin(MultiModalPlugin):
20
    """Plugin for image data."""
21

22
23
    def get_data_key(self) -> str:
        return "image"
24

25
    def _get_hf_image_processor(self, model_config: ModelConfig):
26
        return cached_get_image_processor(
27
28
            model_config.model,
            trust_remote_code=model_config.trust_remote_code)
29

30
31
32
33
34
    def _default_input_mapper(
        self,
        ctx: InputContext,
        data: MultiModalData[object],
    ) -> MultiModalInputs:
35
        model_config = ctx.model_config
36

37
        # PIL image
38
        if isinstance(data, Image.Image) or is_list_of(data, Image.Image):
39
            image_processor = self._get_hf_image_processor(model_config)
40
            if image_processor is None:
41
                raise RuntimeError("No HuggingFace processor is available "
42
43
                                   "to process the image object")
            try:
44
45
46
                batch_data = image_processor \
                    .preprocess(data, return_tensors="pt") \
                    .data
47
            except Exception:
48
                logger.error("Failed to process image (%s)", data)
49
50
                raise

51
            return MultiModalInputs(batch_data)
52
53

        # Image embedding
54
        elif isinstance(data, torch.Tensor) or is_list_of(data, torch.Tensor):
55
            return MultiModalInputs({"image_embeds": data})
56
57

        raise TypeError(f"Invalid image type: {type(data)}")
58
59
60

    def _default_max_multimodal_tokens(self, ctx: InputContext) -> int:
        return 3000