frogml-core 0.0.88__py3-none-any.whl → 0.0.90__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 (52) hide show
  1. frogml_core/__init__.py +1 -1
  2. frogml_core/clients/instance_template/client.py +6 -4
  3. frogml_core/clients/model_version_manager/client.py +3 -3
  4. frogml_core/clients/prompt_manager/model_descriptor_mapper.py +4 -4
  5. frogml_core/feature_store/_common/artifact_utils.py +3 -3
  6. frogml_core/feature_store/data_sources/batch/athena.py +3 -3
  7. frogml_core/feature_store/feature_sets/context.py +2 -6
  8. frogml_core/feature_store/feature_sets/streaming.py +3 -3
  9. frogml_core/feature_store/feature_sets/streaming_backfill.py +1 -1
  10. frogml_core/feature_store/online/client.py +6 -6
  11. frogml_core/feature_store/sinks/streaming/factory.py +1 -1
  12. frogml_core/inner/build_logic/phases/phase_010_fetch_model/fetch_strategy_manager/strategy/git/git_strategy.py +3 -3
  13. frogml_core/inner/tool/grpc/grpc_try_wrapping.py +0 -17
  14. frogml_core/llmops/provider/openai/provider.py +3 -3
  15. frogml_core/model/tools/adapters/output.py +1 -1
  16. frogml_core/tools/logger/logger.py +1 -1
  17. frogml_core-0.0.90.dist-info/METADATA +412 -0
  18. {frogml_core-0.0.88.dist-info → frogml_core-0.0.90.dist-info}/RECORD +51 -19
  19. frogml_core-0.0.90.dist-info/entry_points.txt +3 -0
  20. frogml_storage/__init__.py +0 -0
  21. frogml_storage/_artifactory_api.py +315 -0
  22. frogml_storage/_environment.py +22 -0
  23. frogml_storage/_log_config.py +45 -0
  24. frogml_storage/_storage_utils.py +15 -0
  25. frogml_storage/_utils.py +69 -0
  26. frogml_storage/authentication/_authentication_utils.py +259 -0
  27. frogml_storage/authentication/models/_auth_config.py +70 -0
  28. frogml_storage/cli/_frogml_cli.py +40 -0
  29. frogml_storage/cli/_login_cli.py +240 -0
  30. frogml_storage/cli/commands/_login_command.py +74 -0
  31. frogml_storage/cli/models/_cli_login_arguments.py +22 -0
  32. frogml_storage/cli/utils/_cli_utils.py +19 -0
  33. frogml_storage/cli/utils/_login_checks_utility.py +114 -0
  34. frogml_storage/constants.py +56 -0
  35. frogml_storage/dataset_manifest.py +13 -0
  36. frogml_storage/entity_manifest.py +93 -0
  37. frogml_storage/exceptions/checksum_verification_error.py +3 -0
  38. frogml_storage/exceptions/validation_error.py +4 -0
  39. frogml_storage/frog_ml.py +668 -0
  40. frogml_storage/frogml_entity_type_info.py +46 -0
  41. frogml_storage/http/__init__.py +0 -0
  42. frogml_storage/http/http_client.py +83 -0
  43. frogml_storage/model_manifest.py +60 -0
  44. frogml_storage/models/_download_context.py +54 -0
  45. frogml_storage/models/frogml_dataset_version.py +21 -0
  46. frogml_storage/models/frogml_entity_version.py +34 -0
  47. frogml_storage/models/frogml_model_version.py +21 -0
  48. frogml_storage/serialization_metadata.py +15 -0
  49. frogml_storage/storage.py +140 -0
  50. frogml_storage/utils/_input_checks_utility.py +104 -0
  51. frogml_core-0.0.88.dist-info/METADATA +0 -46
  52. {frogml_core-0.0.88.dist-info → frogml_core-0.0.90.dist-info}/WHEEL +0 -0
@@ -1,4 +1,4 @@
1
- frogml_core/__init__.py,sha256=twgBbCi-3DxE4i01CUscvYdGwQFPLUrynAGv-9irXi0,777
1
+ frogml_core/__init__.py,sha256=c4ggSJQwUvhzyrdVAlFCR4pkh8FQ7g9foccsc0WU-Sg,777
2
2
  frogml_core/automations/__init__.py,sha256=j2gD15MN-xVWhI5rAFsDwhL0CIyICLNT0scXsKvNBkU,1547
3
3
  frogml_core/automations/automation_executions.py,sha256=xpOb9Dq8gPPGNQDJTvBBZbNz4woZDRZY0HqnLSu7pwU,3230
4
4
  frogml_core/automations/automations.py,sha256=GKEQyQMi8sxX5oZn62PaxPi0zD8IaJRjBkhczRJxHNs,13070
@@ -58,7 +58,7 @@ frogml_core/clients/file_versioning/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQ
58
58
  frogml_core/clients/file_versioning/client.py,sha256=GQ0drq_6f3lyKFSEg_tLh6wybGgyiiPF7TsiHCIPPZU,2533
59
59
  frogml_core/clients/file_versioning/file_tag_filter.py,sha256=Ehgr605zPt2x163EQP5IRXKdnmd_8gNwCVBhxUlRcnI,886
60
60
  frogml_core/clients/instance_template/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
61
- frogml_core/clients/instance_template/client.py,sha256=uKTSebOUwb79t8SosxBbjIg6N9PlXXSYfedOVt5-jjc,2541
61
+ frogml_core/clients/instance_template/client.py,sha256=VDGU7IQgrTo3swnAzvRjBFFeyMqCV1i17LodsROchXE,2597
62
62
  frogml_core/clients/integration_management/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
63
63
  frogml_core/clients/integration_management/integration_manager_client.py,sha256=E6VcgbEI6IbP8TLsatWTTvWNXerdSP08er0KkdRmg90,1240
64
64
  frogml_core/clients/integration_management/integration_utils.py,sha256=j5gomMtYi-CYr2flYF8-BKd5pU4tHxAAlB9R2R3iYoo,968
