google-genai 1.55.0__py3-none-any.whl → 1.56.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 (241) hide show
  1. google/genai/_interactions/_base_client.py +8 -2
  2. google/genai/_interactions/resources/interactions.py +6 -6
  3. google/genai/_interactions/types/__init__.py +2 -0
  4. google/genai/_interactions/types/audio_content.py +0 -1
  5. google/genai/_interactions/types/audio_content_param.py +0 -1
  6. google/genai/_interactions/types/code_execution_call_content.py +0 -1
  7. google/genai/_interactions/types/code_execution_call_content_param.py +0 -1
  8. google/genai/_interactions/types/code_execution_result_content.py +0 -1
  9. google/genai/_interactions/types/code_execution_result_content_param.py +0 -1
  10. google/genai/_interactions/types/content_delta.py +7 -23
  11. google/genai/_interactions/types/deep_research_agent_config.py +0 -1
  12. google/genai/_interactions/types/deep_research_agent_config_param.py +0 -1
  13. google/genai/_interactions/types/document_content.py +3 -2
  14. google/genai/_interactions/types/document_content_param.py +3 -2
  15. google/genai/{tests/__init__.py → _interactions/types/document_mime_type.py} +5 -3
  16. google/genai/{tests/afc/__init__.py → _interactions/types/document_mime_type_param.py} +8 -4
  17. google/genai/_interactions/types/dynamic_agent_config.py +0 -1
  18. google/genai/_interactions/types/dynamic_agent_config_param.py +0 -1
  19. google/genai/_interactions/types/file_search_result_content.py +0 -1
  20. google/genai/_interactions/types/file_search_result_content_param.py +0 -1
  21. google/genai/_interactions/types/function_call_content.py +0 -1
  22. google/genai/_interactions/types/function_call_content_param.py +0 -1
  23. google/genai/_interactions/types/function_result_content.py +1 -2
  24. google/genai/_interactions/types/function_result_content_param.py +1 -2
  25. google/genai/_interactions/types/google_search_call_content.py +0 -1
  26. google/genai/_interactions/types/google_search_call_content_param.py +0 -1
  27. google/genai/_interactions/types/google_search_result_content.py +0 -1
  28. google/genai/_interactions/types/google_search_result_content_param.py +0 -1
  29. google/genai/_interactions/types/image_content.py +1 -2
  30. google/genai/_interactions/types/image_content_param.py +1 -2
  31. google/genai/_interactions/types/mcp_server_tool_call_content.py +0 -1
  32. google/genai/_interactions/types/mcp_server_tool_call_content_param.py +0 -1
  33. google/genai/_interactions/types/mcp_server_tool_result_content.py +1 -2
  34. google/genai/_interactions/types/mcp_server_tool_result_content_param.py +1 -2
  35. google/genai/_interactions/types/text_content.py +0 -1
  36. google/genai/_interactions/types/text_content_param.py +0 -1
  37. google/genai/_interactions/types/thinking_level.py +1 -1
  38. google/genai/_interactions/types/thought_content.py +0 -1
  39. google/genai/_interactions/types/thought_content_param.py +0 -1
  40. google/genai/_interactions/types/url_context_call_content.py +0 -1
  41. google/genai/_interactions/types/url_context_call_content_param.py +0 -1
  42. google/genai/_interactions/types/url_context_result_content.py +0 -1
  43. google/genai/_interactions/types/url_context_result_content_param.py +0 -1
  44. google/genai/_interactions/types/video_content.py +1 -2
  45. google/genai/_interactions/types/video_content_param.py +1 -2
  46. google/genai/_live_converters.py +2 -30
  47. google/genai/client.py +3 -1
  48. google/genai/models.py +2 -29
  49. google/genai/tunings.py +1 -27
  50. google/genai/types.py +20 -22
  51. google/genai/version.py +1 -1
  52. {google_genai-1.55.0.dist-info → google_genai-1.56.0.dist-info}/METADATA +224 -22
  53. google_genai-1.56.0.dist-info/RECORD +162 -0
  54. google/genai/tests/afc/test_convert_if_exist_pydantic_model.py +0 -309
  55. google/genai/tests/afc/test_convert_number_values_for_function_call_args.py +0 -63
  56. google/genai/tests/afc/test_find_afc_incompatible_tool_indexes.py +0 -240
  57. google/genai/tests/afc/test_generate_content_stream_afc.py +0 -530
  58. google/genai/tests/afc/test_generate_content_stream_afc_thoughts.py +0 -77
  59. google/genai/tests/afc/test_get_function_map.py +0 -176
  60. google/genai/tests/afc/test_get_function_response_parts.py +0 -277
  61. google/genai/tests/afc/test_get_max_remote_calls_for_afc.py +0 -130
  62. google/genai/tests/afc/test_invoke_function_from_dict_args.py +0 -241
  63. google/genai/tests/afc/test_raise_error_for_afc_incompatible_config.py +0 -159
  64. google/genai/tests/afc/test_should_append_afc_history.py +0 -53
  65. google/genai/tests/afc/test_should_disable_afc.py +0 -214
  66. google/genai/tests/batches/__init__.py +0 -17
  67. google/genai/tests/batches/test_cancel.py +0 -77
  68. google/genai/tests/batches/test_create.py +0 -78
  69. google/genai/tests/batches/test_create_with_bigquery.py +0 -113
  70. google/genai/tests/batches/test_create_with_file.py +0 -82
  71. google/genai/tests/batches/test_create_with_gcs.py +0 -125
  72. google/genai/tests/batches/test_create_with_inlined_requests.py +0 -255
  73. google/genai/tests/batches/test_delete.py +0 -86
  74. google/genai/tests/batches/test_embedding.py +0 -157
  75. google/genai/tests/batches/test_get.py +0 -78
  76. google/genai/tests/batches/test_list.py +0 -79
  77. google/genai/tests/caches/__init__.py +0 -17
  78. google/genai/tests/caches/constants.py +0 -29
  79. google/genai/tests/caches/test_create.py +0 -210
  80. google/genai/tests/caches/test_create_custom_url.py +0 -105
  81. google/genai/tests/caches/test_delete.py +0 -54
  82. google/genai/tests/caches/test_delete_custom_url.py +0 -52
  83. google/genai/tests/caches/test_get.py +0 -94
  84. google/genai/tests/caches/test_get_custom_url.py +0 -52
  85. google/genai/tests/caches/test_list.py +0 -68
  86. google/genai/tests/caches/test_update.py +0 -70
  87. google/genai/tests/caches/test_update_custom_url.py +0 -58
  88. google/genai/tests/chats/__init__.py +0 -1
  89. google/genai/tests/chats/test_get_history.py +0 -597
  90. google/genai/tests/chats/test_send_message.py +0 -844
  91. google/genai/tests/chats/test_validate_response.py +0 -90
  92. google/genai/tests/client/__init__.py +0 -17
  93. google/genai/tests/client/test_async_stream.py +0 -427
  94. google/genai/tests/client/test_client_close.py +0 -197
  95. google/genai/tests/client/test_client_initialization.py +0 -1687
  96. google/genai/tests/client/test_client_requests.py +0 -355
  97. google/genai/tests/client/test_custom_client.py +0 -77
  98. google/genai/tests/client/test_http_options.py +0 -178
  99. google/genai/tests/client/test_replay_client_equality.py +0 -168
  100. google/genai/tests/client/test_retries.py +0 -846
  101. google/genai/tests/client/test_upload_errors.py +0 -136
  102. google/genai/tests/common/__init__.py +0 -17
  103. google/genai/tests/common/test_common.py +0 -954
  104. google/genai/tests/conftest.py +0 -162
  105. google/genai/tests/documents/__init__.py +0 -17
  106. google/genai/tests/documents/test_delete.py +0 -51
  107. google/genai/tests/documents/test_get.py +0 -85
  108. google/genai/tests/documents/test_list.py +0 -72
  109. google/genai/tests/errors/__init__.py +0 -1
  110. google/genai/tests/errors/test_api_error.py +0 -417
  111. google/genai/tests/file_search_stores/__init__.py +0 -17
  112. google/genai/tests/file_search_stores/test_create.py +0 -66
  113. google/genai/tests/file_search_stores/test_delete.py +0 -64
  114. google/genai/tests/file_search_stores/test_get.py +0 -94
  115. google/genai/tests/file_search_stores/test_import_file.py +0 -112
  116. google/genai/tests/file_search_stores/test_list.py +0 -57
  117. google/genai/tests/file_search_stores/test_upload_to_file_search_store.py +0 -141
  118. google/genai/tests/files/__init__.py +0 -17
  119. google/genai/tests/files/test_delete.py +0 -46
  120. google/genai/tests/files/test_download.py +0 -85
  121. google/genai/tests/files/test_get.py +0 -46
  122. google/genai/tests/files/test_list.py +0 -72
  123. google/genai/tests/files/test_upload.py +0 -255
  124. google/genai/tests/imports/test_no_optional_imports.py +0 -28
  125. google/genai/tests/interactions/test_integration.py +0 -80
  126. google/genai/tests/live/__init__.py +0 -16
  127. google/genai/tests/live/test_live.py +0 -2177
  128. google/genai/tests/live/test_live_music.py +0 -362
  129. google/genai/tests/live/test_live_response.py +0 -163
  130. google/genai/tests/live/test_send_client_content.py +0 -147
  131. google/genai/tests/live/test_send_realtime_input.py +0 -268
  132. google/genai/tests/live/test_send_tool_response.py +0 -222
  133. google/genai/tests/local_tokenizer/__init__.py +0 -17
  134. google/genai/tests/local_tokenizer/test_local_tokenizer.py +0 -343
  135. google/genai/tests/local_tokenizer/test_local_tokenizer_loader.py +0 -235
  136. google/genai/tests/mcp/__init__.py +0 -17
  137. google/genai/tests/mcp/test_has_mcp_tool_usage.py +0 -89
  138. google/genai/tests/mcp/test_mcp_to_gemini_tools.py +0 -191
  139. google/genai/tests/mcp/test_parse_config_for_mcp_sessions.py +0 -201
  140. google/genai/tests/mcp/test_parse_config_for_mcp_usage.py +0 -130
  141. google/genai/tests/mcp/test_set_mcp_usage_header.py +0 -72
  142. google/genai/tests/models/__init__.py +0 -17
  143. google/genai/tests/models/constants.py +0 -8
  144. google/genai/tests/models/test_compute_tokens.py +0 -120
  145. google/genai/tests/models/test_count_tokens.py +0 -159
  146. google/genai/tests/models/test_delete.py +0 -107
  147. google/genai/tests/models/test_edit_image.py +0 -264
  148. google/genai/tests/models/test_embed_content.py +0 -94
  149. google/genai/tests/models/test_function_call_streaming.py +0 -442
  150. google/genai/tests/models/test_generate_content.py +0 -2502
  151. google/genai/tests/models/test_generate_content_cached_content.py +0 -132
  152. google/genai/tests/models/test_generate_content_config_zero_value.py +0 -103
  153. google/genai/tests/models/test_generate_content_from_apikey.py +0 -44
  154. google/genai/tests/models/test_generate_content_http_options.py +0 -40
  155. google/genai/tests/models/test_generate_content_image_generation.py +0 -143
  156. google/genai/tests/models/test_generate_content_mcp.py +0 -343
  157. google/genai/tests/models/test_generate_content_media_resolution.py +0 -97
  158. google/genai/tests/models/test_generate_content_model.py +0 -139
  159. google/genai/tests/models/test_generate_content_part.py +0 -821
  160. google/genai/tests/models/test_generate_content_thought.py +0 -76
  161. google/genai/tests/models/test_generate_content_tools.py +0 -1761
  162. google/genai/tests/models/test_generate_images.py +0 -191
  163. google/genai/tests/models/test_generate_videos.py +0 -759
  164. google/genai/tests/models/test_get.py +0 -104
  165. google/genai/tests/models/test_list.py +0 -233
  166. google/genai/tests/models/test_recontext_image.py +0 -189
  167. google/genai/tests/models/test_segment_image.py +0 -148
  168. google/genai/tests/models/test_update.py +0 -95
  169. google/genai/tests/models/test_upscale_image.py +0 -157
  170. google/genai/tests/operations/__init__.py +0 -17
  171. google/genai/tests/operations/test_get.py +0 -38
  172. google/genai/tests/public_samples/__init__.py +0 -17
  173. google/genai/tests/public_samples/test_gemini_text_only.py +0 -34
  174. google/genai/tests/pytest_helper.py +0 -229
  175. google/genai/tests/shared/__init__.py +0 -16
  176. google/genai/tests/shared/batches/__init__.py +0 -14
  177. google/genai/tests/shared/batches/test_create_delete.py +0 -57
  178. google/genai/tests/shared/batches/test_create_get_cancel.py +0 -56
  179. google/genai/tests/shared/batches/test_list.py +0 -40
  180. google/genai/tests/shared/caches/__init__.py +0 -14
  181. google/genai/tests/shared/caches/test_create_get_delete.py +0 -67
  182. google/genai/tests/shared/caches/test_create_update_get.py +0 -71
  183. google/genai/tests/shared/caches/test_list.py +0 -40
  184. google/genai/tests/shared/chats/__init__.py +0 -14
  185. google/genai/tests/shared/chats/test_send_message.py +0 -48
  186. google/genai/tests/shared/chats/test_send_message_stream.py +0 -50
  187. google/genai/tests/shared/files/__init__.py +0 -14
  188. google/genai/tests/shared/files/test_list.py +0 -41
  189. google/genai/tests/shared/files/test_upload_get_delete.py +0 -54
  190. google/genai/tests/shared/models/__init__.py +0 -14
  191. google/genai/tests/shared/models/test_compute_tokens.py +0 -41
  192. google/genai/tests/shared/models/test_count_tokens.py +0 -40
  193. google/genai/tests/shared/models/test_edit_image.py +0 -67
  194. google/genai/tests/shared/models/test_embed.py +0 -40
  195. google/genai/tests/shared/models/test_generate_content.py +0 -39
  196. google/genai/tests/shared/models/test_generate_content_stream.py +0 -54
  197. google/genai/tests/shared/models/test_generate_images.py +0 -40
  198. google/genai/tests/shared/models/test_generate_videos.py +0 -38
  199. google/genai/tests/shared/models/test_list.py +0 -37
  200. google/genai/tests/shared/models/test_recontext_image.py +0 -55
  201. google/genai/tests/shared/models/test_segment_image.py +0 -52
  202. google/genai/tests/shared/models/test_upscale_image.py +0 -52
  203. google/genai/tests/shared/tunings/__init__.py +0 -16
  204. google/genai/tests/shared/tunings/test_create.py +0 -46
  205. google/genai/tests/shared/tunings/test_create_get_cancel.py +0 -56
  206. google/genai/tests/shared/tunings/test_list.py +0 -39
  207. google/genai/tests/tokens/__init__.py +0 -16
  208. google/genai/tests/tokens/test_create.py +0 -154
  209. google/genai/tests/transformers/__init__.py +0 -17
  210. google/genai/tests/transformers/test_blobs.py +0 -71
  211. google/genai/tests/transformers/test_bytes.py +0 -15
  212. google/genai/tests/transformers/test_duck_type.py +0 -96
  213. google/genai/tests/transformers/test_function_responses.py +0 -72
  214. google/genai/tests/transformers/test_schema.py +0 -653
  215. google/genai/tests/transformers/test_t_batch.py +0 -286
  216. google/genai/tests/transformers/test_t_content.py +0 -160
  217. google/genai/tests/transformers/test_t_contents.py +0 -398
  218. google/genai/tests/transformers/test_t_part.py +0 -85
  219. google/genai/tests/transformers/test_t_parts.py +0 -87
  220. google/genai/tests/transformers/test_t_tool.py +0 -157
  221. google/genai/tests/transformers/test_t_tools.py +0 -195
  222. google/genai/tests/tunings/__init__.py +0 -16
  223. google/genai/tests/tunings/test_cancel.py +0 -39
  224. google/genai/tests/tunings/test_end_to_end.py +0 -106
  225. google/genai/tests/tunings/test_get.py +0 -67
  226. google/genai/tests/tunings/test_list.py +0 -75
  227. google/genai/tests/tunings/test_tune.py +0 -268
  228. google/genai/tests/types/__init__.py +0 -16
  229. google/genai/tests/types/test_bytes_internal.py +0 -271
  230. google/genai/tests/types/test_bytes_type.py +0 -152
  231. google/genai/tests/types/test_future.py +0 -101
  232. google/genai/tests/types/test_optional_types.py +0 -36
  233. google/genai/tests/types/test_part_type.py +0 -616
  234. google/genai/tests/types/test_schema_from_json_schema.py +0 -417
  235. google/genai/tests/types/test_schema_json_schema.py +0 -468
  236. google/genai/tests/types/test_types.py +0 -2903
  237. google_genai-1.55.0.dist-info/RECORD +0 -345
  238. /google/genai/{tests/interactions/__init__.py → _interactions/py.typed} +0 -0
  239. {google_genai-1.55.0.dist-info → google_genai-1.56.0.dist-info}/WHEEL +0 -0
  240. {google_genai-1.55.0.dist-info → google_genai-1.56.0.dist-info}/licenses/LICENSE +0 -0
  241. {google_genai-1.55.0.dist-info → google_genai-1.56.0.dist-info}/top_level.txt +0 -0
