video.py 2.79 KB
Newer Older
1
from functools import lru_cache
2
from typing import TYPE_CHECKING, Any, Dict, List, Optional, Union
3
4
5
6
7

import numpy as np

from vllm.inputs.registry import InputContext
from vllm.logger import init_logger
8
from vllm.transformers_utils.processor import get_video_processor
9
from vllm.transformers_utils.tokenizer import get_tokenizer
10
from vllm.utils import is_list_of
11
12
13
14

from .base import MultiModalData, MultiModalInputs
from .image import ImagePlugin

15
16
17
if TYPE_CHECKING:
    from vllm.config import ModelConfig

18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
logger = init_logger(__name__)

cached_get_video_processor = lru_cache(get_video_processor)
cached_get_tokenizer = lru_cache(get_tokenizer)

VideoInput = Union[
    "np.ndarray",  # single video input
    List["np.ndarray"],
    # TODO: support more types
    # List[Image.Image], List[List[Image.Image]],
    # "torch.Tensor",
    # List["torch.Tensor"],
    # List[List["np.ndarrray"]],
    # List[List["torch.Tensor"]],
]


class VideoPlugin(ImagePlugin):
    """Plugin for video data."""

    def get_data_key(self) -> str:
        return "video"

41
42
    def _get_hf_video_processor(
        self,
43
        model_config: "ModelConfig",
44
45
46
47
        mm_processor_kwargs: Optional[Dict[str, Any]] = None,
    ):
        if mm_processor_kwargs is None:
            mm_processor_kwargs = {}
48
49
        return cached_get_video_processor(
            model_config.model,
50
51
            trust_remote_code=model_config.trust_remote_code,
            **mm_processor_kwargs)
52
53
54
55
56

    def _default_input_mapper(
        self,
        ctx: InputContext,
        data: MultiModalData[object],
57
        **mm_processor_kwargs,
58
59
60
    ) -> MultiModalInputs:
        model_config = ctx.model_config

61
62
63
        if isinstance(data, list) and len(data) == 1:
            data = data[0]

64
        if isinstance(data, np.ndarray) or is_list_of(data, np.ndarray):
65
66
67
68
            video_processor = self._get_hf_video_processor(
                model_config,
                mm_processor_kwargs,
            )
69
70
            if video_processor is None:
                raise RuntimeError("No HuggingFace processor is available "
71
                                   "to process the video object")
72
            try:
73
74
75
76
                # NOTE: Similar to image; it may be a good idea to filter and
                # pass mm_processor_kwargs here too, but for now we don't to
                # avoid extra complexity if the initializer and preprocess
                # signatures of the processor don't align
77
78
                batch_data = video_processor(data, return_tensors="pt").data
            except Exception:
79
                logger.error("Failed to process video (%s)", data)
80
81
82
83
84
85
86
87
                raise

            return MultiModalInputs(batch_data)

        raise TypeError(f"Invalid video type: {type(data)}")

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