google-genai 1.53.0__py3-none-any.whl → 1.55.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 (324) hide show
  1. google/genai/__init__.py +1 -0
  2. google/genai/_api_client.py +6 -6
  3. google/genai/_interactions/__init__.py +117 -0
  4. google/genai/_interactions/_base_client.py +2019 -0
  5. google/genai/_interactions/_client.py +511 -0
  6. google/genai/_interactions/_compat.py +234 -0
  7. google/genai/_interactions/_constants.py +29 -0
  8. google/genai/_interactions/_exceptions.py +122 -0
  9. google/genai/_interactions/_files.py +139 -0
  10. google/genai/_interactions/_models.py +873 -0
  11. google/genai/_interactions/_qs.py +165 -0
  12. google/genai/_interactions/_resource.py +58 -0
  13. google/genai/_interactions/_response.py +847 -0
  14. google/genai/_interactions/_streaming.py +354 -0
  15. google/genai/_interactions/_types.py +276 -0
  16. google/genai/_interactions/_utils/__init__.py +79 -0
  17. google/genai/_interactions/_utils/_compat.py +61 -0
  18. google/genai/_interactions/_utils/_datetime_parse.py +151 -0
  19. google/genai/_interactions/_utils/_logs.py +40 -0
  20. google/genai/_interactions/_utils/_proxy.py +80 -0
  21. google/genai/_interactions/_utils/_reflection.py +57 -0
  22. google/genai/_interactions/_utils/_resources_proxy.py +39 -0
  23. google/genai/_interactions/_utils/_streams.py +27 -0
  24. google/genai/_interactions/_utils/_sync.py +73 -0
  25. google/genai/_interactions/_utils/_transform.py +472 -0
  26. google/genai/_interactions/_utils/_typing.py +172 -0
  27. google/genai/_interactions/_utils/_utils.py +437 -0
  28. google/genai/_interactions/_version.py +18 -0
  29. google/genai/_interactions/resources/__init__.py +34 -0
  30. google/genai/_interactions/resources/interactions.py +1350 -0
  31. google/genai/_interactions/types/__init__.py +107 -0
  32. google/genai/_interactions/types/allowed_tools.py +33 -0
  33. google/genai/_interactions/types/allowed_tools_param.py +35 -0
  34. google/genai/_interactions/types/annotation.py +42 -0
  35. google/genai/_interactions/types/annotation_param.py +42 -0
  36. google/genai/_interactions/types/audio_content.py +38 -0
  37. google/genai/_interactions/types/audio_content_param.py +45 -0
  38. google/genai/_interactions/types/audio_mime_type.py +25 -0
  39. google/genai/_interactions/types/audio_mime_type_param.py +27 -0
  40. google/genai/_interactions/types/code_execution_call_arguments.py +33 -0
  41. google/genai/_interactions/types/code_execution_call_arguments_param.py +32 -0
  42. google/genai/_interactions/types/code_execution_call_content.py +37 -0
  43. google/genai/_interactions/types/code_execution_call_content_param.py +37 -0
  44. google/genai/_interactions/types/code_execution_result_content.py +42 -0
  45. google/genai/_interactions/types/code_execution_result_content_param.py +41 -0
  46. google/genai/_interactions/types/content_delta.py +358 -0
  47. google/genai/_interactions/types/content_start.py +79 -0
  48. google/genai/_interactions/types/content_stop.py +35 -0
  49. google/genai/_interactions/types/deep_research_agent_config.py +33 -0
  50. google/genai/_interactions/types/deep_research_agent_config_param.py +32 -0
  51. google/genai/_interactions/types/document_content.py +36 -0
  52. google/genai/_interactions/types/document_content_param.py +43 -0
  53. google/genai/_interactions/types/dynamic_agent_config.py +44 -0
  54. google/genai/_interactions/types/dynamic_agent_config_param.py +33 -0
  55. google/genai/_interactions/types/error_event.py +46 -0
  56. google/genai/_interactions/types/file_search_result_content.py +46 -0
  57. google/genai/_interactions/types/file_search_result_content_param.py +46 -0
  58. google/genai/_interactions/types/function.py +38 -0
  59. google/genai/_interactions/types/function_call_content.py +39 -0
  60. google/genai/_interactions/types/function_call_content_param.py +39 -0
  61. google/genai/_interactions/types/function_param.py +37 -0
  62. google/genai/_interactions/types/function_result_content.py +52 -0
  63. google/genai/_interactions/types/function_result_content_param.py +54 -0
  64. google/genai/_interactions/types/generation_config.py +57 -0
  65. google/genai/_interactions/types/generation_config_param.py +59 -0
  66. google/genai/_interactions/types/google_search_call_arguments.py +29 -0
  67. google/genai/_interactions/types/google_search_call_arguments_param.py +31 -0
  68. google/genai/_interactions/types/google_search_call_content.py +37 -0
  69. google/genai/_interactions/types/google_search_call_content_param.py +37 -0
  70. google/genai/_interactions/types/google_search_result.py +35 -0
  71. google/genai/_interactions/types/google_search_result_content.py +43 -0
  72. google/genai/_interactions/types/google_search_result_content_param.py +44 -0
  73. google/genai/_interactions/types/google_search_result_param.py +35 -0
  74. google/genai/_interactions/types/image_content.py +41 -0
  75. google/genai/_interactions/types/image_content_param.py +48 -0
  76. google/genai/_interactions/types/image_mime_type.py +23 -0
  77. google/genai/_interactions/types/image_mime_type_param.py +25 -0
  78. google/genai/_interactions/types/interaction.py +165 -0
  79. google/genai/_interactions/types/interaction_create_params.py +212 -0
  80. google/genai/_interactions/types/interaction_event.py +37 -0
  81. google/genai/_interactions/types/interaction_get_params.py +46 -0
  82. google/genai/_interactions/types/interaction_sse_event.py +32 -0
  83. google/genai/_interactions/types/interaction_status_update.py +37 -0
  84. google/genai/_interactions/types/mcp_server_tool_call_content.py +42 -0
  85. google/genai/_interactions/types/mcp_server_tool_call_content_param.py +42 -0
  86. google/genai/_interactions/types/mcp_server_tool_result_content.py +52 -0
  87. google/genai/_interactions/types/mcp_server_tool_result_content_param.py +54 -0
  88. google/genai/_interactions/types/model.py +36 -0
  89. google/genai/_interactions/types/model_param.py +38 -0
  90. google/genai/_interactions/types/speech_config.py +35 -0
  91. google/genai/_interactions/types/speech_config_param.py +35 -0
  92. google/genai/_interactions/types/text_content.py +37 -0
  93. google/genai/_interactions/types/text_content_param.py +38 -0
  94. google/genai/_interactions/types/thinking_level.py +22 -0
  95. google/genai/_interactions/types/thought_content.py +41 -0
  96. google/genai/_interactions/types/thought_content_param.py +47 -0
  97. google/genai/_interactions/types/tool.py +100 -0
  98. google/genai/_interactions/types/tool_choice.py +26 -0
  99. google/genai/_interactions/types/tool_choice_config.py +28 -0
  100. google/genai/_interactions/types/tool_choice_config_param.py +29 -0
  101. google/genai/_interactions/types/tool_choice_param.py +28 -0
  102. google/genai/_interactions/types/tool_choice_type.py +22 -0
  103. google/genai/_interactions/types/tool_param.py +97 -0
  104. google/genai/_interactions/types/turn.py +76 -0
  105. google/genai/_interactions/types/turn_param.py +73 -0
  106. google/genai/_interactions/types/url_context_call_arguments.py +29 -0
  107. google/genai/_interactions/types/url_context_call_arguments_param.py +31 -0
  108. google/genai/_interactions/types/url_context_call_content.py +37 -0
  109. google/genai/_interactions/types/url_context_call_content_param.py +37 -0
  110. google/genai/_interactions/types/url_context_result.py +33 -0
  111. google/genai/_interactions/types/url_context_result_content.py +43 -0
  112. google/genai/_interactions/types/url_context_result_content_param.py +44 -0
  113. google/genai/_interactions/types/url_context_result_param.py +32 -0
  114. google/genai/_interactions/types/usage.py +106 -0
  115. google/genai/_interactions/types/usage_param.py +106 -0
  116. google/genai/_interactions/types/video_content.py +41 -0
  117. google/genai/_interactions/types/video_content_param.py +48 -0
  118. google/genai/_interactions/types/video_mime_type.py +36 -0
  119. google/genai/_interactions/types/video_mime_type_param.py +38 -0
  120. google/genai/_live_converters.py +34 -3
  121. google/genai/_tokens_converters.py +5 -0
  122. google/genai/batches.py +62 -55
  123. google/genai/client.py +223 -0
  124. google/genai/errors.py +16 -1
  125. google/genai/file_search_stores.py +60 -60
  126. google/genai/files.py +56 -56
  127. google/genai/interactions.py +17 -0
  128. google/genai/live.py +4 -3
  129. google/genai/models.py +15 -3
  130. google/genai/tests/__init__.py +21 -0
  131. google/genai/tests/afc/__init__.py +21 -0
  132. google/genai/tests/afc/test_convert_if_exist_pydantic_model.py +309 -0
  133. google/genai/tests/afc/test_convert_number_values_for_function_call_args.py +63 -0
  134. google/genai/tests/afc/test_find_afc_incompatible_tool_indexes.py +240 -0
  135. google/genai/tests/afc/test_generate_content_stream_afc.py +530 -0
  136. google/genai/tests/afc/test_generate_content_stream_afc_thoughts.py +77 -0
  137. google/genai/tests/afc/test_get_function_map.py +176 -0
  138. google/genai/tests/afc/test_get_function_response_parts.py +277 -0
  139. google/genai/tests/afc/test_get_max_remote_calls_for_afc.py +130 -0
  140. google/genai/tests/afc/test_invoke_function_from_dict_args.py +241 -0
  141. google/genai/tests/afc/test_raise_error_for_afc_incompatible_config.py +159 -0
  142. google/genai/tests/afc/test_should_append_afc_history.py +53 -0
  143. google/genai/tests/afc/test_should_disable_afc.py +214 -0
  144. google/genai/tests/batches/__init__.py +17 -0
  145. google/genai/tests/batches/test_cancel.py +77 -0
  146. google/genai/tests/batches/test_create.py +78 -0
  147. google/genai/tests/batches/test_create_with_bigquery.py +113 -0
  148. google/genai/tests/batches/test_create_with_file.py +82 -0
  149. google/genai/tests/batches/test_create_with_gcs.py +125 -0
  150. google/genai/tests/batches/test_create_with_inlined_requests.py +255 -0
  151. google/genai/tests/batches/test_delete.py +86 -0
  152. google/genai/tests/batches/test_embedding.py +157 -0
  153. google/genai/tests/batches/test_get.py +78 -0
  154. google/genai/tests/batches/test_list.py +79 -0
  155. google/genai/tests/caches/__init__.py +17 -0
  156. google/genai/tests/caches/constants.py +29 -0
  157. google/genai/tests/caches/test_create.py +210 -0
  158. google/genai/tests/caches/test_create_custom_url.py +105 -0
  159. google/genai/tests/caches/test_delete.py +54 -0
  160. google/genai/tests/caches/test_delete_custom_url.py +52 -0
  161. google/genai/tests/caches/test_get.py +94 -0
  162. google/genai/tests/caches/test_get_custom_url.py +52 -0
  163. google/genai/tests/caches/test_list.py +68 -0
  164. google/genai/tests/caches/test_update.py +70 -0
  165. google/genai/tests/caches/test_update_custom_url.py +58 -0
  166. google/genai/tests/chats/__init__.py +1 -0
  167. google/genai/tests/chats/test_get_history.py +597 -0
  168. google/genai/tests/chats/test_send_message.py +844 -0
  169. google/genai/tests/chats/test_validate_response.py +90 -0
  170. google/genai/tests/client/__init__.py +17 -0
  171. google/genai/tests/client/test_async_stream.py +427 -0
  172. google/genai/tests/client/test_client_close.py +197 -0
  173. google/genai/tests/client/test_client_initialization.py +1687 -0
  174. google/genai/tests/client/test_client_requests.py +355 -0
  175. google/genai/tests/client/test_custom_client.py +77 -0
  176. google/genai/tests/client/test_http_options.py +178 -0
  177. google/genai/tests/client/test_replay_client_equality.py +168 -0
  178. google/genai/tests/client/test_retries.py +846 -0
  179. google/genai/tests/client/test_upload_errors.py +136 -0
  180. google/genai/tests/common/__init__.py +17 -0
  181. google/genai/tests/common/test_common.py +954 -0
  182. google/genai/tests/conftest.py +162 -0
  183. google/genai/tests/documents/__init__.py +17 -0
  184. google/genai/tests/documents/test_delete.py +51 -0
  185. google/genai/tests/documents/test_get.py +85 -0
  186. google/genai/tests/documents/test_list.py +72 -0
  187. google/genai/tests/errors/__init__.py +1 -0
  188. google/genai/tests/errors/test_api_error.py +417 -0
  189. google/genai/tests/file_search_stores/__init__.py +17 -0
  190. google/genai/tests/file_search_stores/test_create.py +66 -0
  191. google/genai/tests/file_search_stores/test_delete.py +64 -0
  192. google/genai/tests/file_search_stores/test_get.py +94 -0
  193. google/genai/tests/file_search_stores/test_import_file.py +112 -0
  194. google/genai/tests/file_search_stores/test_list.py +57 -0
  195. google/genai/tests/file_search_stores/test_upload_to_file_search_store.py +141 -0
  196. google/genai/tests/files/__init__.py +17 -0
  197. google/genai/tests/files/test_delete.py +46 -0
  198. google/genai/tests/files/test_download.py +85 -0
  199. google/genai/tests/files/test_get.py +46 -0
  200. google/genai/tests/files/test_list.py +72 -0
  201. google/genai/tests/files/test_upload.py +255 -0
  202. google/genai/tests/imports/test_no_optional_imports.py +28 -0
  203. google/genai/tests/interactions/__init__.py +0 -0
  204. google/genai/tests/interactions/test_integration.py +80 -0
  205. google/genai/tests/live/__init__.py +16 -0
  206. google/genai/tests/live/test_live.py +2177 -0
  207. google/genai/tests/live/test_live_music.py +362 -0
  208. google/genai/tests/live/test_live_response.py +163 -0
  209. google/genai/tests/live/test_send_client_content.py +147 -0
  210. google/genai/tests/live/test_send_realtime_input.py +268 -0
  211. google/genai/tests/live/test_send_tool_response.py +222 -0
  212. google/genai/tests/local_tokenizer/__init__.py +17 -0
  213. google/genai/tests/local_tokenizer/test_local_tokenizer.py +343 -0
  214. google/genai/tests/local_tokenizer/test_local_tokenizer_loader.py +235 -0
  215. google/genai/tests/mcp/__init__.py +17 -0
  216. google/genai/tests/mcp/test_has_mcp_tool_usage.py +89 -0
  217. google/genai/tests/mcp/test_mcp_to_gemini_tools.py +191 -0
  218. google/genai/tests/mcp/test_parse_config_for_mcp_sessions.py +201 -0
  219. google/genai/tests/mcp/test_parse_config_for_mcp_usage.py +130 -0
  220. google/genai/tests/mcp/test_set_mcp_usage_header.py +72 -0
  221. google/genai/tests/models/__init__.py +17 -0
  222. google/genai/tests/models/constants.py +8 -0
  223. google/genai/tests/models/test_compute_tokens.py +120 -0
  224. google/genai/tests/models/test_count_tokens.py +159 -0
  225. google/genai/tests/models/test_delete.py +107 -0
  226. google/genai/tests/models/test_edit_image.py +264 -0
  227. google/genai/tests/models/test_embed_content.py +94 -0
  228. google/genai/tests/models/test_function_call_streaming.py +442 -0
  229. google/genai/tests/models/test_generate_content.py +2502 -0
  230. google/genai/tests/models/test_generate_content_cached_content.py +132 -0
  231. google/genai/tests/models/test_generate_content_config_zero_value.py +103 -0
  232. google/genai/tests/models/test_generate_content_from_apikey.py +44 -0
  233. google/genai/tests/models/test_generate_content_http_options.py +40 -0
  234. google/genai/tests/models/test_generate_content_image_generation.py +143 -0
  235. google/genai/tests/models/test_generate_content_mcp.py +343 -0
  236. google/genai/tests/models/test_generate_content_media_resolution.py +97 -0
  237. google/genai/tests/models/test_generate_content_model.py +139 -0
  238. google/genai/tests/models/test_generate_content_part.py +821 -0
  239. google/genai/tests/models/test_generate_content_thought.py +76 -0
  240. google/genai/tests/models/test_generate_content_tools.py +1761 -0
  241. google/genai/tests/models/test_generate_images.py +191 -0
  242. google/genai/tests/models/test_generate_videos.py +759 -0
  243. google/genai/tests/models/test_get.py +104 -0
  244. google/genai/tests/models/test_list.py +233 -0
  245. google/genai/tests/models/test_recontext_image.py +189 -0
  246. google/genai/tests/models/test_segment_image.py +148 -0
  247. google/genai/tests/models/test_update.py +95 -0
  248. google/genai/tests/models/test_upscale_image.py +157 -0
  249. google/genai/tests/operations/__init__.py +17 -0
  250. google/genai/tests/operations/test_get.py +38 -0
  251. google/genai/tests/public_samples/__init__.py +17 -0
  252. google/genai/tests/public_samples/test_gemini_text_only.py +34 -0
  253. google/genai/tests/pytest_helper.py +229 -0
  254. google/genai/tests/shared/__init__.py +16 -0
  255. google/genai/tests/shared/batches/__init__.py +14 -0
  256. google/genai/tests/shared/batches/test_create_delete.py +57 -0
  257. google/genai/tests/shared/batches/test_create_get_cancel.py +56 -0
  258. google/genai/tests/shared/batches/test_list.py +40 -0
  259. google/genai/tests/shared/caches/__init__.py +14 -0
  260. google/genai/tests/shared/caches/test_create_get_delete.py +67 -0
  261. google/genai/tests/shared/caches/test_create_update_get.py +71 -0
  262. google/genai/tests/shared/caches/test_list.py +40 -0
  263. google/genai/tests/shared/chats/__init__.py +14 -0
  264. google/genai/tests/shared/chats/test_send_message.py +48 -0
  265. google/genai/tests/shared/chats/test_send_message_stream.py +50 -0
  266. google/genai/tests/shared/files/__init__.py +14 -0
  267. google/genai/tests/shared/files/test_list.py +41 -0
  268. google/genai/tests/shared/files/test_upload_get_delete.py +54 -0
  269. google/genai/tests/shared/models/__init__.py +14 -0
  270. google/genai/tests/shared/models/test_compute_tokens.py +41 -0
  271. google/genai/tests/shared/models/test_count_tokens.py +40 -0
  272. google/genai/tests/shared/models/test_edit_image.py +67 -0
  273. google/genai/tests/shared/models/test_embed.py +40 -0
  274. google/genai/tests/shared/models/test_generate_content.py +39 -0
  275. google/genai/tests/shared/models/test_generate_content_stream.py +54 -0
  276. google/genai/tests/shared/models/test_generate_images.py +40 -0
  277. google/genai/tests/shared/models/test_generate_videos.py +38 -0
  278. google/genai/tests/shared/models/test_list.py +37 -0
  279. google/genai/tests/shared/models/test_recontext_image.py +55 -0
  280. google/genai/tests/shared/models/test_segment_image.py +52 -0
  281. google/genai/tests/shared/models/test_upscale_image.py +52 -0
  282. google/genai/tests/shared/tunings/__init__.py +16 -0
  283. google/genai/tests/shared/tunings/test_create.py +46 -0
  284. google/genai/tests/shared/tunings/test_create_get_cancel.py +56 -0
  285. google/genai/tests/shared/tunings/test_list.py +39 -0
  286. google/genai/tests/tokens/__init__.py +16 -0
  287. google/genai/tests/tokens/test_create.py +154 -0
  288. google/genai/tests/transformers/__init__.py +17 -0
  289. google/genai/tests/transformers/test_blobs.py +71 -0
  290. google/genai/tests/transformers/test_bytes.py +15 -0
  291. google/genai/tests/transformers/test_duck_type.py +96 -0
  292. google/genai/tests/transformers/test_function_responses.py +72 -0
  293. google/genai/tests/transformers/test_schema.py +653 -0
  294. google/genai/tests/transformers/test_t_batch.py +286 -0
  295. google/genai/tests/transformers/test_t_content.py +160 -0
  296. google/genai/tests/transformers/test_t_contents.py +398 -0
  297. google/genai/tests/transformers/test_t_part.py +85 -0
  298. google/genai/tests/transformers/test_t_parts.py +87 -0
  299. google/genai/tests/transformers/test_t_tool.py +157 -0
  300. google/genai/tests/transformers/test_t_tools.py +195 -0
  301. google/genai/tests/tunings/__init__.py +16 -0
  302. google/genai/tests/tunings/test_cancel.py +39 -0
  303. google/genai/tests/tunings/test_end_to_end.py +106 -0
  304. google/genai/tests/tunings/test_get.py +67 -0
  305. google/genai/tests/tunings/test_list.py +75 -0
  306. google/genai/tests/tunings/test_tune.py +268 -0
  307. google/genai/tests/types/__init__.py +16 -0
  308. google/genai/tests/types/test_bytes_internal.py +271 -0
  309. google/genai/tests/types/test_bytes_type.py +152 -0
  310. google/genai/tests/types/test_future.py +101 -0
  311. google/genai/tests/types/test_optional_types.py +36 -0
  312. google/genai/tests/types/test_part_type.py +616 -0
  313. google/genai/tests/types/test_schema_from_json_schema.py +417 -0
  314. google/genai/tests/types/test_schema_json_schema.py +468 -0
  315. google/genai/tests/types/test_types.py +2903 -0
  316. google/genai/tunings.py +57 -57
  317. google/genai/types.py +229 -121
  318. google/genai/version.py +1 -1
  319. {google_genai-1.53.0.dist-info → google_genai-1.55.0.dist-info}/METADATA +4 -2
  320. google_genai-1.55.0.dist-info/RECORD +345 -0
  321. google_genai-1.53.0.dist-info/RECORD +0 -41
  322. {google_genai-1.53.0.dist-info → google_genai-1.55.0.dist-info}/WHEEL +0 -0
  323. {google_genai-1.53.0.dist-info → google_genai-1.55.0.dist-info}/licenses/LICENSE +0 -0
  324. {google_genai-1.53.0.dist-info → google_genai-1.55.0.dist-info}/top_level.txt +0 -0
