pyproject.toml 11.9 KB
Newer Older
1
# SPDX-FileCopyrightText: Copyright (c) 2024-2026 NVIDIA CORPORATION & AFFILIATES. All rights reserved.
2
3
# SPDX-License-Identifier: Apache-2.0

Neelay Shah's avatar
Neelay Shah committed
4
[project]
5
name = "ai-dynamo"
6
version = "1.0.0"
Neelay Shah's avatar
Neelay Shah committed
7
8
9
description = "Distributed Inference Framework"
readme = "README.md"
authors = [
Neelay Shah's avatar
Neelay Shah committed
10
    { name = "NVIDIA Inc.", email = "sw-dl-dynamo@nvidia.com" },
Neelay Shah's avatar
Neelay Shah committed
11
]
12
13
license = { text = "Apache-2.0" }
license-files = ["LICENSE"]
Neelay Shah's avatar
Neelay Shah committed
14
15
requires-python = ">=3.10"
dependencies = [
16
    "ai-dynamo-runtime==1.0.0",
17
    "transformers>=4.56.0",
18
    "kubernetes>=32.0.1,<33.0.0",
19
    "prometheus_client>=0.23.1,<1.0",
20
21
    "msgspec>=0.19.0",
    "pyzmq>=26.0.0",
22
    "msgpack==1.1.2",
Neelay Shah's avatar
Neelay Shah committed
23
]
24

25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
classifiers = [
    "Development Status :: 4 - Beta",
    "Intended Audience :: Developers",
    "Intended Audience :: Science/Research",
    "Intended Audience :: Information Technology",
    "License :: OSI Approved :: Apache Software License",
    "Programming Language :: Python :: 3",
    "Programming Language :: Python :: 3.10",
    "Programming Language :: Python :: 3.11",
    "Programming Language :: Python :: 3.12",
    "Topic :: Scientific/Engineering",
    "Topic :: Scientific/Engineering :: Artificial Intelligence",
    "Operating System :: POSIX :: Linux",
]
keywords = ["llm", "genai", "inference", "nvidia", "distributed", "dynamo"]
Neelay Shah's avatar
Neelay Shah committed
40

41
42
43
[project.urls]
Repository = "https://github.com/ai-dynamo/dynamo.git"

44
[project.optional-dependencies]
45
46
trtllm =[
    "uvloop",
47
    "tensorrt-llm==1.3.0rc8",
48
49
]

50
51
vllm = [
    "uvloop",
52
    "nixl[cu12]<=0.10.1",
53
    "vllm[flashinfer,runai,otel]==0.17.1",
54
    # vllm-omni 0.16.0 is now on PyPI; install only future rc builds from source in container builds
55
56
    # (see container/deps/vllm/install_vllm.sh). pip install ai-dynamo[vllm] will
    # not include vllm-omni — install it separately from source if needed.
57
    "vllm-omni==0.16.0",
58
    "blake3>=1.0.0,<2.0.0",
59
60
]

61
62
sglang = [
    "uvloop",
63
    "sglang[diffusion]==0.5.9",
64
    "nixl[cu12]<=0.10.1",
65
    "cupy-cuda12x>=13.0.0",
66
67
]

68
69
70
71
72
[project.entry-points.pytest11]
vllm_tests = "dynamo.vllm.tests.conftest"
trtllm_tests = "dynamo.trtllm.tests.conftest"
sglang_tests = "dynamo.sglang.tests.conftest"

73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
[dependency-groups]
docs = [
    # Core Sphinx
    "sphinx>=8.1",
    "nvidia-sphinx-theme>=0.0.8",
    # Sphinx extensions
    "ablog>=0.11",
    "sphinx-copybutton>=0.5",
    "sphinx-design>=0.6",
    "sphinx-prompt>=1.9",
    "sphinx-sitemap>=2.6",
    "sphinx-tabs>=3.4",
    "sphinx-book-theme>=1.1",
    "sphinxcontrib-mermaid>=1.0",
    "sphinxcontrib-bibtex>=2.6",
88
    "sphinx-reredirects>=1.0.0",
89
90
91
92
93
94
    # Markdown and notebook support
    "myst-parser>=4.0",
    "myst-nb>=1.2",
    "nbsphinx>=0.9",
]

