anthropic 0.66.0__py3-none-any.whl → 0.68.0__py3-none-any.whl

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (42) hide show
  1. anthropic/__init__.py +3 -0
  2. anthropic/_base_client.py +3 -3
  3. anthropic/_compat.py +48 -48
  4. anthropic/_models.py +54 -45
  5. anthropic/_utils/__init__.py +8 -2
  6. anthropic/_utils/_compat.py +45 -0
  7. anthropic/_utils/_datetime_parse.py +136 -0
  8. anthropic/_utils/_transform.py +5 -1
  9. anthropic/_utils/_typing.py +1 -1
  10. anthropic/_utils/_utils.py +0 -1
  11. anthropic/_version.py +1 -1
  12. anthropic/lib/tools/__init__.py +20 -0
  13. anthropic/lib/tools/_beta_functions.py +289 -0
  14. anthropic/lib/tools/_beta_runner.py +405 -0
  15. anthropic/resources/beta/messages/messages.py +370 -1
  16. anthropic/types/beta/__init__.py +14 -0
  17. anthropic/types/beta/beta_base64_pdf_source.py +15 -0
  18. anthropic/types/beta/beta_citation_config.py +9 -0
  19. anthropic/types/beta/beta_content_block.py +2 -0
  20. anthropic/types/beta/beta_content_block_param.py +4 -0
  21. anthropic/types/beta/beta_document_block.py +26 -0
  22. anthropic/types/beta/beta_plain_text_source.py +15 -0
  23. anthropic/types/beta/beta_raw_content_block_start_event.py +2 -0
  24. anthropic/types/beta/beta_request_document_block_param.py +1 -1
  25. anthropic/types/beta/beta_server_tool_usage.py +3 -0
  26. anthropic/types/beta/beta_server_tool_use_block.py +1 -1
  27. anthropic/types/beta/beta_server_tool_use_block_param.py +3 -1
  28. anthropic/types/beta/beta_tool_union_param.py +2 -0
  29. anthropic/types/beta/beta_web_fetch_block.py +21 -0
  30. anthropic/types/beta/beta_web_fetch_block_param.py +22 -0
  31. anthropic/types/beta/beta_web_fetch_tool_20250910_param.py +46 -0
  32. anthropic/types/beta/beta_web_fetch_tool_result_block.py +20 -0
  33. anthropic/types/beta/beta_web_fetch_tool_result_block_param.py +25 -0
  34. anthropic/types/beta/beta_web_fetch_tool_result_error_block.py +14 -0
  35. anthropic/types/beta/beta_web_fetch_tool_result_error_block_param.py +15 -0
  36. anthropic/types/beta/beta_web_fetch_tool_result_error_code.py +16 -0
  37. anthropic/types/beta/message_count_tokens_params.py +2 -0
  38. anthropic/types/document_block_param.py +1 -1
  39. {anthropic-0.66.0.dist-info → anthropic-0.68.0.dist-info}/METADATA +51 -1
  40. {anthropic-0.66.0.dist-info → anthropic-0.68.0.dist-info}/RECORD +42 -25
  41. {anthropic-0.66.0.dist-info → anthropic-0.68.0.dist-info}/WHEEL +0 -0
  42. {anthropic-0.66.0.dist-info → anthropic-0.68.0.dist-info}/licenses/LICENSE +0 -0
anthropic/__init__.py CHANGED
@@ -87,11 +87,14 @@ __all__ = [
87
87
  "DefaultAioHttpClient",
88
88
  "HUMAN_PROMPT",
89
89
  "AI_PROMPT",
90
+ "beta_tool",
91
+ "beta_async_tool",
90
92
  ]
91
93
 
92
94
  if not _t.TYPE_CHECKING:
93
95
  from ._utils._resources_proxy import resources as resources
94
96
 
97
+ from .lib.tools import beta_tool, beta_async_tool
95
98
  from .lib.vertex import *
96
99
  from .lib.bedrock import *
97
100
  from .lib.streaming import *
anthropic/_base_client.py CHANGED
@@ -62,7 +62,7 @@ from ._types import (
62
62
  ModelBuilderProtocol,
63
63
  )
64
64
  from ._utils import is_dict, is_list, asyncify, is_given, lru_cache, is_mapping
65
- from ._compat import PYDANTIC_V2, model_copy, model_dump
65
+ from ._compat import PYDANTIC_V1, model_copy, model_dump
66
66
  from ._models import GenericModel, FinalRequestOptions, validate_type, construct_type
