Coverage for pydantic/_internal/_mock_val_ser.py: 84.03%

95 statements  

« prev     ^ index     » next       coverage.py v7.9.2, created at 2025-07-22 09:30 +0000

1from __future__ import annotations 1abcdefghijklMmnopqrstuvwxyzNPABCDEFGHIJKLO

2 

3from collections.abc import Iterator, Mapping 1abcdefghijklMmnopqrstuvwxyzNPABCDEFGHIJKLO

4from typing import TYPE_CHECKING, Any, Callable, Generic, Literal, TypeVar, Union 1abcdefghijklMmnopqrstuvwxyzNPABCDEFGHIJKLO

5 

6from pydantic_core import CoreSchema, SchemaSerializer, SchemaValidator 1abcdefghijklMmnopqrstuvwxyzNPABCDEFGHIJKLO

7 

8from ..errors import PydanticErrorCodes, PydanticUserError 1abcdefghijklMmnopqrstuvwxyzNPABCDEFGHIJKLO

9from ..plugin._schema_validator import PluggableSchemaValidator 1abcdefghijklMmnopqrstuvwxyzNPABCDEFGHIJKLO

10 

11if TYPE_CHECKING: 1abcdefghijklMmnopqrstuvwxyzNPABCDEFGHIJKLO

12 from ..dataclasses import PydanticDataclass 

13 from ..main import BaseModel 

14 from ..type_adapter import TypeAdapter 

15 

16 

17ValSer = TypeVar('ValSer', bound=Union[SchemaValidator, PluggableSchemaValidator, SchemaSerializer]) 1abcdefghijklMmnopqrstuvwxyzNPABCDEFGHIJKLO

18T = TypeVar('T') 1abcdefghijklMmnopqrstuvwxyzNPABCDEFGHIJKLO

19 

20 

21class MockCoreSchema(Mapping[str, Any]): 1abcdefghijklMmnopqrstuvwxyzNPABCDEFGHIJKLO

22 """Mocker for `pydantic_core.CoreSchema` which optionally attempts to 

23 rebuild the thing it's mocking when one of its methods is accessed and raises an error if that fails. 

24 """ 

25 

26 __slots__ = '_error_message', '_code', '_attempt_rebuild', '_built_memo' 1abcdefghijklMmnopqrstuvwxyzNPABCDEFGHIJKLO

27 

28 def __init__( 1abcdefghijklMmnopqrstuvwxyzNPABCDEFGHIJKLO

29 self, 

30 error_message: str, 

31 *, 

32 code: PydanticErrorCodes, 

33 attempt_rebuild: Callable[[], CoreSchema | None] | None = None, 

34 ) -> None: 

35 self._error_message = error_message 1abcdefghijklMmnopqrstuvwxyzNPABCDEFGHIJKLO

36 self._code: PydanticErrorCodes = code 1abcdefghijklMmnopqrstuvwxyzNPABCDEFGHIJKLO

37 self._attempt_rebuild = attempt_rebuild 1abcdefghijklMmnopqrstuvwxyzNPABCDEFGHIJKLO

38 self._built_memo: CoreSchema | None = None 1abcdefghijklMmnopqrstuvwxyzNPABCDEFGHIJKLO

39 

40 def __getitem__(self, key: str) -> Any: 1abcdefghijklMmnopqrstuvwxyzNPABCDEFGHIJKLO

41 return self._get_built().__getitem__(key) 1abcdefghijklMmnopqrstuvwxyzNABCDEFGHIJKLO

42 

43 def __len__(self) -> int: 1abcdefghijklMmnopqrstuvwxyzNPABCDEFGHIJKLO

44 return self._get_built().__len__() 

45 

46 def __iter__(self) -> Iterator[str]: 1abcdefghijklMmnopqrstuvwxyzNPABCDEFGHIJKLO

47 return self._get_built().__iter__() 

48 

49 def _get_built(self) -> CoreSchema: 1abcdefghijklMmnopqrstuvwxyzNPABCDEFGHIJKLO

50 if self._built_memo is not None: 50 ↛ 51line 50 didn't jump to line 51 because the condition on line 50 was never true1abcdefghijklMmnopqrstuvwxyzNABCDEFGHIJKLO

51 return self._built_memo 

52 

53 if self._attempt_rebuild: 53 ↛ 58line 53 didn't jump to line 58 because the condition on line 53 was always true1abcdefghijklMmnopqrstuvwxyzNABCDEFGHIJKLO

54 schema = self._attempt_rebuild() 1abcdefghijklMmnopqrstuvwxyzNABCDEFGHIJKLO

