test_tensorizer.py 9.87 KB
Newer Older
1
2
# SPDX-License-Identifier: Apache-2.0

3
import gc
4
import os
5
import pathlib
6
7
8
9
import subprocess
from unittest.mock import MagicMock, patch

import pytest
10
import torch
11
12

from vllm import SamplingParams
13
from vllm.engine.arg_utils import EngineArgs
14
# yapf conflicts with isort for this docstring
15
16
17
18
19
20
# yapf: disable
from vllm.model_executor.model_loader.tensorizer import (TensorizerConfig,
                                                         TensorSerializer,
                                                         is_vllm_tensorized,
                                                         load_with_tensorizer,
                                                         open_stream,
21
                                                         tensorize_vllm_model)
22
# yapf: enable
23
from vllm.utils import PlaceholderModule
24

25
from ..utils import VLLM_PATH
26

27
28
29
30
31
32
try:
    from tensorizer import EncryptionParams
except ImportError:
    tensorizer = PlaceholderModule("tensorizer")  # type: ignore[assignment]
    EncryptionParams = tensorizer.placeholder_attr("EncryptionParams")

33
EXAMPLES_PATH = VLLM_PATH / "examples"
34

35
36
37
38
39
40
41
42
43
44
prompts = [
    "Hello, my name is",
    "The president of the United States is",
    "The capital of France is",
    "The future of AI is",
]
# Create a sampling params object.
sampling_params = SamplingParams(temperature=0.8, top_p=0.95, seed=0)

model_ref = "facebook/opt-125m"
45
46
tensorize_model_for_testing_script = os.path.join(
    os.path.dirname(__file__), "tensorize_vllm_model_for_testing.py")
47

48

49
50
51
52
53
54
55
def is_curl_installed():
    try:
        subprocess.check_call(['curl', '--version'])
        return True
    except (subprocess.CalledProcessError, FileNotFoundError):
        return False

56

57
58
59
60
61
def write_keyfile(keyfile_path: str):
    encryption_params = EncryptionParams.random()
    pathlib.Path(keyfile_path).parent.mkdir(parents=True, exist_ok=True)
    with open(keyfile_path, 'wb') as f:
        f.write(encryption_params.key)
62
63


64
@patch('vllm.model_executor.model_loader.tensorizer.TensorizerAgent')
65
66
67
68
69
70
def test_load_with_tensorizer(mock_agent, tensorizer_config):
    mock_linear_method = MagicMock()
    mock_agent_instance = mock_agent.return_value
    mock_agent_instance.deserialize.return_value = MagicMock()

    result = load_with_tensorizer(tensorizer_config,
71
                                  quant_method=mock_linear_method)
72
73

    mock_agent.assert_called_once_with(tensorizer_config,
74
                                       quant_method=mock_linear_method)
75
76
77
78
79
80
81
82
83
    mock_agent_instance.deserialize.assert_called_once()
    assert result == mock_agent_instance.deserialize.return_value


@pytest.mark.skipif(not is_curl_installed(), reason="cURL is not installed")
def test_can_deserialize_s3(vllm_runner):
    model_ref = "EleutherAI/pythia-1.4b"
    tensorized_path = f"s3://tensorized/{model_ref}/fp16/model.tensors"

84
    with vllm_runner(model_ref,
85
86
87
88
89
90
                     load_format="tensorizer",
                     model_loader_extra_config=TensorizerConfig(
                         tensorizer_uri=tensorized_path,
                         num_readers=1,
                         s3_endpoint="object.ord1.coreweave.com",
                     )) as loaded_hf_model:
91
92
        deserialized_outputs = loaded_hf_model.generate(
            prompts, sampling_params)
93
        # noqa: E501
94

95
        assert deserialized_outputs
96
97
98
99
100


@pytest.mark.skipif(not is_curl_installed(), reason="cURL is not installed")
def test_deserialized_encrypted_vllm_model_has_same_outputs(
        vllm_runner, tmp_path):
101
    args = EngineArgs(model=model_ref)
102
103
104
    with vllm_runner(model_ref) as vllm_model:
        model_path = tmp_path / (model_ref + ".tensors")
        key_path = tmp_path / (model_ref + ".key")
105
106
        write_keyfile(key_path)

107
        outputs = vllm_model.generate(prompts, sampling_params)
108

109
110
    config_for_serializing = TensorizerConfig(tensorizer_uri=str(model_path),
                                              encryption_keyfile=str(key_path))
111

112
    tensorize_vllm_model(args, config_for_serializing)
113

114
115
    config_for_deserializing = TensorizerConfig(
        tensorizer_uri=str(model_path), encryption_keyfile=str(key_path))
116

117
118
119
120
    with vllm_runner(model_ref,
                     load_format="tensorizer",
                     model_loader_extra_config=config_for_deserializing
                     ) as loaded_vllm_model:  # noqa: E501
121

122
123
        deserialized_outputs = loaded_vllm_model.generate(
            prompts, sampling_params)
124
        # noqa: E501
125

126
        assert outputs == deserialized_outputs
127
128
129
130


def test_deserialized_hf_model_has_same_outputs(hf_runner, vllm_runner,
                                                tmp_path):
131
132
133
134
135
136
137
138
    with hf_runner(model_ref) as hf_model:
        model_path = tmp_path / (model_ref + ".tensors")
        max_tokens = 50
        outputs = hf_model.generate_greedy(prompts, max_tokens=max_tokens)
        with open_stream(model_path, "wb+") as stream:
            serializer = TensorSerializer(stream)
            serializer.write_module(hf_model.model)

