test_embedding.py 9.49 KB
Newer Older
1
2
3
import base64

import numpy as np
4
5
import openai
import pytest
6
import os
7
import pytest_asyncio
8
9
import requests

10
from vllm.entrypoints.openai.protocol import EmbeddingResponse
11
from vllm.transformers_utils.tokenizer import get_tokenizer
12

13
from ...utils import RemoteOpenAIServer, models_path_prefix
14

zhuwenwen's avatar
zhuwenwen committed
15
MODEL_NAME = os.path.join(models_path_prefix, "intfloat/e5-mistral-7b-instruct")
16
DUMMY_CHAT_TEMPLATE = """{% for message in messages %}{{message['role'] + ': ' + message['content'] + '\\n'}}{% endfor %}"""  # noqa: E501
17
18
19


@pytest.fixture(scope="module")
20
def server():
21
    args = [
22
23
        "--task",
        "embed",
24
25
26
27
28
29
        # use half precision for speed and memory savings in CI environment
        "--dtype",
        "bfloat16",
        "--enforce-eager",
        "--max-model-len",
        "8192",
30
31
        "--chat-template",
        DUMMY_CHAT_TEMPLATE,
32
33
    ]

34
    with RemoteOpenAIServer(MODEL_NAME, args) as remote_server:
35
        yield remote_server
36
37


38
@pytest_asyncio.fixture
39
40
async def client(server):
    async with server.get_async_client() as async_client:
41
        yield async_client
42
43
44


@pytest.mark.asyncio
45
46
@pytest.mark.parametrize("model_name", [MODEL_NAME])
async def test_single_embedding(client: openai.AsyncOpenAI, model_name: str):
47
48
49
50
51
    input_texts = [
        "The chef prepared a delicious meal.",
    ]

    # test single embedding
52
    embedding_response = await client.embeddings.create(
53
54
55
56
        model=model_name,
        input=input_texts,
        encoding_format="float",
    )
57
58
59
    embeddings = EmbeddingResponse.model_validate(
        embedding_response.model_dump(mode="json"))

60
61
62
63
64
65
66
67
68
    assert embeddings.id is not None
    assert len(embeddings.data) == 1
    assert len(embeddings.data[0].embedding) == 4096
    assert embeddings.usage.completion_tokens == 0
    assert embeddings.usage.prompt_tokens == 9
    assert embeddings.usage.total_tokens == 9

    # test using token IDs
    input_tokens = [1, 1, 1, 1, 1]
69
    embedding_response = await client.embeddings.create(
70
71
72
73
        model=model_name,
        input=input_tokens,
        encoding_format="float",
    )
74
75
76
    embeddings = EmbeddingResponse.model_validate(
        embedding_response.model_dump(mode="json"))

77
78
79
80
81
82
83
84
85
    assert embeddings.id is not None
    assert len(embeddings.data) == 1
    assert len(embeddings.data[0].embedding) == 4096
    assert embeddings.usage.completion_tokens == 0
    assert embeddings.usage.prompt_tokens == 5
    assert embeddings.usage.total_tokens == 5


@pytest.mark.asyncio
86
87
@pytest.mark.parametrize("model_name", [MODEL_NAME])
async def test_batch_embedding(client: openai.AsyncOpenAI, model_name: str):
88
89
90
91
92
    # test List[str]
    input_texts = [
        "The cat sat on the mat.", "A feline was resting on a rug.",
        "Stars twinkle brightly in the night sky."
    ]
93
    embedding_response = await client.embeddings.create(
94
95
96
97
        model=model_name,
        input=input_texts,
        encoding_format="float",
    )
98
99
100
    embeddings = EmbeddingResponse.model_validate(
        embedding_response.model_dump(mode="json"))

101
102
103
    assert embeddings.id is not None
    assert len(embeddings.data) == 3
    assert len(embeddings.data[0].embedding) == 4096
104
105
106
    assert embeddings.usage.completion_tokens == 0
    assert embeddings.usage.prompt_tokens == 32
    assert embeddings.usage.total_tokens == 32
107
108
109
110

    # test List[List[int]]
    input_tokens = [[4, 5, 7, 9, 20], [15, 29, 499], [24, 24, 24, 24, 24],
                    [25, 32, 64, 77]]
111
    embedding_response = await client.embeddings.create(
112
113
114
115
        model=model_name,
        input=input_tokens,
        encoding_format="float",
    )
116
117
118
    embeddings = EmbeddingResponse.model_validate(
        embedding_response.model_dump(mode="json"))

119
120
121
122
123
124
    assert embeddings.id is not None
    assert len(embeddings.data) == 4
    assert len(embeddings.data[0].embedding) == 4096
    assert embeddings.usage.completion_tokens == 0
    assert embeddings.usage.prompt_tokens == 17
    assert embeddings.usage.total_tokens == 17
125
126
127


@pytest.mark.asyncio
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
@pytest.mark.parametrize("model_name", [MODEL_NAME])
async def test_conversation_embedding(server: RemoteOpenAIServer,
                                      client: openai.AsyncOpenAI,
                                      model_name: str):
    messages = [{
        "role": "user",
        "content": "The cat sat on the mat.",
    }, {
        "role": "assistant",
        "content": "A feline was resting on a rug.",
    }, {
        "role": "user",
        "content": "Stars twinkle brightly in the night sky.",
    }]