95
[build-system]
96
97
98
requires = ["hatchling"]
build-backend = "hatchling.build"

99
100
101
[tool.hatch.build.hooks.custom]
path = "hatch_build.py"

102
[tool.hatch.build.targets.wheel]
103
packages = [
104
    "components/src/dynamo",
105
]
106

107
108
109
[tool.hatch.metadata]
allow-direct-references = true

110
111
112
113
114
[tool.codespell]
# note: pre-commit passes explicit lists of files here, which this skip file list doesn't override -
# this is only to allow you to run codespell interactively
# this also overrides the grpc_generated folder, since it is generated

115
116
# Ignore data files and auto-generated files
skip = "./.git,./.github,./lib/llm/tests/data,*.lock,*.sum"
117

118
119
# ignore allowed words used in code
ignore-words-list = "afterall,ser,ende"
120
121
# use the 'clear' dictionary for unambiguous spelling mistakes
builtin = "clear"
122
123
# use custom dictionary in addition to the built-in one
dictionary = "./codespell.txt"
124
125
126
127
128
129
130
131
132
133
134
135
136
137
# disable warnings about binary files and wrong encoding
quiet-level = 3

[tool.isort]
profile = "black"
use_parentheses = true
multi_line_output = 3
include_trailing_comma = true
force_grid_wrap = 0
ensure_newline_before_comments = true
line_length = 88
balanced_wrapping = true
indent = "    "
skip = ["build"]
138
known_first_party = ["dynamo", "deploy"]
139
140
141
142
143
144
# isort may confuse what is 1st or 3rd library. e.g.
# when dynamo/vllm/omni/xx.py import vllm, local isort may treat this `vllm` as first
# party heuristically. This causes local sort differs from GitHub sort and pre-commit
# failure. To mitigate 1) one can install 3rd party lib so that isort is aware of it,
# 2) hardcode 3rd party lib here, 3) add "# isort: skip_file" to problematic files
# as the last resort.
145
known_third_party = ["vllm", "tensorrt_llm", "sglang", "aiconfigurator"]
146
147
148

[tool.pytest.ini_options]
minversion = "8.0"
149
tmp_path_retention_policy = "failed"
Neelay Shah's avatar
Neelay Shah committed
150
151