67
67
  from ._response import (
68
68
  APIResponse,
@@ -237,7 +237,7 @@ class BaseSyncPage(BasePage[_T], Generic[_T]):
237
237
  model: Type[_T],
238
238
  options: FinalRequestOptions,
239
239
  ) -> None:
240
- if PYDANTIC_V2 and getattr(self, "__pydantic_private__", None) is None:
240
+ if (not PYDANTIC_V1) and getattr(self, "__pydantic_private__", None) is None:
241
241
  self.__pydantic_private__ = {}
242
242
 
243
243
  self._model = model
@@ -325,7 +325,7 @@ class BaseAsyncPage(BasePage[_T], Generic[_T]):
325
325
  client: AsyncAPIClient,
326
326
  options: FinalRequestOptions,
327
327
  ) -> None:
328
- if PYDANTIC_V2 and getattr(self, "__pydantic_private__", None) is None:
328
+ if (not PYDANTIC_V1) and getattr(self, "__pydantic_private__", None) is None:
329
329
  self.__pydantic_private__ = {}
330
330
 
331
331
  self._model = model
anthropic/_compat.py CHANGED
@@ -12,14 +12,13 @@ from ._types import IncEx, StrBytesIntFloat
12
12
  _T = TypeVar("_T")
13
13
  _ModelT = TypeVar("_ModelT", bound=pydantic.BaseModel)
14
14
 
15
- # --------------- Pydantic v2 compatibility ---------------
15
+ # --------------- Pydantic v2, v3 compatibility ---------------
16
16
 
17
17
  # Pyright incorrectly reports some of our functions as overriding a method when they don't
18
18
  # pyright: reportIncompatibleMethodOverride=false
19
19
 
20
- PYDANTIC_V2 = pydantic.VERSION.startswith("2.")
20
+ PYDANTIC_V1 = pydantic.VERSION.startswith("1.")
21
21
 
22
- # v1 re-exports
23
22
  if TYPE_CHECKING:
24
23
 
25
24
  def parse_date(value: date | StrBytesIntFloat) -> date: # noqa: ARG001
@@ -44,90 +43,92 @@ if TYPE_CHECKING:
44
43
  ...
45
44
 
46
45
  else:
