video.py 2.72 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
10
11
12
13
from vllm.transformers_utils.tokenizer import get_tokenizer

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

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

17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
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"

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

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

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

        if isinstance(data, np.ndarray):
64
65
66
67
            video_processor = self._get_hf_video_processor(
                model_config,
                mm_processor_kwargs,
            )
68
69
            if video_processor is None:
                raise RuntimeError("No HuggingFace processor is available "
70
                                   "to process the video object")
71
            try:
72
73
74
75
                # 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
76
77
                batch_data = video_processor(data, return_tensors="pt").data
            except Exception:
78
                logger.error("Failed to process video (%s)", data)
79
80
81
82
83
84
85
86
                raise

            return MultiModalInputs(batch_data)

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

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