google-genai 1.56.0__py3-none-any.whl → 1.58.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 (246) hide show
  1. google/genai/_api_client.py +49 -26
  2. google/genai/_interactions/__init__.py +3 -0
  3. google/genai/_interactions/_base_client.py +1 -1
  4. google/genai/_interactions/_client.py +57 -3
  5. google/genai/_interactions/_client_adapter.py +48 -0
  6. google/genai/_interactions/types/__init__.py +6 -0
  7. google/genai/_interactions/types/audio_content.py +2 -0
  8. google/genai/_interactions/types/audio_content_param.py +2 -0
  9. google/genai/_interactions/types/content.py +65 -0
  10. google/genai/_interactions/types/content_delta.py +10 -2
  11. google/genai/_interactions/types/content_param.py +63 -0
  12. google/genai/_interactions/types/content_start.py +5 -46
  13. google/genai/_interactions/types/content_stop.py +1 -2
  14. google/genai/_interactions/types/document_content.py +2 -0
  15. google/genai/_interactions/types/document_content_param.py +2 -0
  16. google/genai/_interactions/types/error_event.py +1 -2
  17. google/genai/_interactions/types/file_search_call_content.py +32 -0
  18. google/genai/_interactions/types/file_search_call_content_param.py +31 -0
  19. google/genai/_interactions/types/generation_config.py +4 -0
  20. google/genai/_interactions/types/generation_config_param.py +4 -0
  21. google/genai/_interactions/types/image_config.py +31 -0
  22. google/genai/_interactions/types/image_config_param.py +30 -0
  23. google/genai/_interactions/types/image_content.py +2 -0
  24. google/genai/_interactions/types/image_content_param.py +2 -0
  25. google/genai/_interactions/types/interaction.py +6 -52
  26. google/genai/_interactions/types/interaction_create_params.py +4 -22
  27. google/genai/_interactions/types/interaction_event.py +1 -2
  28. google/genai/_interactions/types/interaction_sse_event.py +5 -3
  29. google/genai/_interactions/types/interaction_status_update.py +1 -2
  30. google/genai/_interactions/types/model.py +1 -0
  31. google/genai/_interactions/types/model_param.py +1 -0
  32. google/genai/_interactions/types/turn.py +3 -44
  33. google/genai/_interactions/types/turn_param.py +4 -40
  34. google/genai/_interactions/types/usage.py +1 -1
  35. google/genai/_interactions/types/usage_param.py +1 -1
  36. google/genai/_interactions/types/video_content.py +2 -0
  37. google/genai/_interactions/types/video_content_param.py +2 -0
  38. google/genai/_live_converters.py +118 -34
  39. google/genai/_local_tokenizer_loader.py +1 -0
  40. google/genai/_tokens_converters.py +14 -14
  41. google/genai/_transformers.py +15 -21
  42. google/genai/batches.py +27 -22
  43. google/genai/caches.py +42 -42
  44. google/genai/chats.py +0 -2
  45. google/genai/client.py +61 -55
  46. google/genai/files.py +224 -0
  47. google/genai/live.py +1 -1
  48. google/genai/models.py +56 -44
  49. google/genai/tests/__init__.py +21 -0
  50. google/genai/tests/afc/__init__.py +21 -0
  51. google/genai/tests/afc/test_convert_if_exist_pydantic_model.py +309 -0
  52. google/genai/tests/afc/test_convert_number_values_for_function_call_args.py +63 -0
  53. google/genai/tests/afc/test_find_afc_incompatible_tool_indexes.py +240 -0
  54. google/genai/tests/afc/test_generate_content_stream_afc.py +530 -0
  55. google/genai/tests/afc/test_generate_content_stream_afc_thoughts.py +77 -0
  56. google/genai/tests/afc/test_get_function_map.py +176 -0
  57. google/genai/tests/afc/test_get_function_response_parts.py +277 -0
  58. google/genai/tests/afc/test_get_max_remote_calls_for_afc.py +130 -0
  59. google/genai/tests/afc/test_invoke_function_from_dict_args.py +241 -0
  60. google/genai/tests/afc/test_raise_error_for_afc_incompatible_config.py +159 -0
  61. google/genai/tests/afc/test_should_append_afc_history.py +53 -0
  62. google/genai/tests/afc/test_should_disable_afc.py +214 -0
  63. google/genai/tests/batches/__init__.py +17 -0
  64. google/genai/tests/batches/test_cancel.py +77 -0
  65. google/genai/tests/batches/test_create.py +78 -0
  66. google/genai/tests/batches/test_create_with_bigquery.py +113 -0
  67. google/genai/tests/batches/test_create_with_file.py +82 -0
  68. google/genai/tests/batches/test_create_with_gcs.py +125 -0
  69. google/genai/tests/batches/test_create_with_inlined_requests.py +255 -0
  70. google/genai/tests/batches/test_delete.py +86 -0
  71. google/genai/tests/batches/test_embedding.py +157 -0
  72. google/genai/tests/batches/test_get.py +78 -0
  73. google/genai/tests/batches/test_list.py +79 -0
  74. google/genai/tests/caches/__init__.py +17 -0
  75. google/genai/tests/caches/constants.py +29 -0
  76. google/genai/tests/caches/test_create.py +210 -0
  77. google/genai/tests/caches/test_create_custom_url.py +105 -0
  78. google/genai/tests/caches/test_delete.py +54 -0
  79. google/genai/tests/caches/test_delete_custom_url.py +52 -0
  80. google/genai/tests/caches/test_get.py +94 -0
  81. google/genai/tests/caches/test_get_custom_url.py +52 -0
  82. google/genai/tests/caches/test_list.py +68 -0
  83. google/genai/tests/caches/test_update.py +70 -0
  84. google/genai/tests/caches/test_update_custom_url.py +58 -0
  85. google/genai/tests/chats/__init__.py +1 -0
  86. google/genai/tests/chats/test_get_history.py +598 -0
  87. google/genai/tests/chats/test_send_message.py +844 -0
  88. google/genai/tests/chats/test_validate_response.py +90 -0
  89. google/genai/tests/client/__init__.py +17 -0
  90. google/genai/tests/client/test_async_stream.py +427 -0
  91. google/genai/tests/client/test_client_close.py +197 -0
  92. google/genai/tests/client/test_client_initialization.py +1687 -0
  93. google/genai/tests/client/test_client_requests.py +221 -0
  94. google/genai/tests/client/test_custom_client.py +104 -0
  95. google/genai/tests/client/test_http_options.py +178 -0
  96. google/genai/tests/client/test_replay_client_equality.py +168 -0
  97. google/genai/tests/client/test_retries.py +846 -0
  98. google/genai/tests/client/test_upload_errors.py +136 -0
  99. google/genai/tests/common/__init__.py +17 -0
  100. google/genai/tests/common/test_common.py +954 -0
  101. google/genai/tests/conftest.py +162 -0
  102. google/genai/tests/documents/__init__.py +17 -0
  103. google/genai/tests/documents/test_delete.py +51 -0
  104. google/genai/tests/documents/test_get.py +85 -0
  105. google/genai/tests/documents/test_list.py +72 -0
  106. google/genai/tests/errors/__init__.py +1 -0
  107. google/genai/tests/errors/test_api_error.py +417 -0
  108. google/genai/tests/file_search_stores/__init__.py +17 -0
  109. google/genai/tests/file_search_stores/test_create.py +66 -0
  110. google/genai/tests/file_search_stores/test_delete.py +64 -0
  111. google/genai/tests/file_search_stores/test_get.py +94 -0
  112. google/genai/tests/file_search_stores/test_import_file.py +112 -0
  113. google/genai/tests/file_search_stores/test_list.py +57 -0
  114. google/genai/tests/file_search_stores/test_upload_to_file_search_store.py +141 -0
  115. google/genai/tests/files/__init__.py +17 -0
  116. google/genai/tests/files/test_delete.py +46 -0
  117. google/genai/tests/files/test_download.py +85 -0
  118. google/genai/tests/files/test_get.py +46 -0
  119. google/genai/tests/files/test_list.py +72 -0
  120. google/genai/tests/files/test_register.py +272 -0
  121. google/genai/tests/files/test_register_table.py +70 -0
  122. google/genai/tests/files/test_upload.py +255 -0
  123. google/genai/tests/imports/test_no_optional_imports.py +28 -0
  124. google/genai/tests/interactions/test_auth.py +476 -0
  125. google/genai/tests/interactions/test_integration.py +84 -0
  126. google/genai/tests/interactions/test_paths.py +105 -0
  127. google/genai/tests/live/__init__.py +16 -0
  128. google/genai/tests/live/test_live.py +2143 -0
  129. google/genai/tests/live/test_live_music.py +362 -0
  130. google/genai/tests/live/test_live_response.py +163 -0
  131. google/genai/tests/live/test_send_client_content.py +147 -0
  132. google/genai/tests/live/test_send_realtime_input.py +268 -0
  133. google/genai/tests/live/test_send_tool_response.py +222 -0
  134. google/genai/tests/local_tokenizer/__init__.py +17 -0
  135. google/genai/tests/local_tokenizer/test_local_tokenizer.py +343 -0
  136. google/genai/tests/local_tokenizer/test_local_tokenizer_loader.py +235 -0
  137. google/genai/tests/mcp/__init__.py +17 -0
  138. google/genai/tests/mcp/test_has_mcp_tool_usage.py +89 -0
  139. google/genai/tests/mcp/test_mcp_to_gemini_tools.py +191 -0
  140. google/genai/tests/mcp/test_parse_config_for_mcp_sessions.py +201 -0
  141. google/genai/tests/mcp/test_parse_config_for_mcp_usage.py +130 -0
  142. google/genai/tests/mcp/test_set_mcp_usage_header.py +72 -0
  143. google/genai/tests/models/__init__.py +17 -0
  144. google/genai/tests/models/constants.py +8 -0
  145. google/genai/tests/models/test_compute_tokens.py +120 -0
  146. google/genai/tests/models/test_count_tokens.py +159 -0
  147. google/genai/tests/models/test_delete.py +107 -0
  148. google/genai/tests/models/test_edit_image.py +264 -0
  149. google/genai/tests/models/test_embed_content.py +94 -0
  150. google/genai/tests/models/test_function_call_streaming.py +442 -0
  151. google/genai/tests/models/test_generate_content.py +2501 -0
  152. google/genai/tests/models/test_generate_content_cached_content.py +132 -0
  153. google/genai/tests/models/test_generate_content_config_zero_value.py +103 -0
  154. google/genai/tests/models/test_generate_content_from_apikey.py +44 -0
  155. google/genai/tests/models/test_generate_content_http_options.py +40 -0
  156. google/genai/tests/models/test_generate_content_image_generation.py +143 -0
  157. google/genai/tests/models/test_generate_content_mcp.py +343 -0
  158. google/genai/tests/models/test_generate_content_media_resolution.py +97 -0
  159. google/genai/tests/models/test_generate_content_model.py +139 -0
  160. google/genai/tests/models/test_generate_content_part.py +821 -0
  161. google/genai/tests/models/test_generate_content_thought.py +76 -0
  162. google/genai/tests/models/test_generate_content_tools.py +1761 -0
  163. google/genai/tests/models/test_generate_images.py +191 -0
  164. google/genai/tests/models/test_generate_videos.py +759 -0
  165. google/genai/tests/models/test_get.py +104 -0
  166. google/genai/tests/models/test_list.py +233 -0
  167. google/genai/tests/models/test_recontext_image.py +189 -0
  168. google/genai/tests/models/test_segment_image.py +148 -0
  169. google/genai/tests/models/test_update.py +95 -0
  170. google/genai/tests/models/test_upscale_image.py +157 -0
  171. google/genai/tests/operations/__init__.py +17 -0
  172. google/genai/tests/operations/test_get.py +38 -0
  173. google/genai/tests/public_samples/__init__.py +17 -0
  174. google/genai/tests/public_samples/test_gemini_text_only.py +34 -0
  175. google/genai/tests/pytest_helper.py +246 -0
  176. google/genai/tests/shared/__init__.py +16 -0
  177. google/genai/tests/shared/batches/__init__.py +14 -0
  178. google/genai/tests/shared/batches/test_create_delete.py +57 -0
  179. google/genai/tests/shared/batches/test_create_get_cancel.py +56 -0
  180. google/genai/tests/shared/batches/test_list.py +40 -0
  181. google/genai/tests/shared/caches/__init__.py +14 -0
  182. google/genai/tests/shared/caches/test_create_get_delete.py +67 -0
  183. google/genai/tests/shared/caches/test_create_update_get.py +71 -0
  184. google/genai/tests/shared/caches/test_list.py +40 -0
  185. google/genai/tests/shared/chats/__init__.py +14 -0
  186. google/genai/tests/shared/chats/test_send_message.py +48 -0
  187. google/genai/tests/shared/chats/test_send_message_stream.py +50 -0
  188. google/genai/tests/shared/files/__init__.py +14 -0
  189. google/genai/tests/shared/files/test_list.py +41 -0
  190. google/genai/tests/shared/files/test_upload_get_delete.py +54 -0
  191. google/genai/tests/shared/models/__init__.py +14 -0
  192. google/genai/tests/shared/models/test_compute_tokens.py +41 -0
  193. google/genai/tests/shared/models/test_count_tokens.py +40 -0
  194. google/genai/tests/shared/models/test_edit_image.py +67 -0
  195. google/genai/tests/shared/models/test_embed.py +40 -0
  196. google/genai/tests/shared/models/test_generate_content.py +39 -0
  197. google/genai/tests/shared/models/test_generate_content_stream.py +54 -0
  198. google/genai/tests/shared/models/test_generate_images.py +40 -0
  199. google/genai/tests/shared/models/test_generate_videos.py +38 -0
  200. google/genai/tests/shared/models/test_list.py +37 -0
  201. google/genai/tests/shared/models/test_recontext_image.py +55 -0
  202. google/genai/tests/shared/models/test_segment_image.py +52 -0
  203. google/genai/tests/shared/models/test_upscale_image.py +52 -0
  204. google/genai/tests/shared/tunings/__init__.py +16 -0
  205. google/genai/tests/shared/tunings/test_create.py +46 -0
  206. google/genai/tests/shared/tunings/test_create_get_cancel.py +56 -0
  207. google/genai/tests/shared/tunings/test_list.py +39 -0
  208. google/genai/tests/tokens/__init__.py +16 -0
  209. google/genai/tests/tokens/test_create.py +154 -0
  210. google/genai/tests/transformers/__init__.py +17 -0
  211. google/genai/tests/transformers/test_blobs.py +84 -0
  212. google/genai/tests/transformers/test_bytes.py +15 -0
  213. google/genai/tests/transformers/test_duck_type.py +96 -0
  214. google/genai/tests/transformers/test_function_responses.py +72 -0
  215. google/genai/tests/transformers/test_schema.py +653 -0
  216. google/genai/tests/transformers/test_t_batch.py +286 -0
  217. google/genai/tests/transformers/test_t_content.py +160 -0
  218. google/genai/tests/transformers/test_t_contents.py +398 -0
  219. google/genai/tests/transformers/test_t_part.py +85 -0
  220. google/genai/tests/transformers/test_t_parts.py +87 -0
  221. google/genai/tests/transformers/test_t_tool.py +157 -0
  222. google/genai/tests/transformers/test_t_tools.py +195 -0
  223. google/genai/tests/tunings/__init__.py +16 -0
  224. google/genai/tests/tunings/test_cancel.py +39 -0
  225. google/genai/tests/tunings/test_end_to_end.py +106 -0
  226. google/genai/tests/tunings/test_get.py +67 -0
  227. google/genai/tests/tunings/test_list.py +75 -0
  228. google/genai/tests/tunings/test_tune.py +268 -0
  229. google/genai/tests/types/__init__.py +16 -0
  230. google/genai/tests/types/test_bytes_internal.py +271 -0
  231. google/genai/tests/types/test_bytes_type.py +152 -0
  232. google/genai/tests/types/test_future.py +101 -0
  233. google/genai/tests/types/test_optional_types.py +36 -0
  234. google/genai/tests/types/test_part_type.py +616 -0
  235. google/genai/tests/types/test_schema_from_json_schema.py +417 -0
  236. google/genai/tests/types/test_schema_json_schema.py +468 -0
  237. google/genai/tests/types/test_types.py +2903 -0
  238. google/genai/types.py +631 -488
  239. google/genai/version.py +1 -1
  240. {google_genai-1.56.0.dist-info → google_genai-1.58.0.dist-info}/METADATA +6 -11
  241. google_genai-1.58.0.dist-info/RECORD +358 -0
  242. google_genai-1.56.0.dist-info/RECORD +0 -162
  243. /google/genai/{_interactions/py.typed → tests/interactions/__init__.py} +0 -0
  244. {google_genai-1.56.0.dist-info → google_genai-1.58.0.dist-info}/WHEEL +0 -0
  245. {google_genai-1.56.0.dist-info → google_genai-1.58.0.dist-info}/licenses/LICENSE +0 -0
  246. {google_genai-1.56.0.dist-info → google_genai-1.58.0.dist-info}/top_level.txt +0 -0
