"tests/lora/data/__init__.py" did not exist on "376725ce74d2d75490eed1840b41de00c0e4acd6"
protocol.py 25.7 KB
Newer Older
1
2
# Adapted from
# https://github.com/lm-sys/FastChat/blob/168ccc29d3f7edc50823016105c024fe2282732a/fastchat/protocol/openai_api_protocol.py
Zhuohan Li's avatar
Zhuohan Li committed
3
import time
4
from argparse import Namespace
5
from typing import Any, Dict, List, Literal, Optional, Union
Zhuohan Li's avatar
Zhuohan Li committed
6

7
import torch
8
from pydantic import BaseModel, ConfigDict, Field, model_validator
9
from transformers import PreTrainedTokenizer
10
from typing_extensions import Annotated
Zhuohan Li's avatar
Zhuohan Li committed
11

12
from vllm.entrypoints.chat_utils import ChatCompletionMessageParam
13
from vllm.entrypoints.openai.logits_processors import get_logits_processors
14
from vllm.pooling_params import PoolingParams
15
from vllm.sampling_params import LogitsProcessor, SamplingParams
16
from vllm.sequence import Logprob
17
from vllm.utils import random_uuid
18

19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
# torch is mocked during docs generation,
# so we have to provide the values as literals
_MOCK_LONG_INFO = Namespace(min=-9223372036854775808, max=9223372036854775807)

try:
    from sphinx.ext.autodoc.mock import _MockModule

    if isinstance(torch, _MockModule):
        _LONG_INFO = _MOCK_LONG_INFO
    else:
        _LONG_INFO = torch.iinfo(torch.long)
except ModuleNotFoundError:
    _LONG_INFO = torch.iinfo(torch.long)

assert _LONG_INFO.min == _MOCK_LONG_INFO.min
assert _LONG_INFO.max == _MOCK_LONG_INFO.max

Zhuohan Li's avatar
Zhuohan Li committed
36

37
38
39
40
41
42
class OpenAIBaseModel(BaseModel):
    # OpenAI API does not allow extra fields
    model_config = ConfigDict(extra="forbid")


class ErrorResponse(OpenAIBaseModel):
Zhuohan Li's avatar
Zhuohan Li committed
43
44
45
46
    object: str = "error"
    message: str
    type: str
    param: Optional[str] = None
47
    code: int
Zhuohan Li's avatar
Zhuohan Li committed
48
49


50
class ModelPermission(OpenAIBaseModel):
Zhuohan Li's avatar
Zhuohan Li committed
51
52
53
54
55
56
57
58
59
60
61
    id: str = Field(default_factory=lambda: f"modelperm-{random_uuid()}")
    object: str = "model_permission"
    created: int = Field(default_factory=lambda: int(time.time()))
    allow_create_engine: bool = False
    allow_sampling: bool = True
    allow_logprobs: bool = True
    allow_search_indices: bool = False
    allow_view: bool = True
    allow_fine_tuning: bool = False
    organization: str = "*"
    group: Optional[str] = None
62
    is_blocking: bool = False
Zhuohan Li's avatar
Zhuohan Li committed
63
64


65
class ModelCard(OpenAIBaseModel):
Zhuohan Li's avatar
Zhuohan Li committed
66
67
68
    id: str
    object: str = "model"
    created: int = Field(default_factory=lambda: int(time.time()))
Woosuk Kwon's avatar
Woosuk Kwon committed
69
    owned_by: str = "vllm"
Zhuohan Li's avatar
Zhuohan Li committed
70
71
    root: Optional[str] = None
    parent: Optional[str] = None
72
    max_model_len: Optional[int] = None
Zhuohan Li's avatar
Zhuohan Li committed
73
74
75
    permission: List[ModelPermission] = Field(default_factory=list)


76
class ModelList(OpenAIBaseModel):
Zhuohan Li's avatar
Zhuohan Li committed
77
78
79
80
    object: str = "list"
    data: List[ModelCard] = Field(default_factory=list)


81
class UsageInfo(OpenAIBaseModel):
Zhuohan Li's avatar
Zhuohan Li committed
82
83
84
85
86
    prompt_tokens: int = 0
    total_tokens: int = 0
    completion_tokens: Optional[int] = 0


87
class ResponseFormat(OpenAIBaseModel):
88
    # type must be "json_object" or "text"
89
    type: Literal["text", "json_object"]