55 if schema is not None: 1abcdefghijklmnopqrstuvwxyzABCDEFGHIJKL

56 self._built_memo = schema 1abcdefghijklmnopqrstuvwxyzABCDEFGHIJKL

57 return schema 1abcdefghijklmnopqrstuvwxyzABCDEFGHIJKL

58 raise PydanticUserError(self._error_message, code=self._code) 1abcdefghijklmnopqrstuvwxyzABCDEFGHIJKL

59 

60 def rebuild(self) -> CoreSchema | None: 1abcdefghijklMmnopqrstuvwxyzNPABCDEFGHIJKLO

61 self._built_memo = None 1abcdefghijklMmnopqrstuvwxyzNABCDEFGHIJKLO

62 if self._attempt_rebuild: 62 ↛ 68line 62 didn't jump to line 68 because the condition on line 62 was always true1abcdefghijklMmnopqrstuvwxyzNABCDEFGHIJKLO

63 schema = self._attempt_rebuild() 1abcdefghijklMmnopqrstuvwxyzNABCDEFGHIJKLO

64 if schema is not None: 1abcdefghijklMmnopqrstuvwxyzNABCDEFGHIJKLO

65 return schema 1abcdefghijklMmnopqrstuvwxyzNABCDEFGHIJKLO

66 else: 

67 raise PydanticUserError(self._error_message, code=self._code) 1abcdefghijklmnopqrstuvwxyzABCDEFGHIJKL

68 return None 

69 

70 

71class MockValSer(Generic[ValSer]): 1abcdefghijklMmnopqrstuvwxyzNPABCDEFGHIJKLO

72 """Mocker for `pydantic_core.SchemaValidator` or `pydantic_core.SchemaSerializer` which optionally attempts to 

73 rebuild the thing it's mocking when one of its methods is accessed and raises an error if that fails. 

74 """ 

75 

76 __slots__ = '_error_message', '_code', '_val_or_ser', '_attempt_rebuild' 1abcdefghijklMmnopqrstuvwxyzNPABCDEFGHIJKLO

77 

78 def __init__( 1abcdefghijklMmnopqrstuvwxyzNPABCDEFGHIJKLO

79 self, 

80 error_message: str, 

81 *, 

82 code: PydanticErrorCodes, 

83 val_or_ser: Literal['validator', 'serializer'], 

84 attempt_rebuild: Callable[[], ValSer | None] | None = None, 

85 ) -> None: 

86 self._error_message = error_message 1abcdefghijklMmnopqrstuvwxyzNPABCDEFGHIJKLO

87 self._val_or_ser = SchemaValidator if val_or_ser == 'validator' else SchemaSerializer 1abcdefghijklMmnopqrstuvwxyzNPABCDEFGHIJKLO

88 self._code: PydanticErrorCodes = code 1abcdefghijklMmnopqrstuvwxyzNPABCDEFGHIJKLO

89 self._attempt_rebuild = attempt_rebuild 1abcdefghijklMmnopqrstuvwxyzNPABCDEFGHIJKLO

90 

91 def __getattr__(self, item: str) -> None: 1abcdefghijklMmnopqrstuvwxyzNPABCDEFGHIJKLO

92 __tracebackhide__ = True 1abcdefghijklMmnopqrstuvwxyzNPABCDEFGHIJKLO

93 if self._attempt_rebuild: 1abcdefghijklMmnopqrstuvwxyzNPABCDEFGHIJKLO

94 val_ser = self._attempt_rebuild() 1abcdefghijklMmnopqrstuvwxyzNABCDEFGHIJKLO

95 if val_ser is not None: 1abcdefghijklMmnopqrstuvwxyzNABCDEFGHIJKLO

96 return getattr(val_ser, item) 1abcdefghijklMmnopqrstuvwxyzNABCDEFGHIJKLO

97 

98 # raise an AttributeError if `item` doesn't exist 

99 getattr(self._val_or_ser, item) 1abcdefghijklMmnopqrstuvwxyzNPABCDEFGHIJKLO

100 raise PydanticUserError(self._error_message, code=self._code) 1abcdefghijklMmnopqrstuvwxyzNABCDEFGHIJKLO

101 

102 def rebuild(self) -> ValSer | None: 1abcdefghijklMmnopqrstuvwxyzNPABCDEFGHIJKLO

103 if self._attempt_rebuild: 

104 val_ser = self._attempt_rebuild() 

105 if val_ser is not None: 

106 return val_ser 

107 else: 