# NOTE
152
# We ignore model.py explicitly here to avoid mypy errors with duplicate modules
Neelay Shah's avatar
Neelay Shah committed
153
# pytest overrides the default mypy exclude configuration and so we exclude here as well
154
155
156
157
158
159
addopts = [
    "-ra",
    "--showlocals",
    "--strict-markers",
    "--strict-config",
    "--ignore-glob=*model.py",
Ryan Olson's avatar
Ryan Olson committed
160
    "--ignore-glob=*vllm_integration*",
161
    "--ignore-glob=*trtllm_integration*",
Richard Huo's avatar
Richard Huo committed
162
    "--ignore-glob=*kvbm/python/kvbm*",
163
    "--ignore-glob=*_inc.py",
164
    "--ignore-glob=*/llm/tensorrtllm*",
165
    "--ignore-glob=docs/*",
166
    "--ignore-glob=components/src/dynamo/sglang/request_handlers/*",
167
    "--ignore-glob=components/src/dynamo/sglang/multimodal_utils/*",
168
    "--ignore-glob=components/src/dynamo/vllm/multimodal_utils/*",
169
    "--ignore-glob=examples/backends/sglang/slurm_jobs/*",
170
171
    # FIXME: Get relative/generic blob paths to work here
]
172
173
174
xfail_strict = true
log_cli_level = "INFO"
filterwarnings = [
175
    "error",
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
    # CUDA deprecation warnings from tensorrt_llm
    "ignore:.*cuda*:DeprecationWarning",
    # protobuf C extension warning
    "ignore:.*PyType_Spec.*custom tp_new.*:DeprecationWarning",
    # unclosed socket/event loop warnings
    "ignore:.*unclosed.*socket.*:ResourceWarning",
    "ignore:.*unclosed event loop.*:ResourceWarning",
    # unraisable exception warnings
    "ignore:.*Exception ignored in.*:pytest.PytestUnraisableExceptionWarning",
    # pynvml deprecation, temporary until upstream migrates to nvidia-ml-py
    "ignore:The pynvml package is deprecated.*:FutureWarning",
    # Dynamo's own KV events deprecation warning
    "ignore:Automatic KV events configuration is deprecated.*:FutureWarning",
    # Python 3.12 SWIG extension warning from third-party tokenizer deps
    "ignore:builtin type (SwigPyPacked|SwigPyObject|swigvarlink) has no __module__ attribute:DeprecationWarning",
    # Pydantic V2 deprecation warnings from TRTLLM dependencies
192
193
194
    "ignore:Support for class-based `config`.*:pydantic.warnings.PydanticDeprecatedSince20",
    "ignore:Using extra keyword arguments on `Field`.*:pydantic.warnings.PydanticDeprecatedSince20",
    "ignore:The `schema` method is deprecated.*:pydantic.warnings.PydanticDeprecatedSince20",
195
    # Pydantic field shadowing in tensorrt_llm.serve.openai_protocol.ResponseFormat
196
    'ignore:Field name "schema" in "ResponseFormat" shadows an attribute in parent:UserWarning',
197
    # pytest-benchmark automatically disables when xdist is active
198
    "ignore:.*Benchmarks are automatically disabled.*:pytest_benchmark.logger.PytestBenchmarkWarning",
199
    # torchao invalid escape sequences in docstrings at import time
200
    "ignore:.*invalid escape sequence.*:SyntaxWarning",
201
    # torchao import path changes (https://github.com/pytorch/ao/issues/2752)
202
    "ignore:Importing.*torchao\\.dtypes.*:DeprecationWarning",
203
204
205
206
207
    # Triton CPU fallback warning on CPU-only runners
    "ignore:Triton is not supported on current platform.*:UserWarning",
    # torch.jit.script_method deprecation from torch.utils.mkldnn
    "ignore:.*torch\\.jit\\.script_method.*is deprecated.*:DeprecationWarning",
    # nvidia-modelopt warning about transformers version (transitive dep from TRT-LLM)
208
    "ignore:transformers version .* is incompatible with nvidia-modelopt.*:UserWarning",
209
210
211
    # SGLang quantization warnings on CPU-only runners
    "ignore:Only CUDA, HIP and XPU support AWQ currently.*:UserWarning",
    "ignore:Only CUDA support GGUF quantization currently.*:UserWarning",
212
]
213
214


215
216
# NOTE: Can also manually mark tests with @pytest.mark.asyncio
asyncio_mode = "auto"
Neelay Shah's avatar
Neelay Shah committed
217
markers = [
218
    "pre_merge: marks tests to run before merging",
219
    "post_merge: marks tests to run after merge",
220
    "parallel: marks tests that can run in parallel with pytest-xdist",
221
222
    "nightly: marks tests to run nightly",
    "weekly: marks tests to run weekly",
223
    "release: marks tests to run on release pipelines",
224
    "gpu_0: marks tests that don't require GPU",
Neelay Shah's avatar
Neelay Shah committed
225
226
    "gpu_1: marks tests to run on GPU",
    "gpu_2: marks tests to run on 2GPUs",
227
228
    "gpu_4: marks tests to run on 4GPUs",
    "gpu_8: marks tests to run on 8GPUs",
229
    "max_vram_gib(N): peak VRAM in GiB (with 10% safety). Filter with --max-vram-gib=N",
Neelay Shah's avatar
Neelay Shah committed
230
231
232
233
    "e2e: marks tests as end-to-end tests",
    "integration: marks tests as integration tests",
    "unit: marks tests as unit tests",
    "stress: marks tests as stress tests",
234
    "performance: marks tests as performance tests",
235
    "benchmark: marks tests as benchmark tests",
Neelay Shah's avatar
Neelay Shah committed
236
    "vllm: marks tests as requiring vllm",
237
    "trtllm: marks tests as requiring trtllm",
Neelay Shah's avatar
Neelay Shah committed
238
    "sglang: marks tests as requiring sglang",
239
    "lmcache: marks tests as requiring lmcache",
240
    "multimodal: marks tests as multimodal (image/video) tests",
Ryan Olson's avatar
Ryan Olson committed
241
    "slow: marks tests as known to be slow",
242
    "h100: marks tests to run on H100",
243
    "aiconfigurator: marks e2e tests that cover aiconfigurator functionality",
244
245
    "router: marks tests for router component",
    "planner: marks tests for planner component",
Alec's avatar
Alec committed
246
    "kvbm: marks tests for KV behavior and model determinism",
247
    "kvbm_concurrency: marks concurrency stress tests for KVBM (runs separately)",
248
    "model: model id used by a test or parameter",
249
250
    "custom_build: marks tests that require custom builds or special setup (e.g., MoE models)",
    "k8s: marks tests as requiring Kubernetes",
251
    "fault_tolerance: marks tests as fault tolerance tests",
252
    "deploy: marks tests as deployment tests",
253
254
255
256
257
258
259
    # Built-in markers
    "skip: skip this test",
    "skipif: skip if condition is true",
    "xfail: expected failure",
    "usefixtures: use fixtures",
    "parametrize: parameterized test",
    "filterwarnings: filter warnings",
260
261
262
    "asyncio: asyncio test marker",
    # Third-party plugin markers
    "timeout: test timeout in seconds (pytest-timeout plugin)",
Neelay Shah's avatar
Neelay Shah committed
263
]
264
265
266
267
268
269
270