90
91


92
class StreamOptions(OpenAIBaseModel):
93
94
    include_usage: Optional[bool] = True
    continuous_usage_stats: Optional[bool] = True
95
96


97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
class FunctionDefinition(OpenAIBaseModel):
    name: str
    description: Optional[str] = None
    parameters: Optional[Dict[str, Any]] = None


class ChatCompletionToolsParam(OpenAIBaseModel):
    type: Literal["function"] = "function"
    function: FunctionDefinition


class ChatCompletionNamedFunction(OpenAIBaseModel):
    name: str


class ChatCompletionNamedToolChoiceParam(OpenAIBaseModel):
    function: ChatCompletionNamedFunction
    type: Literal["function"] = "function"


117
class ChatCompletionRequest(OpenAIBaseModel):
118
119
    # Ordered by official OpenAI API documentation
    # https://platform.openai.com/docs/api-reference/chat/create
120
    messages: List[ChatCompletionMessageParam]
121
122
123
124
    model: str
    frequency_penalty: Optional[float] = 0.0
    logit_bias: Optional[Dict[str, float]] = None
    logprobs: Optional[bool] = False
125
    top_logprobs: Optional[int] = 0
126
    max_tokens: Optional[int] = None
127
128
129
    n: Optional[int] = 1
    presence_penalty: Optional[float] = 0.0
    response_format: Optional[ResponseFormat] = None
130
    seed: Optional[int] = Field(None, ge=_LONG_INFO.min, le=_LONG_INFO.max)
131
    stop: Optional[Union[str, List[str]]] = Field(default_factory=list)
Zhuohan Li's avatar
Zhuohan Li committed
132
    stream: Optional[bool] = False
133
    stream_options: Optional[StreamOptions] = None
134
135
    temperature: Optional[float] = 0.7
    top_p: Optional[float] = 1.0
136
137
138
    tools: Optional[List[ChatCompletionToolsParam]] = None
    tool_choice: Optional[Union[Literal["none"],
                                ChatCompletionNamedToolChoiceParam]] = "none"
Zhuohan Li's avatar
Zhuohan Li committed
139
    user: Optional[str] = None
140
141

    # doc: begin-chat-completion-sampling-params
142
    best_of: Optional[int] = None
143
144
145
146
147
148
    use_beam_search: bool = False
    top_k: int = -1
    min_p: float = 0.0
    repetition_penalty: float = 1.0
    length_penalty: float = 1.0
    early_stopping: bool = False
149
    stop_token_ids: Optional[List[int]] = Field(default_factory=list)
150
151
152
153
154
155
    include_stop_str_in_output: bool = False
    ignore_eos: bool = False
    min_tokens: int = 0
    skip_special_tokens: bool = True
    spaces_between_special_tokens: bool = True
    truncate_prompt_tokens: Optional[Annotated[int, Field(ge=1)]] = None
156
    prompt_logprobs: Optional[int] = None
157
158
159
    # doc: end-chat-completion-sampling-params

    # doc: begin-chat-completion-extra-params
160
    echo: bool = Field(
161
162
163
164
165
        default=False,
        description=(
            "If true, the new message will be prepended with the last message "
            "if they belong to the same role."),
    )
166
    add_generation_prompt: bool = Field(
167
168
169
170
171
172
        default=True,
        description=
        ("If true, the generation prompt will be added to the chat template. "
         "This is a parameter used by chat template in tokenizer config of the "
         "model."),
    )
173
    add_special_tokens: bool = Field(
174
175
176
177
178
        default=False,
        description=(
            "If true, special tokens (e.g. BOS) will be added to the prompt "
            "on top of what is added by the chat template. "
            "For most models, the chat template takes care of adding the "
179
            "special tokens so this should be set to false (as is the "
180
181
            "default)."),
    )
