forked from pydantic/pydantic-ai
-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathpyproject.toml
More file actions
361 lines (331 loc) · 12.7 KB
/
pyproject.toml
File metadata and controls
361 lines (331 loc) · 12.7 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
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
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
[build-system]
requires = ["hatchling", "uv-dynamic-versioning>=0.7.0"]
build-backend = "hatchling.build"
[tool.hatch.version]
source = "uv-dynamic-versioning"
[tool.uv-dynamic-versioning]
vcs = "git"
style = "pep440"
bump = true
[project]
name = "pydantic-ai"
dynamic = ["version", "dependencies", "optional-dependencies"]
description = "Agent Framework / shim to use Pydantic with LLMs"
authors = [
{ name = "Samuel Colvin", email = "samuel@pydantic.dev" },
{ name = "Marcelo Trylesinski", email = "marcelotryle@gmail.com" },
{ name = "David Montague", email = "david@pydantic.dev" },
{ name = "Alex Hall", email = "alex@pydantic.dev" },
{ name = "Douwe Maan", email = "douwe@pydantic.dev" },
]
license = "MIT"
license-files = ["LICENSE"]
readme = "README.md"
classifiers = [
"Development Status :: 5 - Production/Stable",
"Programming Language :: Python",
"Programming Language :: Python :: 3",
"Programming Language :: Python :: 3 :: Only",
"Programming Language :: Python :: 3.10",
"Programming Language :: Python :: 3.11",
"Programming Language :: Python :: 3.12",
"Programming Language :: Python :: 3.13",
"Programming Language :: Python :: 3.14",
"Intended Audience :: Developers",
"Intended Audience :: Information Technology",
"Operating System :: OS Independent",
"Topic :: Internet",
"Topic :: Scientific/Engineering :: Artificial Intelligence",
"Topic :: Software Development :: Libraries :: Python Modules",
"Framework :: Pydantic",
"Framework :: Pydantic :: 2",
]
requires-python = ">=3.10"
[tool.hatch.metadata.hooks.uv-dynamic-versioning]
dependencies = [
"pydantic-ai-slim[openai,vertexai,google,xai,groq,anthropic,mistral,cohere,bedrock,huggingface,cli,mcp,fastmcp,evals,ag-ui,retries,temporal,logfire,ui,spec]=={{ version }}",
]
[tool.hatch.metadata.hooks.uv-dynamic-versioning.optional-dependencies]
examples = ["pydantic-ai-examples=={{ version }}"]
a2a = ["fasta2a>=0.4.1"]
dbos = ["pydantic-ai-slim[dbos]=={{ version }}"]
prefect = ["pydantic-ai-slim[prefect]=={{ version }}"]
outlines-transformers = ["pydantic-ai-slim[outlines-transformers]=={{ version }}"]
outlines-llamacpp = ["pydantic-ai-slim[outlines-llamacpp]=={{ version }}"]
outlines-mlxlm = ["pydantic-ai-slim[outlines-mlxlm]=={{ version }}; platform_system == 'Darwin' and platform_machine == 'arm64'"]
outlines-sglang = ["pydantic-ai-slim[outlines-sglang]=={{ version }}"]
outlines-vllm-offline = ["pydantic-ai-slim[outlines-vllm-offline]=={{ version }}"]
sentence-transformers = ["pydantic-ai-slim[sentence-transformers]=={{ version }}"]
voyageai = ["pydantic-ai-slim[voyageai]=={{ version }}"]
[project.urls]
Homepage = "https://ai.pydantic.dev"
Source = "https://github.com/pydantic/pydantic-ai"
Documentation = "https://ai.pydantic.dev"
Changelog = "https://github.com/pydantic/pydantic-ai/releases"
[project.scripts]
pai = "pydantic_ai._cli:cli_exit" # TODO remove this when clai has been out for a while
[tool.uv.sources]
pydantic-ai = { workspace = true }
pydantic-ai-slim = { workspace = true }
pydantic-evals = { workspace = true }
pydantic-graph = { workspace = true }
pydantic-ai-examples = { workspace = true }
pydantic-docs = { git = "https://github.com/pydantic/pydantic-docs" }
[tool.uv.workspace]
members = [
"pydantic_ai_slim",
"pydantic_evals",
"pydantic_graph",
"clai",
"examples",
]
[tool.uv]
default-groups = ["dev", "lint", "docs"]
constraint-dependencies = ["ray>=2.54.0", "authlib>=1.6.7"]
[dependency-groups]
dev = [
"anyio>=4.5.0",
"asgi-lifespan>=2.1.0",
"devtools>=0.12.2",
"coverage[toml]>=7.10.7",
"dirty-equals>=0.9.0",
"duckduckgo-search>=7.0.0",
"exa-py>=2.0.0",
"tavily-python>=0.5.0",
"markdownify>=1.2",
"inline-snapshot>=0.32.5",
"pytest>=9.0.0",
"pytest-examples>=0.0.18",
"pytest-mock>=3.14.0",
"pytest-pretty>=1.3.0",
"pytest-recording>=0.13.2",
"diff-cover>=9.2.0",
"boto3-stubs[bedrock-runtime]>=1.42.13",
"strict-no-cover @ git+https://github.com/pydantic/strict-no-cover.git@7fc59da2c4dff919db2095a0f0e47101b657131d",
"pytest-xdist>=3.6.1",
# Needed for PyCharm users
"pip>=26.0",
"genai-prices>=0.0.28",
"brotli>=1.2.0",
]
lint = ["mypy>=1.11.2", "pyright>=1.1.408", "ruff>=0.14.14"]
docs = [
"pydantic-ai[a2a]",
"black>=24.10.0",
"mkdocs>=1.6.1",
"mkdocs-glightbox>=0.4.0",
"mkdocs-llmstxt>=0.2.0",
"mkdocs-redirects>=1.2.2",
"mkdocs-material[imaging]>=9.7.0",
"mkdocstrings-python>=2.0.0",
"griffe-warnings-deprecated>=1.1.0",
"pydantic-docs",
]
docs-upload = ["algoliasearch>=4.12.0", "pydantic>=2.10.1"]
[tool.hatch.build.targets.wheel]
bypass-selection = true
[tool.hatch.build.targets.sdist]
include = ["/README.md", "/Makefile"]
# We are excluding the tests because we have 100MB of test data in this repository.
# Obviously that's a problem, but for now, we'll just exclude them.
exclude = ["/tests"]
[tool.ruff]
line-length = 120
target-version = "py310"
include = [
"pydantic_ai_slim/**/*.py",
"pydantic_evals/**/*.py",
"pydantic_graph/**/*.py",
"examples/**/*.py",
"clai/**/*.py",
"tests/**/*.py",
"docs/**/*.py",
]
[tool.ruff.lint]
preview = true # For `PLW1514`
extend-select = [
"Q",
"RUF100",
"RUF018", # https://docs.astral.sh/ruff/rules/assignment-in-assert/
"C90",
"UP",
"I",
"D",
"TID251",
"PLW1514", # https://docs.astral.sh/ruff/rules/unspecified-encoding/
]
flake8-quotes = { inline-quotes = "single", multiline-quotes = "double" }
mccabe = { max-complexity = 15 }
ignore = [
"D100", # ignore missing docstring in module
"D102", # ignore missing docstring in public method
"D104", # ignore missing docstring in public package
"D105", # ignore missing docstring in magic methods
"D107", # ignore missing docstring in __init__ methods
]
[tool.ruff.lint.isort]
combine-as-imports = true
known-first-party = ["pydantic_ai", "pydantic_evals", "pydantic_graph"]
# weird issue with ruff thinking fasta2a is still editable
known-third-party = ["fasta2a"]
[tool.ruff.lint.pydocstyle]
convention = "google"
[tool.ruff.lint.flake8-tidy-imports.banned-api]
"typing.TypedDict".msg = "Use typing_extensions.TypedDict instead."
"typing.assert_never".msg = "Use typing_extensions.assert_never instead."
[tool.ruff.format]
# don't format python in docstrings, pytest-examples takes care of it
docstring-code-format = false
quote-style = "single"
[tool.ruff.lint.per-file-ignores]
"examples/**/*.py" = ["D101", "D103"]
"tests/**/*.py" = ["D"]
"docs/**/*.py" = ["D"]
[tool.pyright]
pythonVersion = "3.10"
typeCheckingMode = "strict"
reportMissingTypeStubs = false
reportUnnecessaryIsInstance = false
reportUnnecessaryTypeIgnoreComment = true
reportMissingModuleSource = false
include = [
"pydantic_ai_slim",
"pydantic_evals",
"pydantic_graph",
"tests",
"examples",
"clai",
]
venvPath = '.'
venv = ".venv"
# see https://github.com/microsoft/pyright/issues/7771 - we don't want to error on decorated functions in tests
# which are not otherwise used
executionEnvironments = [
{ root = "tests", reportUnusedFunction = false, reportPrivateImportUsage = false },
]
exclude = [
"examples/pydantic_ai_examples/weather_agent_gradio.py",
"pydantic_ai_slim/pydantic_ai/ext/aci.py", # aci-sdk is too niche to be added as an (optional) dependency
"pydantic_ai_slim/pydantic_ai/embeddings/voyageai.py", # voyageai package has no type stubs
]
[tool.mypy]
files = "tests/typed_agent.py,tests/typed_graph.py"
strict = true
[tool.pytest.ini_options]
testpaths = ["tests", "docs/.hooks"]
xfail_strict = true
filterwarnings = [
"error",
# Issue with python-multipart - we don't want to bump the minimum version of starlette.
"ignore::PendingDeprecationWarning:starlette",
# boto3
"ignore::DeprecationWarning:botocore.*",
"ignore::RuntimeWarning:pydantic_ai.mcp",
# ag-ui-protocol >= 0.1.15 deprecated BinaryInputContent
"ignore:BinaryInputContent is deprecated:DeprecationWarning",
# uvicorn (mcp server)
"ignore:websockets.legacy is deprecated.*:DeprecationWarning:websockets.legacy",
"ignore:websockets.server.WebSocketServerProtocol is deprecated:DeprecationWarning",
# Provider.__del__ emits ResourceWarning when GC'd with an open HTTP client.
# Most tests create providers without async context managers, so this is expected.
"ignore:.*was garbage collected with an open HTTP client:ResourceWarning",
# random resource warnings; I suspect these are coming from vendor SDKs when running examples..
"ignore:unclosed <socket:ResourceWarning",
"ignore:unclosed <ssl.SSLSocket:ResourceWarning",
"ignore:unclosed event loop:ResourceWarning",
# prefect has a Pydantic v2 compatibility issue in RRuleSchedule
# happens only for python < 3.12 because of vllms pins
"ignore:The 'default' attribute with value 'UTC' was provided.*:UserWarning",
# This is because of cohere.
"ignore: 'asyncio.iscoroutinefunction' is deprecated:DeprecationWarning:cohere",
# This is because of google.
"ignore: '_UnionGenericAlias' is deprecated and slated for removal:DeprecationWarning:google.genai",
# TODO(Marcelo): Drop this once voyageai stops using Pydantic V1 functionality.
"ignore: Core Pydantic V1 functionality:UserWarning",
# prefect has a Pydantic v2 compatibility issue in RRuleSchedule
# happens only for python < 3.12 because of vllms pins
"ignore:UnsupportedFieldAttributeWarning:UserWarning",
# coolname (prefect dep) uses codecs.open(), deprecated in Python 3.14
"ignore:codecs.open\\(\\) is deprecated:DeprecationWarning",
]
# https://coverage.readthedocs.io/en/latest/config.html#run
[tool.coverage.run]
patch = ["subprocess"]
concurrency = ["multiprocessing", "thread"]
# We use a subdirectory for coverage data to avoid noisy coverage data files.
data_file = ".coverage/.coverage"
# required to avoid warnings about files created by create_module fixture
include = [
"pydantic_ai_slim/**/*.py",
"pydantic_evals/**/*.py",
"pydantic_graph/**/*.py",
"tests/**/*.py",
]
omit = [
"tests/example_modules/*.py",
"pydantic_ai_slim/pydantic_ai/ext/aci.py", # aci-sdk is too niche to be added as an (optional) dependency
"pydantic_ai_slim/pydantic_ai/common_tools/exa.py", # exa-py integration with external API calls
# TODO(Marcelo): Enable prefect coverage again.
"pydantic_ai_slim/pydantic_ai/durable_exec/prefect/*.py",
"tests/test_prefect.py",
"tests/models/xai_proto_cassettes.py", # dev-only helpers for recording/replaying xAI gRPC protobuf cassettes
"tests/_inline_snapshot.py", # test utility wrapper; branches depend on CLI flags
"tests/providers/test_gateway_catalog.py", # live gateway catalog smoke test; execution depends on opt-in flag and credentials
]
branch = true
# Disable include-ignored warnings as --source is enabled automatically causing a self conflict as per:
# https://github.com/pytest-dev/pytest-cov/issues/532
# https://github.com/pytest-dev/pytest-cov/issues/369
# This prevents coverage being generated by pytest-cov which has direct editor support in VS Code,
# making it super useful to check coverage while writing tests.
disable_warnings = ["include-ignored"]
[tool.coverage.paths]
# Allow CI run assets to be downloaded an replicated locally.
source = [
".",
"/home/runner/work/pydantic-ai/pydantic-ai",
"/System/Volumes/Data/home/runner/work/pydantic-ai/pydantic-ai",
]
# https://coverage.readthedocs.io/en/latest/config.html#report
[tool.coverage.report]
fail_under = 100
skip_covered = true
show_missing = true
ignore_errors = true
precision = 2
exclude_lines = [
# `# pragma: no cover` is standard marker for code that's not covered, this will error if code is covered
'pragma: no cover',
# use `# pragma: lax no cover` if you want to ignore cases where (some of) the code is covered
'pragma: lax no cover',
'raise NotImplementedError',
'if TYPE_CHECKING:',
'if typing.TYPE_CHECKING:',
'@overload',
'@deprecated',
'@typing.overload',
'@abstractmethod',
'\(Protocol\):$',
'typing.assert_never',
'$\s*assert_never\(',
'if __name__ == .__main__.:',
'except ImportError as _import_error:',
'$\s*pass$',
'assert False',
'@pytest\.mark\.skip',
'@pytest\.mark\.xfail',
]
[tool.logfire]
ignore_no_config = true
[tool.inline-snapshot]
format-command = "ruff format --stdin-filename {filename}"
[tool.inline-snapshot.shortcuts]
snap-fix = ["create", "fix"]
snap = ["create"]
[tool.codespell]
# Ref: https://github.com/codespell-project/codespell#using-a-config-file
skip = '.git*,*.svg,*.lock,*.css,*.yaml'
check-hidden = true
# Ignore "formatting" like **L**anguage
ignore-regex = '\*\*[A-Z]\*\*[a-z]+\b'
ignore-words-list = 'asend,aci,fpr'