@@ -75,11 +75,11 @@ frogml_core/clients/logging_client/client.py,sha256=A7qQJWW54Ve7O0DuJX9gmSJlUQyR
75
75
  frogml_core/clients/model_management/__init__.py,sha256=vjWVP8MjmK4_A70WOgJqa6x24AeLK-ABjGJtogGzw9w,43
76
76
  frogml_core/clients/model_management/client.py,sha256=hOWbpO6KkcScT6sn3eNHSp7n2HfQIPug7lm-e-bkzEs,4915
77
77
  frogml_core/clients/model_version_manager/__init__.py,sha256=4Pnfzj4Egps48__dProdbSKQl5pNip0hGJd75w67BfY,46
78
- frogml_core/clients/model_version_manager/client.py,sha256=WeU9XXLu5K0ViS-2YeENq121qV4ESXmI6pzc_K8QkvM,9457
78
+ frogml_core/clients/model_version_manager/client.py,sha256=_qIfPNg-eZLpf9axOHuTUAveK0IQ3gtZGnBOZsskat8,9459
79
79
  frogml_core/clients/project/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
80
80
  frogml_core/clients/project/client.py,sha256=2L4LJS3xIGTAeWuL2HOgW0OyUNiQUHZ4mXfukRLjX2U,2527
81
81
  frogml_core/clients/prompt_manager/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
82
- frogml_core/clients/prompt_manager/model_descriptor_mapper.py,sha256=-j97vEEx0RUNBH245vIIMYw_xrFE-4AgV1hmg7g14U8,7856
82
+ frogml_core/clients/prompt_manager/model_descriptor_mapper.py,sha256=IaIR4mseRiql0mgBrjz0rmioglSGgAR6OTEJeFXlatg,7866
83
83
  frogml_core/clients/prompt_manager/prompt_manager_client.py,sha256=pl5SZ0j81PBMBeJc7jOkDJtJ61TGMqeq3Splg6Tsr90,7958
84
84
  frogml_core/clients/prompt_manager/prompt_proto_mapper.py,sha256=WqIrL1wq_8LkLiLnyVO6vVh7qDLHnZNaHZ7n3s94qgw,10073
85
85
  frogml_core/clients/secret_service/__init__.py,sha256=TdQl1lgplXCKVHYSN4feRIAoonZ7XDz50zALjwVDcM4,40
@@ -111,7 +111,7 @@ frogml_core/exceptions/frogml_suggestion_exception.py,sha256=saqST0umnKRFCscoEBH
111
111
  frogml_core/exceptions/quiet_error.py,sha256=ePdCGP6ta8afjzprMiGoJFY-gxf8albRwuY0t1WF2lY,559
112
112
  frogml_core/feature_store/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
113
113
  frogml_core/feature_store/_common/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
114
- frogml_core/feature_store/_common/artifact_utils.py,sha256=STiDfiYOOX57zYSm2O8oUKIiRNP07w2NaxNcR0CuqWk,2005
114
+ frogml_core/feature_store/_common/artifact_utils.py,sha256=g_DDeeBQ2JyqAuDiAj5utI742Jfyzy0woDW5QOy4vkI,2007
115
115
  frogml_core/feature_store/_common/feature_set_utils.py,sha256=s_GGSHqgMD88c1ij62ptgMo0ar9iEBi4yd6T1wuocwQ,9222
116
116
  frogml_core/feature_store/_common/featureset_asterisk_handler.py,sha256=3kygt0HM6QxDSGIRWcWoDwhOr126wwtv0Xr8W7AUGnk,4721
117
117
  frogml_core/feature_store/_common/functions.py,sha256=kSNYJ7dy48NN09HG9asm4ibQh0JaCGcZYsRDjRWlUHE,659
@@ -125,7 +125,7 @@ frogml_core/feature_store/data_sources/base.py,sha256=sCJ1CzbhRX-fgsw_Y3ucUu-tKC
125
125
  frogml_core/feature_store/data_sources/batch/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
126
126
  frogml_core/feature_store/data_sources/batch/_batch.py,sha256=xnrC1SZ4OH4b7gYLRajEBFQ2U00cA3JxKT8bKNTNHLg,204
127
127
  frogml_core/feature_store/data_sources/batch/_jdbc.py,sha256=lC-m3ZypER5vLtYo4h6ghYWeXvdfGSOdTadqS-sTb58,676
128
- frogml_core/feature_store/data_sources/batch/athena.py,sha256=2brkewwDRKEPiUW27oER1dwbjWOg9Xq3ik1tPSsHBxY,11290
128
+ frogml_core/feature_store/data_sources/batch/athena.py,sha256=fKKdKE1B59dbCiOQQ5amJECoyETwxswkBhDLQ-Pctsg,11292
129
129
  frogml_core/feature_store/data_sources/batch/big_query.py,sha256=h9iscw9I-WyCj1BeyesASNoIiNwPty9acQ37KGD0Ulo,3112
130
130
  frogml_core/feature_store/data_sources/batch/clickhouse.py,sha256=mUSE139-ACnInctQNZqETbnWgFyZidvffVJpwQu0T3Q,2149
131
131
  frogml_core/feature_store/data_sources/batch/csv.py,sha256=saxFeP--CL4h5j2toyxSuOl4kN_WtEEg8pumaYOjCHg,2070
@@ -161,12 +161,12 @@ frogml_core/feature_store/feature_sets/_utils/_featureset_utils.py,sha256=mnznpQ
161
161
  frogml_core/feature_store/feature_sets/backfill.py,sha256=oZqGb8PA1BQxVOxf9-MwzeZyjJqgwulbA-flmBmXAj4,1981
162
162
  frogml_core/feature_store/feature_sets/base_feature_set.py,sha256=1niCItXApA6JZB7tviXJJVaw91sFwSBwiA1f2xOQnwM,5378
163
163
  frogml_core/feature_store/feature_sets/batch.py,sha256=VADm8KVYhyxV7Ibt2luOlnrbrqd774_ICG0iKPs82cU,17422