182
183
184
185
186
187
188
189
190
191
192
193
194
    documents: Optional[List[Dict[str, str]]] = Field(
        default=None,
        description=
        ("A list of dicts representing documents that will be accessible to "
         "the model if it is performing RAG (retrieval-augmented generation)."
         " If the template does not support RAG, this argument will have no "
         "effect. We recommend that each document should be a dict containing "
         "\"title\" and \"text\" keys."),
    )
    chat_template: Optional[str] = Field(
        default=None,
        description=(
            "A Jinja template to use for this conversion. "
195
196
197
            "As of transformers v4.44, default chat template is no longer "
            "allowed, so you must provide a chat template if the tokenizer "
            "does not define one."),
198
199
200
201
202
203
    )
    chat_template_kwargs: Optional[Dict[str, Any]] = Field(
        default=None,
        description=("Additional kwargs to pass to the template renderer. "
                     "Will be accessible by the chat template."),
    )
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
    guided_json: Optional[Union[str, dict, BaseModel]] = Field(
        default=None,
        description=("If specified, the output will follow the JSON schema."),
    )
    guided_regex: Optional[str] = Field(
        default=None,
        description=(
            "If specified, the output will follow the regex pattern."),
    )
    guided_choice: Optional[List[str]] = Field(
        default=None,
        description=(
            "If specified, the output will be exactly one of the choices."),
    )
    guided_grammar: Optional[str] = Field(
        default=None,
        description=(
            "If specified, the output will follow the context free grammar."),
    )
223
224
225
226
227
228
    guided_decoding_backend: Optional[str] = Field(
        default=None,
        description=(
            "If specified, will override the default guided decoding backend "
            "of the server for this specific request. If set, must be either "
            "'outlines' / 'lm-format-enforcer'"))
229
230
231
232
233
    guided_whitespace_pattern: Optional[str] = Field(
        default=None,
        description=(
            "If specified, will override the default whitespace pattern "
            "for guided json decoding."))
234
235

    # doc: end-chat-completion-extra-params
Zhuohan Li's avatar
Zhuohan Li committed
236

237
238
239
240
241
242
243
    def to_sampling_params(
            self, tokenizer: PreTrainedTokenizer,
            guided_decode_logits_processor: Optional[LogitsProcessor],
            default_max_tokens: int) -> SamplingParams:
        max_tokens = self.max_tokens
        if max_tokens is None:
            max_tokens = default_max_tokens
244

245
        # We now allow logprobs being true without top_logrobs.
246
247
248
249
250
        logits_processors = get_logits_processors(
            logit_bias=self.logit_bias,
            allowed_token_ids=None,
            tokenizer=tokenizer,
        )
251
252
        if guided_decode_logits_processor:
            logits_processors.append(guided_decode_logits_processor)
253

254
255
        return SamplingParams(
            n=self.n,
256
            best_of=self.best_of,
257
258
259
260
261
            presence_penalty=self.presence_penalty,
            frequency_penalty=self.frequency_penalty,
            repetition_penalty=self.repetition_penalty,
            temperature=self.temperature,
            top_p=self.top_p,
262
            top_k=self.top_k,
263
            min_p=self.min_p,
Nick Hill's avatar
Nick Hill committed
264
            seed=self.seed,
265
266
            stop=self.stop,
            stop_token_ids=self.stop_token_ids,
267
            logprobs=self.top_logprobs if self.logprobs else None,
268
269
            prompt_logprobs=self.prompt_logprobs if self.prompt_logprobs else
            (self.top_logprobs if self.echo else None),
270
            ignore_eos=self.ignore_eos,
271
            max_tokens=max_tokens,
272
            min_tokens=self.min_tokens,
273
            use_beam_search=self.use_beam_search,
274
            early_stopping=self.early_stopping,
275
276
            skip_special_tokens=self.skip_special_tokens,
            spaces_between_special_tokens=self.spaces_between_special_tokens,
277
278
            include_stop_str_in_output=self.include_stop_str_in_output,
            length_penalty=self.length_penalty,
279
            logits_processors=logits_processors,
280
            truncate_prompt_tokens=self.truncate_prompt_tokens,
281
282
        )

283
284
285
286
287
288
289
290
291
    @model_validator(mode='before')
    @classmethod
    def validate_stream_options(cls, values):
        if (values.get('stream_options') is not None
                and not values.get('stream')):
            raise ValueError(
                "stream_options can only be set if stream is true")
        return values

292
293
294
295
296
297
298
299
    @model_validator(mode="before")
    @classmethod
    def check_guided_decoding_count(cls, data):
        guide_count = sum([
            "guided_json" in data and data["guided_json"] is not None,
            "guided_regex" in data and data["guided_regex"] is not None,
            "guided_choice" in data and data["guided_choice"] is not None
        ])
300
        # you can only use one kind of guided decoding
