Coverage for pydantic/plugin/_loader.py: 87.50%
32 statements
« prev ^ index » next coverage.py v7.5.4, created at 2024-07-03 19:29 +0000
« prev ^ index » next coverage.py v7.5.4, created at 2024-07-03 19:29 +0000
1from __future__ import annotations 1abcdefghijklmnopqrstuvwxyzMNOPQRSTUVABCDEFGHIJKL
3import importlib.metadata as importlib_metadata 1abcdefghijklmnopqrstuvwxyzMNOPQRSTUVABCDEFGHIJKL
4import os 1abcdefghijklmnopqrstuvwxyzMNOPQRSTUVABCDEFGHIJKL
5import warnings 1abcdefghijklmnopqrstuvwxyzMNOPQRSTUVABCDEFGHIJKL
6from typing import TYPE_CHECKING, Final, Iterable 1abcdefghijklmnopqrstuvwxyzMNOPQRSTUVABCDEFGHIJKL
8if TYPE_CHECKING: 1abcdefghijklmnopqrstuvwxyzMNOPQRSTUVABCDEFGHIJKL
9 from . import PydanticPluginProtocol
12PYDANTIC_ENTRY_POINT_GROUP: Final[str] = 'pydantic' 1abcdefghijklmnopqrstuvwxyzMNOPQRSTUVABCDEFGHIJKL
14# cache of plugins
15_plugins: dict[str, PydanticPluginProtocol] | None = None 1abcdefghijklmnopqrstuvwxyzMNOPQRSTUVABCDEFGHIJKL
16# return no plugins while loading plugins to avoid recursion and errors while import plugins
17# this means that if plugins use pydantic
18_loading_plugins: bool = False 1abcdefghijklmnopqrstuvwxyzMNOPQRSTUVABCDEFGHIJKL
21def get_plugins() -> Iterable[PydanticPluginProtocol]: 1abcdefghijklmnopqrstuvwxyzMNOPQRSTUVABCDEFGHIJKL
22 """Load plugins for Pydantic.
24 Inspired by: https://github.com/pytest-dev/pluggy/blob/1.3.0/src/pluggy/_manager.py#L376-L402
25 """
26 disabled_plugins = os.getenv('PYDANTIC_DISABLE_PLUGINS') 1abcdefghijklmnopqrstuvwxyzMNOPQRSTUVABCDEFGHIJKL
27 global _plugins, _loading_plugins
28 if _loading_plugins: 28 ↛ 30line 28 didn't jump to line 30 because the condition on line 28 was never true1abcdefghijklmnopqrstuvwxyzMNOPQRSTUVABCDEFGHIJKL
29 # this happens when plugins themselves use pydantic, we return no plugins
30 return ()
31 elif disabled_plugins in ('__all__', '1', 'true'): 1abcdefghijklmnopqrstuvwxyzMNOPQRSTUVABCDEFGHIJKL
32 return () 1abcdefghijklmnopqrstuvwxyzABCDEFGHIJKL
33 elif _plugins is None: 1abcdefghijklmnopqrstuvwxyzMNOPQRSTUVABCDEFGHIJKL
34 _plugins = {} 1abcdefghijklmnopqrstuvwxyzMNOPQRSTUVABCDEFGHIJKL
35 # set _loading_plugins so any plugins that use pydantic don't themselves use plugins
36 _loading_plugins = True 1abcdefghijklmnopqrstuvwxyzMNOPQRSTUVABCDEFGHIJKL
37 try: 1abcdefghijklmnopqrstuvwxyzMNOPQRSTUVABCDEFGHIJKL
38 for dist in importlib_metadata.distributions(): 1abcdefghijklmnopqrstuvwxyzMNOPQRSTUVABCDEFGHIJKL
39 for entry_point in dist.entry_points: 1abcdefghijklmnopqrstuvwxyzMNOPQRSTUVABCDEFGHIJKL
40 if entry_point.group != PYDANTIC_ENTRY_POINT_GROUP: 1abcdefghijklmnopqrstuvwxyzMNOPQRSTUVABCDEFGHIJKL
41 continue 1abcdefghijklmnopqrstuvwxyzMNOPQRSTUVABCDEFGHIJKL
42 if entry_point.value in _plugins: 42 ↛ 43line 42 didn't jump to line 43 because the condition on line 42 was never true1abcdefghijklmnopqrstuvwxyzABCDEFGHIJKL
43 continue
44 if disabled_plugins is not None and entry_point.name in disabled_plugins.split(','): 1abcdefghijklmnopqrstuvwxyzABCDEFGHIJKL
45 continue 1abcdefghijklmnopqrstuvwxyzABCDEFGHIJKL
46 try: 1abcdefghijklmnopqrstuvwxyzABCDEFGHIJKL
47 _plugins[entry_point.value] = entry_point.load() 1abcdefghijklmnopqrstuvwxyzABCDEFGHIJKL
48 except (ImportError, AttributeError) as e:
49 warnings.warn(
50 f'{e.__class__.__name__} while loading the `{entry_point.name}` Pydantic plugin, '
51 f'this plugin will not be installed.\n\n{e!r}'
52 )
53 finally:
54 _loading_plugins = False 1abcdefghijklmnopqrstuvwxyzMNOPQRSTUVABCDEFGHIJKL
56 return _plugins.values() 1abcdefghijklmnopqrstuvwxyzMNOPQRSTUVABCDEFGHIJKL