139
    with vllm_runner(model_ref,
140
141
142
143
144
                     load_format="tensorizer",
                     model_loader_extra_config=TensorizerConfig(
                         tensorizer_uri=model_path,
                         num_readers=1,
                     )) as loaded_hf_model:
145
146
        deserialized_outputs = loaded_hf_model.generate_greedy(
            prompts, max_tokens=max_tokens)
147

148
        assert outputs == deserialized_outputs
149
150


151
def test_load_without_tensorizer_load_format(vllm_runner, capfd):
152
    model = None
153
    try:
154
        model = vllm_runner(
155
156
            model_ref,
            model_loader_extra_config=TensorizerConfig(tensorizer_uri="test"))
157
158
159
160
161
162
163
164
165
166
167
168
169
    except RuntimeError:
        out, err = capfd.readouterr()
        combined_output = out + err
        assert ("ValueError: Model loader extra config "
                "is not supported for load "
                "format LoadFormat.AUTO") in combined_output
    finally:
        del model
        gc.collect()
        torch.cuda.empty_cache()


def test_raise_value_error_on_invalid_load_format(vllm_runner, capfd):
170
    model = None
171
    try:
172
        model = vllm_runner(
173
174
175
            model_ref,
            load_format="safetensors",
            model_loader_extra_config=TensorizerConfig(tensorizer_uri="test"))
176
177
178
179
180
181
182
183
184
185
    except RuntimeError:
        out, err = capfd.readouterr()

        combined_output = out + err
        assert ("ValueError: Model loader extra config is not supported "
                "for load format LoadFormat.SAFETENSORS") in combined_output
    finally:
        del model
        gc.collect()
        torch.cuda.empty_cache()
186
187


188
@pytest.mark.skipif(torch.cuda.device_count() < 2, reason="Requires 2 GPUs")
189
190
def test_tensorizer_with_tp_path_without_template(vllm_runner, capfd):
    try:
191
192
193
194
195
196
        model_ref = "EleutherAI/pythia-1.4b"
        tensorized_path = f"s3://tensorized/{model_ref}/fp16/model.tensors"

        vllm_runner(
            model_ref,
            load_format="tensorizer",
197
198
199
200
201
            model_loader_extra_config=TensorizerConfig(
                tensorizer_uri=tensorized_path,
                num_readers=1,
                s3_endpoint="object.ord1.coreweave.com",
            ),
202
            tensor_parallel_size=2,
203
            disable_custom_all_reduce=True,
204
        )
205
206
207
208
209
210
211
    except RuntimeError:
        out, err = capfd.readouterr()
        combined_output = out + err
        assert ("ValueError: For a sharded model, tensorizer_uri "
                "should include a string format template like '%04d' "
                "to be formatted with the rank "
                "of the shard") in combined_output
212

213

214
215
216
@pytest.mark.skipif(torch.cuda.device_count() < 2, reason="Requires 2 GPUs")
def test_deserialized_encrypted_vllm_model_with_tp_has_same_outputs(
        vllm_runner, tmp_path):
217
218
    model_ref = "EleutherAI/pythia-1.4b"
    # record outputs from un-sharded un-tensorized model
219
220
221
222
223
224
    with vllm_runner(
            model_ref,
            disable_custom_all_reduce=True,
            enforce_eager=True,
    ) as base_model:
        outputs = base_model.generate(prompts, sampling_params)
225
226
227
228
229
230
231

    # load model with two shards and serialize with encryption
    model_path = str(tmp_path / (model_ref + "-%02d.tensors"))
    key_path = tmp_path / (model_ref + ".key")

    tensorizer_config = TensorizerConfig(
        tensorizer_uri=model_path,
232
        encryption_keyfile=str(key_path),
233
234
235
236
    )

    tensorize_vllm_model(
        engine_args=EngineArgs(
237
238
239
240
241
            model=model_ref,
            tensor_parallel_size=2,
            disable_custom_all_reduce=True,
            enforce_eager=True,
        ),
242
243
244
245
246
        tensorizer_config=tensorizer_config,
    )
    assert os.path.isfile(model_path % 0), "Serialization subprocess failed"
    assert os.path.isfile(model_path % 1), "Serialization subprocess failed"

247
248
249
250
251
252
253
    with vllm_runner(
            model_ref,
            tensor_parallel_size=2,
            load_format="tensorizer",
            disable_custom_all_reduce=True,
            enforce_eager=True,
            model_loader_extra_config=tensorizer_config) as loaded_vllm_model:
254
255
        deserialized_outputs = loaded_vllm_model.generate(
            prompts, sampling_params)
256
257
258

    assert outputs == deserialized_outputs

259

260
@pytest.mark.flaky(reruns=3)
261
def test_vllm_tensorized_model_has_same_outputs(vllm_runner, tmp_path):
262
263
    gc.collect()
    torch.cuda.empty_cache()
264
265
266
    model_ref = "facebook/opt-125m"
    model_path = tmp_path / (model_ref + ".tensors")
    config = TensorizerConfig(tensorizer_uri=str(model_path))
267
    args = EngineArgs(model=model_ref, device="cuda")
268

269
270
    with vllm_runner(model_ref) as vllm_model:
        outputs = vllm_model.generate(prompts, sampling_params)
271

272
273
    tensorize_vllm_model(args, config)
    assert is_vllm_tensorized(config)
274

275
    with vllm_runner(model_ref,
276
277
                     load_format="tensorizer",
                     model_loader_extra_config=config) as loaded_vllm_model:
278
279
        deserialized_outputs = loaded_vllm_model.generate(
            prompts, sampling_params)
280
        # noqa: E501
281

282
        assert outputs == deserialized_outputs