301
302
303
304
        if guide_count > 1:
            raise ValueError(
                "You can only use one kind of guided decoding "
                "('guided_json', 'guided_regex' or 'guided_choice').")
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
        # you can only either use guided decoding or tools, not both
        if guide_count > 1 and "tool_choice" in data and data[
                "tool_choice"] != "none":
            raise ValueError(
                "You can only either use guided decoding or tools, not both.")
        return data

    @model_validator(mode="before")
    @classmethod
    def check_tool_choice(cls, data):
        if "tool_choice" in data and data["tool_choice"] != "none":
            if not isinstance(data["tool_choice"], dict):
                raise ValueError("Currently only named tools are supported.")
            if "tools" not in data or data["tools"] is None:
                raise ValueError(
                    "When using `tool_choice`, `tools` must be set.")
321
322
        return data

323
324
325
326
327
328
329
330
    @model_validator(mode="before")
    @classmethod
    def check_logprobs(cls, data):
        if "top_logprobs" in data and data["top_logprobs"] is not None:
            if "logprobs" not in data or data["logprobs"] is False:
                raise ValueError(
                    "when using `top_logprobs`, `logprobs` must be set to true."
                )
331
            elif data["top_logprobs"] < 0:
332
                raise ValueError(
333
                    "`top_logprobs` must be a value a positive value.")
334
335
        return data

Zhuohan Li's avatar
Zhuohan Li committed
336

337
class CompletionRequest(OpenAIBaseModel):
338
339
    # Ordered by official OpenAI API documentation
    # https://platform.openai.com/docs/api-reference/completions/create
Zhuohan Li's avatar
Zhuohan Li committed
340
    model: str
341
    prompt: Union[List[int], List[List[int]], str, List[str]]
342
    best_of: Optional[int] = None
Zhuohan Li's avatar
Zhuohan Li committed
343
344
345
    echo: Optional[bool] = False
    frequency_penalty: Optional[float] = 0.0
    logit_bias: Optional[Dict[str, float]] = None
346
347
    logprobs: Optional[int] = None
    max_tokens: Optional[int] = 16
348
    n: int = 1
349
    presence_penalty: Optional[float] = 0.0
350
    seed: Optional[int] = Field(None, ge=_LONG_INFO.min, le=_LONG_INFO.max)
351
352
    stop: Optional[Union[str, List[str]]] = Field(default_factory=list)
    stream: Optional[bool] = False
353
    stream_options: Optional[StreamOptions] = None
354
355
356
    suffix: Optional[str] = None
    temperature: Optional[float] = 1.0
    top_p: Optional[float] = 1.0
Zhuohan Li's avatar
Zhuohan Li committed
357
    user: Optional[str] = None
358
359

    # doc: begin-completion-sampling-params
360
361
362
363
364
365
    use_beam_search: bool = False
    top_k: int = -1
    min_p: float = 0.0
    repetition_penalty: float = 1.0
    length_penalty: float = 1.0
    early_stopping: bool = False
366
    stop_token_ids: Optional[List[int]] = Field(default_factory=list)
367
368
369
370
371
    include_stop_str_in_output: bool = False
    ignore_eos: bool = False
    min_tokens: int = 0
    skip_special_tokens: bool = True
    spaces_between_special_tokens: bool = True
372
    truncate_prompt_tokens: Optional[Annotated[int, Field(ge=1)]] = None
373
    allowed_token_ids: Optional[List[int]] = None
374
    prompt_logprobs: Optional[int] = None
375
376
377
    # doc: end-completion-sampling-params

    # doc: begin-completion-extra-params
378
379
    add_special_tokens: bool = Field(
        default=True,
380
        description=(
381
382
            "If true (the default), special tokens (e.g. BOS) will be added to "
            "the prompt."),
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
    )
    response_format: Optional[ResponseFormat] = Field(
        default=None,
        description=
        ("Similar to chat completion, this parameter specifies the format of "
         "output. Only {'type': 'json_object'} or {'type': 'text' } is "
         "supported."),
    )
    guided_json: Optional[Union[str, dict, BaseModel]] = Field(
        default=None,
        description=("If specified, the output will follow the JSON schema."),
    )
    guided_regex: Optional[str] = Field(
        default=None,
        description=(
            "If specified, the output will follow the regex pattern."),
    )
    guided_choice: Optional[List[str]] = Field(
        default=None,
        description=(
            "If specified, the output will be exactly one of the choices."),
    )
    guided_grammar: Optional[str] = Field(
        default=None,
        description=(
            "If specified, the output will follow the context free grammar."),
    )