164
- frogml_core/feature_store/feature_sets/context.py,sha256=RfHXPI6hD7AI7_hZWoLGEhYROeDDsfXTW0OToRB33-s,349
164
+ frogml_core/feature_store/feature_sets/context.py,sha256=zV6r0O70cfM4pmxlfC6xxAtro-wBhenXWwYwF3KwfTY,263
165
165
  frogml_core/feature_store/feature_sets/execution_spec.py,sha256=zKQd7U-PdYkZMqBpA9eIRhhWff-8xxKB_Qo4IDolwGI,2348
166
166
  frogml_core/feature_store/feature_sets/metadata.py,sha256=Vv2pyBbwaJZRFhWKRhxdFyN3AsV-DvTQzLs9nyRMWK0,1888
167
167
  frogml_core/feature_store/feature_sets/read_policies.py,sha256=BQu6B6IZuKJt8Ff5RYeADdqpHmSkec790RIYeSl6Ulo,6844
168
- frogml_core/feature_store/feature_sets/streaming.py,sha256=7IRumnjXgjwnmL7RU8RcW2E8-vLmmL4wDMP5snORt_s,25293
169
- frogml_core/feature_store/feature_sets/streaming_backfill.py,sha256=Nl4q41NvnW8eHtus2aBNdViYc8WqP5he_z9xQgX5i08,9833
168
+ frogml_core/feature_store/feature_sets/streaming.py,sha256=78ItVSojMfLoTY3k0SH6UD40XS7dnVZFn9cKfYUcS3Y,25295
169
+ frogml_core/feature_store/feature_sets/streaming_backfill.py,sha256=u-tjq86AaXAusLUwPYtdCKhTuySQFAtRrUvbPrY3CtI,9834
170
170
  frogml_core/feature_store/feature_sets/transformations/__init__.py,sha256=ozc50AI9RBY71nhNiJDu1-vSWJL2Bdgstyh7GGUW2ig,902
171
171
  frogml_core/feature_store/feature_sets/transformations/aggregations/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
172
172
  frogml_core/feature_store/feature_sets/transformations/aggregations/aggregations.py,sha256=f81U0xiE-TBPqtqGabKb-3ApkIyE0tGU0yIxbF4j-D4,15308
@@ -181,14 +181,14 @@ frogml_core/feature_store/offline/_offline_serving_validations.py,sha256=yfkV8UV
181
181
  frogml_core/feature_store/offline/client_v2.py,sha256=kTFyHAYIsKBe3wcuE1S_LyD9CLQ_yCorOLcHVR3Emms,14966
182
182
  frogml_core/feature_store/offline/feature_set_features.py,sha256=MjrQrXNhzk7QBdCojdpLfy1fuGdP3GcpOgcc7n7H0G8,740
183
183
  frogml_core/feature_store/online/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
184
- frogml_core/feature_store/online/client.py,sha256=JM7GLS8TeXW-jhEbecu2wp-tHoFHxPePqCw1XrsM6lw,13554
184
+ frogml_core/feature_store/online/client.py,sha256=_NCVJngFgh8D3IVBW8u8estPqZ7mSwg5IxmuTndFY0A,13572
185
185
  frogml_core/feature_store/online/endpoint_utils.py,sha256=lGssZR-r8kJpcSozVxQAk1_JpVXgRLqOVrK6fw8flPg,2242
186
186
  frogml_core/feature_store/sinks/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
187
187
  frogml_core/feature_store/sinks/base.py,sha256=QdIutDlO_8IBRr-zKfHBRHJ1-DjDmFfR_Yuad193kg0,361
188
188
  frogml_core/feature_store/sinks/kafka.py,sha256=zlawE62TSShX1D4RWHMowLdVaw5FFThl7kcDBkN2LH0,1923
189
189
  frogml_core/feature_store/sinks/streaming/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
190
190
  frogml_core/feature_store/sinks/streaming/attachment.py,sha256=oHHaxoMtSALI2udHk59joizs15di6OXX-BN_rruWF-c,1158
191
- frogml_core/feature_store/sinks/streaming/factory.py,sha256=tpBkJmc1EC7OdhDO9bNiiVq2XtV7MVLfZJs1RiHt1Fk,2529
191
+ frogml_core/feature_store/sinks/streaming/factory.py,sha256=nal7FWQEZWIKHsVZ4gIwPzCz3WClxL7tDBhkkuCtlsQ,2543
192
192
  frogml_core/feature_store/validations/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
193
193
  frogml_core/feature_store/validations/validation_options.py,sha256=2EbOVC8pmnToP0GUwiPHuz3ew7Vrqr3IdRpUdiLqa0Y,2847
194
194
  frogml_core/feature_store/validations/validation_response.py,sha256=N7lh7h187wBkeuEfVTooE_cGDt_Wo29mEZjmiHR6Pq0,3953
@@ -248,7 +248,7 @@ frogml_core/inner/build_logic/phases/phase_010_fetch_model/fetch_strategy_manage
248
248
  frogml_core/inner/build_logic/phases/phase_010_fetch_model/fetch_strategy_manager/strategy/folder/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
249
249
  frogml_core/inner/build_logic/phases/phase_010_fetch_model/fetch_strategy_manager/strategy/folder/folder_strategy.py,sha256=Uy3HJuiIaGjqVdFyMhyB33QLQxZEHR22K_wx9KrFSaY,5081
250
250
  frogml_core/inner/build_logic/phases/phase_010_fetch_model/fetch_strategy_manager/strategy/git/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
251
- frogml_core/inner/build_logic/phases/phase_010_fetch_model/fetch_strategy_manager/strategy/git/git_strategy.py,sha256=0ZbCGBJ6pxjIBi8t8fT-g2fK-es_vSHboZiPT_hgOkI,5947
251
+ frogml_core/inner/build_logic/phases/phase_010_fetch_model/fetch_strategy_manager/strategy/git/git_strategy.py,sha256=iqNqD79iKY9kgj3e5G2aZwjeLJnOIE4BC9i0Dl0aaII,5949
252
252
  frogml_core/inner/build_logic/phases/phase_010_fetch_model/fetch_strategy_manager/strategy/strategy.py,sha256=MocvnsB_0NAy8ab6NOWRhmNbApGnO_zttyBXvtx8CPE,1447
253
253
  frogml_core/inner/build_logic/phases/phase_010_fetch_model/fetch_strategy_manager/strategy/zip/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