@@ -1,653 +0,0 @@
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 schema processing methods in the _transformers module."""
18
-
19
- import copy
20
- from typing import Union
21
-
22
- import pydantic
23
- import pytest
24
-
25
- from ... import _transformers
26
- from ... import client as google_genai_client_module
27
- from ... import types
28
-
29
-
30
- class CurrencyInfo(pydantic.BaseModel):
31
- name: str
32
- code: str
33
- symbol: str
34
-
35
-
36
- currency_info_fields = CurrencyInfo.model_fields
37
-
38
-
39
- class CountryInfo(pydantic.BaseModel):
40
- name: str
41
- population: int
42
- capital: str
43
- continent: str
44
- gdp: int
45
- official_language: str
46
- total_area_sq_mi: int
47
-
48
-
49
- country_info_fields = CountryInfo.model_fields
50
-
51
-
52
- class CountryInfoWithCurrency(pydantic.BaseModel):
53
- name: str
54
- population: int
55
- capital: str
56
- continent: str
57
- gdp: int
58
- official_language: str
59
- total_area_sq_mi: int
60
- currency: CurrencyInfo
61
-
62
-
63
- nested_country_info_fields = CountryInfoWithCurrency.model_fields
64
-
65
-
66
- class CountryInfoWithNullFields(pydantic.BaseModel):
67
- name: str
68
- population: Union[int, None] = None
69
-
70
-
71
- class CountryInfoWithDefaultValue(pydantic.BaseModel):
72
- name: str
73
- population: int = 0
74
-
75
-
76
- class CountryInfoWithAnyOf(pydantic.BaseModel):
77
- name: str
78
- restaurants_per_capita: Union[int, float]
79
-
80
-
81
- @pytest.fixture
82
- @pytest.mark.parametrize('use_vertex', [True, False])
83
- def client(use_vertex):
84
- if use_vertex:
85
- yield google_genai_client_module.Client(
86
- vertexai=use_vertex, project='test-project', location='test-location'
87
- )
88
- else:
89
- yield google_genai_client_module.Client(
90
- vertexai=use_vertex, api_key='test-api-key'
91
- )
92
-
93
-
94
- def test_build_schema_for_list_of_pydantic_schema(client):
95
- """Tests _build_schema() when list[pydantic.BaseModel] is provided to response_schema."""
96
-
97
- list_schema = _transformers.t_schema(client, CountryInfo).model_dump()
98
-
99
- assert isinstance(list_schema, dict)
100
-
101
- for field_name in list_schema['properties']:
102
- assert 'title' in list_schema['properties'][field_name]
103
- assert 'type' in list_schema['properties'][field_name]
104
- assert field_name in country_info_fields
105
- field_type_str = country_info_fields[field_name].annotation.__name__
106
- assert (
107
- list_schema['properties'][field_name]['type']
108
- .lower()
109
- .startswith(field_type_str.lower())
110
- )
111
- assert 'required' in list_schema
112
- assert list_schema['required'] == list(country_info_fields.keys())
113
-
114
-
115
- def test_build_schema_for_list_of_nested_pydantic_schema(client):
116
- """Tests _build_schema() when list[pydantic.BaseModel] is provided to response_schema and the pydantic.BaseModel has nested pydantic fields."""
117
- list_schema = _transformers.t_schema(
118
- client, CountryInfoWithCurrency
119
- ).model_dump()
120
-
121
- assert isinstance(list_schema, dict)
122
-
123
- for field_name in list_schema['properties']:
124
- assert 'title' in list_schema['properties'][field_name]
125
- assert 'type' in list_schema['properties'][field_name]
126
- assert field_name in nested_country_info_fields
127
-
128
- # Tested nested schema was created
129
- assert 'properties' in list_schema['properties']['currency']
130
-
131
- for field_name in list_schema['properties']['currency']['properties']:
132
- assert field_name in currency_info_fields
133
-
134
-
135
- def test_t_schema_for_pydantic_schema(client):
136
- """Tests t_schema when pydantic.BaseModel is passed to response_schema."""
137
- transformed_schema = _transformers.t_schema(client, CountryInfo)
138
- assert isinstance(transformed_schema, types.Schema)
139
- for schema_property in transformed_schema.properties:
140
- assert schema_property in country_info_fields
141
- assert isinstance(
142
- transformed_schema.properties[schema_property], types.Schema
143
- )
144
-
145
-
146
- def test_t_schema_for_list_of_pydantic_schema(client):
147
- """Tests t_schema when list[pydantic.BaseModel] is passed to response_schema."""
148
- transformed_schema = _transformers.t_schema(client, list[CountryInfo])
149
- assert isinstance(transformed_schema, types.Schema)
150
- assert isinstance(transformed_schema.items, types.Schema)
151
-
152
- for schema_property in transformed_schema.items.properties:
153
- assert schema_property in country_info_fields
154
- assert isinstance(
155
- transformed_schema.items.properties[schema_property], types.Schema
156
- )
157
-
158
-
159
- def test_t_schema_for_null_fields(client):
160
- """Tests t_schema when null fields are present."""
161
- transformed_schema = _transformers.t_schema(client, CountryInfoWithNullFields)
162
- assert isinstance(transformed_schema, types.Schema)
163
- assert transformed_schema.properties['population'].nullable
164
-
165
-
166
- def test_schema_with_no_null_fields_is_unchanged():
167
- """Tests handle_null_fields() doesn't change anything when no null fields are present."""
168
- test_properties = {
169
- 'name': {'title': 'Name', 'type': 'string'},
170
- 'total_area_sq_mi': {
171
- 'anyOf': [{'type': 'integer'}, {'type': 'float'}],
172
- 'default': 'null',
173
- 'title': 'Total Area Sq Mi',
174
- },
175
- }
176
-
177
- for _, schema in test_properties.items():
178
- schema_before = copy.deepcopy(schema)
179
- _transformers.handle_null_fields(schema)
180
- assert schema_before == schema
181
-
182
-
183
- @pytest.mark.parametrize('use_vertex', [True, False])
184
- def test_schema_with_default_value(client):
185
-
186
- transformed_schema = _transformers.t_schema(
187
- client._api_client, CountryInfoWithDefaultValue
188
- )
189
- expected_schema = types.Schema(
190
- properties={
191
- 'name': types.Schema(
192
- type='STRING',
193
- title='Name',
194
- ),
195
- 'population': types.Schema(
196
- type='INTEGER',
197
- default=0,
198
- title='Population',
199
- ),
200
- },
201
- type='OBJECT',
202
- required=['name'],
203
- title='CountryInfoWithDefaultValue',
204
- property_ordering=['name', 'population'],
205
- )
206
-
207
- assert transformed_schema == expected_schema
208
-
209
-
210
- def test_schema_with_any_of(client):
211
- transformed_schema = _transformers.t_schema(client, CountryInfoWithAnyOf)
212
- expected_schema = types.Schema(
213
- properties={
214
- 'name': types.Schema(
215
- type='STRING',
216
- title='Name',
217
- ),
218
- 'restaurants_per_capita': types.Schema(
219
- any_of=[
220
- types.Schema(type='INTEGER'),
221
- types.Schema(type='NUMBER'),
222
- ],
223
- title='Restaurants Per Capita',
224
- ),
225
- },
226
- type='OBJECT',
227
- required=['name', 'restaurants_per_capita'],
228
- title='CountryInfoWithAnyOf',
229
- property_ordering=['name', 'restaurants_per_capita'],
230
- )
231
-
232
- assert transformed_schema == expected_schema
233
-
234
-
235
- @pytest.mark.parametrize('use_vertex', [True, False])
236
- def test_complex_dict_schema_with_anyof_is_unchanged(client):
237
- """When a dict schema is passed to process_schema, the only change should be camel-casing anyOf."""
238
- if client.vertexai:
239
- dict_schema = {
240
- 'type': 'OBJECT',
241
- 'title': 'Fruit Basket',
242
- 'description': 'A structured representation of a fruit basket',
243
- 'required': ['fruit'],
244
- 'properties': {
245
- 'fruit': {
246
- 'type': 'ARRAY',
247
- 'description': 'An ordered list of the fruit in the basket',
248
- 'items': {
249
- 'description': 'A piece of fruit',
250
- 'anyOf': [
251
- {
252
- 'title': 'Apple',
253
- 'description': 'Describes an apple',
254
- 'type': 'OBJECT',
255
- 'properties': {
256
- 'type': {
257
- 'type': 'STRING',
258
- 'description': "Always 'apple'",
259
- },
260
- 'color': {
261
- 'type': 'STRING',
262
- 'description': (
263
- "The color of the apple (e.g., 'red')"
264
- ),
265
- },
266
- },
267
- 'propertyOrdering': ['type', 'color'],
268
- 'required': ['type', 'color'],
269
- },
270
- {
271
- 'title': 'Orange',
272
- 'description': 'Describes an orange',
273
- 'type': 'OBJECT',
274
- 'properties': {
275
- 'type': {
276
- 'type': 'STRING',
277
- 'description': "Always 'orange'",
278
- },
279
- 'size': {
280
- 'type': 'STRING',
281
- 'description': (
282
- 'The size of the orange (e.g.,'
283
- " 'medium')"
284
- ),
285
- },
286
- },
287
- 'propertyOrdering': ['type', 'size'],
288
- 'required': ['type', 'size'],
289
- },
290
- ],
291
- },
292
- }
293
- },
294
- }
295
-
296
- schema_before = copy.deepcopy(dict_schema)
297
- _transformers.process_schema(dict_schema, client)
298
-
299
- assert schema_before == dict_schema
300
-
301
-
302
- @pytest.mark.parametrize('use_vertex', [True, False])
303
- def test_process_schema_converts_const_to_enum(client):
304
- """The 'const' field should be converted to a singleton 'enum'."""
305
- schema = {
306
- 'type': 'STRING',
307
- 'const': 'FOO',
308
- }
309
- expected_schema = {
310
- 'type': 'STRING',
311
- 'enum': ['FOO'],
312
- }
313
-
314
- _transformers.process_schema(schema, client)
315
-
316
- assert schema == expected_schema
317
-
318
-
319
- @pytest.mark.parametrize('use_vertex', [True, False])
320
- def test_process_schema_forbids_non_string_const(client):
321
- """The 'const' field only works for strings."""
322
- schema = {
323
- 'type': 'INTEGER',
324
- 'const': 123,
325
- }
326
-
327
- with pytest.raises(ValueError, match='.*Literal values must be strings.*'):
328
- _transformers.process_schema(schema, client)
329
-
330
-
331
- @pytest.mark.parametrize(
332
- 'use_vertex,order_properties',
333
- [(False, False), (False, True), (True, False), (True, True)],
334
- )
335
- def test_process_schema_order_properties_propagates_into_defs(
336
- client, order_properties
337
- ):
338
- """The `order_properties` setting should apply to '$defs'."""
339
- schema = {
340
- '$ref': '#/$defs/Foo',
341
- '$defs': {
342
- 'Foo': {
343
- 'type': 'OBJECT',
344
- 'properties': {
345
- 'foo': {'type': 'STRING'},
346
- 'bar': {'type': 'STRING'},
347
- },
348
- },
349
- },
350
- }
351
- schema_without_property_ordering = {
352
- 'type': 'OBJECT',
353
- 'properties': {
354
- 'foo': {'type': 'STRING'},
355
- 'bar': {'type': 'STRING'},
356
- },
357
- }
358
- schema_with_property_ordering = {
359
- 'type': 'OBJECT',
360
- 'properties': {
361
- 'foo': {'type': 'STRING'},
362
- 'bar': {'type': 'STRING'},
363
- },
364
- 'property_ordering': ['foo', 'bar'],
365
- }
366
-
367
- _transformers.process_schema(
368
- schema, client, order_properties=order_properties
369
- )
370
-
371
- if order_properties:
372
- assert schema == schema_with_property_ordering
373
- else:
374
- assert schema == schema_without_property_ordering
375
-
376
-
377
- @pytest.mark.parametrize(
378
- 'use_vertex,order_properties',
379
- [(False, False), (False, True), (True, False), (True, True)],
380
- )
381
- def test_process_schema_order_properties_propagates_into_items(
382
- client, order_properties
383
- ):
384
- """The `order_properties` setting should apply to 'items'."""
385
- schema = {
386
- 'type': 'ARRAY',
387
- 'items': {
388
- 'type': 'OBJECT',
389
- 'properties': {
390
- 'foo': {'type': 'STRING'},
391
- 'bar': {'type': 'STRING'},
392
- },
393
- },
394
- }
395
- schema_without_property_ordering = copy.deepcopy(schema)
396
- schema_with_property_ordering = {
397
- 'type': 'ARRAY',
398
- 'items': {
399
- 'type': 'OBJECT',
400
- 'properties': {
401
- 'foo': {'type': 'STRING'},
402
- 'bar': {'type': 'STRING'},
403
- },
404
- 'property_ordering': ['foo', 'bar'],
405
- },
406
- }
407
-
408
- _transformers.process_schema(
409
- schema, client, order_properties=order_properties
410
- )
411
-
412
- if order_properties:
413
- assert schema == schema_with_property_ordering
414
- else:
415
- assert schema == schema_without_property_ordering
416
-
417
-
418
- @pytest.mark.parametrize(
419
- 'use_vertex,order_properties',
420
- [(False, False), (False, True), (True, False), (True, True)],
421
- )
422
- def test_process_schema_order_properties_propagates_into_prefix_items(
423
- client, order_properties
424
- ):
425
- """The `order_properties` setting should apply to 'prefixItems'."""
426
- schema = {
427
- 'type': 'ARRAY',
428
- 'prefixItems': [
429
- {
430
- 'type': 'OBJECT',
431
- 'properties': {
432
- 'foo': {'type': 'STRING'},
433
- 'bar': {'type': 'STRING'},
434
- },
435
- },
436
- ],
437
- }
438
- schema_without_property_ordering = copy.deepcopy(schema)
439
- schema_with_property_ordering = {
440
- 'type': 'ARRAY',
441
- 'prefixItems': [
442
- {
443
- 'type': 'OBJECT',
444
- 'properties': {
445
- 'foo': {'type': 'STRING'},
446
- 'bar': {'type': 'STRING'},
447
- },
448
- 'property_ordering': ['foo', 'bar'],
449
- },
450
- ],
451
- }
452
-
453
- _transformers.process_schema(
454
- schema, client, order_properties=order_properties
455
- )
456
-
457
- if order_properties:
458
- assert schema == schema_with_property_ordering
459
- else:
460
- assert schema == schema_without_property_ordering
461
-
462
-
463
- @pytest.mark.parametrize(
464
- 'use_vertex,order_properties',
465
- [(False, False), (False, True), (True, False), (True, True)],
466
- )
467
- def test_process_schema_order_properties_propagates_into_properties(
468
- client, order_properties
469
- ):
470
- """The `order_properties` setting should apply to 'properties'."""
471
- schema = {
472
- 'type': 'OBJECT',
473
- 'properties': {
474
- 'xyz': {
475
- 'type': 'OBJECT',
476
- 'properties': {
477
- 'foo': {'type': 'STRING'},
478
- 'bar': {'type': 'STRING'},
479
- },
480
- },
481
- 'abc': {'type': 'STRING'},
482
- },
483
- }
484
- schema_without_property_ordering = copy.deepcopy(schema)
485
- schema_with_property_ordering = {
486
- 'type': 'OBJECT',
487
- 'properties': {
488
- 'xyz': {
489
- 'type': 'OBJECT',
490
- 'properties': {
491
- 'foo': {'type': 'STRING'},
492
- 'bar': {'type': 'STRING'},
493
- },
494
- 'property_ordering': ['foo', 'bar'],
495
- },
496
- 'abc': {'type': 'STRING'},
497
- },
498
- 'property_ordering': ['xyz', 'abc'],
499
- }
500
-
501
- _transformers.process_schema(
502
- schema, client, order_properties=order_properties
503
- )
504
-
505
- if order_properties:
506
- assert schema == schema_with_property_ordering
507
- else:
508
- assert schema == schema_without_property_ordering
509
-
510
-
511
- @pytest.mark.parametrize(
512
- 'use_vertex,order_properties',
513
- [(False, False), (False, True), (True, False), (True, True)],
514
- )
515
- def test_process_schema_order_properties_propagates_into_additional_properties(
516
- client, order_properties
517
- ):
518
- """The `order_properties` setting should apply to 'additionalProperties'."""
519
- schema = {
520
- 'type': 'OBJECT',
521
- 'additionalProperties': {
522
- 'type': 'OBJECT',
523
- 'properties': {
524
- 'foo': {'type': 'STRING'},
525
- 'bar': {'type': 'STRING'},
526
- },
527
- },
528
- }
529
- schema_without_property_ordering = copy.deepcopy(schema)
530
- schema_with_property_ordering = {
531
- 'type': 'OBJECT',
532
- 'additionalProperties': {
533
- 'type': 'OBJECT',
534
- 'properties': {
535
- 'foo': {'type': 'STRING'},
536
- 'bar': {'type': 'STRING'},
537
- },
538
- 'property_ordering': ['foo', 'bar'],
539
- },
540
- }
541
-
542
- if client.vertexai:
543
- _transformers.process_schema(
544
- schema, client, order_properties=order_properties
545
- )
546
-
547
- if order_properties:
548
- assert schema == schema_with_property_ordering
549
- else:
550
- assert schema == schema_without_property_ordering
551
- else:
552
- with pytest.raises(ValueError) as e:
553
- _transformers.process_schema(
554
- schema, client, order_properties=order_properties
555
- )
556
- assert 'additionalProperties is not supported in the Gemini API.' in str(e)
557
-
558
-
559
- @pytest.mark.parametrize(
560
- 'use_vertex,order_properties',
561
- [(False, False), (False, True), (True, False), (True, True)],
562
- )
563
- def test_process_schema_order_properties_propagates_into_any_of(
564
- client, order_properties
565
- ):
566
- """The `order_properties` setting should apply to 'anyOf'."""
567
- schema = {
568
- 'anyOf': [
569
- {
570
- 'type': 'OBJECT',
571
- 'properties': {
572
- 'foo': {'type': 'STRING'},
573
- 'bar': {'type': 'STRING'},
574
- },
575
- },
576
- {'type': 'STRING'},
577
- ]
578
- }
579
- schema_without_property_ordering = copy.deepcopy(schema)
580
- schema_with_property_ordering = {
581
- 'anyOf': [
582
- {
583
- 'type': 'OBJECT',
584
- 'properties': {
585
- 'foo': {'type': 'STRING'},
586
- 'bar': {'type': 'STRING'},
587
- },
588
- 'property_ordering': ['foo', 'bar'],
589
- },
590
- {'type': 'STRING'},
591
- ]
592
- }
593
-
594
- _transformers.process_schema(
595
- schema, client, order_properties=order_properties
596
- )
597
-
598
- if order_properties:
599
- assert schema == schema_with_property_ordering
600
- else:
601
- assert schema == schema_without_property_ordering
602
-
603
-
604
- def test_t_schema_does_not_change_property_ordering_if_set(client):
605
- """Tests t_schema doesn't overwrite the property_ordering field if already set."""
606
-
607
- schema = CountryInfo.model_json_schema()
608
- custom_property_ordering = ['code', 'symbol', 'name']
609
- schema['property_ordering'] = custom_property_ordering.copy()
610
-
611
- transformed_schema = _transformers.t_schema(client, schema)
612
- assert transformed_schema.property_ordering == custom_property_ordering
613
-
614
-
615
- def test_t_schema_sets_property_ordering_for_json_schema(client):
616
- """Tests t_schema sets the property_ordering field for json schemas."""
617
-
618
- schema = CountryInfo.model_json_schema()
619
-
620
- transformed_schema = _transformers.t_schema(client, schema)
621
- assert transformed_schema.property_ordering == [
622
- 'name',
623
- 'population',
624
- 'capital',
625
- 'continent',
626
- 'gdp',
627
- 'official_language',
628
- 'total_area_sq_mi',
629
- ]
630
-
631
-
632
- def test_t_schema_sets_property_ordering_for_schema_type(client):
633
- """Tests t_schema sets the property_ordering field for Schema types."""
634
-
635
- schema = types.Schema(
636
- properties={
637
- 'name': types.Schema(
638
- type='STRING',
639
- title='Name',
640
- ),
641
- 'population': types.Schema(
642
- type='INTEGER',
643
- default=0,
644
- title='Population',
645
- ),
646
- },
647
- type='OBJECT',
648
- required=['name'],
649
- title='CountryInfoWithDefaultValue',
650
- )
651
-
652
- transformed_schema = _transformers.t_schema(client, schema)
653
- assert transformed_schema.property_ordering == ['name', 'population']