410
411
412
413
414
415
    guided_decoding_backend: Optional[str] = Field(
        default=None,
        description=(
            "If specified, will override the default guided decoding backend "
            "of the server for this specific request. If set, must be one of "
            "'outlines' / 'lm-format-enforcer'"))
416
417
418
419
420
    guided_whitespace_pattern: Optional[str] = Field(
        default=None,
        description=(
            "If specified, will override the default whitespace pattern "
            "for guided json decoding."))
421
422

    # doc: end-completion-extra-params
Zhuohan Li's avatar
Zhuohan Li committed
423

424
425
426
427
428
429
430
431
    def to_sampling_params(
            self, tokenizer: PreTrainedTokenizer,
            guided_decode_logits_processor: Optional[LogitsProcessor],
            default_max_tokens: int) -> SamplingParams:
        max_tokens = self.max_tokens
        if max_tokens is None:
            max_tokens = default_max_tokens

432
433
        echo_without_generation = self.echo and self.max_tokens == 0

434
435
436
437
438
        logits_processors = get_logits_processors(
            logit_bias=self.logit_bias,
            allowed_token_ids=self.allowed_token_ids,
            tokenizer=tokenizer,
        )
439
440
        if guided_decode_logits_processor:
            logits_processors.append(guided_decode_logits_processor)
441

442
443
444
445
446
447
448
449
450
451
        return SamplingParams(
            n=self.n,
            best_of=self.best_of,
            presence_penalty=self.presence_penalty,
            frequency_penalty=self.frequency_penalty,
            repetition_penalty=self.repetition_penalty,
            temperature=self.temperature,
            top_p=self.top_p,
            top_k=self.top_k,
            min_p=self.min_p,
Nick Hill's avatar
Nick Hill committed
452
            seed=self.seed,
453
454
            stop=self.stop,
            stop_token_ids=self.stop_token_ids,
455
            logprobs=self.logprobs,
456
            ignore_eos=self.ignore_eos,
457
            max_tokens=max_tokens if not echo_without_generation else 1,
458
            min_tokens=self.min_tokens,
459
            use_beam_search=self.use_beam_search,
460
            early_stopping=self.early_stopping,
461
462
            prompt_logprobs=self.prompt_logprobs
            if self.prompt_logprobs else self.logprobs if self.echo else None,
463
            skip_special_tokens=self.skip_special_tokens,
464
            spaces_between_special_tokens=self.spaces_between_special_tokens,
465
466
            include_stop_str_in_output=self.include_stop_str_in_output,
            length_penalty=self.length_penalty,
467
            logits_processors=logits_processors,
468
            truncate_prompt_tokens=self.truncate_prompt_tokens,
469
470
        )

471
472
473
474
475
476
477
478
479
480
481
482
483
484
    @model_validator(mode="before")
    @classmethod
    def check_guided_decoding_count(cls, data):
        guide_count = sum([
            "guided_json" in data and data["guided_json"] is not None,
            "guided_regex" in data and data["guided_regex"] is not None,
            "guided_choice" in data and data["guided_choice"] is not None
        ])
        if guide_count > 1:
            raise ValueError(
                "You can only use one kind of guided decoding "
                "('guided_json', 'guided_regex' or 'guided_choice').")
        return data

485
486
487
488
    @model_validator(mode="before")
    @classmethod
    def check_logprobs(cls, data):
        if "logprobs" in data and data[
489
490
                "logprobs"] is not None and not data["logprobs"] >= 0:
            raise ValueError("if passed, `logprobs` must be a positive value.")
491
492
        return data

493
494
495
496
497
    @model_validator(mode="before")
    @classmethod
    def validate_stream_options(cls, data):
        if data.get("stream_options") and not data.get("stream"):
            raise ValueError(
498
                "Stream options can only be defined when stream is true.")
499
500
        return data

Zhuohan Li's avatar
Zhuohan Li committed
501

502
class EmbeddingRequest(OpenAIBaseModel):
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
    # Ordered by official OpenAI API documentation
    # https://platform.openai.com/docs/api-reference/embeddings
    model: str
    input: Union[List[int], List[List[int]], str, List[str]]
    encoding_format: Optional[str] = Field('float', pattern='^(float|base64)$')
    dimensions: Optional[int] = None
    user: Optional[str] = None

    # doc: begin-embedding-pooling-params
    additional_data: Optional[Any] = None

    # doc: end-embedding-pooling-params

    def to_pooling_params(self):
        return PoolingParams(additional_data=self.additional_data)