254
254
  frogml_core/inner/build_logic/phases/phase_010_fetch_model/fetch_strategy_manager/strategy/zip/zip_strategy.py,sha256=wCDk5wQlEtCLVDFdC3lXEebpLIMECovHHTvyLsorki8,2276
@@ -285,7 +285,7 @@ frogml_core/inner/tool/auth.py,sha256=H0tQTc8JgEGLxv79IS_28jahi8cq8NNoP3lpxlQshP
285
285
  frogml_core/inner/tool/grpc/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
286
286
  frogml_core/inner/tool/grpc/grpc_auth.py,sha256=WPFWn7CfGp-pSqlnUTplHB6lfuJdzpS6jeidmpyondo,1421
287
287
  frogml_core/inner/tool/grpc/grpc_tools.py,sha256=8sXDWBd_kVRuwmUSdpQNobylT6u8H_83Q8WlZJYOi0c,7247
288
- frogml_core/inner/tool/grpc/grpc_try_wrapping.py,sha256=nMNzmHQcgROBfd_QOTTUCZy-bIcpxPGPiUwsseqIrLg,1172
288
+ frogml_core/inner/tool/grpc/grpc_try_wrapping.py,sha256=8ustbZBz2uYzsz1zFnQydoEGOwK4XhOdHbqojkJ0niY,662
289
289
  frogml_core/inner/tool/protobuf_factory.py,sha256=QBk7ySxHRkxvrC8ICZR7sYizDHIZKQHOcGfo6qXnrDA,1699
290
290
  frogml_core/inner/tool/retry_utils.py,sha256=KcSFJuj02RKF-H9INpCmdiTNXlywEMJ2ClBa00N9aNM,435
291
291
  frogml_core/inner/tool/run_config/__init__.py,sha256=krOWmfbiUyMxa4Z7FHZk3gGZBbMiJINxLxD7XwyUefE,277
@@ -345,7 +345,7 @@ frogml_core/llmops/provider/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMp
345
345
  frogml_core/llmops/provider/chat.py,sha256=n0WY2OhdX__w20DO7vvP53xVIqmr3cVF_G0yiCJsAzU,1626
346
346
  frogml_core/llmops/provider/openai/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
347
347
  frogml_core/llmops/provider/openai/client.py,sha256=JwI9SMUu3bpMoixBAl10iX0J5xKdGFXV0SloPMI6gv8,4534
348
- frogml_core/llmops/provider/openai/provider.py,sha256=cWM3KoxFNOKgMtkKAgSpcFO_D9MipXRBwwC-sQe3uZM,3129
348
+ frogml_core/llmops/provider/openai/provider.py,sha256=2dQQSBxr6hqJFhT4U-KJmsZd-K_vLnOPFeyBMdRntxU,3131
349
349
  frogml_core/model/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
350
350
  frogml_core/model/_entity_extraction.py,sha256=gFeHgKgnz-MpB61w0OfJe30PUh5Cm390nZWSKy0JIls,4669
351
351
  frogml_core/model/adapters/__init__.py,sha256=jMM_0Nf6_vCKccsRB0izs8odCXYlXt-u1sXGh4PcpX4,1739
@@ -393,7 +393,7 @@ frogml_core/model/tools/adapters/input_adapters/image_input.py,sha256=Kh8RHMxqHO
393
393
  frogml_core/model/tools/adapters/input_adapters/json_input.py,sha256=kF3n6zFOjRtaU34PZuDoM9cIHKUe-N8Xx5v3GTW__OA,621
394
394
  frogml_core/model/tools/adapters/input_adapters/string_input.py,sha256=K2SIJy92C-1eX1ReT8iqeFAU8UQWF2d88Zv79jsMrac,151
395
395
  frogml_core/model/tools/adapters/input_adapters/tf_tensor_input.py,sha256=vP0DWrCzn2dh48cN16872QZzyTfWSRKPs5FAUxjZgHk,1415
396
- frogml_core/model/tools/adapters/output.py,sha256=xkYikB2iTcm9oZ8l9pJ3W-vLlORRYy7wPsigkdOB4VQ,2741
396
+ frogml_core/model/tools/adapters/output.py,sha256=JtmNkpPvJRuTqDrfjWo67GdEC9C4j6oZxOW7wKlVmrE,2745
397
397
  frogml_core/model/tools/adapters/output_adapters/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
398
398
  frogml_core/model/tools/adapters/output_adapters/base_output.py,sha256=VXiwnA1499RQSfdFbEU769lRhuOqWTBPfXglKaymhRc,343
399
399
  frogml_core/model/tools/adapters/output_adapters/dataframe_output.py,sha256=f1stPSAEJoceOO4rblmaid3cbN2PmzL-JV408hFQRLY,809
@@ -412,7 +412,7 @@ frogml_core/testing/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSu
412
412
  frogml_core/testing/fixtures.py,sha256=tjWIvdZ2nIfNPs6VtUeGx5coJepQVMUWemKGtqUYPzM,318
413
413
  frogml_core/tools/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
414
414
  frogml_core/tools/logger/__init__.py,sha256=3FfLlYKajB15QA7d2AeDUTfv_UPGz7s2hucPRe_GXdA,115
415
- frogml_core/tools/logger/logger.py,sha256=m38m6niR7QJgcd90t7_5N6ZTKnO-cP2cRmshcbLRa5k,9677
415
+ frogml_core/tools/logger/logger.py,sha256=y4RaCRV8YVFtfUhj8KpSg1WukNoxFM61O6zpt8mv85A,9685
416
416
  frogml_core/tools/logger/logging.yml,sha256=UWC2i3NVKT3j5S8_SapzqClDzXLIEAurzNIXa2tS4UA,1941
417
417
  frogml_core/utils/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
418
418
  frogml_core/utils/datetime_utils.py,sha256=3zK7PUrerMlwB7U6WeuYwMAclVMfPqBNZihNRyIVMs4,581
@@ -1103,6 +1103,38 @@ frogml_services_mock/mocks/workspace_manager_service_mock.py,sha256=WbOiWgOyr-xT
1103
1103
  frogml_services_mock/services_mock.py,sha256=sgKgwhu2W0YOHtzil8x7f1znK_sZr_i27XSeiF4xqVE,21200