@@ -0,0 +1,221 @@
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
+
17
+ """Tests for client behavior when issuing requests."""
18
+
19
+ from ... import _api_client as api_client
20
+ from ... import Client
21
+ from ... import types
22
+
23
+
24
+ def build_test_client(monkeypatch):
25
+ monkeypatch.setenv('GOOGLE_API_KEY', 'google_api_key')
26
+ return Client()
27
+
28
+
29
+ def test_join_url_path_base_url_with_trailing_slash_and_path_with_leading_slash():
30
+ base_url = 'https://fake-url.com/some_path/'
31
+ path = '/v1beta/models'
32
+ assert (
33
+ api_client.join_url_path(base_url, path)
34
+ == 'https://fake-url.com/some_path/v1beta/models'
35
+ )
36
+
37
+
38
+ def test_join_url_path_with_base_url_with_trailing_slash_and_path_without_leading_slash():
39
+ base_url = 'https://fake-url.com/some_path/'
40
+ path = 'v1beta/models'
41
+ assert (
42
+ api_client.join_url_path(base_url, path)
43
+ == 'https://fake-url.com/some_path/v1beta/models'
44
+ )
45
+
46
+
47
+ def test_join_url_path_with_base_url_without_trailing_slash_and_path_with_leading_slash():
48
+ base_url = 'https://fake-url.com/some_path'
49
+ path = '/v1beta/models'
50
+ assert (
51
+ api_client.join_url_path(base_url, path)
52
+ == 'https://fake-url.com/some_path/v1beta/models'
53
+ )
54
+
55
+
56
+ def test_join_url_path_with_base_url_without_trailing_slash_and_path_without_leading_slash():
57
+ base_url = 'https://fake-url.com/some_path'
58
+ path = 'v1beta/models'
59
+ assert (
60
+ api_client.join_url_path(base_url, path)
61
+ == 'https://fake-url.com/some_path/v1beta/models'
62
+ )
63
+
64
+
65
+ def test_join_url_path_base_url_without_path_with_trailing_slash():
66
+ base_url = 'https://fake-url.com/'
67
+ path = 'v1beta/models'
68
+ assert (
69
+ api_client.join_url_path(base_url, path)
70
+ == 'https://fake-url.com/v1beta/models'
71
+ )
72
+
73
+
74
+ def test_join_url_path_base_url_without_path_without_trailing_slash():
75
+ base_url = 'https://fake-url.com'
76
+ path = 'v1beta/models'
77
+ assert (
78
+ api_client.join_url_path(base_url, path)
79
+ == 'https://fake-url.com/v1beta/models'
80
+ )
81
+
82
+
83
+ def test_build_request_sets_library_version_headers(monkeypatch):
84
+ request_client = build_test_client(monkeypatch).models._api_client
85
+ request = request_client._build_request('GET', 'test/path', {'key': 'value'})
86
+ assert 'google-genai-sdk/' in request.headers['user-agent']
87
+ assert 'gl-python/' in request.headers['user-agent']
88
+ assert 'google-genai-sdk/' in request.headers['x-goog-api-client']
89
+ assert 'gl-python/' in request.headers['x-goog-api-client']
90
+
91
+
92
+ def test_build_request_appends_to_user_agent_headers(monkeypatch):
93
+ request_client = build_test_client(monkeypatch).models._api_client
94
+ request = request_client._build_request(
95
+ 'GET',
96
+ 'test/path',
97
+ {'key': 'value'},
98
+ types.HttpOptionsDict(
99
+ base_url='test/url',
100
+ api_version='1',
101
+ headers={'user-agent': 'test-user-agent'},
102
+ ),
103
+ )
104
+ assert 'test-user-agent' in request.headers['user-agent']
105
+ assert 'google-genai-sdk/' in request.headers['user-agent']
106
+ assert 'gl-python/' in request.headers['user-agent']
107
+ assert 'google-genai-sdk/' in request.headers['x-goog-api-client']
108
+
109
+
110
+ def test_build_request_appends_to_goog_api_client_headers(monkeypatch):
111
+ request_client = build_test_client(monkeypatch).models._api_client
112
+ request = request_client._build_request(
113
+ 'GET',
114
+ 'test/path',
115
+ {'key': 'value'},
116
+ types.HttpOptionsDict(
117
+ base_url='test/url',
118
+ api_version='1',
119
+ headers={'x-goog-api-client': 'test-goog-api-client'},
120
+ ),
121
+ )
122
+ assert 'google-genai-sdk/' in request.headers['user-agent']
123
+ assert 'test-goog-api-client' in request.headers['x-goog-api-client']
124
+ assert 'google-genai-sdk/' in request.headers['x-goog-api-client']
125
+ assert 'gl-python/' in request.headers['x-goog-api-client']
126
+
127
+
128
+ def test_build_request_keeps_sdk_version_headers(monkeypatch):
129
+ headers_to_inject = {}
130
+ api_client.append_library_version_headers(headers_to_inject)
131
+ assert 'google-genai-sdk/' in headers_to_inject['user-agent']
132
+ request_client = build_test_client(monkeypatch).models._api_client
133
+ request = request_client._build_request(
134
+ 'GET',
135
+ 'test/path',
136
+ {'key': 'value'},
137
+ types.HttpOptionsDict(
138
+ base_url='test/url',
139
+ api_version='1',
140
+ headers=headers_to_inject,
141
+ ),
142
+ )
143
+ assert 'google-genai-sdk/' in request.headers['user-agent']
144
+ assert 'gl-python/' in request.headers['x-goog-api-client']
145
+ assert 'google-genai-sdk/' in request.headers['x-goog-api-client']
146
+ assert 'gl-python/' in request.headers['x-goog-api-client']
147
+
148
+
149
+ def test_build_request_with_resource_scope(monkeypatch):
150
+ monkeypatch.delenv('GOOGLE_API_KEY', raising=False)
151
+ monkeypatch.delenv('GEMINI_API_KEY', raising=False)
152
+ monkeypatch.delenv('GOOGLE_CLOUD_PROJECT', raising=False)
153
+ monkeypatch.delenv('GOOGLE_CLOUD_LOCATION', raising=False)
154
+
155
+ client = Client(
156
+ vertexai=True,
157
+ http_options=types.HttpOptionsDict(
158
+ base_url='https://custom-base-url.com',
159
+ base_url_resource_scope=types.ResourceScope.COLLECTION,
160
+ ),
161
+ )
162
+
163
+ request = client.models._api_client._build_request(
164
+ 'post',
165
+ 'publishers/google/models/gemini-3-pro-preview',
166
+ {'key': 'value'},
167
+ )
168
+ assert request.url == 'https://custom-base-url.com/publishers/google/models/gemini-3-pro-preview'
169
+
170
+
171
+ def test_build_request_with_resource_scope_with_project_and_location(
172
+ monkeypatch,
173
+ ):
174
+ monkeypatch.delenv('GOOGLE_API_KEY', raising=False)
175
+ monkeypatch.delenv('GEMINI_API_KEY', raising=False)
176
+ monkeypatch.delenv('GOOGLE_CLOUD_PROJECT', raising=False)
177
+ monkeypatch.delenv('GOOGLE_CLOUD_LOCATION', raising=False)
178
+
179
+ client = Client(
180
+ vertexai=True,
181
+ project='test-project',
182
+ location='test-location',
183
+ http_options=types.HttpOptionsDict(
184
+ base_url='https://custom-base-url.com',
185
+ base_url_resource_scope=types.ResourceScope.COLLECTION,
186
+ ),
187
+ )
188
+
189
+ request = client.models._api_client._build_request(
190
+ 'post',
191
+ 'publishers/google/models/gemini-3-pro-preview',
192
+ {'key': 'value'},
193
+ )
194
+ assert request.url == 'https://custom-base-url.com/publishers/google/models/gemini-3-pro-preview'
195
+
196
+
197
+
198
+ def build_test_client_no_env_vars(monkeypatch):
199
+ monkeypatch.delenv('GOOGLE_API_KEY', raising=False)
200
+ monkeypatch.delenv('GEMINI_API_KEY', raising=False)
201
+ monkeypatch.delenv('GOOGLE_CLOUD_PROJECT', raising=False)
202
+ monkeypatch.delenv('GOOGLE_CLOUD_LOCATION', raising=False)
203
+ return Client(
204
+ vertexai=True,
205
+ http_options=types.HttpOptionsDict(
206
+ base_url='https://custom-base-url.com',
207
+ headers={'Authorization': 'Bearer fake_access_token'},
208
+ ),
209
+ )
210
+
211
+
212
+ def test_build_request_with_custom_base_url_no_env_vars(monkeypatch):
213
+ request_client = (
214
+ build_test_client_no_env_vars(monkeypatch).models._api_client
215
+ )
216
+ request = request_client._build_request(
217
+ 'GET',
218
+ 'test/path',
219
+ {'key': 'value'},
220
+ )
221
+ assert request.url == 'https://custom-base-url.com'
@@ -0,0 +1,104 @@
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
+ """Tests for custom clients."""
17
+ import asyncio
18
+ from unittest import mock
19
+
20
+ from google.oauth2 import credentials
21
+ import httpx
22
+ import pytest
23
+
24
+ from ... import _api_client as api_client
25
+ from ... import Client
26
+
27
+
28
+ try:
29
+ import aiohttp
30
+
31
+ AIOHTTP_NOT_INSTALLED = False
32
+ except ImportError:
33
+ AIOHTTP_NOT_INSTALLED = True
34
+ aiohttp = mock.MagicMock()
35
+
36
+ requires_aiohttp = pytest.mark.skipif(
37
+ AIOHTTP_NOT_INSTALLED, reason='aiohttp is not installed, skipping test.'
38
+ )
39
+
40
+
41
+ # Httpx
42
+ def test_constructor_with_httpx_clients():
43
+ mldev_http_options = {
44
+ 'httpx_client': httpx.Client(trust_env=False),
45
+ 'httpx_async_client': httpx.AsyncClient(trust_env=False),
46
+ }
47
+ vertexai_http_options = {
48
+ 'httpx_client': httpx.Client(trust_env=False),
49
+ 'httpx_async_client': httpx.AsyncClient(trust_env=False),
50
+ }
51
+
52
+ # Even if aiohttp is installed, expect it to be disabled when httpx clients
53
+ # are provided.
54
+ api_client.has_aiohttp = True
55
+
56
+ mldev_client = Client(
57
+ api_key='google_api_key', http_options=mldev_http_options
58
+ )
59
+ assert not mldev_client.models._api_client._httpx_client.trust_env
60
+ assert not mldev_client.models._api_client._async_httpx_client.trust_env
61
+ # Expect aiohttp to be disabled when httpx clients are provided, regardless of
62
+ # whether aiohttp is installed.
63
+ assert not mldev_client.models._api_client._use_aiohttp()
64
+
65
+ vertexai_client = Client(
66
+ vertexai=True,
67
+ project='fake_project_id',
68
+ location='fake-location',
69
+ http_options=vertexai_http_options,
70
+ )
71
+ assert not vertexai_client.models._api_client._httpx_client.trust_env
72
+ assert not vertexai_client.models._api_client._async_httpx_client.trust_env
73
+ # Expect aiohttp to be disabled when httpx clients are provided, regardless of
74
+ # whether aiohttp is installed.
75
+ assert not mldev_client.models._api_client._use_aiohttp()
76
+
77
+
78
+ # Aiohttp
79
+ @requires_aiohttp
80
+ @pytest.mark.asyncio
81
+ @pytest.mark.skipif(
82
+ AIOHTTP_NOT_INSTALLED, reason='aiohttp is not installed, skipping test.'
83
+ )
84
+ async def test_constructor_with_aiohttp_clients():
85
+ api_client.has_aiohttp = True
86
+ mldev_http_options = {
87
+ 'aiohttp_client': aiohttp.ClientSession(trust_env=False),
88
+ }
89
+ vertexai_http_options = {
90
+ 'aiohttp_client': aiohttp.ClientSession(trust_env=False),
91
+ }
92
+ mldev_client = Client(
93
+ api_key='google_api_key', http_options=mldev_http_options
94
+ )
95
+ assert not mldev_client.models._api_client._aiohttp_session.trust_env
96
+
97
+ vertexai_client = Client(
98
+ vertexai=True,
99
+ project='fake_project_id',
100
+ location='fake-location',
101
+ http_options=vertexai_http_options,
102
+ )
103
+ assert not vertexai_client.models._api_client._aiohttp_session.trust_env
104
+
@@ -0,0 +1,178 @@
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
+
17
+ """Tests for client behavior when issuing requests."""
18
+
19
+ import pytest
20
+
21
+ from ... import _api_client
22
+ from ... import types
23
+
24
+
25
+ def test_patch_http_options_with_copies_all_fields():
26
+ patch_options = types.HttpOptions(
27
+ base_url='https://fake-url.com/',
28
+ api_version='v1',
29
+ headers={'X-Custom-Header': 'custom_value'},
30
+ timeout=10000,
31
+ client_args={'http2': True},
32
+ async_client_args={'http1': True},
33
+ extra_body={'key': 'value'},
34
+ retry_options=types.HttpRetryOptions(attempts=10),
35
+ base_url_resource_scope=types.ResourceScope.COLLECTION,
36
+ )
37
+ options = types.HttpOptions()
38
+ patched = _api_client.patch_http_options(options, patch_options)
39
+ http_options_keys = types.HttpOptions.model_fields.keys()
40
+
41
+ for key in http_options_keys:
42
+ assert hasattr(patched, key)
43
+ if key not in ['httpx_client', 'httpx_async_client', 'aiohttp_client']:
44
+ assert getattr(patched, key) is not None
45
+ assert patched.base_url == 'https://fake-url.com/'
46
+ assert patched.api_version == 'v1'
47
+ assert patched.headers['X-Custom-Header'] == 'custom_value'
48
+ assert patched.timeout == 10000
49
+ assert patched.retry_options.attempts == 10
50
+ assert patched.client_args['http2']
51
+ assert patched.async_client_args['http1']
52
+
53
+
54
+ def test_patch_http_options_merges_headers():
55
+ original_options = types.HttpOptions(
56
+ headers={
57
+ 'X-Custom-Header': 'different_value',
58
+ 'X-different-header': 'different_value',
59
+ }
60
+ )
61
+ patch_options = types.HttpOptions(
62
+ base_url='https://fake-url.com/',
63
+ api_version='v1',
64
+ headers={'X-Custom-Header': 'custom_value'},
65
+ timeout=10000,
66
+ )
67
+ patched = _api_client.patch_http_options(original_options, patch_options)
68
+ # If the header is present in both the original and patch options, the patch
69
+ # options value should be used
70
+ assert patched.headers['X-Custom-Header'] == 'custom_value'
71
+ assert patched.headers['X-different-header'] == 'different_value'
72
+ assert patched.base_url == 'https://fake-url.com/'
73
+ assert patched.api_version == 'v1'
74
+ assert patched.timeout == 10000
75
+
76
+
77
+ def test_patch_http_options_appends_version_headers():
78
+ original_options = types.HttpOptions(
79
+ headers={
80
+ 'X-Custom-Header': 'different_value',
81
+ 'X-different-header': 'different_value',
82
+ }
83
+ )
84
+ patch_options = types.HttpOptions(
85
+ base_url='https://fake-url.com/',
86
+ api_version='v1',
87
+ headers={'X-Custom-Header': 'custom_value'},
88
+ timeout=10000,
89
+ )
90
+ patched = _api_client.patch_http_options(original_options, patch_options)
91
+ assert 'user-agent' in patched.headers
92
+ assert 'x-goog-api-client' in patched.headers
93
+
94
+
95
+ def test_setting_timeout_populates_server_timeout_header():
96
+ api_client = _api_client.BaseApiClient(
97
+ vertexai=False,
98
+ api_key='test_api_key',
99
+ http_options=types.HttpOptions(timeout=10000),
100
+ )
101
+ request = api_client._build_request(
102
+ http_method='POST',
103
+ path='sample/path',
104
+ request_dict={},
105
+ )
106
+ assert 'X-Server-Timeout' in request.headers
107
+ assert request.headers['X-Server-Timeout'] == '10'
108
+
109
+
110
+ def test_timeout_rounded_to_nearest_second():
111
+ api_client = _api_client.BaseApiClient(
112
+ vertexai=False,
113
+ api_key='test_api_key',
114
+ )
115
+ http_options = types.HttpOptions(timeout=7300)
116
+ request = api_client._build_request(
117
+ http_method='POST',
118
+ path='sample/path',
119
+ request_dict={},
120
+ http_options=http_options,
121
+ )
122
+ assert request.headers['X-Server-Timeout'] == '8'
123
+
124
+
125
+ def test_server_timeout_not_overwritten():
126
+ api_client = _api_client.BaseApiClient(
127
+ vertexai=False,
128
+ api_key='test_api_key',
129
+ )
130
+ http_options = types.HttpOptions(
131
+ headers={'X-Server-Timeout': '3'},
132
+ timeout=11000)
133
+ request = api_client._build_request(
134
+ http_method='POST',
135
+ path='sample/path',
136
+ request_dict={},
137
+ http_options=http_options,
138
+ )
139
+ assert request.headers['X-Server-Timeout'] == '3'
140
+
141
+
142
+ def test_server_timeout_not_set_by_default():
143
+ api_client = _api_client.BaseApiClient(
144
+ vertexai=False,
145
+ api_key='test_api_key',
146
+ )
147
+ request = api_client._build_request(
148
+ http_method='POST',
149
+ path='sample/path',
150
+ request_dict={},
151
+ )
152
+ assert not 'X-Server-Timeout' in request.headers
153
+
154
+
155
+ def test_resource_scope_without_base_url_raises_error():
156
+ with pytest.raises(ValueError):
157
+ _api_client.BaseApiClient(
158
+ vertexai=True,
159
+ http_options=types.HttpOptions(
160
+ base_url_resource_scope=types.ResourceScope.COLLECTION,
161
+ ),
162
+ )
163
+
164
+
165
+ def test_base_url_resource_scope_not_set_by_default():
166
+ api_client = _api_client.BaseApiClient(
167
+ vertexai=True,
168
+ http_options=types.HttpOptions(
169
+ base_url='https://fake-url.com/',
170
+ ),
171
+ )
172
+
173
+ assert api_client._http_options.base_url_resource_scope is None
174
+
175
+
176
+ def test_retry_options_not_set_by_default():
177
+ options = types.HttpOptions()
178
+ assert options.retry_options is None
@@ -0,0 +1,168 @@
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
+
17
+ """Tests replay client correctly compares requests ignoring key casing."""
18
+
19
+ from ... import _replay_api_client
20
+ from ... import types
21
+
22
+
23
+ def test_equal_objects_with_same_casing_returns_true():
24
+ obj1 = [{
25
+ 'contents': [{'parts': [{'text': 'What is your name?'}], 'role': 'user'}],
26
+ 'generationConfig': {'topK': 2.0, 'maxOutputTokens': 3},
27
+ }]
28
+ obj2 = [{
29
+ 'contents': [{'parts': [{'text': 'What is your name?'}], 'role': 'user'}],
30
+ 'generationConfig': {'topK': 2.0, 'maxOutputTokens': 3},
31
+ }]
32
+
33
+ assert _replay_api_client._equals_ignore_key_case(obj1, obj2)
34
+
35
+
36
+ def test_equal_objects_with_different_casing_returns_true():
37
+ obj1 = [{
38
+ 'contents': [{'parts': [{'text': 'What is your name?'}], 'role': 'user'}],
39
+ 'generationConfig': {'topK': 2.0, 'maxOutputTokens': 3},
40
+ }]
41
+ obj2 = [{
42
+ 'contents': [{'parts': [{'text': 'What is your name?'}], 'role': 'user'}],
43
+ 'generation_config': {'topK': 2.0, 'max_output_tokens': 3},
44
+ }]
45
+
46
+ assert _replay_api_client._equals_ignore_key_case(obj1, obj2)
47
+
48
+
49
+ def test_equal_objects_with_different_nested_casing_returns_true():
50
+ obj1 = [{
51
+ 'contents': [{'parts': [{'text': 'What is your name?'}], 'role': 'user'}],
52
+ 'generationConfig': {'topK': 2.0, 'maxOutputTokens': 3},
53
+ }]
54
+ obj2 = [{
55
+ 'contents': [{'parts': [{'text': 'What is your name?'}], 'role': 'user'}],
56
+ 'generationConfig': {'top_k': 2.0, 'maxOutputTokens': 3},
57
+ }]
58
+
59
+ assert _replay_api_client._equals_ignore_key_case(obj1, obj2)
60
+
61
+ def test_equal_int_float_values_returns_true():
62
+ obj1 = [{
63
+ 'contents': [{'parts': [{'text': 'What is your name?'}], 'role': 'user'}],
64
+ 'generationConfig': {'topK': 2.0, 'maxOutputTokens': 3},
65
+ }]
66
+ obj2 = [{
67
+ 'contents': [{'parts': [{'text': 'What is your name?'}], 'role': 'user'}],
68
+ 'generation_config': {'topK': 2, 'max_output_tokens': 3},
69
+ }]
70
+
71
+ assert _replay_api_client._equals_ignore_key_case(obj1, obj2)
72
+
73
+
74
+ def test_equal_enum_string_values_returns_true():
75
+ obj1 = [{
76
+ 'contents': [{'parts': [{'text': 'What is your name?'}], 'role': 'user'}],
77
+ 'generationConfig': {
78
+ 'topK': 2.0,
79
+ 'maxOutputTokens': 3,
80
+ 'type': types.Type.STRING,
81
+ },
82
+ }]
83
+ obj2 = [{
84
+ 'contents': [{'parts': [{'text': 'What is your name?'}], 'role': 'user'}],
85
+ 'generation_config': {
86
+ 'topK': 2,
87
+ 'max_output_tokens': 3,
88
+ 'type': 'STRING',
89
+ },
90
+ }]
91
+
92
+ assert _replay_api_client._equals_ignore_key_case(obj1, obj2)
93
+
94
+
95
+ def test_equal_enum_values_returns_true():
96
+ obj1 = [{
97
+ 'contents': [{'parts': [{'text': 'What is your name?'}], 'role': 'user'}],
98
+ 'generationConfig': {
99
+ 'topK': 2.0,
100
+ 'maxOutputTokens': 3,
101
+ 'type': types.Type.STRING,
102
+ },
103
+ }]
104
+ obj2 = [{
105
+ 'contents': [{'parts': [{'text': 'What is your name?'}], 'role': 'user'}],
106
+ 'generation_config': {
107
+ 'topK': 2,
108
+ 'max_output_tokens': 3,
109
+ 'type': types.Type.STRING,
110
+ },
111
+ }]
112
+
113
+ assert _replay_api_client._equals_ignore_key_case(obj1, obj2)
114
+
115
+
116
+ def test_different_number_value_returns_false():
117
+ obj1 = [{
118
+ 'contents': [{'parts': [{'text': 'What is your name?'}], 'role': 'user'}],
119
+ 'generationConfig': {'topK': 3, 'maxOutputTokens': 3},
120
+ }]
121
+ obj2 = [{
122
+ 'contents': [{'parts': [{'text': 'What is your name?'}], 'role': 'user'}],
123
+ 'generation_config': {'topK': 2.0, 'max_output_tokens': 3},
124
+ }]
125
+
126
+ assert not _replay_api_client._equals_ignore_key_case(
127
+ obj1, obj2
128
+ )
129
+
130
+
131
+ def test_different_string_value_returns_false():
132
+ obj1 = [{
133
+ 'contents': [{'parts': [{'text': 'What is your name?'}], 'role': 'user'}],
134
+ 'generationConfig': {'topK': 3, 'maxOutputTokens': 3},
135
+ }]
136
+ obj2 = [{
137
+ 'contents': [
138
+ {'parts': [{'text': 'What is your name?'}], 'role': 'model'}
139
+ ],
140
+ 'generation_config': {'topK': 3, 'max_output_tokens': 3},
141
+ }]
142
+
143
+ assert not _replay_api_client._equals_ignore_key_case(
144
+ obj1, obj2
145
+ )
146
+
147
+
148
+ def test_different_enum_values_returns_false():
149
+ obj1 = [{
150
+ 'contents': [{'parts': [{'text': 'What is your name?'}], 'role': 'user'}],
151
+ 'generationConfig': {
152
+ 'topK': 2.0,
153
+ 'maxOutputTokens': 3,
154
+ 'type': types.Type.STRING,
155
+ },
156
+ }]
157
+ obj2 = [{
158
+ 'contents': [{'parts': [{'text': 'What is your name?'}], 'role': 'user'}],
159
+ 'generation_config': {
160
+ 'topK': 2,
161
+ 'max_output_tokens': 3,
162
+ 'type': types.Type.OBJECT,
163
+ },
164
+ }]
165
+
166
+ assert not _replay_api_client._equals_ignore_key_case(
167
+ obj1, obj2
168
+ )