520
class CompletionLogProbs(OpenAIBaseModel):
Zhuohan Li's avatar
Zhuohan Li committed
521
522
523
    text_offset: List[int] = Field(default_factory=list)
    token_logprobs: List[Optional[float]] = Field(default_factory=list)
    tokens: List[str] = Field(default_factory=list)
524
525
    top_logprobs: List[Optional[Dict[str,
                                     float]]] = Field(default_factory=list)
Zhuohan Li's avatar
Zhuohan Li committed
526
527


528
class CompletionResponseChoice(OpenAIBaseModel):
Zhuohan Li's avatar
Zhuohan Li committed
529
530
    index: int
    text: str
531
    logprobs: Optional[CompletionLogProbs] = None
532
533
    finish_reason: Optional[str] = None
    stop_reason: Optional[Union[int, str]] = Field(
534
535
536
537
538
539
        default=None,
        description=(
            "The stop string or token id that caused the completion "
            "to stop, None if the completion finished for some other reason "
            "including encountering the EOS token"),
    )
540
    prompt_logprobs: Optional[List[Optional[Dict[int, Logprob]]]] = None
Zhuohan Li's avatar
Zhuohan Li committed
541
542


543
class CompletionResponse(OpenAIBaseModel):
Zhuohan Li's avatar
Zhuohan Li committed
544
545
546
547
548
549
550
551
    id: str = Field(default_factory=lambda: f"cmpl-{random_uuid()}")
    object: str = "text_completion"
    created: int = Field(default_factory=lambda: int(time.time()))
    model: str
    choices: List[CompletionResponseChoice]
    usage: UsageInfo


552
class CompletionResponseStreamChoice(OpenAIBaseModel):
Zhuohan Li's avatar
Zhuohan Li committed
553
554
    index: int
    text: str
555
    logprobs: Optional[CompletionLogProbs] = None
556
557
    finish_reason: Optional[str] = None
    stop_reason: Optional[Union[int, str]] = Field(
558
559
560
561
562
563
        default=None,
        description=(
            "The stop string or token id that caused the completion "
            "to stop, None if the completion finished for some other reason "
            "including encountering the EOS token"),
    )
Zhuohan Li's avatar
Zhuohan Li committed
564
565


566
class CompletionStreamResponse(OpenAIBaseModel):
Zhuohan Li's avatar
Zhuohan Li committed
567
568
569
570
571
    id: str = Field(default_factory=lambda: f"cmpl-{random_uuid()}")
    object: str = "text_completion"
    created: int = Field(default_factory=lambda: int(time.time()))
    model: str
    choices: List[CompletionResponseStreamChoice]
572
    usage: Optional[UsageInfo] = Field(default=None)
573
574


575
class EmbeddingResponseData(OpenAIBaseModel):
576
577
    index: int
    object: str = "embedding"
578
    embedding: Union[List[float], str]
579
580


581
class EmbeddingResponse(OpenAIBaseModel):
582
583
584
585
586
587
588
589
    id: str = Field(default_factory=lambda: f"cmpl-{random_uuid()}")
    object: str = "list"
    created: int = Field(default_factory=lambda: int(time.time()))
    model: str
    data: List[EmbeddingResponseData]
    usage: UsageInfo


590
591
592
593
594
595
596
597
598
599
600
class FunctionCall(OpenAIBaseModel):
    name: str
    arguments: str


class ToolCall(OpenAIBaseModel):
    id: str = Field(default_factory=lambda: f"chatcmpl-tool-{random_uuid()}")
    type: Literal["function"] = "function"
    function: FunctionCall


601
class ChatMessage(OpenAIBaseModel):
602
603
    role: str
    content: str
604
    tool_calls: List[ToolCall] = Field(default_factory=list)
605
606


607
608
609
610
611
612
613
614
615
616
617
618
619
620
class ChatCompletionLogProb(OpenAIBaseModel):
    token: str
    logprob: float = -9999.0
    bytes: Optional[List[int]] = None


class ChatCompletionLogProbsContent(ChatCompletionLogProb):
    top_logprobs: List[ChatCompletionLogProb] = Field(default_factory=list)