108 raise PydanticUserError(self._error_message, code=self._code) 

109 return None 

110 

111 

112def set_type_adapter_mocks(adapter: TypeAdapter) -> None: 1abcdefghijklMmnopqrstuvwxyzNPABCDEFGHIJKLO

113 """Set `core_schema`, `validator` and `serializer` to mock core types on a type adapter instance. 

114 

115 Args: 

116 adapter: The type adapter instance to set the mocks on 

117 """ 

118 type_repr = str(adapter._type) 1abcdefghijklMmnopqrstuvwxyzNABCDEFGHIJKLO

119 undefined_type_error_message = ( 1abcdefghijklMmnopqrstuvwxyzNABCDEFGHIJKLO

120 f'`TypeAdapter[{type_repr}]` is not fully defined; you should define `{type_repr}` and all referenced types,' 

121 f' then call `.rebuild()` on the instance.' 

122 ) 

123 

124 def attempt_rebuild_fn(attr_fn: Callable[[TypeAdapter], T]) -> Callable[[], T | None]: 1abcdefghijklMmnopqrstuvwxyzNABCDEFGHIJKLO

125 def handler() -> T | None: 1abcdefghijklMmnopqrstuvwxyzNABCDEFGHIJKLO

126 if adapter.rebuild(raise_errors=False, _parent_namespace_depth=5) is not False: 126 ↛ 128line 126 didn't jump to line 128 because the condition on line 126 was always true1abcdefghijklMmnopqrstuvwxyzNABCDEFGHIJKLO

127 return attr_fn(adapter) 1abcdefghijklMmnopqrstuvwxyzNABCDEFGHIJKLO

128 return None 

129 

130 return handler 1abcdefghijklMmnopqrstuvwxyzNABCDEFGHIJKLO

131 

132 adapter.core_schema = MockCoreSchema( # pyright: ignore[reportAttributeAccessIssue] 1abcdefghijklMmnopqrstuvwxyzNABCDEFGHIJKLO

133 undefined_type_error_message, 

134 code='class-not-fully-defined', 

135 attempt_rebuild=attempt_rebuild_fn(lambda ta: ta.core_schema), 

136 ) 

137 adapter.validator = MockValSer( # pyright: ignore[reportAttributeAccessIssue] 1abcdefghijklMmnopqrstuvwxyzNABCDEFGHIJKLO

138 undefined_type_error_message, 

139 code='class-not-fully-defined', 

140 val_or_ser='validator', 

141 attempt_rebuild=attempt_rebuild_fn(lambda ta: ta.validator), 

142 ) 

143 adapter.serializer = MockValSer( # pyright: ignore[reportAttributeAccessIssue] 1abcdefghijklMmnopqrstuvwxyzNABCDEFGHIJKLO

144 undefined_type_error_message, 

145 code='class-not-fully-defined', 

146 val_or_ser='serializer', 

147 attempt_rebuild=attempt_rebuild_fn(lambda ta: ta.serializer), 

148 ) 

149 

150 

151def set_model_mocks(cls: type[BaseModel], undefined_name: str = 'all referenced types') -> None: 1abcdefghijklMmnopqrstuvwxyzNPABCDEFGHIJKLO

152 """Set `__pydantic_core_schema__`, `__pydantic_validator__` and `__pydantic_serializer__` to mock core types on a model. 

153 

154 Args: 

155 cls: The model class to set the mocks on 

156 undefined_name: Name of the undefined thing, used in error messages 

157 """ 

158 undefined_type_error_message = ( 1abcdefghijklMmnopqrstuvwxyzNPABCDEFGHIJKLO

159 f'`{cls.__name__}` is not fully defined; you should define {undefined_name},' 

160 f' then call `{cls.__name__}.model_rebuild()`.' 

161 ) 

162 

163 def attempt_rebuild_fn(attr_fn: Callable[[type[BaseModel]], T]) -> Callable[[], T | None]: 1abcdefghijklMmnopqrstuvwxyzNPABCDEFGHIJKLO

164 def handler() -> T | None: 1abcdefghijklMmnopqrstuvwxyzNPABCDEFGHIJKLO

165 if cls.model_rebuild(raise_errors=False, _parent_namespace_depth=5) is not False: 1abcdefghijklMmnopqrstuvwxyzNABCDEFGHIJKLO

166 return attr_fn(cls) 1abcdefghijklMmnopqrstuvwxyzNABCDEFGHIJKLO

167 return None 1abcdefghijklMmnopqrstuvwxyzNABCDEFGHIJKLO

168 