47
- if PYDANTIC_V2:
48
- from pydantic.v1.typing import (
46
+ # v1 re-exports
47
+ if PYDANTIC_V1:
48
+ from pydantic.typing import (
49
49
  get_args as get_args,
50
50
  is_union as is_union,
51
51
  get_origin as get_origin,
52
52
  is_typeddict as is_typeddict,
53
53
  is_literal_type as is_literal_type,
54
54
  )
55
- from pydantic.v1.datetime_parse import parse_date as parse_date, parse_datetime as parse_datetime
55
+ from pydantic.datetime_parse import parse_date as parse_date, parse_datetime as parse_datetime
56
56
  else:
57
- from pydantic.typing import (
57
+ from ._utils import (
58
58
  get_args as get_args,
59
59
  is_union as is_union,
60
60
  get_origin as get_origin,
61
+ parse_date as parse_date,
61
62
  is_typeddict as is_typeddict,
63
+ parse_datetime as parse_datetime,
62
64
  is_literal_type as is_literal_type,
63
65
  )
64
- from pydantic.datetime_parse import parse_date as parse_date, parse_datetime as parse_datetime
65
66
 
66
67
 
67
68
  # refactored config
68
69
  if TYPE_CHECKING:
69
70
  from pydantic import ConfigDict as ConfigDict
70
71
  else:
71
- if PYDANTIC_V2:
72
- from pydantic import ConfigDict
73
- else:
72
+ if PYDANTIC_V1:
74
73
  # TODO: provide an error message here?
75
74
  ConfigDict = None
75
+ else:
76
+ from pydantic import ConfigDict as ConfigDict
76
77
 
77
78
 
78
79
  # renamed methods / properties
79
80
  def parse_obj(model: type[_ModelT], value: object) -> _ModelT:
80
- if PYDANTIC_V2:
81
- return model.model_validate(value)
82
- else:
81
+ if PYDANTIC_V1:
83
82
  return cast(_ModelT, model.parse_obj(value)) # pyright: ignore[reportDeprecated, reportUnnecessaryCast]
83
+ else:
84
+ return model.model_validate(value)
84
85
 
85
86
 
86
87
  def field_is_required(field: FieldInfo) -> bool:
87
- if PYDANTIC_V2:
88
- return field.is_required()
89
- return field.required # type: ignore
88
+ if PYDANTIC_V1:
89
+ return field.required # type: ignore
90
+ return field.is_required()
90
91
 
91
92
 
92
93
  def field_get_default(field: FieldInfo) -> Any:
93
94
  value = field.get_default()
94
- if PYDANTIC_V2:
95
- from pydantic_core import PydanticUndefined
96
-
97
- if value == PydanticUndefined:
98
- return None
95
+ if PYDANTIC_V1:
99
96
  return value
97
+ from pydantic_core import PydanticUndefined
98
+
99
+ if value == PydanticUndefined:
100
+ return None
100
101
  return value
101
102
 
102
103
 
103
104
  def field_outer_type(field: FieldInfo) -> Any:
104
- if PYDANTIC_V2:
105
- return field.annotation
106
- return field.outer_type_ # type: ignore
105
+ if PYDANTIC_V1:
106
+ return field.outer_type_ # type: ignore
107
+ return field.annotation
107
108
 
108
109
 
109
110
  def get_model_config(model: type[pydantic.BaseModel]) -> Any:
110
- if PYDANTIC_V2:
111
- return model.model_config
112
- return model.__config__ # type: ignore
111
+ if PYDANTIC_V1:
112
+ return model.__config__ # type: ignore
113
+ return model.model_config
113
114
 
114
115
 
115
116
  def get_model_fields(model: type[pydantic.BaseModel]) -> dict[str, FieldInfo]:
116
- if PYDANTIC_V2:
117
- return model.model_fields
118
- return model.__fields__ # type: ignore
117
+ if PYDANTIC_V1:
118
+ return model.__fields__ # type: ignore
119
+ return model.model_fields
119
120
 
120
121
 
121
122
  def model_copy(model: _ModelT, *, deep: bool = False) -> _ModelT:
122
- if PYDANTIC_V2:
123
- return model.model_copy(deep=deep)
124
- return model.copy(deep=deep) # type: ignore
123
+ if PYDANTIC_V1:
124
+ return model.copy(deep=deep) # type: ignore
125
+ return model.model_copy(deep=deep)
125
126
 
126
127
 
127
128
  def model_json(model: pydantic.BaseModel, *, indent: int | None = None) -> str:
128
- if PYDANTIC_V2:
129
- return model.model_dump_json(indent=indent)
130
- return model.json(indent=indent) # type: ignore
129
+ if PYDANTIC_V1:
130
+ return model.json(indent=indent) # type: ignore
131
+ return model.model_dump_json(indent=indent)
131
132
 
132
133
 
133
134
  def model_dump(
@@ -139,14 +140,14 @@ def model_dump(
139
140
  warnings: bool = True,
140
141
  mode: Literal["json", "python"] = "python",
141
142
  ) -> dict[str, Any]:
142
- if PYDANTIC_V2 or hasattr(model, "model_dump"):
143
+ if (not PYDANTIC_V1) or hasattr(model, "model_dump"):
143
144
  return model.model_dump(
144
145
  mode=mode,
145
146
  exclude=exclude,
146
147
  exclude_unset=exclude_unset,
147
148
  exclude_defaults=exclude_defaults,
148
149
  # warnings are not supported in Pydantic v1
149
- warnings=warnings if PYDANTIC_V2 else True,
150
+ warnings=True if PYDANTIC_V1 else warnings,
150
151
  )
151
152
  return cast(
152
153
  "dict[str, Any]",
@@ -159,9 +160,9 @@ def model_dump(
159
160
 
160
161
 
161
162
  def model_parse(model: type[_ModelT], data: Any) -> _ModelT:
162
- if PYDANTIC_V2:
163
- return model.model_validate(data)
164
- return model.parse_obj(data) # pyright: ignore[reportDeprecated]
163
+ if PYDANTIC_V1:
164
+ return model.parse_obj(data) # pyright: ignore[reportDeprecated]
165
+ return model.model_validate(data)
165
166
 
166
167
 
167
168
  # generic models
@@ -170,17 +171,16 @@ if TYPE_CHECKING:
170
171
  class GenericModel(pydantic.BaseModel): ...
171
172
 
172
173
  else:
173
- if PYDANTIC_V2:
174
+ if PYDANTIC_V1:
175
+ import pydantic.generics
176
+
177
+ class GenericModel(pydantic.generics.GenericModel, pydantic.BaseModel): ...
178
+ else:
174
179
  # there no longer needs to be a distinction in v2 but
175
180
  # we still have to create our own subclass to avoid
176
181
  # inconsistent MRO ordering errors
177
182
  class GenericModel(pydantic.BaseModel): ...
178
183
 
179
- else:
180
- import pydantic.generics
181
-
182
- class GenericModel(pydantic.generics.GenericModel, pydantic.BaseModel): ...
183
-
184
184
 
185
185
  # cached properties
186
186
  if TYPE_CHECKING:
anthropic/_models.py CHANGED
@@ -50,7 +50,7 @@ from ._utils import (
50
50
  strip_annotated_type,
51
51
  )
52
52
  from ._compat import (
53
- PYDANTIC_V2,
53
+ PYDANTIC_V1,
54
54
  ConfigDict,
55
55
  GenericModel as BaseGenericModel,
56
56
  get_args,
@@ -81,11 +81,7 @@ class _ConfigProtocol(Protocol):
81
81
 
82
82
 
83
83
  class BaseModel(pydantic.BaseModel):
84
- if PYDANTIC_V2:
85
- model_config: ClassVar[ConfigDict] = ConfigDict(
86
- extra="allow", defer_build=coerce_boolean(os.environ.get("DEFER_PYDANTIC_BUILD", "true"))
87
- )
88
- else:
84
+ if PYDANTIC_V1:
89
85
 
90
86
  @property
91
87
  @override
@@ -95,6 +91,10 @@ class BaseModel(pydantic.BaseModel):
95
91
 
96
92
  class Config(pydantic.BaseConfig): # pyright: ignore[reportDeprecated]
97
93
  extra: Any = pydantic.Extra.allow # type: ignore
94
+ else:
95
+ model_config: ClassVar[ConfigDict] = ConfigDict(
96
+ extra="allow", defer_build=coerce_boolean(os.environ.get("DEFER_PYDANTIC_BUILD", "true"))
97
+ )
98
98
 
99
99
  if TYPE_CHECKING:
100
100
  _request_id: Optional[str] = None
@@ -231,25 +231,25 @@ class BaseModel(pydantic.BaseModel):
231
231
  if key not in model_fields:
232
232
  parsed = construct_type(value=value, type_=extra_field_type) if extra_field_type is not None else value
233
233
 
234
- if PYDANTIC_V2:
235
- _extra[key] = parsed
236
- else:
234
+ if PYDANTIC_V1:
237
235
  _fields_set.add(key)
238
236
  fields_values[key] = parsed
237
+ else:
238
+ _extra[key] = parsed
239
239
 
240
240
  object.__setattr__(m, "__dict__", fields_values)
241
241
 
242
- if PYDANTIC_V2:
243
- # these properties are copied from Pydantic's `model_construct()` method
244
- object.__setattr__(m, "__pydantic_private__", None)
245
- object.__setattr__(m, "__pydantic_extra__", _extra)
246
- object.__setattr__(m, "__pydantic_fields_set__", _fields_set)
247
- else:
242
+ if PYDANTIC_V1:
248
243
  # init_private_attributes() does not exist in v2
249
244
  m._init_private_attributes() # type: ignore
250
245
 
251
246
  # copied from Pydantic v1's `construct()` method
252
247
  object.__setattr__(m, "__fields_set__", _fields_set)
248
+ else:
249
+ # these properties are copied from Pydantic's `model_construct()` method
250
+ object.__setattr__(m, "__pydantic_private__", None)
251
+ object.__setattr__(m, "__pydantic_extra__", _extra)
252
+ object.__setattr__(m, "__pydantic_fields_set__", _fields_set)
253
253
 
254
254
  return m
255
255
 
@@ -259,7 +259,7 @@ class BaseModel(pydantic.BaseModel):
259
259
  # although not in practice
260
260
  model_construct = construct
261
261
 
262
- if not PYDANTIC_V2:
262
+ if PYDANTIC_V1:
263
263
  # we define aliases for some of the new pydantic v2 methods so
264
264
  # that we can just document these methods without having to specify
265
265
  # a specific pydantic version as some users may not know which
@@ -272,7 +272,7 @@ class BaseModel(pydantic.BaseModel):
272
272
  mode: Literal["json", "python"] | str = "python",
273
273
  include: IncEx | None = None,
274
274
  exclude: IncEx | None = None,
275
- by_alias: bool = False,
275
+ by_alias: bool | None = None,
276
276
  exclude_unset: bool = False,
277
277
  exclude_defaults: bool = False,
278
278
  exclude_none: bool = False,
@@ -280,6 +280,7 @@ class BaseModel(pydantic.BaseModel):
280
280
  warnings: bool | Literal["none", "warn", "error"] = True,
281
281
  context: dict[str, Any] | None = None,
282
282
  serialize_as_any: bool = False,
283
+ fallback: Callable[[Any], Any] | None = None,
283
284
  ) -> dict[str, Any]:
284
285
  """Usage docs: https://docs.pydantic.dev/2.4/concepts/serialization/#modelmodel_dump
285
286
 
@@ -311,10 +312,12 @@ class BaseModel(pydantic.BaseModel):
311
312
  raise ValueError("context is only supported in Pydantic v2")
312
313
  if serialize_as_any != False:
313
314
  raise ValueError("serialize_as_any is only supported in Pydantic v2")
315
+ if fallback is not None:
316
+ raise ValueError("fallback is only supported in Pydantic v2")
314
317
  dumped = super().dict( # pyright: ignore[reportDeprecated]
315
318
  include=include,
316
319
  exclude=exclude,
317
- by_alias=by_alias,
320
+ by_alias=by_alias if by_alias is not None else False,
318
321
  exclude_unset=exclude_unset,
319
322
  exclude_defaults=exclude_defaults,
320
323
  exclude_none=exclude_none,
@@ -329,13 +332,14 @@ class BaseModel(pydantic.BaseModel):
329
332
  indent: int | None = None,
330
333
  include: IncEx | None = None,
331
334
  exclude: IncEx | None = None,
332
- by_alias: bool = False,
335
+ by_alias: bool | None = None,
333
336
  exclude_unset: bool = False,
334
337
  exclude_defaults: bool = False,
335
338
  exclude_none: bool = False,
336
339
  round_trip: bool = False,
337
340
  warnings: bool | Literal["none", "warn", "error"] = True,
338
341
  context: dict[str, Any] | None = None,
342
+ fallback: Callable[[Any], Any] | None = None,
339
343
  serialize_as_any: bool = False,
340
344
  ) -> str:
341
345
  """Usage docs: https://docs.pydantic.dev/2.4/concepts/serialization/#modelmodel_dump_json
@@ -364,11 +368,13 @@ class BaseModel(pydantic.BaseModel):
364
368
  raise ValueError("context is only supported in Pydantic v2")
365
369
  if serialize_as_any != False:
366
370
  raise ValueError("serialize_as_any is only supported in Pydantic v2")
371
+ if fallback is not None:
372
+ raise ValueError("fallback is only supported in Pydantic v2")
367
373
  return super().json( # type: ignore[reportDeprecated]
368
374
  indent=indent,
369
375
  include=include,
370
376
  exclude=exclude,
371
- by_alias=by_alias,
377
+ by_alias=by_alias if by_alias is not None else False,
372
378
  exclude_unset=exclude_unset,
373
379
  exclude_defaults=exclude_defaults,
374
380
  exclude_none=exclude_none,
@@ -379,10 +385,10 @@ def _construct_field(value: object, field: FieldInfo, key: str) -> object:
379
385
  if value is None:
380
386
  return field_get_default(field)
381
387
 
382
- if PYDANTIC_V2:
383
- type_ = field.annotation
384
- else:
388
+ if PYDANTIC_V1:
385
389
  type_ = cast(type, field.outer_type_) # type: ignore
390
+ else:
391
+ type_ = field.annotation # type: ignore
386
392
 
387
393
  if type_ is None:
388
394
  raise RuntimeError(f"Unexpected field type is None for {key}")
@@ -391,7 +397,7 @@ def _construct_field(value: object, field: FieldInfo, key: str) -> object:
391
397
 
392
398
 
393
399
  def _get_extra_fields_type(cls: type[pydantic.BaseModel]) -> type | None:
394
- if not PYDANTIC_V2:
400
+ if PYDANTIC_V1:
395
401
  # TODO
396
402
  return None
397
403
 
@@ -644,30 +650,30 @@ def _build_discriminated_union_meta(*, union: type, meta_annotations: tuple[Any,
644
650
  for variant in get_args(union):
645
651
  variant = strip_annotated_type(variant)
646
652
  if is_basemodel_type(variant):
647
- if PYDANTIC_V2:
648
- field = _extract_field_schema_pv2(variant, discriminator_field_name)
649
- if not field:
653
+ if PYDANTIC_V1:
654
+ field_info = cast("dict[str, FieldInfo]", variant.__fields__).get(discriminator_field_name) # pyright: ignore[reportDeprecated, reportUnnecessaryCast]
655
+ if not field_info:
650
656
  continue
651
657
 
652
658
  # Note: if one variant defines an alias then they all should
653
- discriminator_alias = field.get("serialization_alias")
654
-
655
- field_schema = field["schema"]
659
+ discriminator_alias = field_info.alias
656
660
 
657
- if field_schema["type"] == "literal":
658
- for entry in cast("LiteralSchema", field_schema)["expected"]:
661
+ if (annotation := getattr(field_info, "annotation", None)) and is_literal_type(annotation):
662
+ for entry in get_args(annotation):
659
663
  if isinstance(entry, str):
660
664
  mapping[entry] = variant
661
665
  else:
662
- field_info = cast("dict[str, FieldInfo]", variant.__fields__).get(discriminator_field_name) # pyright: ignore[reportDeprecated, reportUnnecessaryCast]
663
- if not field_info:
666
+ field = _extract_field_schema_pv2(variant, discriminator_field_name)
667
+ if not field:
664
668
  continue
665
669
 
666
670
  # Note: if one variant defines an alias then they all should
667
- discriminator_alias = field_info.alias
671
+ discriminator_alias = field.get("serialization_alias")
668
672
 
669
- if (annotation := getattr(field_info, "annotation", None)) and is_literal_type(annotation):
670
- for entry in get_args(annotation):
673
+ field_schema = field["schema"]
674
+
675
+ if field_schema["type"] == "literal":
676
+ for entry in cast("LiteralSchema", field_schema)["expected"]:
671
677
  if isinstance(entry, str):
672
678
  mapping[entry] = variant
673
679
 
@@ -726,7 +732,7 @@ def add_request_id(obj: BaseModel, request_id: str | None) -> None:
726
732
  # in Pydantic v1, using setattr like we do above causes the attribute
727
733
  # to be included when serializing the model which we don't want in this
728
734
  # case so we need to explicitly exclude it
729
- if not PYDANTIC_V2:
735
+ if PYDANTIC_V1:
730
736
  try:
731
737
  exclude_fields = obj.__exclude_fields__ # type: ignore
732
738
  except AttributeError:
@@ -745,7 +751,7 @@ else:
745
751
  pass
746
752
 
747
753
 
748
- if PYDANTIC_V2:
754
+ if not PYDANTIC_V1:
749
755
  from pydantic import TypeAdapter as _TypeAdapter
750
756
 
751
757
  _CachedTypeAdapter = cast("TypeAdapter[object]", lru_cache(maxsize=None)(_TypeAdapter))
@@ -780,6 +786,9 @@ elif not TYPE_CHECKING: # TODO: condition is weird
780
786
  def _create_pydantic_model(type_: _T) -> Type[RootModel[_T]]:
781
787
  return RootModel[type_] # type: ignore
782
788
 
789
+ def TypeAdapter(*_args: Any, **_kwargs: Any) -> Any:
790
+ raise RuntimeError("attempted to use TypeAdapter in pydantic v1")
791
+
783
792
 
784
793
  class FinalRequestOptionsInput(TypedDict, total=False):
785
794
  method: Required[str]
@@ -813,12 +822,12 @@ class FinalRequestOptions(pydantic.BaseModel):
813
822
  json_data: Union[Body, None] = None
814
823
  extra_json: Union[AnyMapping, None] = None
815
824
 
816
- if PYDANTIC_V2:
817
- model_config: ClassVar[ConfigDict] = ConfigDict(arbitrary_types_allowed=True)
818
- else:
825
+ if PYDANTIC_V1:
819
826
 
820
827
  class Config(pydantic.BaseConfig): # pyright: ignore[reportDeprecated]
821
828
  arbitrary_types_allowed: bool = True
829
+ else:
830
+ model_config: ClassVar[ConfigDict] = ConfigDict(arbitrary_types_allowed=True)
822
831
 
823
832
  def get_max_retries(self, max_retries: int) -> int:
824
833
  if isinstance(self.max_retries, NotGiven):
@@ -851,9 +860,9 @@ class FinalRequestOptions(pydantic.BaseModel):
851
860
  key: strip_not_given(value)
852
861
  for key, value in values.items()
853
862
  }
854
- if PYDANTIC_V2:
855
- return super().model_construct(_fields_set, **kwargs)
856
- return cast(FinalRequestOptions, super().construct(_fields_set, **kwargs)) # pyright: ignore[reportDeprecated]
863
+ if PYDANTIC_V1:
864
+ return cast(FinalRequestOptions, super().construct(_fields_set, **kwargs)) # pyright: ignore[reportDeprecated]
865
+ return super().model_construct(_fields_set, **kwargs)
857
866
 
858
867
  if not TYPE_CHECKING:
859
868
  # type checkers incorrectly complain about this assignment
@@ -10,7 +10,6 @@ from ._utils import (
10
10
  lru_cache as lru_cache,
11
11
  is_mapping as is_mapping,
12
12
  is_tuple_t as is_tuple_t,
13
- parse_date as parse_date,
14
13
  is_iterable as is_iterable,
15
14
  is_sequence as is_sequence,
16
15
  coerce_float as coerce_float,
@@ -23,7 +22,6 @@ from ._utils import (
23
22
  coerce_boolean as coerce_boolean,
24
23
  coerce_integer as coerce_integer,
25
24
  file_from_path as file_from_path,
26
- parse_datetime as parse_datetime,
27
25
  strip_not_given as strip_not_given,
28
26
  deepcopy_minimal as deepcopy_minimal,
29
27
  get_async_library as get_async_library,
@@ -32,6 +30,13 @@ from ._utils import (
32
30
  maybe_coerce_boolean as maybe_coerce_boolean,
33
31
  maybe_coerce_integer as maybe_coerce_integer,
34
32
  )
33
+ from ._compat import (
34
+ get_args as get_args,
35
+ is_union as is_union,
36
+ get_origin as get_origin,
37
+ is_typeddict as is_typeddict,
38
+ is_literal_type as is_literal_type,
39
+ )
35
40
  from ._typing import (
36
41
  is_list_type as is_list_type,
37
42
  is_union_type as is_union_type,
@@ -56,3 +61,4 @@ from ._reflection import (
56
61
  function_has_argument as function_has_argument,
57
62
  assert_signatures_in_sync as assert_signatures_in_sync,
58
63
  )
64
+ from ._datetime_parse import parse_date as parse_date, parse_datetime as parse_datetime
@@ -0,0 +1,45 @@
1
+ from __future__ import annotations
2
+
3
+ import sys
4
+ import typing_extensions
5
+ from typing import Any, Type, Union, Literal, Optional
6
+ from datetime import date, datetime
7
+ from typing_extensions import get_args as _get_args, get_origin as _get_origin
8
+
9
+ from .._types import StrBytesIntFloat
10
+ from ._datetime_parse import parse_date as _parse_date, parse_datetime as _parse_datetime
11
+
12
+ _LITERAL_TYPES = {Literal, typing_extensions.Literal}
13
+
14
+
15
+ def get_args(tp: type[Any]) -> tuple[Any, ...]:
16
+ return _get_args(tp)
17
+
18
+
19
+ def get_origin(tp: type[Any]) -> type[Any] | None:
20
+ return _get_origin(tp)
21
+
22
+
23
+ def is_union(tp: Optional[Type[Any]]) -> bool:
24
+ if sys.version_info < (3, 10):
25
+ return tp is Union # type: ignore[comparison-overlap]
26
+ else:
27
+ import types
28
+
29
+ return tp is Union or tp is types.UnionType
30
+
31
+
32
+ def is_typeddict(tp: Type[Any]) -> bool:
33
+ return typing_extensions.is_typeddict(tp)
34
+
35
+
36
+ def is_literal_type(tp: Type[Any]) -> bool:
37
+ return get_origin(tp) in _LITERAL_TYPES
38
+
39
+
40
+ def parse_date(value: Union[date, StrBytesIntFloat]) -> date:
41
+ return _parse_date(value)
42
+
43
+
44
+ def parse_datetime(value: Union[datetime, StrBytesIntFloat]) -> datetime:
45
+ return _parse_datetime(value)