video.py 2.54 KB
Newer Older
1
from functools import lru_cache
2
from typing import TYPE_CHECKING, Any, Dict, Optional
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
from .base import MultiModalData
13
from .image import ImagePlugin
14
from .inputs import MultiModalKwargs, VideoItem
15

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

19
20
21
22
23
24
25
26
27
28
29
30
logger = init_logger(__name__)

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


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

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

31
32
    def _get_hf_video_processor(
        self,
33
        model_config: "ModelConfig",
34
35
36
37
        mm_processor_kwargs: Optional[Dict[str, Any]] = None,
    ):
        if mm_processor_kwargs is None:
            mm_processor_kwargs = {}
38
39
        return cached_get_video_processor(
            model_config.model,
40
41
            trust_remote_code=model_config.trust_remote_code,
            **mm_processor_kwargs)
42
43
44
45

    def _default_input_mapper(
        self,
        ctx: InputContext,
46
        data: MultiModalData[VideoItem],
47
        **mm_processor_kwargs,
48
    ) -> MultiModalKwargs:
49
50
        model_config = ctx.model_config

51
        if isinstance(data, list) and len(data) == 1:
52
            data = data[0]  # type: ignore
53

54
        if isinstance(data, np.ndarray) or is_list_of(data, np.ndarray):
55
56
57
58
            video_processor = self._get_hf_video_processor(
                model_config,
                mm_processor_kwargs,
            )
59
60
            if video_processor is None:
                raise RuntimeError("No HuggingFace processor is available "
61
                                   "to process the video object")
62
            try:
63
64
65
66
                # 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
67
68
                batch_data = video_processor(data, return_tensors="pt").data
            except Exception:
69
                logger.error("Failed to process video (%s)", data)
70
71
                raise

72
            return MultiModalKwargs(batch_data)
73
74
75
76
77

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

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