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,268 @@
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 tunings.tune()."""
18
+
19
+ from ... import types as genai_types
20
+ from .. import pytest_helper
21
+ import pytest
22
+
23
+ evaluation_config=genai_types.EvaluationConfig(
24
+ metrics=[
25
+ genai_types.Metric(name="bleu", prompt_template="test prompt template")
26
+ ],
27
+ output_config=genai_types.OutputConfig(
28
+ gcs_destination=genai_types.GcsDestination(
29
+ output_uri_prefix="gs://sararob_test/"
30
+ )
31
+ ),
32
+ )
33
+
34
+ test_table: list[pytest_helper.TestTableItem] = [
35
+ pytest_helper.TestTableItem(
36
+ name="test_dataset_gcs_uri",
37
+ parameters=genai_types.CreateTuningJobParameters(
38
+ base_model="gemini-2.5-flash",
39
+ training_dataset=genai_types.TuningDataset(
40
+ gcs_uri="gs://cloud-samples-data/ai-platform/generative_ai/gemini-1_5/text/sft_train_data.jsonl",
41
+ ),
42
+ ),
43
+ exception_if_mldev="gcs_uri parameter is not supported in Gemini API.",
44
+ ),
45
+ pytest_helper.TestTableItem(
46
+ name="test_tune_pretuned_model",
47
+ parameters=genai_types.CreateTuningJobParameters(
48
+ base_model="projects/801452371447/locations/us-central1/models/9030969596621881344",
49
+ training_dataset=genai_types.TuningDataset(
50
+ gcs_uri="gs://cloud-samples-data/ai-platform/generative_ai/gemini-2_0/text/sft_train_data.jsonl",
51
+ ),
52
+ ),
53
+ exception_if_mldev="is not supported in Gemini API",
54
+ ),
55
+ pytest_helper.TestTableItem(
56
+ name="test_tune_pretuned_model_with_checkpoint_id",
57
+ parameters=genai_types.CreateTuningJobParameters(
58
+ base_model="projects/801452371447/locations/us-central1/models/9030969596621881344",
59
+ training_dataset=genai_types.TuningDataset(
60
+ gcs_uri="gs://cloud-samples-data/ai-platform/generative_ai/gemini-2_0/text/sft_train_data.jsonl",
61
+ ),
62
+ config=genai_types.CreateTuningJobConfig(
63
+ pre_tuned_model_checkpoint_id="3",
64
+ ),
65
+ ),
66
+ exception_if_mldev="is not supported in Gemini API",
67
+ ),
68
+ pytest_helper.TestTableItem(
69
+ name="test_tune_simple_dpo",
70
+ parameters=genai_types.CreateTuningJobParameters(
71
+ base_model="gemini-2.5-flash",
72
+ training_dataset=genai_types.TuningDataset(
73
+ gcs_uri="gs://cloud-samples-data/ai-platform/generative_ai/gemini-1_5/text/sft_train_data.jsonl",
74
+ ),
75
+ config=genai_types.CreateTuningJobConfig(
76
+ tuned_model_display_name="Model display name",
77
+ epoch_count=1,
78
+ method="PREFERENCE_TUNING",
79
+ ),
80
+ ),
81
+ exception_if_mldev="parameter is not supported in Gemini API.",
82
+ ),
83
+ pytest_helper.TestTableItem(
84
+ name="test_tune_dpo_with_beta",
85
+ parameters=genai_types.CreateTuningJobParameters(
86
+ base_model="gemini-2.5-flash",
87
+ training_dataset=genai_types.TuningDataset(
88
+ gcs_uri="gs://cloud-samples-data/ai-platform/generative_ai/gemini-1_5/text/sft_train_data.jsonl",
89
+ ),
90
+ config=genai_types.CreateTuningJobConfig(
91
+ tuned_model_display_name="Model display name",
92
+ epoch_count=1,
93
+ method=genai_types.TuningMethod.PREFERENCE_TUNING,
94
+ beta=0.5,
95
+ ),
96
+ ),
97
+ exception_if_mldev="parameter is not supported in Gemini API.",
98
+ ),
99
+ pytest_helper.TestTableItem(
100
+ name="test_non_pretuned_model_with_checkpoint_id",
101
+ parameters=genai_types.CreateTuningJobParameters(
102
+ base_model="gemini-2.5-flash",
103
+ training_dataset=genai_types.TuningDataset(
104
+ gcs_uri="gs://cloud-samples-data/ai-platform/generative_ai/gemini-1_5/text/sft_train_data.jsonl",
105
+ ),
106
+ config=genai_types.CreateTuningJobConfig(
107
+ pre_tuned_model_checkpoint_id="3",
108
+ ),
109
+ ),
110
+ exception_if_mldev="is not supported in Gemini API.",
111
+ ),
112
+ pytest_helper.TestTableItem(
113
+ name="test_dataset_gcs_uri_all_parameters",
114
+ parameters=genai_types.CreateTuningJobParameters(
115
+ base_model="gemini-2.5-flash",
116
+ training_dataset=genai_types.TuningDataset(
117
+ gcs_uri="gs://cloud-samples-data/ai-platform/generative_ai/gemini-1_5/text/sft_train_data.jsonl",
118
+ ),
119
+ config=genai_types.CreateTuningJobConfig(
120
+ tuned_model_display_name="Model display name",
121
+ epoch_count=1,
122
+ learning_rate_multiplier=1.0,
123
+ adapter_size="ADAPTER_SIZE_ONE",
124
+ validation_dataset=genai_types.TuningValidationDataset(
125
+ gcs_uri="gs://cloud-samples-data/ai-platform/generative_ai/gemini-1_5/text/sft_validation_data.jsonl",
126
+ ),
127
+ ),
128
+ ),
129
+ exception_if_mldev="gcs_uri parameter is not supported in Gemini API.",
130
+ ),
131
+ pytest_helper.TestTableItem(
132
+ name="test_dataset_vertex_dataset_resource",
133
+ parameters=genai_types.CreateTuningJobParameters(
134
+ base_model="gemini-2.5-flash",
135
+ training_dataset=genai_types.TuningDataset(
136
+ vertex_dataset_resource="projects/801452371447/locations/us-central1/datasets/5779918772206829568",
137
+ ),
138
+ ),
139
+ exception_if_mldev="vertex_dataset_resource parameter is not supported in Gemini API.",
140
+ ),
141
+ pytest_helper.TestTableItem(
142
+ name="test_dataset_dataset_resource_all_parameters",
143
+ parameters=genai_types.CreateTuningJobParameters(
144
+ base_model="gemini-2.5-flash",
145
+ training_dataset=genai_types.TuningDataset(
146
+ vertex_dataset_resource="projects/801452371447/locations/us-central1/datasets/5779918772206829568",
147
+ ),
148
+ config=genai_types.CreateTuningJobConfig(
149
+ tuned_model_display_name="Model display name",
150
+ epoch_count=1,
151
+ learning_rate_multiplier=1.0,
152
+ adapter_size="ADAPTER_SIZE_ONE",
153
+ validation_dataset=genai_types.TuningValidationDataset(
154
+ vertex_dataset_resource="projects/801452371447/locations/us-central1/datasets/1168232753779441664",
155
+ ),
156
+ labels={"testlabelkey": "testlabelvalue"},
157
+ ),
158
+ ),
159
+ exception_if_mldev="vertex_dataset_resource parameter is not supported in Gemini API.",
160
+ ),
161
+ ]
162
+
163
+ pytestmark = pytest_helper.setup(
164
+ file=__file__,
165
+ globals_for_file=globals(),
166
+ test_method="tunings.tune",
167
+ test_table=test_table,
168
+ )
169
+
170
+ pytest_plugins = ("pytest_asyncio",)
171
+
172
+
173
+ @pytest.mark.skipif(
174
+ "config.getoption('--private')",
175
+ reason="Skipping in pre-public tests"
176
+ )
177
+ def test_eval_config(client):
178
+ """Tests tuning with eval config."""
179
+
180
+ if client._api_client.vertexai:
181
+ tuning_job = client.tunings.tune(
182
+ base_model="gemini-2.5-flash",
183
+ training_dataset=genai_types.TuningDataset(gcs_uri="gs://cloud-samples-data/ai-platform/generative_ai/gemini-2_0/text/sft_train_data.jsonl"),
184
+ config=genai_types.CreateTuningJobConfig(
185
+ tuned_model_display_name="tuning job with eval config",
186
+ epoch_count=1,
187
+ learning_rate_multiplier=1.0,
188
+ adapter_size="ADAPTER_SIZE_ONE",
189
+ validation_dataset=genai_types.TuningValidationDataset(
190
+ gcs_uri="gs://cloud-samples-data/ai-platform/generative_ai/gemini-2_0/text/sft_validation_data.jsonl"
191
+ ),
192
+ evaluation_config=evaluation_config,
193
+ ),
194
+ )
195
+
196
+ assert tuning_job.state == genai_types.JobState.JOB_STATE_PENDING
197
+ assert tuning_job.evaluation_config == evaluation_config
198
+
199
+
200
+ @pytest.mark.skipif(
201
+ "config.getoption('--private')",
202
+ reason="Skipping in pre-public tests"
203
+ )
204
+ def test_eval_config_with_metrics(client):
205
+ """Tests tuning with eval config metrics."""
206
+ if client._api_client.vertexai:
207
+ evaluation_config=genai_types.EvaluationConfig(
208
+ metrics=[
209
+ genai_types.Metric(
210
+ name="prompt-relevance",
211
+ prompt_template="How well does the response address the prompt?: PROMPT: {request}\n RESPONSE: {response}\n",
212
+ return_raw_output=True,
213
+ judge_model_system_instruction="You are a cat. Make all evaluations from this perspective.",
214
+ ),
215
+ genai_types.Metric(name='bleu'),
216
+ genai_types.Metric(name='rouge_1'),
217
+ ],
218
+ output_config=genai_types.OutputConfig(
219
+ gcs_destination=genai_types.GcsDestination(
220
+ output_uri_prefix="gs://sararob_test/"
221
+ )
222
+ ),
223
+ autorater_config=genai_types.AutoraterConfig(
224
+ sampling_count=1,
225
+ autorater_model="test-model",
226
+ ),
227
+ )
228
+ tuning_job = client.tunings.tune(
229
+ base_model="gemini-2.5-flash",
230
+ training_dataset=genai_types.TuningDataset(gcs_uri="gs://cloud-samples-data/ai-platform/generative_ai/gemini-2_0/text/sft_train_data.jsonl"),
231
+ config=genai_types.CreateTuningJobConfig(
232
+ tuned_model_display_name="tuning job with eval config",
233
+ epoch_count=1,
234
+ learning_rate_multiplier=1.0,
235
+ adapter_size="ADAPTER_SIZE_ONE",
236
+ validation_dataset=genai_types.TuningValidationDataset(
237
+ gcs_uri="gs://cloud-samples-data/ai-platform/generative_ai/gemini-2_0/text/sft_validation_data.jsonl"
238
+ ),
239
+ evaluation_config=evaluation_config,
240
+ ),
241
+ )
242
+ assert tuning_job.state == genai_types.JobState.JOB_STATE_PENDING
243
+
244
+
245
+ @pytest.mark.skipif(
246
+ "config.getoption('--private')",
247
+ reason="Skipping in pre-public tests"
248
+ )
249
+ @pytest.mark.asyncio
250
+ async def test_eval_config_async(client):
251
+ """Tests tuning with eval config."""
252
+ if client._api_client.vertexai:
253
+ tuning_job = await client.aio.tunings.tune(
254
+ base_model="gemini-2.5-flash",
255
+ training_dataset=genai_types.TuningDataset(gcs_uri="gs://cloud-samples-data/ai-platform/generative_ai/gemini-2_0/text/sft_train_data.jsonl"),
256
+ config=genai_types.CreateTuningJobConfig(
257
+ tuned_model_display_name="tuning job with eval config",
258
+ epoch_count=1,
259
+ learning_rate_multiplier=1.0,
260
+ adapter_size="ADAPTER_SIZE_ONE",
261
+ validation_dataset=genai_types.TuningValidationDataset(
262
+ gcs_uri="gs://cloud-samples-data/ai-platform/generative_ai/gemini-2_0/text/sft_validation_data.jsonl"
263
+ ),
264
+ evaluation_config=evaluation_config,
265
+ ),
266
+ )
267
+
268
+ assert tuning_job.state == genai_types.JobState.JOB_STATE_PENDING
@@ -0,0 +1,16 @@
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
+
@@ -0,0 +1,271 @@
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
+ import base64
18
+ import json
19
+ from unittest import mock
20
+
21
+ import pytest
22
+
23
+ from .. import pytest_helper
24
+
25
+ from ... import _api_client as google_genai_api_client_module
26
+ from ... import _common as common_module
27
+ from ... import client as google_genai_client_module
28
+ from ... import types
29
+
30
+
31
+ # 64 chars in url safe base64.
32
+ _BASE64_URL_SAFE = (
33
+ '-_abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789'
34
+ )
35
+ _RAW_BYTES = (
36
+ b'\xfb\xf6\x9bq\xd7\x9f\x82\x18\xa3\x92Y\xa7\xa2\x9a\xab\xb2\xdb\xaf\xc3\x1c\xb3\x00\x10\x83\x10Q\x87'
37
+ b' \x92\x8b0\xd3\x8fA\x14\x93QU\x97a\x9d5\xdb~9\xeb\xbf='
38
+ )
39
+ # 64 chars in normal base64 is invalid format for pydantic `val_json_bytes`.
40
+ _BASE64_NOT_URL_SAFE = (
41
+ '+/abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789==='
42
+ )
43
+ assert base64.b64decode(_BASE64_NOT_URL_SAFE) == _RAW_BYTES
44
+
45
+
46
+ @pytest.fixture
47
+ def client(use_vertex):
48
+ if use_vertex:
49
+ yield google_genai_client_module.Client(
50
+ vertexai=use_vertex, project='test-project', location='test-location'
51
+ )
52
+ else:
53
+ yield google_genai_client_module.Client(
54
+ vertexai=use_vertex, api_key='test-api-key'
55
+ )
56
+
57
+ @pytest.fixture
58
+ def encode_unserializable_types_method():
59
+ with mock.patch.object(common_module, 'encode_unserializable_types', wraps=common_module.encode_unserializable_types) as method:
60
+ yield method
61
+
62
+
63
+ pytestmark = [pytest.mark.parametrize('use_vertex', [True, False])]
64
+
65
+
66
+ @pytest.fixture
67
+ def mock_request_method():
68
+ with mock.patch.object(
69
+ google_genai_api_client_module.BaseApiClient, 'request'
70
+ ) as request_method:
71
+ yield request_method
72
+
73
+
74
+ # This test checks if user pass in valid base64 string(url safe base64)
75
+ # via pydantic type, then SDK will return the raw bytes in pydantic type.
76
+ @pytest.mark.usefixtures('client', 'mock_request_method', 'encode_unserializable_types_method')
77
+ @pytest.mark.parametrize('bytes_input', [_RAW_BYTES, _BASE64_URL_SAFE])
78
+ def test_base64_pydantic_input_success(
79
+ client, mock_request_method, encode_unserializable_types_method, bytes_input
80
+ ):
81
+ mock_request_method.return_value = types.HttpResponse(
82
+ headers={'header_key': 'header_value'},
83
+ body=json.dumps({
84
+ 'candidates': [
85
+ {'content': {'parts': [{'text': 'Hello World'}], 'role': 'model'}}
86
+ ]
87
+ }),
88
+ )
89
+
90
+ response = client.models.generate_content(
91
+ model='gemini-2.5-flash-001',
92
+ contents=types.Content(
93
+ role='user',
94
+ parts=[
95
+ types.Part(
96
+ inline_data=types.Blob(
97
+ mime_type='image/png',
98
+ data=bytes_input,
99
+ )
100
+ )
101
+ ],
102
+ ),
103
+ )
104
+
105
+ encode_unserializable_types_method.assert_called()
106
+ assert mock_request_method.call_count == 1
107
+ assert (
108
+ pytest_helper.get_value_ignore_key_case(
109
+ mock_request_method.call_args[0][2]['contents'][0]['parts'][0],
110
+ 'inlineData'
111
+ )['data']
112
+ == _BASE64_URL_SAFE
113
+ )
114
+ assert response.candidates[0].content == types.Content(
115
+ role='model',
116
+ parts=[types.Part(text='Hello World')],
117
+ )
118
+
119
+
120
+ # This test checks if user pass in valid base64 string(url safe base64)
121
+ # via dict type, then SDK will return the raw bytes in pydantic type.
122
+ @pytest.mark.usefixtures('client', 'mock_request_method', 'encode_unserializable_types_method')
123
+ @pytest.mark.parametrize('bytes_input', [_RAW_BYTES, _BASE64_URL_SAFE])
124
+ def test_base64_dict_input_success(client, mock_request_method, encode_unserializable_types_method, bytes_input):
125
+ mock_request_method.return_value = types.HttpResponse(
126
+ headers={'header_key': 'header_value'},
127
+ body = json.dumps({
128
+ 'candidates': [
129
+ {'content': {'parts': [{'text': 'Hello World'}], 'role': 'model'}}
130
+ ]
131
+ }),
132
+ )
133
+
134
+ response = client.models.generate_content(
135
+ model='gemini-2.5-flash-001',
136
+ contents={
137
+ 'role': 'user',
138
+ 'parts': [
139
+ {
140
+ 'inlineData': {
141
+ 'mimeType': 'image/png',
142
+ 'data': bytes_input,
143
+ }
144
+ }
145
+ ],
146
+ },
147
+ )
148
+
149
+ encode_unserializable_types_method.assert_called()
150
+ assert mock_request_method.call_count == 1
151
+ assert (
152
+ pytest_helper.get_value_ignore_key_case(
153
+ mock_request_method.call_args[0][2]['contents'][0]['parts'][0],
154
+ 'inlineData'
155
+ )['data']
156
+ == _BASE64_URL_SAFE
157
+ )
158
+ assert response.candidates[0].content == types.Content(
159
+ role='model',
160
+ parts=[types.Part(text='Hello World')],
161
+ )
162
+
163
+
164
+ # This test checks if user pass in invalid base64 string(normal base64 but not
165
+ # url safe) via pydantic type, then SDK will raise ValueError.
166
+ @pytest.mark.usefixtures('client', 'mock_request_method')
167
+ def test_base64_pydantic_input_failure(client):
168
+ with pytest.raises(ValueError, match='Data should be valid base64'):
169
+ client.models.generate_content(
170
+ model='gemini-2.5-flash-001',
171
+ contents=types.Content(
172
+ role='user',
173
+ parts=[
174
+ types.Part(
175
+ inline_data=types.Blob(
176
+ mime_type='image/png',
177
+ data=_BASE64_NOT_URL_SAFE,
178
+ )
179
+ )
180
+ ],
181
+ ),
182
+ )
183
+
184
+
185
+ # This test checks if user pass in invalid base64 string(normal base64 but not
186
+ # url safe) via dict type, then SDK will raise ValueError.
187
+ @pytest.mark.usefixtures('client', 'mock_request_method')
188
+ def test_base64_dict_input_failure(client):
189
+ with pytest.raises(ValueError, match='Data should be valid base64'):
190
+ client.models.generate_content(
191
+ model='gemini-2.5-flash-001',
192
+ contents={
193
+ 'role': 'user',
194
+ 'parts': [{
195
+ 'inlineData': {
196
+ 'mimeType': 'image/png',
197
+ 'data': _BASE64_NOT_URL_SAFE,
198
+ }
199
+ }],
200
+ },
201
+ )
202
+
203
+
204
+ # This test checks if server returns valid base64 string(url safe base64),
205
+ # then SDK will return the raw bytes in pydantic type.
206
+ @pytest.mark.usefixtures('client', 'mock_request_method',)
207
+ def test_base64_pydantic_output_success(client, mock_request_method):
208
+ mock_request_method.return_value = types.HttpResponse(
209
+ headers={'header_key': 'header_value'},
210
+ body=json.dumps({
211
+ 'candidates': [{
212
+ 'content': {
213
+ 'parts': [{
214
+ 'inlineData': {
215
+ 'data': _BASE64_URL_SAFE,
216
+ 'mimeType': 'image/png',
217
+ }
218
+ }],
219
+ 'role': 'model',
220
+ }
221
+ }]
222
+ }),
223
+ )
224
+
225
+ response = client.models.generate_content(
226
+ model='gemini-2.5-flash-001',
227
+ contents=types.Content(
228
+ role='user',
229
+ parts=[types.Part(text='Hello World')],
230
+ ),
231
+ )
232
+
233
+ assert response.candidates[0].content == types.Content(
234
+ role='model',
235
+ parts=[
236
+ types.Part(
237
+ inline_data=types.Blob(data=_RAW_BYTES, mime_type='image/png')
238
+ )
239
+ ],
240
+ )
241
+
242
+
243
+ # This test checks if server returns invalid base64 string(normal base64 but
244
+ # not url safe), then SDK will raise ValueError.
245
+ @pytest.mark.usefixtures('client', 'mock_request_method')
246
+ def test_base64_pydantic_output_failure(client, mock_request_method):
247
+ mock_request_method.return_value = types.HttpResponse(
248
+ headers={'header_key': 'header_value'},
249
+ body=json.dumps({
250
+ 'candidates': [{
251
+ 'content': {
252
+ 'parts': [{
253
+ 'inlineData': {
254
+ 'data': _BASE64_NOT_URL_SAFE,
255
+ 'mimeType': 'image/png',
256
+ }
257
+ }],
258
+ 'role': 'model',
259
+ }
260
+ }]
261
+ }),
262
+ )
263
+
264
+ with pytest.raises(ValueError, match='Data should be valid base64'):
265
+ client.models.generate_content(
266
+ model='gemini-2.5-flash-001',
267
+ contents=types.Content(
268
+ role='user',
269
+ parts=[types.Part(text='Hello World')],
270
+ ),
271
+ )