1104
1104
  frogml_services_mock/utils/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
1105
1105
  frogml_services_mock/utils/service_utils.py,sha256=ZlB0CnB1J6oBn6_m7fQO2U8tKoboHdUa6ljjkRMYNXU,265
1106
- frogml_core-0.0.88.dist-info/METADATA,sha256=d-_5vh5OvKAIllVEA8nyZrJWJ1zdLFT4gBDIRKxeEvc,2028
1107
- frogml_core-0.0.88.dist-info/WHEEL,sha256=b4K_helf-jlQoXBBETfwnf4B04YC67LOev0jo4fX5m8,88
1108
- frogml_core-0.0.88.dist-info/RECORD,,
1106
+ frogml_storage/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
1107
+ frogml_storage/_artifactory_api.py,sha256=Hn8y7l61fhu9KZJBUr31F5m5gPIOz1UbnvthV4N_6-0,11209
1108
+ frogml_storage/_environment.py,sha256=zuzOJBtBwFaguwn_JkKjfhXStZoustgP30KzOP3mYv8,707
1109
+ frogml_storage/_log_config.py,sha256=ytwyRLYr3SyRlo4UpTjLBtlNqVApP9URTVO5m5DN3uU,1162
1110
+ frogml_storage/_storage_utils.py,sha256=HB2g7uY5A3b33yIcAUM1OjHb5jWsnpESsiDrEviQwrI,366
1111
+ frogml_storage/_utils.py,sha256=JkjJMObF-foFEA693KyrdDoVDHAyI21SW4UPlYrVYvs,1956
1112
+ frogml_storage/authentication/_authentication_utils.py,sha256=RQUNd2c88cOpcRJ-khsmI_XotwvzQ473pde6oyVSUPc,9227
1113
+ frogml_storage/authentication/models/_auth_config.py,sha256=uDcPXaPRXKDEtAhHEuN-GQTeDg8AUL8FE4aiAByHU_I,2024
1114
+ frogml_storage/cli/_frogml_cli.py,sha256=KAv0TrwwHazXoCb_cIScnbSG7NwZLf1nCjxEel642fI,1045
1115
+ frogml_storage/cli/_login_cli.py,sha256=n9sXUd4McD_0nULQErV8vrDmFJQvQTGKRMFEf8U-j64,8564
1116
+ frogml_storage/cli/commands/_login_command.py,sha256=RyeChctCB_wSAoLa2MdvSVKJLYkBxK9oDLoD7PaxvCA,2899
1117
+ frogml_storage/cli/models/_cli_login_arguments.py,sha256=jcFdiWRqjaVAGG9hiYpPHF0WUNT4DqMqVp1FPFGuwOM,723
1118
+ frogml_storage/cli/utils/_cli_utils.py,sha256=hffNsDeA8WnjLTDS2lE_HZLSdYgi7xEElSjuWDFWJrk,581
1119
+ frogml_storage/cli/utils/_login_checks_utility.py,sha256=sRS4-5MZQa3_Vcv7pSylUNZ_AcU8YW64PSCByDqGBCM,3014
1120
+ frogml_storage/constants.py,sha256=Fx-Dyzkl9e28O2qCvIgyEf2aBuhftXAdbn2BIfXFaGw,1542
1121
+ frogml_storage/dataset_manifest.py,sha256=x-1hLFXcX09T_drpCeN19JKZeI4WLD0HF1rXc8jcoa8,296
1122
+ frogml_storage/entity_manifest.py,sha256=xal9plqq7QNhR0hEaISKi2zUfiNmB4ENbyuG_7W4gsU,2796
1123
+ frogml_storage/exceptions/checksum_verification_error.py,sha256=t1muLRYvqza7Q4jo1mYHck9ZoDwlGE7WxcheAFAnIQs,159
1124
+ frogml_storage/exceptions/validation_error.py,sha256=_twu_xcL-O7D26qkskuuR-_NcnPPTADIUO9HGY4g0x0,156
1125
+ frogml_storage/frog_ml.py,sha256=ltESsR1ux8SU-cHNgiUQZ2Ua-Rr4m1eVKrG2Zmzll4k,24238
1126
+ frogml_storage/frogml_entity_type_info.py,sha256=EnPP49U-PdMrTZpFUJzXib5WHPs8Vh4AwIEPC3x2iqM,1376
1127
+ frogml_storage/http/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
1128
+ frogml_storage/http/http_client.py,sha256=B9KwfbX2UdzEDxWCgP0RqUE9BpyxSiyMjm8X7IcMrUo,2418
1129
+ frogml_storage/model_manifest.py,sha256=BVDw_41GVajpWj5WhCoqAk6lIiNMMnIMFKTFHkLuUbU,2186
1130
+ frogml_storage/models/_download_context.py,sha256=pc9KnepPQB9F-MaIe6JkRF0ineKj4B2ycdgr5e37zc4,1660
1131
+ frogml_storage/models/frogml_dataset_version.py,sha256=Uwe50e8eD9XqwARCsa4-iJJqyMQsDqs9E-iuAInvJuA,637
1132
+ frogml_storage/models/frogml_entity_version.py,sha256=-3lRFQSES76YbBAzKuOOlWGcT4T1v2RNQ-54bN8-GH0,921
1133
+ frogml_storage/models/frogml_model_version.py,sha256=2pVHW-BhowBcIScLZzWj5SH0abAirP3cDvXeq334lJI,631
1134
+ frogml_storage/serialization_metadata.py,sha256=KOJ9Yj0XGbYudpcUBMh67Hd7VohmoWsYOvhD-L2yzyY,347
1135
+ frogml_storage/storage.py,sha256=gfkaNzQBgR7-uMHc9-prdiFwyIGb-rO4dKQ49r-TQ3I,5595
1136
+ frogml_storage/utils/_input_checks_utility.py,sha256=bn3_xQcTnsASIHgmvoCyWCFu0akgWCiI8RdXn1AHqsE,3225
1137
+ frogml_core-0.0.90.dist-info/METADATA,sha256=F2lmBOif_Q0HN9FmMh6sAcEZXrhV5WXJsZ_-cAoXtYA,14996
1138
+ frogml_core-0.0.90.dist-info/WHEEL,sha256=b4K_helf-jlQoXBBETfwnf4B04YC67LOev0jo4fX5m8,88
1139
+ frogml_core-0.0.90.dist-info/entry_points.txt,sha256=VvrKJ6Zw6M14z5ynvrzWY05fmyxgWrMm3Ti0r8eeCv0,62
1140
+ frogml_core-0.0.90.dist-info/RECORD,,
@@ -0,0 +1,3 @@
1
+ [console_scripts]
2
+ frogml=frogml_storage.cli._frogml_cli:main
3
+
File without changes
@@ -0,0 +1,315 @@
1
+ import json
2
+ import os
3
+ from typing import Optional
4
+
5
+ from requests import Response
6
+ from tqdm.auto import tqdm
7
+ from tqdm.utils import CallbackIOWrapper
8
+ from urllib3 import Retry
9
+
10
+ from frogml_storage._log_config import logger
11
+ from frogml_storage._utils import join_url
12
+ from frogml_storage.constants import CHECKSUM_SHA2_HEADER
13
+ from frogml_storage.entity_manifest import Checksums, EntityManifest
14
+ from frogml_storage.frogml_entity_type_info import FrogMLEntityTypeInfo
15
+ from frogml_storage.http.http_client import HTTPClient
16
+ from frogml_storage.models._download_context import DownloadContext
17
+
18
+
19
+ class StartTransactionResponse:
20
+ files_upload_path: str
21
+ lead_upload_path: str
22
+ dependencies_upload_path: str
23
+ code_upload_path: str
24
+ transaction_id: str
25
+
26
+ def __init__(
27
+ self,
28
+ files_upload_path,
29
+ lead_upload_path,
30
+ dependencies_upload_path,
31
+ code_upload_path,
32
+ transaction_id,
33
+ ):
34
+ self.files_upload_path = files_upload_path
35
+ self.lead_upload_path = lead_upload_path
36
+ self.dependencies_upload_path = dependencies_upload_path
37
+ self.code_upload_path = code_upload_path
38
+ self.transaction_id = transaction_id
39
+
40
+
41
+ class ArtifactoryApi:
42
+ def __init__(self, uri, auth=None, http_client=None):
43
+ self.uri = uri
44
+ if http_client is not None:
45
+ self.http_client = http_client
46
+ else:
47
+ self.auth = auth
48
+ self.http_client = HTTPClient(auth=auth)
49
+
50
+ def start_transaction(
51
+ self,
52
+ entity_type_info: FrogMLEntityTypeInfo,
53
+ repository: str,
54
+ entity_name: str,
55
+ version: Optional[str],
56
+ ) -> StartTransactionResponse:
57
+ """
58
+ Initializes an upload. Returns transaction ID and upload path
59
+ """
60
+ if version is None:
61
+ start_transaction_url = (
62
+ f"{self.uri}/api/machinelearning/{repository}/"
63
+ f"{entity_type_info.entity_type}/{entity_name}/start-transaction"
64
+ )
65
+ else:
66
+ start_transaction_url = (
67
+ f"{self.uri}/api/machinelearning/{repository}/{entity_type_info.entity_type}"
68
+ f"/{entity_name}/start-transaction/{version}"
69
+ )
70
+ try:
71
+ response = self.http_client.post(start_transaction_url)
72
+ response.raise_for_status()
73
+ files_upload_path = response.json()["filesUploadPath"]
74
+ lead_upload_path = response.json()["leadUploadPath"]
75
+ dependencies_upload_path = response.json()["dependenciesUploadPath"]
76
+ code_upload_path = response.json()["codeUploadPath"]
77
+ transaction_id = response.json()["transactionId"]
78
+ except Exception as exception:
79
+ err = (
80
+ f"Error occurred while trying to start an upload transaction for "
81
+ f"{entity_type_info.entity_type}: '{entity_name}'"
82
+ f" Error: '{exception}'"
83
+ )
84
+ logger.error(err, exc_info=False)
85
+ raise exception
86
+ return StartTransactionResponse(
87
+ files_upload_path=files_upload_path,
88
+ lead_upload_path=lead_upload_path,
89
+ dependencies_upload_path=dependencies_upload_path,
90
+ code_upload_path=code_upload_path,
91
+ transaction_id=transaction_id,
92
+ )
93
+
94
+ def end_transaction(
95
+ self,
96
+ entity_type: FrogMLEntityTypeInfo,
97
+ repository: str,
98
+ entity_name: str,
99
+ entity_manifest: EntityManifest,
100
+ transaction_id: str,
101
+ version: str,
102
+ properties: Optional[dict[str, str]],
103
+ ) -> None:
104
+ """
105
+ Upload model-manifest.json | dataset-manifest.json file, makes the model | dataset available in the repository
106
+ """
107
+ filename = entity_type.metadata_file_name
108
+
109
+ url = join_url(
110
+ self.uri,
111
+ "api",
112
+ "machinelearning",
113
+ repository,
114
+ entity_type.entity_type,
115
+ "entity-manifest",
116
+ entity_name,
117
+ version,
118
+ transaction_id,
119
+ )
120
+
121
+ json_entity_manifest = entity_manifest.to_json()
122
+ self.upload_entity_manifest(
123
+ entity_type=entity_type,
124
+ filename=filename,
125
+ payload=json_entity_manifest,
126
+ url=url,
127
+ properties=properties,
128
+ )
129
+
130
+ def download_file(self, args: DownloadContext) -> None:
131
+ filename = os.path.basename(args.target_path)
132
+ try:
133
+ url = f"{self.uri}/{args.repo_key}/{args.source_url}"
134
+ with self.http_client.get(url=url, stream=True) as response:
135
+ response.raise_for_status()
136
+ total_size = int(response.headers.get("content-length", 0))
137
+ with open(args.target_path, "wb") as file:
138
+ with self.__initialize_progress_bar(total_size, filename) as pbar:
139
+ for chunk in response.iter_content(chunk_size=8192):
140
+ if chunk:
141
+ file.write(chunk)
142
+ pbar.update(len(chunk))
143
+
144
+ except Exception as exception:
145
+ self.__handle_download_exception(exception, args.target_path, filename)
146
+
147
+ def get_entity_manifest(
148
+ self,
149
+ entity_type_info: FrogMLEntityTypeInfo,
150
+ repository: str,
151
+ entity_name: str,
152
+ namespace: Optional[str],
153
+ version: Optional[str],
154
+ ) -> dict:
155
+ url = join_url(
156
+ self.uri,
157
+ "api",
158
+ "machinelearning",
159
+ repository,
160
+ entity_type_info.entity_type,
161
+ "entity-manifest",
162
+ namespace,
163
+ entity_name,
164
+ version,
165
+ )
166
+ try:
167
+ with self.http_client.get(url=url) as r:
168
+ r.raise_for_status()
169
+ return r.json()
170
+ except Exception as exception:
171
+ err = f"Error occurred while trying to get {entity_type_info.entity_type} info file. Error: '{exception}'"
172
+ logger.error(err, exc_info=False)
173
+ raise exception
174
+
175
+ @staticmethod
176
+ def __handle_download_exception(
177
+ exception: Exception, target_path: str, filename: str
178
+ ) -> None:
179
+ if os.path.exists(target_path):
180
+ os.remove(target_path)
181
+ err = f"Error occurred while trying to download file: '{filename}' Error: '{exception}'"
182
+ logger.error(err, exc_info=False)
183
+ raise exception
184
+
185
+ def get_artifact_checksum(self, download_context: DownloadContext) -> str:
186
+ url = f"{self.uri}/{download_context.repo_key}/{download_context.source_url}"
187
+ try:
188
+ with self.http_client.head(url=url, stream=True) as response:
189
+ response.raise_for_status()
190
+ return response.headers.get(CHECKSUM_SHA2_HEADER)
191
+
192
+ except Exception as exception:
193
+ logger.error(exception.__cause__, exc_info=False)
194
+ raise exception
195
+
196
+ def upload_entity_manifest(
197
+ self,
198
+ entity_type: FrogMLEntityTypeInfo,
199
+ filename: str,
200
+ payload: str,
201
+ url: str,
202
+ properties: Optional[dict[str, str]],
203
+ stream: bool = False,
204
+ ) -> None:
205
+ body_part_name = f"{entity_type.body_part_stream}"
206
+
207
+ try:
208
+ files = {
209
+ f"{body_part_name}": (
210
+ f"{body_part_name}",
211
+ payload,
212
+ "application/octet-stream",
213
+ ), # Include the InputStream
214
+ "additionalData": (
215
+ "additionalData",
216
+ json.dumps(properties),
217
+ "application/octet-stream",
218
+ ), # Include the object
219
+ }
220
+ with self.http_client.put(url=url, files=files, stream=stream) as response:
221
+ response.raise_for_status()
222
+ except Exception as exception:
223
+ err = f"Error occurred while trying to upload file: '{filename}' Error: '{exception}'"
224
+ logger.error(err, exc_info=False)
225
+ raise exception
226
+
227
+ def upload_file(self, file_path: str, url: str) -> None:
228
+ wrapped_file = None
229
+ try:
230
+ file_size = os.stat(file_path).st_size
231
+ with (
232
+ self.__initialize_progress_bar(file_size, file_path) as pbar,
233
+ open(file_path, "rb") as file,
234
+ ):
235
+ wrapped_file = CallbackIOWrapper(pbar.update, file, "read")
236
+ with self.http_client.put(url=url, payload=wrapped_file) as response:
237
+ response.raise_for_status()
238
+ except Exception as exception:
239
+ err = f"Error occurred while trying to upload file: '{file_path}' Error: '{exception}'"
240
+ logger.error(err, exc_info=False)
241
+ raise type(exception)(f"{err} File: {file_path}") from exception
242
+ finally:
243
+ if wrapped_file is not None:
244
+ wrapped_file.close()
245
+
246
+ def checksum_deployment(
247
+ self,
248
+ checksum: Checksums,
249
+ url: str,
250
+ full_path: str,
251
+ stream: bool = False,
252
+ ) -> bool:
253
+ response = self.http_client.put(
254
+ url=url,
255
+ headers={"X-Checksum-Sha256": checksum.sha2, "X-Checksum-Deploy": "true"},
256
+ stream=stream,
257
+ )
258
+ if response.status_code != 200 and response.status_code != 201:
259
+ return False
260
+ else:
261
+ file_size = os.path.getsize(full_path)
262
+ pbar = self.__initialize_progress_bar(file_size, full_path)
263
+ pbar.update(file_size)
264
+ pbar.close()
265
+ return True
266
+
267
+ @staticmethod
268
+ def __initialize_progress_bar(total_size: int, filename: str) -> tqdm:
269
+ return tqdm(
270
+ total=total_size, unit="B", unit_scale=True, desc=filename, initial=0
271
+ )
272
+
273
+ def encrypt_password(self) -> Response:
274
+ """
275
+ returns encrypted password as text
276
+ """
277
+ return self.http_client.get(
278
+ url=join_url(self.uri, "/api/security/encryptedPassword")
279
+ )
280
+
281
+ def ping(self) -> Response:
282
+ """
283
+ Sends a ping to Artifactory to validate login status
284
+ """
285
+ url = join_url(self.uri, "api/system/ping")
286
+ return self.http_client.get(url=url)
287
+
288
+ def get_artifactory_version(self) -> Response:
289
+ return self.http_client.get(url=join_url(self.uri, "/api/system/version"))
290
+
291
+ def create_machinelearning_local_repo(self, repo_name: str) -> Response:
292
+ data = {
293
+ "rclass": "local",
294
+ "packageType": "machinelearning",
295
+ }
296
+ return self.http_client.put(
297
+ url=join_url(self.uri, "/api/repositories/" + repo_name), json=data
298
+ )
299
+
300
+ def delete_frogml_local_repo(self, repo_name: str) -> Response:
301
+ return self.http_client.delete(
302
+ url=join_url(self.uri, "/api/repositories/" + repo_name)
303
+ )
304
+
305
+
306
+ class RetryWithLog(Retry):
307
+ """
308
+ Adding extra logs before making a retry request
309
+ """
310
+
311
+ def __init__(self, *args, **kwargs):
312
+ history = kwargs.get("history")
313
+ if history is not None:
314
+ logger.debug(f"Error: ${history[-1].error}\nretrying...")
315
+ super().__init__(*args, **kwargs)
@@ -0,0 +1,22 @@
1
+ import importlib.metadata
2
+ import platform
3
+
4
+
5
+ class Recorder:
6
+ @staticmethod
7
+ def get_environment_dependencies():
8
+ distributions = importlib.metadata.distributions()
9
+ return sorted(
10
+ [f"{dist.metadata['Name']}=={dist.version}" for dist in distributions]
11
+ )
12
+
13
+ @staticmethod
14
+ def get_environment_details():
15
+ return [
16
+ f"arch={platform.architecture()[0]}",
17
+ f"cpu={platform.processor()}",
18
+ f"platform={platform.platform()}",
19
+ f"python_version={platform.python_version()}",
20
+ f"python_implementation={platform.python_implementation()}",
21
+ f"python_compiler={platform.python_compiler()}",
22
+ ]
@@ -0,0 +1,45 @@
1
+ import logging.config
2
+ import os
3
+ import sys
4
+
5
+ log_level = (
6
+ "DEBUG"
7
+ if os.getenv("JFML_DEBUG", "false").casefold() == "true".casefold()
8
+ else "INFO"
9
+ )
10
+ log_file = f'{os.path.expanduser("~")}/.frogml/frogml-log-history.log'
11
+ os.makedirs(os.path.dirname(log_file), exist_ok=True)
12
+
13
+ DEFAULT_LOGGING = {
14
+ "version": 1,
15
+ "formatters": {
16
+ "standard": {
17
+ "format": "%(asctime)s - %(levelname)s - %(name)s.%(module)s.%(funcName)s:%(lineno)d - %(message)s"
18
+ },
19
+ },
20
+ "handlers": {
21
+ "console": {
22
+ "class": "logging.StreamHandler",
23
+ "formatter": "standard",
24
+ "stream": sys.stdout,
25
+ },
26
+ "file": {
27
+ "class": "logging.FileHandler",
28
+ "formatter": "standard",
29
+ "filename": log_file,
30
+ },
31
+ },
32
+ "loggers": {
33
+ __name__: {
34
+ "level": log_level,
35
+ "handlers": ["console", "file"],
36
+ "propagate": False,
37
+ },
38
+ },
39
+ }
40
+
41
+ if os.getenv("IS_LOGGER_SHADED") is not None:
42
+ logger = logging.getLogger(__name__)
43
+ else:
44
+ logging.config.dictConfig(DEFAULT_LOGGING)
45
+ logger = logging.getLogger(__name__)
@@ -0,0 +1,15 @@
1
+ import hashlib
2
+
3
+
4
+ def calculate_sha2(path: str) -> str:
5
+ sha256 = hashlib.sha256()
6
+ with open(path, "rb") as f:
7
+ while chunk := f.read(8192):
8
+ sha256.update(chunk)
9
+ return sha256.hexdigest()
10
+
11
+
12
+ def calc_content_sha2(content: str) -> str:
13
+ sha256 = hashlib.sha256()
14
+ sha256.update(content.encode("utf-8"))
15
+ return sha256.hexdigest()
@@ -0,0 +1,69 @@
1
+ from typing import Optional
2
+ from urllib.parse import urlparse
3
+
4
+ from requests.auth import AuthBase
5
+
6
+ from frogml_storage.frogml_entity_type_info import FrogMLEntityTypeInfo
7
+
8
+
9
+ class BearerAuth(AuthBase):
10
+ def __init__(self, token):
11
+ self.token = token
12
+
13
+ def __call__(self, r):
14
+ r.headers["Authorization"] = f"Bearer {self.token}"
15
+ return r
16
+
17
+ def __eq__(self, other):
18
+ if not isinstance(other, BearerAuth):
19
+ return False
20
+ return self.token == other.token
21
+
22
+
23
+ class EmptyAuth(AuthBase):
24
+ def __call__(self, r):
25
+ return r
26
+
27
+
28
+ def join_url(base_uri: str, *parts: Optional[str]) -> str:
29
+ if base_uri.endswith("/"):
30
+ base_uri = base_uri[:-1]
31
+
32
+ cleaned_parts = [
33
+ part.strip("/") for part in parts if part is not None and part.strip("/")
34
+ ]
35
+ return f"{base_uri}/{'/'.join(cleaned_parts)}"
36
+
37
+
38
+ def assemble_artifact_url(uri: Optional[str]) -> str:
39
+ if uri is None:
40
+ raise Exception("Artifactory URI is required")
41
+
42
+ parsed_url = urlparse(uri)
43
+ if parsed_url.scheme not in ["http", "https"]:
44
+ raise Exception(
45
+ f"Not a valid Artifactory URI: {uri}. "
46
+ f"Artifactory URI example: `https://frogger.jfrog.io/artifactory/ml-local`"
47
+ )
48
+
49
+ return f"{parsed_url.scheme}://{parsed_url.netloc}/artifactory"
50
+
51
+
52
+ # The following method affect e2e tests.
53
+ def build_download_success_log(
54
+ entity_type_info: FrogMLEntityTypeInfo, entity_name: str, version: str
55
+ ) -> str:
56
+ return (
57
+ f'{entity_type_info.entity_type.capitalize()}: "{entity_name}", version: "{version}"'
58
+ f" has been downloaded successfully"
59
+ )
60
+
61
+
62
+ # The following method affect e2e tests.
63
+ def build_upload_success_log(
64
+ entity_type_info: FrogMLEntityTypeInfo, entity_name: str, version: str
65
+ ) -> str:
66
+ return (
67
+ f'{entity_type_info.entity_type.capitalize()}: "{entity_name}", version: "{version}"'
68
+ f" has been uploaded successfully"
69
+ )