# Linting/formatting
[tool.ruff]
# Same as Black.
line-length = 88
indent-width = 4

Blazej's avatar
Blazej committed
271
272
[tool.ruff.lint.extend-per-file-ignores]
"icp/tests/**/test_*.py" = ["F811", "F401"]
273
"*_inc.py" = ["F821"]
Blazej's avatar
Blazej committed
274

275
276
277
278
279
280
281
282
283
# This is IDE (e.g. Cursor's default Python language server)
# Configure it so that developers can use "go-to-definition", "hover types" and other
# features.
[tool.basedpyright]
extraPaths = ["components/src", "lib/bindings/python/src"]
# This is for external dependencies.
venvPath = "."
venv = ".venv"

284
[tool.mypy]
Neelay Shah's avatar
Neelay Shah committed
285

286
287
# --disable-error-code: WAR large set of errors due to mypy not being run
#   previously. We can slowly enable sets of errors to fix over time.
Neelay Shah's avatar
Neelay Shah committed
288
289
# disable_error_code = []

290
# --explicit-package-bases: WAR errors about duplicate module names used
291
292
293
#   throughout the llm examples. For example, the common module in
#   tensorrt_llm and vllm are both named common.
explicit_package_bases = true
Neelay Shah's avatar
Neelay Shah committed
294

295
296
297
298
# --ignore-missing-imports: WAR too many errors when developing outside
#   of container environment with PYTHONPATH set and packages installed.
#   NOTE: Can possibly move mypy from pre-commit to a github action run only in
#   a container with the expected environment and PYTHONPATH setup.
Neelay Shah's avatar
Neelay Shah committed
299
300
ignore_missing_imports = true

301
302
303
304
check_untyped_defs = true

[[tool.mypy.overrides]]
# Skip mypy analysis on internal dependencies of vllm
305
module = ["vllm.*"]
306
follow_imports = "skip"
307
ignore_missing_imports = true
308

309
310
311
312
313
314
[[tool.mypy.overrides]]
# WAR mypy 1.18.x crash with numpy 1.26.x stubs:
# "Should never get here in normal mode, got TypeAlias:numpy.float64 instead of TypeInfo"
module = ["numpy", "numpy.*"]
follow_imports = "skip"

315
316
[tool.sphinx]

317
# extra-content-head
318
319
320
321
322
323
324
325
326
327
328
329
extra_content_head = [
   '''
   <script src="https://assets.adobedtm.com/5d4962a43b79/c1061d2c5e7b/launch-191c2462b890.min.js" ></script>
   ''',
]

#extra-content-footer
extra_content_footer = [
   '''
   <script type="text/javascript">if (typeof _satellite !== "undefined") {_satellite.pageBottom();}</script>
   ''',
]