143
144
145
146
147
148
149
150
    chat_response = requests.post(
        server.url_for("v1/embeddings"),
        json={
            "model": model_name,
            "messages": messages,
            "encoding_format": "float",
        },
    )
151
    chat_response.raise_for_status()
152
    chat_embeddings = EmbeddingResponse.model_validate(chat_response.json())
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168

    tokenizer = get_tokenizer(tokenizer_name=model_name, tokenizer_mode="fast")
    prompt = tokenizer.apply_chat_template(
        messages,
        chat_template=DUMMY_CHAT_TEMPLATE,
        add_generation_prompt=True,
        continue_final_message=False,
        tokenize=False,
    )
    completion_response = await client.embeddings.create(
        model=model_name,
        input=prompt,
        encoding_format="float",
        # To be consistent with chat
        extra_body={"add_special_tokens": False},
    )
169
170
    completion_embeddings = EmbeddingResponse.model_validate(
        completion_response.model_dump(mode="json"))
171

172
173
174
175
176
177
    assert chat_embeddings.id is not None
    assert completion_embeddings.id is not None
    assert chat_embeddings.created <= completion_embeddings.created
    assert chat_embeddings.model_dump(
        exclude={"id", "created"}) == (completion_embeddings.model_dump(
            exclude={"id", "created"}))
178
179
180
181
182


@pytest.mark.asyncio
@pytest.mark.parametrize("model_name", [MODEL_NAME])
async def test_batch_base64_embedding(client: openai.AsyncOpenAI,
183
184
185
186
187
188
                                      model_name: str):
    input_texts = [
        "Hello my name is",
        "The best thing about vLLM is that it supports many different models"
    ]

189
190
191
    responses_float = await client.embeddings.create(input=input_texts,
                                                     model=model_name,
                                                     encoding_format="float")
192

193
194
195
    responses_base64 = await client.embeddings.create(input=input_texts,
                                                      model=model_name,
                                                      encoding_format="base64")
196
197
198
199
200

    decoded_responses_base64_data = []
    for data in responses_base64.data:
        decoded_responses_base64_data.append(
            np.frombuffer(base64.b64decode(data.embedding),
201
                          dtype="float32").tolist())
202
203
204
205
206

    assert responses_float.data[0].embedding == decoded_responses_base64_data[
        0]
    assert responses_float.data[1].embedding == decoded_responses_base64_data[
        1]
207
208

    # Default response is float32 decoded from base64 by OpenAI Client
209
210
    responses_default = await client.embeddings.create(input=input_texts,
                                                       model=model_name)
211
212
213
214
215

    assert responses_float.data[0].embedding == responses_default.data[
        0].embedding
    assert responses_float.data[1].embedding == responses_default.data[
        1].embedding
216
217
218


@pytest.mark.asyncio
219
220
221
@pytest.mark.parametrize("model_name", [MODEL_NAME])
async def test_single_embedding_truncation(client: openai.AsyncOpenAI,
                                           model_name: str):
222
223
224
225
226
    input_texts = [
        "Como o Brasil pode fomentar o desenvolvimento de modelos de IA?",
    ]

    # test single embedding
227
    embedding_response = await client.embeddings.create(
228
229
230
        model=model_name,
        input=input_texts,
        extra_body={"truncate_prompt_tokens": 10})
231
232
233
    embeddings = EmbeddingResponse.model_validate(
        embedding_response.model_dump(mode="json"))

234
235
236
237
238
239
240
241
242
243
244
    assert embeddings.id is not None
    assert len(embeddings.data) == 1
    assert len(embeddings.data[0].embedding) == 4096
    assert embeddings.usage.completion_tokens == 0
    assert embeddings.usage.prompt_tokens == 10
    assert embeddings.usage.total_tokens == 10

    input_tokens = [
        1, 24428, 289, 18341, 26165, 285, 19323, 283, 289, 26789, 3871, 28728,
        9901, 340, 2229, 385, 340, 315, 28741, 28804, 2
    ]
245
    embedding_response = await client.embeddings.create(
246
247
248
        model=model_name,
        input=input_tokens,
        extra_body={"truncate_prompt_tokens": 10})
249
250
    embeddings = EmbeddingResponse.model_validate(
        embedding_response.model_dump(mode="json"))
251
252
253
254
255
256
257
258
259
260

    assert embeddings.id is not None
    assert len(embeddings.data) == 1
    assert len(embeddings.data[0].embedding) == 4096
    assert embeddings.usage.completion_tokens == 0
    assert embeddings.usage.prompt_tokens == 10
    assert embeddings.usage.total_tokens == 10


@pytest.mark.asyncio
261
262
263
@pytest.mark.parametrize("model_name", [MODEL_NAME])
async def test_single_embedding_truncation_invalid(client: openai.AsyncOpenAI,
                                                   model_name: str):
264
265
266
267
268
    input_texts = [
        "Como o Brasil pode fomentar o desenvolvimento de modelos de IA?",
    ]

    with pytest.raises(openai.BadRequestError):
269
        response = await client.embeddings.create(
270
271
272
            model=model_name,
            input=input_texts,
            extra_body={"truncate_prompt_tokens": 8193})
273
        assert "error" in response.object
274
        assert "truncate_prompt_tokens value is greater than max_model_len. "\
275
               "Please, select a smaller truncation size." in response.message