processor.py 7.22 KB
Newer Older
1
2
3
4
import time
from typing import Any, Dict, Mapping, Optional, Tuple, Union

from vllm.config import LoRAConfig, ModelConfig
5
6
7
from vllm.inputs import (INPUT_REGISTRY, InputRegistry, ProcessorInputs,
                         PromptType, SingletonInputsAdapter)
from vllm.inputs.parse import is_encoder_decoder_inputs
8
9
from vllm.inputs.preprocess import InputPreprocessor
from vllm.lora.request import LoRARequest
10
11
from vllm.multimodal import (MULTIMODAL_REGISTRY, MultiModalKwargs,
                             MultiModalRegistry)
12
13
14
15
from vllm.pooling_params import PoolingParams
from vllm.prompt_adapter.request import PromptAdapterRequest
from vllm.sampling_params import SamplingParams
from vllm.transformers_utils.config import try_get_generation_config
16
from vllm.transformers_utils.tokenizer_group import BaseTokenizerGroup
17
from vllm.v1.engine import DetokenizerRequest, EngineCoreRequest
18
from vllm.v1.engine.mm_input_mapper import MMInputMapper
19
20
21
22
23
24
25
26


class Processor:

    def __init__(
        self,
        model_config: ModelConfig,
        lora_config: Optional[LoRAConfig],
27
        tokenizer: BaseTokenizerGroup,
28
        input_registry: InputRegistry = INPUT_REGISTRY,
29
        mm_registry: MultiModalRegistry = MULTIMODAL_REGISTRY,
30
31
32
33
34
35
36
37
38
    ):

        self.model_config = model_config
        self.lora_config = lora_config
        self.tokenizer = tokenizer

        self.generation_config_fields = _load_generation_config_dict(
            model_config)
        self.input_preprocessor = InputPreprocessor(model_config,
39
40
                                                    self.tokenizer,
                                                    mm_registry)
41
42
43
        self.input_processor = input_registry.create_input_processor(
            model_config)

44
45
46
        # Multi-modal (huggingface) input mapper
        self.mm_input_mapper = MMInputMapper(model_config)

47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
    # TODO: run in an ThreadpoolExecutor or BackgroundProcess.
    # This ideally should releases the GIL, so we should not block the
    # asyncio loop while this is running.
    def process_inputs(
        self,
        request_id: str,
        prompt: PromptType,
        params: Union[SamplingParams, PoolingParams],
        arrival_time: float,
        lora_request: Optional[LoRARequest] = None,
        trace_headers: Optional[Mapping[str, str]] = None,
        prompt_adapter_request: Optional[PromptAdapterRequest] = None,
        priority: int = 0,
    ) -> Tuple[DetokenizerRequest, EngineCoreRequest]:

        # TODO(woosuk): Support embedding mode.
        # TODO(woosuk): Check max_logprobs
        # TODO(woosuk): Support encoder-decoder models.

        if lora_request is not None and not self.lora_config:
            raise ValueError(f"Got lora_request {lora_request} but LoRA is "
                             "not enabled!")
        if arrival_time is None:
            arrival_time = time.time()
        assert priority == 0, "vLLM V1 does not support priority at the moment."
        assert trace_headers is None, "vLLM V1 does not support tracing yet."

        # Process inputs.
        preprocessed_inputs = self.input_preprocessor.preprocess(
            prompt,
            request_id=request_id,
            lora_request=lora_request,
            prompt_adapter_request=prompt_adapter_request,
        )
        processed_inputs = self.input_processor(preprocessed_inputs)
        self._validate_model_inputs(processed_inputs)
        eos_token_id = self.input_preprocessor.get_eos_token_id(lora_request)

85
86
87
88
89
90
91
92
93
94
95
96
97
        if is_encoder_decoder_inputs(processed_inputs):
            decoder_inputs = SingletonInputsAdapter(
                processed_inputs["decoder"])
            encoder_inputs = SingletonInputsAdapter(
                processed_inputs["encoder"])
        else:
            decoder_inputs = SingletonInputsAdapter(processed_inputs)
            encoder_inputs = None

        # TODO: Impl encoder-decoder
        if encoder_inputs is not None:
            raise NotImplementedError

98
99
100
101
102
103
        assert isinstance(params, SamplingParams)
        # TODO: can we avoid cloning here in multiproc case
        sampling_params = params.clone()
        sampling_params.update_from_generation_config(
            self.generation_config_fields, eos_token_id)

104
        # Preprocess multi-modal data
105
106
107
108
109
110
111
112
113
        if len(decoder_inputs.multi_modal_data) == 0:
            mm_inputs = None
        elif isinstance(decoder_inputs.multi_modal_data, MultiModalKwargs):
            mm_inputs = [decoder_inputs.multi_modal_data]
        else:
            mm_inputs = self.mm_input_mapper.process_inputs(
                decoder_inputs.multi_modal_data,
                decoder_inputs.mm_processor_kwargs,
            )
114

115
116
        # Make Request for Detokenizer.
        detokenizer_request = DetokenizerRequest(
117
118
119
            request_id,
            decoder_inputs.prompt,
            decoder_inputs.prompt_token_ids,
120
121
            sampling_params.skip_special_tokens,
            sampling_params.spaces_between_special_tokens,
122
123
124
125
            sampling_params.output_kind,
            sampling_params.stop,
            sampling_params.include_stop_str_in_output,
        )
126
127
128

        # Make Request for EngineCore.
        engine_core_request = EngineCoreRequest(
129
130
131
            request_id,
            decoder_inputs.prompt,
            decoder_inputs.prompt_token_ids,
132
            mm_inputs,
133
134
135
136
137
138
            decoder_inputs.multi_modal_placeholders,
            sampling_params,
            eos_token_id,
            arrival_time,
            lora_request,
        )
139
140
141

        return detokenizer_request, engine_core_request

142
143
144
145
146
147
148
149
150
151
152
    def _validate_model_inputs(self, inputs: ProcessorInputs):
        if is_encoder_decoder_inputs(inputs):
            # For encoder-decoder multimodal models, the max_prompt_len
            # restricts the decoder prompt length
            prompt_inputs = inputs["decoder" if self.model_config.
                                   is_multimodal_model else "encoder"]
        else:
            prompt_inputs = inputs

        prompt_ids = SingletonInputsAdapter(prompt_inputs).prompt_token_ids

153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
        if prompt_ids is None or len(prompt_ids) == 0:
            raise ValueError("Prompt cannot be empty")

        if self.model_config.is_multimodal_model:
            max_prompt_len = self.model_config.max_model_len

            if len(prompt_ids) > max_prompt_len:
                raise ValueError(
                    f"The prompt (total length {len(prompt_ids)}) is too long "
                    f"to fit into the model (context length {max_prompt_len}). "
                    "Make sure that `max_model_len` is no smaller than the "
                    "number of text tokens plus multimodal tokens. For image "
                    "inputs, the number of image tokens depends on the number "
                    "of images, and possibly their aspect ratios as well.")

168
169
170
171
            # TODO: Find out how many placeholder tokens are there so we can
            # check that chunked prefill does not truncate them
            # max_batch_len = self.scheduler_config.max_num_batched_tokens

172
173
174
175
176
177
178
179
180
181
182
183

def _load_generation_config_dict(model_config: ModelConfig) -> Dict[str, Any]:
    config = try_get_generation_config(
        model_config.model,
        trust_remote_code=model_config.trust_remote_code,
        revision=model_config.revision,
    )

    if config is None:
        return {}

    return config.to_diff_dict()