class ChatCompletionLogProbs(OpenAIBaseModel):
    content: Optional[List[ChatCompletionLogProbsContent]] = None


621
class ChatCompletionResponseChoice(OpenAIBaseModel):
622
623
    index: int
    message: ChatMessage
624
    logprobs: Optional[ChatCompletionLogProbs] = None
625
    finish_reason: Optional[str] = None
626
    stop_reason: Optional[Union[int, str]] = None
627
628


629
class ChatCompletionResponse(OpenAIBaseModel):
630
    id: str = Field(default_factory=lambda: f"chatcmpl-{random_uuid()}")
631
    object: Literal["chat.completion"] = "chat.completion"
632
633
634
635
    created: int = Field(default_factory=lambda: int(time.time()))
    model: str
    choices: List[ChatCompletionResponseChoice]
    usage: UsageInfo
636
    prompt_logprobs: Optional[List[Optional[Dict[int, Logprob]]]] = None
637
638


639
class DeltaMessage(OpenAIBaseModel):
640
641
    role: Optional[str] = None
    content: Optional[str] = None
642
    tool_calls: List[ToolCall] = Field(default_factory=list)
643
644


645
class ChatCompletionResponseStreamChoice(OpenAIBaseModel):
646
647
    index: int
    delta: DeltaMessage
648
    logprobs: Optional[ChatCompletionLogProbs] = None
649
    finish_reason: Optional[str] = None
650
    stop_reason: Optional[Union[int, str]] = None
651
652


653
class ChatCompletionStreamResponse(OpenAIBaseModel):
654
    id: str = Field(default_factory=lambda: f"chatcmpl-{random_uuid()}")
655
    object: Literal["chat.completion.chunk"] = "chat.completion.chunk"
656
657
658
    created: int = Field(default_factory=lambda: int(time.time()))
    model: str
    choices: List[ChatCompletionResponseStreamChoice]
659
    usage: Optional[UsageInfo] = Field(default=None)
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680


class BatchRequestInput(OpenAIBaseModel):
    """
    The per-line object of the batch input file.

    NOTE: Currently only the `/v1/chat/completions` endpoint is supported.
    """

    # A developer-provided per-request id that will be used to match outputs to
    # inputs. Must be unique for each request in a batch.
    custom_id: str

    # The HTTP method to be used for the request. Currently only POST is
    # supported.
    method: str

    # The OpenAI API relative URL to be used for the request. Currently
    # /v1/chat/completions is supported.
    url: str

681
    # The parameters of the request.
682
    body: Union[ChatCompletionRequest, EmbeddingRequest]
683
684


685
686
687
688
689
690
691
692
class BatchResponseData(OpenAIBaseModel):
    # HTTP status code of the response.
    status_code: int = 200

    # An unique identifier for the API request.
    request_id: str

    # The body of the response.
693
    body: Optional[Union[ChatCompletionResponse, EmbeddingResponse]] = None
694
695


696
697
698
699
700
701
702
703
704
705
706
class BatchRequestOutput(OpenAIBaseModel):
    """
    The per-line object of the batch output and error files
    """

    id: str

    # A developer-provided per-request id that will be used to match outputs to
    # inputs.
    custom_id: str

707
    response: Optional[BatchResponseData]
708
709
710
711

    # For requests that failed with a non-HTTP error, this will contain more
    # information on the cause of the failure.
    error: Optional[Any]
712
713


714
715
716
717
718
719
720
721
722
723
724
class TokenizeCompletionRequest(OpenAIBaseModel):
    model: str
    prompt: str

    add_special_tokens: bool = Field(default=True)


class TokenizeChatRequest(OpenAIBaseModel):
    model: str
    messages: List[ChatCompletionMessageParam]

725
726
    add_generation_prompt: bool = Field(default=True)
    add_special_tokens: bool = Field(default=False)
727
728
729


TokenizeRequest = Union[TokenizeCompletionRequest, TokenizeChatRequest]
730
731
732
733
734


class TokenizeResponse(OpenAIBaseModel):
    count: int
    max_model_len: int
735
    tokens: List[int]
736
737
738
739
740
741
742
743
744


class DetokenizeRequest(OpenAIBaseModel):
    model: str
    tokens: List[int]


class DetokenizeResponse(OpenAIBaseModel):
    prompt: str