@@ -0,0 +1,472 @@
1
+ # Copyright 2025 Google LLC
2
+ #
3
+ # Licensed under the Apache License, Version 2.0 (the "License");
4
+ # you may not use this file except in compliance with the License.
5
+ # You may obtain a copy of the License at
6
+ #
7
+ # http://www.apache.org/licenses/LICENSE-2.0
8
+ #
9
+ # Unless required by applicable law or agreed to in writing, software
10
+ # distributed under the License is distributed on an "AS IS" BASIS,
11
+ # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12
+ # See the License for the specific language governing permissions and
13
+ # limitations under the License.
14
+ #
15
+
16
+ from __future__ import annotations
17
+
18
+ import io
19
+ import base64
20
+ import pathlib
21
+ from typing import Any, Mapping, TypeVar, cast
22
+ from datetime import date, datetime
23
+ from typing_extensions import Literal, get_args, override, get_type_hints as _get_type_hints
24
+
25
+ import anyio
26
+ import pydantic
27
+
28
+ from ._utils import (
29
+ is_list,
30
+ is_given,
31
+ lru_cache,
32
+ is_mapping,
33
+ is_iterable,
34
+ is_sequence,
35
+ )
36
+ from .._files import is_base64_file_input
37
+ from ._compat import get_origin, is_typeddict
38
+ from ._typing import (
39
+ is_list_type,
40
+ is_union_type,
41
+ extract_type_arg,
42
+ is_iterable_type,
43
+ is_required_type,
44
+ is_sequence_type,
45
+ is_annotated_type,
46
+ strip_annotated_type,
47
+ )
48
+
49
+ _T = TypeVar("_T")
50
+
51
+
52
+ # TODO: support for drilling globals() and locals()
53
+ # TODO: ensure works correctly with forward references in all cases
54
+
55
+
56
+ PropertyFormat = Literal["iso8601", "base64", "custom"]
57
+
58
+
59
+ class PropertyInfo:
60
+ """Metadata class to be used in Annotated types to provide information about a given type.
61
+
62
+ For example:
63
+
64
+ class MyParams(TypedDict):
65
+ account_holder_name: Annotated[str, PropertyInfo(alias='accountHolderName')]
66
+
67
+ This means that {'account_holder_name': 'Robert'} will be transformed to {'accountHolderName': 'Robert'} before being sent to the API.
68
+ """
69
+
70
+ alias: str | None
71
+ format: PropertyFormat | None
72
+ format_template: str | None
73
+ discriminator: str | None
74
+
75
+ def __init__(
76
+ self,
77
+ *,
78
+ alias: str | None = None,
79
+ format: PropertyFormat | None = None,
80
+ format_template: str | None = None,
81
+ discriminator: str | None = None,
82
+ ) -> None:
83
+ self.alias = alias
84
+ self.format = format
85
+ self.format_template = format_template
86
+ self.discriminator = discriminator
87
+
88
+ @override
89
+ def __repr__(self) -> str:
90
+ return f"{self.__class__.__name__}(alias='{self.alias}', format={self.format}, format_template='{self.format_template}', discriminator='{self.discriminator}')"
91
+
92
+
93
+ def maybe_transform(
94
+ data: object,
95
+ expected_type: object,
96
+ ) -> Any | None:
97
+ """Wrapper over `transform()` that allows `None` to be passed.
98
+
99
+ See `transform()` for more details.
100
+ """
101
+ if data is None:
102
+ return None
103
+ return transform(data, expected_type)
104
+
105
+
106
+ # Wrapper over _transform_recursive providing fake types
107
+ def transform(
108
+ data: _T,
109
+ expected_type: object,
110
+ ) -> _T:
111
+ """Transform dictionaries based off of type information from the given type, for example:
112
+
113
+ ```py
114
+ class Params(TypedDict, total=False):
115
+ card_id: Required[Annotated[str, PropertyInfo(alias="cardID")]]
116
+
117
+
118
+ transformed = transform({"card_id": "<my card ID>"}, Params)
119
+ # {'cardID': '<my card ID>'}
120
+ ```
121
+
122
+ Any keys / data that does not have type information given will be included as is.
123
+
124
+ It should be noted that the transformations that this function does are not represented in the type system.
125
+ """
126
+ transformed = _transform_recursive(data, annotation=cast(type, expected_type))
127
+ return cast(_T, transformed)
128
+
129
+
130
+ @lru_cache(maxsize=8096)
131
+ def _get_annotated_type(type_: type) -> type | None:
132
+ """If the given type is an `Annotated` type then it is returned, if not `None` is returned.
133
+
134
+ This also unwraps the type when applicable, e.g. `Required[Annotated[T, ...]]`
135
+ """
136
+ if is_required_type(type_):
137
+ # Unwrap `Required[Annotated[T, ...]]` to `Annotated[T, ...]`
138
+ type_ = get_args(type_)[0]
139
+
140
+ if is_annotated_type(type_):
141
+ return type_
142
+
143
+ return None
144
+
145
+
146
+ def _maybe_transform_key(key: str, type_: type) -> str:
147
+ """Transform the given `data` based on the annotations provided in `type_`.
148
+
149
+ Note: this function only looks at `Annotated` types that contain `PropertyInfo` metadata.
150
+ """
151
+ annotated_type = _get_annotated_type(type_)
152
+ if annotated_type is None:
153
+ # no `Annotated` definition for this type, no transformation needed
154
+ return key
155
+
156
+ # ignore the first argument as it is the actual type
157
+ annotations = get_args(annotated_type)[1:]
158
+ for annotation in annotations:
159
+ if isinstance(annotation, PropertyInfo) and annotation.alias is not None:
160
+ return annotation.alias
161
+
162
+ return key
163
+
164
+
165
+ def _no_transform_needed(annotation: type) -> bool:
166
+ return annotation == float or annotation == int
167
+
168
+
169
+ def _transform_recursive(
170
+ data: object,
171
+ *,
172
+ annotation: type,
173
+ inner_type: type | None = None,
174
+ ) -> object:
175
+ """Transform the given data against the expected type.
176
+
177
+ Args:
178
+ annotation: The direct type annotation given to the particular piece of data.
179
+ This may or may not be wrapped in metadata types, e.g. `Required[T]`, `Annotated[T, ...]` etc
180
+
181
+ inner_type: If applicable, this is the "inside" type. This is useful in certain cases where the outside type
182
+ is a container type such as `List[T]`. In that case `inner_type` should be set to `T` so that each entry in
183
+ the list can be transformed using the metadata from the container type.
184
+
185
+ Defaults to the same value as the `annotation` argument.
186
+ """
187
+ from .._compat import model_dump
188
+
189
+ if inner_type is None:
190
+ inner_type = annotation
191
+
192
+ stripped_type = strip_annotated_type(inner_type)
193
+ origin = get_origin(stripped_type) or stripped_type
194
+ if is_typeddict(stripped_type) and is_mapping(data):
195
+ return _transform_typeddict(data, stripped_type)
196
+
197
+ if origin == dict and is_mapping(data):
198
+ items_type = get_args(stripped_type)[1]
199
+ return {key: _transform_recursive(value, annotation=items_type) for key, value in data.items()}
200
+
201
+ if (
202
+ # List[T]
203
+ (is_list_type(stripped_type) and is_list(data))
204
+ # Iterable[T]
205
+ or (is_iterable_type(stripped_type) and is_iterable(data) and not isinstance(data, str))
206
+ # Sequence[T]
207
+ or (is_sequence_type(stripped_type) and is_sequence(data) and not isinstance(data, str))
208
+ ):
209
+ # dicts are technically iterable, but it is an iterable on the keys of the dict and is not usually
210
+ # intended as an iterable, so we don't transform it.
211
+ if isinstance(data, dict):
212
+ return cast(object, data)
213
+
214
+ inner_type = extract_type_arg(stripped_type, 0)
215
+ if _no_transform_needed(inner_type):
216
+ # for some types there is no need to transform anything, so we can get a small
217
+ # perf boost from skipping that work.
218
+ #
219
+ # but we still need to convert to a list to ensure the data is json-serializable
220
+ if is_list(data):
221
+ return data
222
+ return list(data)
223
+
224
+ return [_transform_recursive(d, annotation=annotation, inner_type=inner_type) for d in data]
225
+
226
+ if is_union_type(stripped_type):
227
+ # For union types we run the transformation against all subtypes to ensure that everything is transformed.
228
+ #
229
+ # TODO: there may be edge cases where the same normalized field name will transform to two different names
230
+ # in different subtypes.
231
+ for subtype in get_args(stripped_type):
232
+ data = _transform_recursive(data, annotation=annotation, inner_type=subtype)
233
+ return data
234
+
235
+ if isinstance(data, pydantic.BaseModel):
236
+ return model_dump(data, exclude_unset=True, mode="json")
237
+
238
+ annotated_type = _get_annotated_type(annotation)
239
+ if annotated_type is None:
240
+ return data
241
+
242
+ # ignore the first argument as it is the actual type
243
+ annotations = get_args(annotated_type)[1:]
244
+ for annotation in annotations:
245
+ if isinstance(annotation, PropertyInfo) and annotation.format is not None:
246
+ return _format_data(data, annotation.format, annotation.format_template)
247
+
248
+ return data
249
+
250
+
251
+ def _format_data(data: object, format_: PropertyFormat, format_template: str | None) -> object:
252
+ if isinstance(data, (date, datetime)):
253
+ if format_ == "iso8601":
254
+ return data.isoformat()
255
+
256
+ if format_ == "custom" and format_template is not None:
257
+ return data.strftime(format_template)
258
+
259
+ if format_ == "base64" and is_base64_file_input(data):
260
+ binary: str | bytes | None = None
261
+
262
+ if isinstance(data, pathlib.Path):
263
+ binary = data.read_bytes()
264
+ elif isinstance(data, io.IOBase):
265
+ binary = data.read()
266
+
267
+ if isinstance(binary, str): # type: ignore[unreachable]
268
+ binary = binary.encode()
269
+
270
+ if not isinstance(binary, bytes):
271
+ raise RuntimeError(f"Could not read bytes from {data}; Received {type(binary)}")
272
+
273
+ return base64.b64encode(binary).decode("ascii")
274
+
275
+ return data
276
+
277
+
278
+ def _transform_typeddict(
279
+ data: Mapping[str, object],
280
+ expected_type: type,
281
+ ) -> Mapping[str, object]:
282
+ result: dict[str, object] = {}
283
+ annotations = get_type_hints(expected_type, include_extras=True)
284
+ for key, value in data.items():
285
+ if not is_given(value):
286
+ # we don't need to include omitted values here as they'll
287
+ # be stripped out before the request is sent anyway
288
+ continue
289
+
290
+ type_ = annotations.get(key)
291
+ if type_ is None:
292
+ # we do not have a type annotation for this field, leave it as is
293
+ result[key] = value
294
+ else:
295
+ result[_maybe_transform_key(key, type_)] = _transform_recursive(value, annotation=type_)
296
+ return result
297
+
298
+
299
+ async def async_maybe_transform(
300
+ data: object,
301
+ expected_type: object,
302
+ ) -> Any | None:
303
+ """Wrapper over `async_transform()` that allows `None` to be passed.
304
+
305
+ See `async_transform()` for more details.
306
+ """
307
+ if data is None:
308
+ return None
309
+ return await async_transform(data, expected_type)
310
+
311
+
312
+ async def async_transform(
313
+ data: _T,
314
+ expected_type: object,
315
+ ) -> _T:
316
+ """Transform dictionaries based off of type information from the given type, for example:
317
+
318
+ ```py
319
+ class Params(TypedDict, total=False):
320
+ card_id: Required[Annotated[str, PropertyInfo(alias="cardID")]]
321
+
322
+
323
+ transformed = transform({"card_id": "<my card ID>"}, Params)
324
+ # {'cardID': '<my card ID>'}
325
+ ```
326
+
327
+ Any keys / data that does not have type information given will be included as is.
328
+
329
+ It should be noted that the transformations that this function does are not represented in the type system.
330
+ """
331
+ transformed = await _async_transform_recursive(data, annotation=cast(type, expected_type))
332
+ return cast(_T, transformed)
333
+
334
+
335
+ async def _async_transform_recursive(
336
+ data: object,
337
+ *,
338
+ annotation: type,
339
+ inner_type: type | None = None,
340
+ ) -> object:
341
+ """Transform the given data against the expected type.
342
+
343
+ Args:
344
+ annotation: The direct type annotation given to the particular piece of data.
345
+ This may or may not be wrapped in metadata types, e.g. `Required[T]`, `Annotated[T, ...]` etc
346
+
347
+ inner_type: If applicable, this is the "inside" type. This is useful in certain cases where the outside type
348
+ is a container type such as `List[T]`. In that case `inner_type` should be set to `T` so that each entry in
349
+ the list can be transformed using the metadata from the container type.
350
+
351
+ Defaults to the same value as the `annotation` argument.
352
+ """
353
+ from .._compat import model_dump
354
+
355
+ if inner_type is None:
356
+ inner_type = annotation
357
+
358
+ stripped_type = strip_annotated_type(inner_type)
359
+ origin = get_origin(stripped_type) or stripped_type
360
+ if is_typeddict(stripped_type) and is_mapping(data):
361
+ return await _async_transform_typeddict(data, stripped_type)
362
+
363
+ if origin == dict and is_mapping(data):
364
+ items_type = get_args(stripped_type)[1]
365
+ return {key: _transform_recursive(value, annotation=items_type) for key, value in data.items()}
366
+
367
+ if (
368
+ # List[T]
369
+ (is_list_type(stripped_type) and is_list(data))
370
+ # Iterable[T]
371
+ or (is_iterable_type(stripped_type) and is_iterable(data) and not isinstance(data, str))
372
+ # Sequence[T]
373
+ or (is_sequence_type(stripped_type) and is_sequence(data) and not isinstance(data, str))
374
+ ):
375
+ # dicts are technically iterable, but it is an iterable on the keys of the dict and is not usually
376
+ # intended as an iterable, so we don't transform it.
377
+ if isinstance(data, dict):
378
+ return cast(object, data)
379
+
380
+ inner_type = extract_type_arg(stripped_type, 0)
381
+ if _no_transform_needed(inner_type):
382
+ # for some types there is no need to transform anything, so we can get a small
383
+ # perf boost from skipping that work.
384
+ #
385
+ # but we still need to convert to a list to ensure the data is json-serializable
386
+ if is_list(data):
387
+ return data
388
+ return list(data)
389
+
390
+ return [await _async_transform_recursive(d, annotation=annotation, inner_type=inner_type) for d in data]
391
+
392
+ if is_union_type(stripped_type):
393
+ # For union types we run the transformation against all subtypes to ensure that everything is transformed.
394
+ #
395
+ # TODO: there may be edge cases where the same normalized field name will transform to two different names
396
+ # in different subtypes.
397
+ for subtype in get_args(stripped_type):
398
+ data = await _async_transform_recursive(data, annotation=annotation, inner_type=subtype)
399
+ return data
400
+
401
+ if isinstance(data, pydantic.BaseModel):
402
+ return model_dump(data, exclude_unset=True, mode="json")
403
+
404
+ annotated_type = _get_annotated_type(annotation)
405
+ if annotated_type is None:
406
+ return data
407
+
408
+ # ignore the first argument as it is the actual type
409
+ annotations = get_args(annotated_type)[1:]
410
+ for annotation in annotations:
411
+ if isinstance(annotation, PropertyInfo) and annotation.format is not None:
412
+ return await _async_format_data(data, annotation.format, annotation.format_template)
413
+
414
+ return data
415
+
416
+
417
+ async def _async_format_data(data: object, format_: PropertyFormat, format_template: str | None) -> object:
418
+ if isinstance(data, (date, datetime)):
419
+ if format_ == "iso8601":
420
+ return data.isoformat()
421
+
422
+ if format_ == "custom" and format_template is not None:
423
+ return data.strftime(format_template)
424
+
425
+ if format_ == "base64" and is_base64_file_input(data):
426
+ binary: str | bytes | None = None
427
+
428
+ if isinstance(data, pathlib.Path):
429
+ binary = await anyio.Path(data).read_bytes()
430
+ elif isinstance(data, io.IOBase):
431
+ binary = data.read()
432
+
433
+ if isinstance(binary, str): # type: ignore[unreachable]
434
+ binary = binary.encode()
435
+
436
+ if not isinstance(binary, bytes):
437
+ raise RuntimeError(f"Could not read bytes from {data}; Received {type(binary)}")
438
+
439
+ return base64.b64encode(binary).decode("ascii")
440
+
441
+ return data
442
+
443
+
444
+ async def _async_transform_typeddict(
445
+ data: Mapping[str, object],
446
+ expected_type: type,
447
+ ) -> Mapping[str, object]:
448
+ result: dict[str, object] = {}
449
+ annotations = get_type_hints(expected_type, include_extras=True)
450
+ for key, value in data.items():
451
+ if not is_given(value):
452
+ # we don't need to include omitted values here as they'll
453
+ # be stripped out before the request is sent anyway
454
+ continue
455
+
456
+ type_ = annotations.get(key)
457
+ if type_ is None:
458
+ # we do not have a type annotation for this field, leave it as is
459
+ result[key] = value
460
+ else:
461
+ result[_maybe_transform_key(key, type_)] = await _async_transform_recursive(value, annotation=type_)
462
+ return result
463
+
464
+
465
+ @lru_cache(maxsize=8096)
466
+ def get_type_hints(
467
+ obj: Any,
468
+ globalns: dict[str, Any] | None = None,
469
+ localns: Mapping[str, Any] | None = None,
470
+ include_extras: bool = False,
471
+ ) -> dict[str, Any]:
472
+ return _get_type_hints(obj, globalns=globalns, localns=localns, include_extras=include_extras)
@@ -0,0 +1,172 @@
1
+ # Copyright 2025 Google LLC
2
+ #
3
+ # Licensed under the Apache License, Version 2.0 (the "License");
4
+ # you may not use this file except in compliance with the License.
5
+ # You may obtain a copy of the License at
6
+ #
7
+ # http://www.apache.org/licenses/LICENSE-2.0
8
+ #
9
+ # Unless required by applicable law or agreed to in writing, software
10
+ # distributed under the License is distributed on an "AS IS" BASIS,
11
+ # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12
+ # See the License for the specific language governing permissions and
13
+ # limitations under the License.
14
+ #
15
+
16
+ # mypy: ignore-errors
17
+ from __future__ import annotations
18
+
19
+ import sys
20
+ import typing
21
+ import typing_extensions
22
+ from typing import Any, TypeVar, Iterable, cast
23
+ from collections import abc as _c_abc
24
+ from typing_extensions import (
25
+ TypeIs,
26
+ Required,
27
+ Annotated,
28
+ get_args,
29
+ get_origin,
30
+ )
31
+
32
+ from ._utils import lru_cache
33
+ from .._types import InheritsGeneric
34
+ from ._compat import is_union as _is_union
35
+
36
+
37
+ def is_annotated_type(typ: type) -> bool:
38
+ return get_origin(typ) == Annotated
39
+
40
+
41
+ def is_list_type(typ: type) -> bool:
42
+ return (get_origin(typ) or typ) == list
43
+
44
+
45
+ def is_sequence_type(typ: type) -> bool:
46
+ origin = get_origin(typ) or typ
47
+ return origin == typing_extensions.Sequence or origin == typing.Sequence or origin == _c_abc.Sequence
48
+
49
+
50
+ def is_iterable_type(typ: type) -> bool:
51
+ """If the given type is `typing.Iterable[T]`"""
52
+ origin = get_origin(typ) or typ
53
+ return origin == Iterable or origin == _c_abc.Iterable
54
+
55
+
56
+ def is_union_type(typ: type) -> bool:
57
+ return _is_union(get_origin(typ))
58
+
59
+
60
+ def is_required_type(typ: type) -> bool:
61
+ return get_origin(typ) == Required
62
+
63
+
64
+ def is_typevar(typ: type) -> bool:
65
+ # type ignore is required because type checkers
66
+ # think this expression will always return False
67
+ return type(typ) == TypeVar # type: ignore
68
+
69
+
70
+ _TYPE_ALIAS_TYPES: tuple[type[typing_extensions.TypeAliasType], ...] = (typing_extensions.TypeAliasType,)
71
+ if sys.version_info >= (3, 12):
72
+ _TYPE_ALIAS_TYPES = (*_TYPE_ALIAS_TYPES, typing.TypeAliasType)
73
+
74
+
75
+ def is_type_alias_type(tp: Any, /) -> TypeIs[typing_extensions.TypeAliasType]:
76
+ """Return whether the provided argument is an instance of `TypeAliasType`.
77
+
78
+ ```python
79
+ type Int = int
80
+ is_type_alias_type(Int)
81
+ # > True
82
+ Str = TypeAliasType("Str", str)
83
+ is_type_alias_type(Str)
84
+ # > True
85
+ ```
86
+ """
87
+ return isinstance(tp, _TYPE_ALIAS_TYPES)
88
+
89
+
90
+ # Extracts T from Annotated[T, ...] or from Required[Annotated[T, ...]]
91
+ @lru_cache(maxsize=8096)
92
+ def strip_annotated_type(typ: type) -> type:
93
+ if is_required_type(typ) or is_annotated_type(typ):
94
+ return strip_annotated_type(cast(type, get_args(typ)[0]))
95
+
96
+ return typ
97
+
98
+
99
+ def extract_type_arg(typ: type, index: int) -> type:
100
+ args = get_args(typ)
101
+ try:
102
+ return cast(type, args[index])
103
+ except IndexError as err:
104
+ raise RuntimeError(f"Expected type {typ} to have a type argument at index {index} but it did not") from err
105
+
106
+
107
+ def extract_type_var_from_base(
108
+ typ: type,
109
+ *,
110
+ generic_bases: tuple[type, ...],
111
+ index: int,
112
+ failure_message: str | None = None,
113
+ ) -> type:
114
+ """Given a type like `Foo[T]`, returns the generic type variable `T`.
115
+
116
+ This also handles the case where a concrete subclass is given, e.g.
117
+ ```py
118
+ class MyResponse(Foo[bytes]):
119
+ ...
120
+
121
+ extract_type_var(MyResponse, bases=(Foo,), index=0) -> bytes
122
+ ```
123
+
124
+ And where a generic subclass is given:
125
+ ```py
126
+ _T = TypeVar('_T')
127
+ class MyResponse(Foo[_T]):
128
+ ...
129
+
130
+ extract_type_var(MyResponse[bytes], bases=(Foo,), index=0) -> bytes
131
+ ```
132
+ """
133
+ cls = cast(object, get_origin(typ) or typ)
134
+ if cls in generic_bases: # pyright: ignore[reportUnnecessaryContains]
135
+ # we're given the class directly
136
+ return extract_type_arg(typ, index)
137
+
138
+ # if a subclass is given
139
+ # ---
140
+ # this is needed as __orig_bases__ is not present in the typeshed stubs
141
+ # because it is intended to be for internal use only, however there does
142
+ # not seem to be a way to resolve generic TypeVars for inherited subclasses
143
+ # without using it.
144
+ if isinstance(cls, InheritsGeneric):
145
+ target_base_class: Any | None = None
146
+ for base in cls.__orig_bases__:
147
+ if base.__origin__ in generic_bases:
148
+ target_base_class = base
149
+ break
150
+
151
+ if target_base_class is None:
152
+ raise RuntimeError(
153
+ "Could not find the generic base class;\n"
154
+ "This should never happen;\n"
155
+ f"Does {cls} inherit from one of {generic_bases} ?"
156
+ )
157
+
158
+ extracted = extract_type_arg(target_base_class, index)
159
+ if is_typevar(extracted):
160
+ # If the extracted type argument is itself a type variable
161
+ # then that means the subclass itself is generic, so we have
162
+ # to resolve the type argument from the class itself, not
163
+ # the base class.
164
+ #
165
+ # Note: if there is more than 1 type argument, the subclass could
166
+ # change the ordering of the type arguments, this is not currently
167
+ # supported.
168
+ return extract_type_arg(typ, index)
169
+
170
+ return extracted
171
+
172
+ raise RuntimeError(failure_message or f"Could not resolve inner type variable at index {index} for {typ}")