169 return handler 1abcdefghijklMmnopqrstuvwxyzNPABCDEFGHIJKLO

170 

171 cls.__pydantic_core_schema__ = MockCoreSchema( # pyright: ignore[reportAttributeAccessIssue] 1abcdefghijklMmnopqrstuvwxyzNPABCDEFGHIJKLO

172 undefined_type_error_message, 

173 code='class-not-fully-defined', 

174 attempt_rebuild=attempt_rebuild_fn(lambda c: c.__pydantic_core_schema__), 

175 ) 

176 cls.__pydantic_validator__ = MockValSer( # pyright: ignore[reportAttributeAccessIssue] 1abcdefghijklMmnopqrstuvwxyzNPABCDEFGHIJKLO

177 undefined_type_error_message, 

178 code='class-not-fully-defined', 

179 val_or_ser='validator', 

180 attempt_rebuild=attempt_rebuild_fn(lambda c: c.__pydantic_validator__), 

181 ) 

182 cls.__pydantic_serializer__ = MockValSer( # pyright: ignore[reportAttributeAccessIssue] 1abcdefghijklMmnopqrstuvwxyzNPABCDEFGHIJKLO

183 undefined_type_error_message, 

184 code='class-not-fully-defined', 

185 val_or_ser='serializer', 

186 attempt_rebuild=attempt_rebuild_fn(lambda c: c.__pydantic_serializer__), 

187 ) 

188 

189 

190def set_dataclass_mocks(cls: type[PydanticDataclass], undefined_name: str = 'all referenced types') -> None: 1abcdefghijklMmnopqrstuvwxyzNPABCDEFGHIJKLO

191 """Set `__pydantic_validator__` and `__pydantic_serializer__` to `MockValSer`s on a dataclass. 

192 

193 Args: 

194 cls: The model class to set the mocks on 

195 undefined_name: Name of the undefined thing, used in error messages 

196 """ 

197 from ..dataclasses import rebuild_dataclass 1abcdefghijklMmnopqrstuvwxyzNABCDEFGHIJKLO

198 

199 undefined_type_error_message = ( 1abcdefghijklMmnopqrstuvwxyzNABCDEFGHIJKLO

200 f'`{cls.__name__}` is not fully defined; you should define {undefined_name},' 

201 f' then call `pydantic.dataclasses.rebuild_dataclass({cls.__name__})`.' 

202 ) 

203 

204 def attempt_rebuild_fn(attr_fn: Callable[[type[PydanticDataclass]], T]) -> Callable[[], T | None]: 1abcdefghijklMmnopqrstuvwxyzNABCDEFGHIJKLO

205 def handler() -> T | None: 1abcdefghijklMmnopqrstuvwxyzNABCDEFGHIJKLO

206 if rebuild_dataclass(cls, raise_errors=False, _parent_namespace_depth=5) is not False: 1abcdefghijklMmnopqrstuvwxyzNABCDEFGHIJKLO

207 return attr_fn(cls) 1abcdefghijklMmnopqrstuvwxyzNABCDEFGHIJKLO

208 return None 1abcdefghijklMmnopqrstuvwxyzNABCDEFGHIJKLO

209 

210 return handler 1abcdefghijklMmnopqrstuvwxyzNABCDEFGHIJKLO

211 

212 cls.__pydantic_core_schema__ = MockCoreSchema( # pyright: ignore[reportAttributeAccessIssue] 1abcdefghijklMmnopqrstuvwxyzNABCDEFGHIJKLO

213 undefined_type_error_message, 

214 code='class-not-fully-defined', 

215 attempt_rebuild=attempt_rebuild_fn(lambda c: c.__pydantic_core_schema__), 

216 ) 

217 cls.__pydantic_validator__ = MockValSer( # pyright: ignore[reportAttributeAccessIssue] 1abcdefghijklMmnopqrstuvwxyzNABCDEFGHIJKLO

218 undefined_type_error_message, 

219 code='class-not-fully-defined', 

220 val_or_ser='validator', 

221 attempt_rebuild=attempt_rebuild_fn(lambda c: c.__pydantic_validator__), 

222 ) 

223 cls.__pydantic_serializer__ = MockValSer( # pyright: ignore[reportAttributeAccessIssue] 1abcdefghijklMmnopqrstuvwxyzNABCDEFGHIJKLO

224 undefined_type_error_message, 

225 code='class-not-fully-defined', 

226 val_or_ser='serializer', 

227 attempt_rebuild=attempt_rebuild_fn(lambda c: c.__pydantic_serializer__), 

228 )