lionagi 0.4.0__py3-none-any.whl → 0.5.1__py3-none-any.whl

Sign up to get free protection for your applications and to get access to all the features.
Files changed (584) hide show
  1. lionagi/__init__.py +14 -46
  2. lionagi/core/__init__.py +3 -1
  3. lionagi/core/_class_registry.py +69 -0
  4. lionagi/core/action/__init__.py +3 -13
  5. lionagi/core/action/action_manager.py +287 -0
  6. lionagi/core/action/base.py +109 -0
  7. lionagi/core/action/function_calling.py +127 -92
  8. lionagi/core/action/tool.py +172 -70
  9. lionagi/core/action/types.py +16 -0
  10. lionagi/core/communication/__init__.py +3 -0
  11. lionagi/core/communication/action_request.py +163 -0
  12. lionagi/core/communication/action_response.py +149 -0
  13. lionagi/core/communication/assistant_response.py +161 -0
  14. lionagi/core/communication/base_mail.py +49 -0
  15. lionagi/core/communication/instruction.py +376 -0
  16. lionagi/core/communication/message.py +286 -0
  17. lionagi/core/communication/message_manager.py +530 -0
  18. lionagi/core/communication/system.py +116 -0
  19. lionagi/core/communication/templates/README.md +28 -0
  20. lionagi/core/communication/templates/action_request.jinja2 +5 -0
  21. lionagi/core/communication/templates/action_response.jinja2 +9 -0
  22. lionagi/core/communication/templates/assistant_response.jinja2 +2 -0
  23. lionagi/core/communication/templates/instruction_message.jinja2 +61 -0
  24. lionagi/core/communication/templates/system_message.jinja2 +11 -0
  25. lionagi/core/communication/templates/tool_schemas.jinja2 +7 -0
  26. lionagi/core/communication/types.py +27 -0
  27. lionagi/core/communication/utils.py +254 -0
  28. lionagi/core/forms/__init__.py +3 -0
  29. lionagi/core/forms/base.py +232 -0
  30. lionagi/core/forms/form.py +791 -0
  31. lionagi/core/forms/report.py +321 -0
  32. lionagi/core/forms/types.py +13 -0
  33. lionagi/core/forms/utils.py +26 -0
  34. lionagi/core/generic/__init__.py +3 -6
  35. lionagi/core/generic/component.py +422 -0
  36. lionagi/core/generic/edge.py +143 -101
  37. lionagi/core/generic/element.py +195 -0
  38. lionagi/core/generic/graph.py +297 -180
  39. lionagi/core/generic/log.py +151 -0
  40. lionagi/core/generic/log_manager.py +320 -0
  41. lionagi/core/generic/node.py +7 -229
  42. lionagi/core/generic/pile.py +1017 -0
  43. lionagi/core/generic/progression.py +388 -0
  44. lionagi/core/generic/types.py +23 -0
  45. lionagi/core/generic/utils.py +50 -0
  46. lionagi/core/models/__init__.py +5 -0
  47. lionagi/core/models/base.py +85 -0
  48. lionagi/core/models/field_model.py +122 -0
  49. lionagi/core/models/new_model_params.py +195 -0
  50. lionagi/core/models/note.py +351 -0
  51. lionagi/core/models/operable_model.py +392 -0
  52. lionagi/core/models/schema_model.py +50 -0
  53. lionagi/core/models/types.py +10 -0
  54. lionagi/core/session/__init__.py +3 -0
  55. lionagi/core/session/branch.py +115 -415
  56. lionagi/core/session/branch_mixins.py +545 -0
  57. lionagi/core/session/session.py +122 -257
  58. lionagi/core/session/types.py +8 -0
  59. lionagi/core/typing/__init__.py +9 -0
  60. lionagi/core/typing/concepts.py +132 -0
  61. lionagi/core/typing/config.py +15 -0
  62. lionagi/core/typing/id.py +221 -0
  63. lionagi/core/typing/pydantic_.py +33 -0
  64. lionagi/core/typing/typing_.py +54 -0
  65. lionagi/integrations/__init__.py +0 -1
  66. lionagi/integrations/anthropic_/AnthropicModel.py +268 -0
  67. lionagi/integrations/anthropic_/AnthropicService.py +117 -0
  68. lionagi/integrations/anthropic_/__init__.py +3 -0
  69. lionagi/integrations/anthropic_/anthropic_max_output_token_data.yaml +7 -0
  70. lionagi/integrations/anthropic_/anthropic_price_data.yaml +14 -0
  71. lionagi/integrations/anthropic_/api_endpoints/__init__.py +3 -0
  72. lionagi/integrations/anthropic_/api_endpoints/api_request.py +277 -0
  73. lionagi/integrations/anthropic_/api_endpoints/data_models.py +40 -0
  74. lionagi/integrations/anthropic_/api_endpoints/match_response.py +119 -0
  75. lionagi/integrations/anthropic_/api_endpoints/messages/__init__.py +3 -0
  76. lionagi/integrations/anthropic_/api_endpoints/messages/request/__init__.py +3 -0
  77. lionagi/integrations/anthropic_/api_endpoints/messages/request/message_models.py +14 -0
  78. lionagi/integrations/anthropic_/api_endpoints/messages/request/request_body.py +74 -0
  79. lionagi/integrations/anthropic_/api_endpoints/messages/response/content_models.py +32 -0
  80. lionagi/integrations/anthropic_/api_endpoints/messages/response/response_body.py +101 -0
  81. lionagi/integrations/anthropic_/api_endpoints/messages/response/usage_models.py +25 -0
  82. lionagi/integrations/anthropic_/version.py +5 -0
  83. lionagi/integrations/groq_/GroqModel.py +318 -0
  84. lionagi/integrations/groq_/GroqService.py +151 -0
  85. lionagi/integrations/groq_/__init__.py +3 -0
  86. lionagi/integrations/groq_/api_endpoints/data_models.py +187 -0
  87. lionagi/integrations/groq_/api_endpoints/groq_request.py +288 -0
  88. lionagi/integrations/groq_/api_endpoints/match_response.py +106 -0
  89. lionagi/integrations/groq_/api_endpoints/response_utils.py +105 -0
  90. lionagi/integrations/groq_/groq_max_output_token_data.yaml +21 -0
  91. lionagi/integrations/groq_/groq_price_data.yaml +58 -0
  92. lionagi/integrations/groq_/groq_rate_limits.yaml +105 -0
  93. lionagi/integrations/groq_/version.py +5 -0
  94. lionagi/integrations/litellm_/__init__.py +3 -0
  95. lionagi/integrations/litellm_/imodel.py +73 -0
  96. lionagi/integrations/ollama_/OllamaModel.py +244 -0
  97. lionagi/integrations/ollama_/OllamaService.py +142 -0
  98. lionagi/integrations/ollama_/__init__.py +3 -0
  99. lionagi/integrations/ollama_/api_endpoints/__init__.py +3 -0
  100. lionagi/integrations/ollama_/api_endpoints/api_request.py +179 -0
  101. lionagi/integrations/ollama_/api_endpoints/chat_completion/__init__.py +3 -0
  102. lionagi/integrations/ollama_/api_endpoints/chat_completion/message_models.py +31 -0
  103. lionagi/integrations/ollama_/api_endpoints/chat_completion/request_body.py +46 -0
  104. lionagi/integrations/ollama_/api_endpoints/chat_completion/response_body.py +67 -0
  105. lionagi/integrations/ollama_/api_endpoints/chat_completion/tool_models.py +49 -0
  106. lionagi/integrations/ollama_/api_endpoints/completion/request_body.py +72 -0
  107. lionagi/integrations/ollama_/api_endpoints/completion/response_body.py +59 -0
  108. lionagi/integrations/ollama_/api_endpoints/data_models.py +15 -0
  109. lionagi/integrations/ollama_/api_endpoints/embedding/request_body.py +33 -0
  110. lionagi/integrations/ollama_/api_endpoints/embedding/response_body.py +29 -0
  111. lionagi/integrations/ollama_/api_endpoints/match_data_model.py +62 -0
  112. lionagi/integrations/ollama_/api_endpoints/match_response.py +190 -0
  113. lionagi/integrations/ollama_/api_endpoints/model/__init__.py +3 -0
  114. lionagi/integrations/ollama_/api_endpoints/model/copy_model.py +13 -0
  115. lionagi/integrations/ollama_/api_endpoints/model/create_model.py +28 -0
  116. lionagi/integrations/ollama_/api_endpoints/model/delete_model.py +11 -0
  117. lionagi/integrations/ollama_/api_endpoints/model/list_model.py +60 -0
  118. lionagi/integrations/ollama_/api_endpoints/model/pull_model.py +34 -0
  119. lionagi/integrations/ollama_/api_endpoints/model/push_model.py +35 -0
  120. lionagi/integrations/ollama_/api_endpoints/model/show_model.py +36 -0
  121. lionagi/integrations/ollama_/api_endpoints/option_models.py +68 -0
  122. lionagi/integrations/openai_/OpenAIModel.py +418 -0
  123. lionagi/integrations/openai_/OpenAIService.py +426 -0
  124. lionagi/integrations/openai_/api_endpoints/__init__.py +3 -0
  125. lionagi/integrations/openai_/api_endpoints/api_request.py +277 -0
  126. lionagi/integrations/openai_/api_endpoints/audio/__init__.py +9 -0
  127. lionagi/integrations/openai_/api_endpoints/audio/speech_models.py +34 -0
  128. lionagi/integrations/openai_/api_endpoints/audio/transcription_models.py +136 -0
  129. lionagi/integrations/openai_/api_endpoints/audio/translation_models.py +41 -0
  130. lionagi/integrations/openai_/api_endpoints/audio/types.py +41 -0
  131. lionagi/integrations/openai_/api_endpoints/batch/__init__.py +17 -0
  132. lionagi/integrations/openai_/api_endpoints/batch/batch_models.py +146 -0
  133. lionagi/integrations/openai_/api_endpoints/batch/cancel_batch.py +7 -0
  134. lionagi/integrations/openai_/api_endpoints/batch/create_batch.py +26 -0
  135. lionagi/integrations/openai_/api_endpoints/batch/list_batch.py +37 -0
  136. lionagi/integrations/openai_/api_endpoints/batch/request_object_models.py +65 -0
  137. lionagi/integrations/openai_/api_endpoints/batch/retrieve_batch.py +7 -0
  138. lionagi/integrations/openai_/api_endpoints/batch/types.py +4 -0
  139. lionagi/integrations/openai_/api_endpoints/chat_completions/__init__.py +1 -0
  140. lionagi/integrations/openai_/api_endpoints/chat_completions/request/__init__.py +39 -0
  141. lionagi/integrations/openai_/api_endpoints/chat_completions/request/message_models.py +121 -0
  142. lionagi/integrations/openai_/api_endpoints/chat_completions/request/request_body.py +221 -0
  143. lionagi/integrations/openai_/api_endpoints/chat_completions/request/response_format.py +71 -0
  144. lionagi/integrations/openai_/api_endpoints/chat_completions/request/stream_options.py +14 -0
  145. lionagi/integrations/openai_/api_endpoints/chat_completions/request/tool_choice_models.py +17 -0
  146. lionagi/integrations/openai_/api_endpoints/chat_completions/request/tool_models.py +54 -0
  147. lionagi/integrations/openai_/api_endpoints/chat_completions/request/types.py +18 -0
  148. lionagi/integrations/openai_/api_endpoints/chat_completions/response/choice_models.py +62 -0
  149. lionagi/integrations/openai_/api_endpoints/chat_completions/response/function_models.py +16 -0
  150. lionagi/integrations/openai_/api_endpoints/chat_completions/response/log_prob_models.py +47 -0
  151. lionagi/integrations/openai_/api_endpoints/chat_completions/response/message_models.py +25 -0
  152. lionagi/integrations/openai_/api_endpoints/chat_completions/response/response_body.py +99 -0
  153. lionagi/integrations/openai_/api_endpoints/chat_completions/response/types.py +8 -0
  154. lionagi/integrations/openai_/api_endpoints/chat_completions/response/usage_models.py +24 -0
  155. lionagi/integrations/openai_/api_endpoints/chat_completions/util.py +46 -0
  156. lionagi/integrations/openai_/api_endpoints/data_models.py +23 -0
  157. lionagi/integrations/openai_/api_endpoints/embeddings/__init__.py +3 -0
  158. lionagi/integrations/openai_/api_endpoints/embeddings/request_body.py +79 -0
  159. lionagi/integrations/openai_/api_endpoints/embeddings/response_body.py +67 -0
  160. lionagi/integrations/openai_/api_endpoints/files/__init__.py +11 -0
  161. lionagi/integrations/openai_/api_endpoints/files/delete_file.py +20 -0
  162. lionagi/integrations/openai_/api_endpoints/files/file_models.py +56 -0
  163. lionagi/integrations/openai_/api_endpoints/files/list_files.py +27 -0
  164. lionagi/integrations/openai_/api_endpoints/files/retrieve_file.py +9 -0
  165. lionagi/integrations/openai_/api_endpoints/files/upload_file.py +38 -0
  166. lionagi/integrations/openai_/api_endpoints/fine_tuning/__init__.py +37 -0
  167. lionagi/integrations/openai_/api_endpoints/fine_tuning/cancel_jobs.py +9 -0
  168. lionagi/integrations/openai_/api_endpoints/fine_tuning/create_jobs.py +133 -0
  169. lionagi/integrations/openai_/api_endpoints/fine_tuning/fine_tuning_job_checkpoint_models.py +58 -0
  170. lionagi/integrations/openai_/api_endpoints/fine_tuning/fine_tuning_job_event_models.py +31 -0
  171. lionagi/integrations/openai_/api_endpoints/fine_tuning/fine_tuning_job_models.py +140 -0
  172. lionagi/integrations/openai_/api_endpoints/fine_tuning/list_fine_tuning_checkpoints.py +51 -0
  173. lionagi/integrations/openai_/api_endpoints/fine_tuning/list_fine_tuning_events.py +42 -0
  174. lionagi/integrations/openai_/api_endpoints/fine_tuning/list_fine_tuning_jobs.py +31 -0
  175. lionagi/integrations/openai_/api_endpoints/fine_tuning/retrieve_jobs.py +9 -0
  176. lionagi/integrations/openai_/api_endpoints/fine_tuning/training_format.py +30 -0
  177. lionagi/integrations/openai_/api_endpoints/images/__init__.py +9 -0
  178. lionagi/integrations/openai_/api_endpoints/images/image_edit_models.py +69 -0
  179. lionagi/integrations/openai_/api_endpoints/images/image_models.py +56 -0
  180. lionagi/integrations/openai_/api_endpoints/images/image_variation_models.py +56 -0
  181. lionagi/integrations/openai_/api_endpoints/images/response_body.py +30 -0
  182. lionagi/integrations/openai_/api_endpoints/match_data_model.py +197 -0
  183. lionagi/integrations/openai_/api_endpoints/match_response.py +336 -0
  184. lionagi/integrations/openai_/api_endpoints/models/__init__.py +7 -0
  185. lionagi/integrations/openai_/api_endpoints/models/delete_fine_tuned_model.py +17 -0
  186. lionagi/integrations/openai_/api_endpoints/models/models_models.py +31 -0
  187. lionagi/integrations/openai_/api_endpoints/models/retrieve_model.py +9 -0
  188. lionagi/integrations/openai_/api_endpoints/moderations/__init__.py +3 -0
  189. lionagi/integrations/openai_/api_endpoints/moderations/request_body.py +20 -0
  190. lionagi/integrations/openai_/api_endpoints/moderations/response_body.py +139 -0
  191. lionagi/integrations/openai_/api_endpoints/uploads/__init__.py +19 -0
  192. lionagi/integrations/openai_/api_endpoints/uploads/add_upload_part.py +11 -0
  193. lionagi/integrations/openai_/api_endpoints/uploads/cancel_upload.py +7 -0
  194. lionagi/integrations/openai_/api_endpoints/uploads/complete_upload.py +18 -0
  195. lionagi/integrations/openai_/api_endpoints/uploads/create_upload.py +17 -0
  196. lionagi/integrations/openai_/api_endpoints/uploads/uploads_models.py +52 -0
  197. lionagi/integrations/openai_/image_token_calculator/image_token_calculator.py +92 -0
  198. lionagi/integrations/openai_/image_token_calculator/openai_image_token_data.yaml +15 -0
  199. lionagi/integrations/openai_/openai_max_output_token_data.yaml +12 -0
  200. lionagi/integrations/openai_/openai_price_data.yaml +26 -0
  201. lionagi/integrations/openai_/version.py +1 -0
  202. lionagi/integrations/pandas_/__init__.py +24 -0
  203. lionagi/integrations/pandas_/extend_df.py +61 -0
  204. lionagi/integrations/pandas_/read.py +103 -0
  205. lionagi/integrations/pandas_/remove_rows.py +61 -0
  206. lionagi/integrations/pandas_/replace_keywords.py +65 -0
  207. lionagi/integrations/pandas_/save.py +131 -0
  208. lionagi/integrations/pandas_/search_keywords.py +69 -0
  209. lionagi/integrations/pandas_/to_df.py +196 -0
  210. lionagi/integrations/pandas_/update_cells.py +54 -0
  211. lionagi/integrations/perplexity_/PerplexityModel.py +269 -0
  212. lionagi/integrations/perplexity_/PerplexityService.py +113 -0
  213. lionagi/integrations/perplexity_/__init__.py +3 -0
  214. lionagi/integrations/perplexity_/api_endpoints/api_request.py +171 -0
  215. lionagi/integrations/perplexity_/api_endpoints/chat_completions/request/request_body.py +121 -0
  216. lionagi/integrations/perplexity_/api_endpoints/chat_completions/response/response_body.py +146 -0
  217. lionagi/integrations/perplexity_/api_endpoints/data_models.py +63 -0
  218. lionagi/integrations/perplexity_/api_endpoints/match_response.py +26 -0
  219. lionagi/integrations/perplexity_/perplexity_max_output_token_data.yaml +3 -0
  220. lionagi/integrations/perplexity_/perplexity_price_data.yaml +10 -0
  221. lionagi/integrations/perplexity_/version.py +1 -0
  222. lionagi/integrations/pydantic_/__init__.py +8 -0
  223. lionagi/integrations/pydantic_/break_down_annotation.py +81 -0
  224. lionagi/integrations/pydantic_/new_model.py +208 -0
  225. lionagi/integrations/services.py +17 -0
  226. lionagi/libs/__init__.py +0 -55
  227. lionagi/libs/compress/models.py +62 -0
  228. lionagi/libs/compress/utils.py +81 -0
  229. lionagi/libs/constants.py +98 -0
  230. lionagi/libs/file/chunk.py +265 -0
  231. lionagi/libs/file/file_ops.py +114 -0
  232. lionagi/libs/file/params.py +212 -0
  233. lionagi/libs/file/path.py +301 -0
  234. lionagi/libs/file/process.py +139 -0
  235. lionagi/libs/file/save.py +90 -0
  236. lionagi/libs/file/types.py +22 -0
  237. lionagi/libs/func/async_calls/__init__.py +21 -0
  238. lionagi/libs/func/async_calls/alcall.py +157 -0
  239. lionagi/libs/func/async_calls/bcall.py +82 -0
  240. lionagi/libs/func/async_calls/mcall.py +134 -0
  241. lionagi/libs/func/async_calls/pcall.py +149 -0
  242. lionagi/libs/func/async_calls/rcall.py +185 -0
  243. lionagi/libs/func/async_calls/tcall.py +114 -0
  244. lionagi/libs/func/async_calls/ucall.py +85 -0
  245. lionagi/libs/func/decorators.py +277 -0
  246. lionagi/libs/func/lcall.py +57 -0
  247. lionagi/libs/func/params.py +64 -0
  248. lionagi/libs/func/throttle.py +119 -0
  249. lionagi/libs/func/types.py +39 -0
  250. lionagi/libs/func/utils.py +96 -0
  251. lionagi/libs/package/imports.py +162 -0
  252. lionagi/libs/package/management.py +58 -0
  253. lionagi/libs/package/params.py +26 -0
  254. lionagi/libs/package/system.py +18 -0
  255. lionagi/libs/package/types.py +26 -0
  256. lionagi/libs/parse/__init__.py +1 -0
  257. lionagi/libs/parse/flatten/__init__.py +9 -0
  258. lionagi/libs/parse/flatten/flatten.py +168 -0
  259. lionagi/libs/parse/flatten/params.py +52 -0
  260. lionagi/libs/parse/flatten/unflatten.py +79 -0
  261. lionagi/libs/parse/json/__init__.py +27 -0
  262. lionagi/libs/parse/json/as_readable.py +104 -0
  263. lionagi/libs/parse/json/extract.py +102 -0
  264. lionagi/libs/parse/json/parse.py +179 -0
  265. lionagi/libs/parse/json/schema.py +227 -0
  266. lionagi/libs/parse/json/to_json.py +71 -0
  267. lionagi/libs/parse/nested/__init__.py +33 -0
  268. lionagi/libs/parse/nested/nfilter.py +55 -0
  269. lionagi/libs/parse/nested/nget.py +40 -0
  270. lionagi/libs/parse/nested/ninsert.py +103 -0
  271. lionagi/libs/parse/nested/nmerge.py +155 -0
  272. lionagi/libs/parse/nested/npop.py +66 -0
  273. lionagi/libs/parse/nested/nset.py +89 -0
  274. lionagi/libs/parse/nested/to_flat_list.py +64 -0
  275. lionagi/libs/parse/nested/utils.py +185 -0
  276. lionagi/libs/parse/string_parse/__init__.py +11 -0
  277. lionagi/libs/parse/string_parse/code_block.py +73 -0
  278. lionagi/libs/parse/string_parse/docstring.py +179 -0
  279. lionagi/libs/parse/string_parse/function_.py +92 -0
  280. lionagi/libs/parse/type_convert/__init__.py +19 -0
  281. lionagi/libs/parse/type_convert/params.py +145 -0
  282. lionagi/libs/parse/type_convert/to_dict.py +333 -0
  283. lionagi/libs/parse/type_convert/to_list.py +186 -0
  284. lionagi/libs/parse/type_convert/to_num.py +358 -0
  285. lionagi/libs/parse/type_convert/to_str.py +195 -0
  286. lionagi/libs/parse/types.py +9 -0
  287. lionagi/libs/parse/validate/__init__.py +14 -0
  288. lionagi/libs/parse/validate/boolean.py +96 -0
  289. lionagi/libs/parse/validate/keys.py +150 -0
  290. lionagi/libs/parse/validate/mapping.py +109 -0
  291. lionagi/libs/parse/validate/params.py +62 -0
  292. lionagi/libs/parse/xml/__init__.py +10 -0
  293. lionagi/libs/parse/xml/convert.py +56 -0
  294. lionagi/libs/parse/xml/parser.py +93 -0
  295. lionagi/libs/string_similarity/__init__.py +32 -0
  296. lionagi/libs/string_similarity/algorithms.py +219 -0
  297. lionagi/libs/string_similarity/matcher.py +102 -0
  298. lionagi/libs/string_similarity/utils.py +15 -0
  299. lionagi/libs/utils.py +255 -0
  300. lionagi/operations/__init__.py +3 -6
  301. lionagi/operations/brainstorm/__init__.py +3 -0
  302. lionagi/operations/brainstorm/brainstorm.py +204 -0
  303. lionagi/operations/brainstorm/prompt.py +1 -0
  304. lionagi/operations/plan/__init__.py +3 -0
  305. lionagi/operations/plan/plan.py +172 -0
  306. lionagi/operations/plan/prompt.py +21 -0
  307. lionagi/operations/select/__init__.py +3 -0
  308. lionagi/operations/select/prompt.py +1 -0
  309. lionagi/operations/select/select.py +100 -0
  310. lionagi/operations/select/utils.py +107 -0
  311. lionagi/operations/utils.py +35 -0
  312. lionagi/protocols/adapters/adapter.py +79 -0
  313. lionagi/protocols/adapters/json_adapter.py +43 -0
  314. lionagi/protocols/adapters/pandas_adapter.py +96 -0
  315. lionagi/protocols/configs/__init__.py +15 -0
  316. lionagi/protocols/configs/branch_config.py +86 -0
  317. lionagi/protocols/configs/id_config.py +15 -0
  318. lionagi/protocols/configs/imodel_config.py +73 -0
  319. lionagi/protocols/configs/log_config.py +93 -0
  320. lionagi/protocols/configs/retry_config.py +29 -0
  321. lionagi/protocols/operatives/__init__.py +15 -0
  322. lionagi/protocols/operatives/action.py +181 -0
  323. lionagi/protocols/operatives/instruct.py +196 -0
  324. lionagi/protocols/operatives/operative.py +182 -0
  325. lionagi/protocols/operatives/prompts.py +232 -0
  326. lionagi/protocols/operatives/reason.py +56 -0
  327. lionagi/protocols/operatives/step.py +217 -0
  328. lionagi/protocols/registries/_component_registry.py +19 -0
  329. lionagi/protocols/registries/_pile_registry.py +26 -0
  330. lionagi/service/__init__.py +13 -0
  331. lionagi/service/complete_request_info.py +11 -0
  332. lionagi/service/imodel.py +125 -0
  333. lionagi/service/rate_limiter.py +108 -0
  334. lionagi/service/service.py +41 -0
  335. lionagi/service/service_match_util.py +131 -0
  336. lionagi/service/service_util.py +72 -0
  337. lionagi/service/token_calculator.py +51 -0
  338. lionagi/settings.py +136 -0
  339. lionagi/strategies/base.py +53 -0
  340. lionagi/strategies/concurrent.py +71 -0
  341. lionagi/strategies/concurrent_chunk.py +43 -0
  342. lionagi/strategies/concurrent_sequential_chunk.py +104 -0
  343. lionagi/strategies/params.py +128 -0
  344. lionagi/strategies/sequential.py +23 -0
  345. lionagi/strategies/sequential_chunk.py +89 -0
  346. lionagi/strategies/sequential_concurrent_chunk.py +100 -0
  347. lionagi/strategies/types.py +21 -0
  348. lionagi/strategies/utils.py +49 -0
  349. lionagi/version.py +1 -1
  350. lionagi-0.5.1.dist-info/METADATA +545 -0
  351. lionagi-0.5.1.dist-info/RECORD +373 -0
  352. {lionagi-0.4.0.dist-info → lionagi-0.5.1.dist-info}/WHEEL +1 -1
  353. lionagi/core/_setting/_setting.py +0 -59
  354. lionagi/core/action/README.md +0 -20
  355. lionagi/core/action/manual.py +0 -1
  356. lionagi/core/action/node.py +0 -94
  357. lionagi/core/action/tool_manager.py +0 -342
  358. lionagi/core/agent/README.md +0 -1
  359. lionagi/core/agent/base_agent.py +0 -82
  360. lionagi/core/agent/eval/README.md +0 -1
  361. lionagi/core/agent/eval/evaluator.py +0 -1
  362. lionagi/core/agent/eval/vote.py +0 -40
  363. lionagi/core/agent/learn/learner.py +0 -59
  364. lionagi/core/agent/plan/unit_template.py +0 -1
  365. lionagi/core/collections/README.md +0 -23
  366. lionagi/core/collections/__init__.py +0 -16
  367. lionagi/core/collections/_logger.py +0 -312
  368. lionagi/core/collections/abc/README.md +0 -63
  369. lionagi/core/collections/abc/__init__.py +0 -53
  370. lionagi/core/collections/abc/component.py +0 -620
  371. lionagi/core/collections/abc/concepts.py +0 -277
  372. lionagi/core/collections/abc/exceptions.py +0 -136
  373. lionagi/core/collections/abc/util.py +0 -45
  374. lionagi/core/collections/exchange.py +0 -146
  375. lionagi/core/collections/flow.py +0 -416
  376. lionagi/core/collections/model.py +0 -465
  377. lionagi/core/collections/pile.py +0 -1232
  378. lionagi/core/collections/progression.py +0 -221
  379. lionagi/core/collections/util.py +0 -73
  380. lionagi/core/director/README.md +0 -1
  381. lionagi/core/director/direct.py +0 -298
  382. lionagi/core/director/director.py +0 -2
  383. lionagi/core/director/operations/select.py +0 -3
  384. lionagi/core/director/operations/utils.py +0 -6
  385. lionagi/core/engine/branch_engine.py +0 -361
  386. lionagi/core/engine/instruction_map_engine.py +0 -213
  387. lionagi/core/engine/sandbox_.py +0 -16
  388. lionagi/core/engine/script_engine.py +0 -89
  389. lionagi/core/executor/base_executor.py +0 -97
  390. lionagi/core/executor/graph_executor.py +0 -335
  391. lionagi/core/executor/neo4j_executor.py +0 -394
  392. lionagi/core/generic/README.md +0 -0
  393. lionagi/core/generic/edge_condition.py +0 -17
  394. lionagi/core/generic/hyperedge.py +0 -1
  395. lionagi/core/generic/tree.py +0 -49
  396. lionagi/core/generic/tree_node.py +0 -85
  397. lionagi/core/mail/__init__.py +0 -11
  398. lionagi/core/mail/mail.py +0 -26
  399. lionagi/core/mail/mail_manager.py +0 -185
  400. lionagi/core/mail/package.py +0 -49
  401. lionagi/core/mail/start_mail.py +0 -36
  402. lionagi/core/message/__init__.py +0 -18
  403. lionagi/core/message/action_request.py +0 -114
  404. lionagi/core/message/action_response.py +0 -121
  405. lionagi/core/message/assistant_response.py +0 -80
  406. lionagi/core/message/instruction.py +0 -194
  407. lionagi/core/message/message.py +0 -86
  408. lionagi/core/message/system.py +0 -71
  409. lionagi/core/message/util.py +0 -274
  410. lionagi/core/report/__init__.py +0 -4
  411. lionagi/core/report/base.py +0 -201
  412. lionagi/core/report/form.py +0 -212
  413. lionagi/core/report/report.py +0 -150
  414. lionagi/core/report/util.py +0 -15
  415. lionagi/core/rule/_default.py +0 -17
  416. lionagi/core/rule/action.py +0 -87
  417. lionagi/core/rule/base.py +0 -234
  418. lionagi/core/rule/boolean.py +0 -56
  419. lionagi/core/rule/choice.py +0 -48
  420. lionagi/core/rule/mapping.py +0 -82
  421. lionagi/core/rule/number.py +0 -73
  422. lionagi/core/rule/rulebook.py +0 -45
  423. lionagi/core/rule/string.py +0 -43
  424. lionagi/core/rule/util.py +0 -0
  425. lionagi/core/session/directive_mixin.py +0 -307
  426. lionagi/core/structure/__init__.py +0 -1
  427. lionagi/core/structure/chain.py +0 -1
  428. lionagi/core/structure/forest.py +0 -1
  429. lionagi/core/structure/graph.py +0 -1
  430. lionagi/core/structure/tree.py +0 -1
  431. lionagi/core/unit/__init__.py +0 -4
  432. lionagi/core/unit/parallel_unit.py +0 -234
  433. lionagi/core/unit/template/action.py +0 -65
  434. lionagi/core/unit/template/base.py +0 -35
  435. lionagi/core/unit/template/plan.py +0 -69
  436. lionagi/core/unit/template/predict.py +0 -95
  437. lionagi/core/unit/template/score.py +0 -108
  438. lionagi/core/unit/template/select.py +0 -91
  439. lionagi/core/unit/unit.py +0 -452
  440. lionagi/core/unit/unit_form.py +0 -290
  441. lionagi/core/unit/unit_mixin.py +0 -1166
  442. lionagi/core/unit/util.py +0 -103
  443. lionagi/core/validator/validator.py +0 -376
  444. lionagi/core/work/work.py +0 -59
  445. lionagi/core/work/work_edge.py +0 -102
  446. lionagi/core/work/work_function.py +0 -114
  447. lionagi/core/work/work_function_node.py +0 -50
  448. lionagi/core/work/work_queue.py +0 -90
  449. lionagi/core/work/work_task.py +0 -151
  450. lionagi/core/work/worker.py +0 -410
  451. lionagi/core/work/worker_engine.py +0 -208
  452. lionagi/core/work/worklog.py +0 -108
  453. lionagi/experimental/compressor/base.py +0 -47
  454. lionagi/experimental/compressor/llm_compressor.py +0 -265
  455. lionagi/experimental/compressor/llm_summarizer.py +0 -61
  456. lionagi/experimental/compressor/util.py +0 -70
  457. lionagi/experimental/directive/README.md +0 -1
  458. lionagi/experimental/directive/__init__.py +0 -19
  459. lionagi/experimental/directive/parser/base_parser.py +0 -294
  460. lionagi/experimental/directive/parser/base_syntax.txt +0 -200
  461. lionagi/experimental/directive/template/base_template.py +0 -71
  462. lionagi/experimental/directive/template/schema.py +0 -36
  463. lionagi/experimental/directive/tokenizer.py +0 -59
  464. lionagi/experimental/evaluator/README.md +0 -1
  465. lionagi/experimental/evaluator/ast_evaluator.py +0 -119
  466. lionagi/experimental/evaluator/base_evaluator.py +0 -213
  467. lionagi/experimental/knowledge/__init__.py +0 -0
  468. lionagi/experimental/knowledge/base.py +0 -10
  469. lionagi/experimental/knowledge/graph.py +0 -0
  470. lionagi/experimental/memory/__init__.py +0 -0
  471. lionagi/experimental/strategies/__init__.py +0 -0
  472. lionagi/experimental/strategies/base.py +0 -1
  473. lionagi/integrations/bridge/__init__.py +0 -4
  474. lionagi/integrations/bridge/autogen_/__init__.py +0 -0
  475. lionagi/integrations/bridge/autogen_/autogen_.py +0 -127
  476. lionagi/integrations/bridge/langchain_/__init__.py +0 -0
  477. lionagi/integrations/bridge/langchain_/documents.py +0 -138
  478. lionagi/integrations/bridge/langchain_/langchain_bridge.py +0 -68
  479. lionagi/integrations/bridge/llamaindex_/__init__.py +0 -0
  480. lionagi/integrations/bridge/llamaindex_/index.py +0 -36
  481. lionagi/integrations/bridge/llamaindex_/llama_index_bridge.py +0 -108
  482. lionagi/integrations/bridge/llamaindex_/llama_pack.py +0 -256
  483. lionagi/integrations/bridge/llamaindex_/node_parser.py +0 -92
  484. lionagi/integrations/bridge/llamaindex_/reader.py +0 -201
  485. lionagi/integrations/bridge/llamaindex_/textnode.py +0 -59
  486. lionagi/integrations/bridge/pydantic_/__init__.py +0 -0
  487. lionagi/integrations/bridge/pydantic_/pydantic_bridge.py +0 -7
  488. lionagi/integrations/bridge/transformers_/__init__.py +0 -0
  489. lionagi/integrations/bridge/transformers_/install_.py +0 -39
  490. lionagi/integrations/chunker/__init__.py +0 -0
  491. lionagi/integrations/chunker/chunk.py +0 -314
  492. lionagi/integrations/config/__init__.py +0 -4
  493. lionagi/integrations/config/mlx_configs.py +0 -1
  494. lionagi/integrations/config/oai_configs.py +0 -154
  495. lionagi/integrations/config/ollama_configs.py +0 -1
  496. lionagi/integrations/config/openrouter_configs.py +0 -74
  497. lionagi/integrations/langchain_/__init__.py +0 -0
  498. lionagi/integrations/llamaindex_/__init__.py +0 -0
  499. lionagi/integrations/loader/__init__.py +0 -0
  500. lionagi/integrations/loader/load.py +0 -257
  501. lionagi/integrations/loader/load_util.py +0 -214
  502. lionagi/integrations/provider/__init__.py +0 -11
  503. lionagi/integrations/provider/_mapping.py +0 -47
  504. lionagi/integrations/provider/litellm.py +0 -53
  505. lionagi/integrations/provider/mistralai.py +0 -1
  506. lionagi/integrations/provider/mlx_service.py +0 -55
  507. lionagi/integrations/provider/oai.py +0 -196
  508. lionagi/integrations/provider/ollama.py +0 -55
  509. lionagi/integrations/provider/openrouter.py +0 -170
  510. lionagi/integrations/provider/services.py +0 -138
  511. lionagi/integrations/provider/transformers.py +0 -108
  512. lionagi/integrations/storage/__init__.py +0 -3
  513. lionagi/integrations/storage/neo4j.py +0 -681
  514. lionagi/integrations/storage/storage_util.py +0 -302
  515. lionagi/integrations/storage/structure_excel.py +0 -291
  516. lionagi/integrations/storage/to_csv.py +0 -70
  517. lionagi/integrations/storage/to_excel.py +0 -91
  518. lionagi/libs/ln_api.py +0 -944
  519. lionagi/libs/ln_async.py +0 -208
  520. lionagi/libs/ln_context.py +0 -37
  521. lionagi/libs/ln_convert.py +0 -671
  522. lionagi/libs/ln_dataframe.py +0 -187
  523. lionagi/libs/ln_func_call.py +0 -1328
  524. lionagi/libs/ln_image.py +0 -114
  525. lionagi/libs/ln_knowledge_graph.py +0 -422
  526. lionagi/libs/ln_nested.py +0 -822
  527. lionagi/libs/ln_parse.py +0 -750
  528. lionagi/libs/ln_queue.py +0 -107
  529. lionagi/libs/ln_tokenize.py +0 -179
  530. lionagi/libs/ln_validate.py +0 -299
  531. lionagi/libs/special_tokens.py +0 -172
  532. lionagi/libs/sys_util.py +0 -710
  533. lionagi/lions/__init__.py +0 -0
  534. lionagi/lions/coder/__init__.py +0 -0
  535. lionagi/lions/coder/add_feature.py +0 -20
  536. lionagi/lions/coder/base_prompts.py +0 -22
  537. lionagi/lions/coder/code_form.py +0 -15
  538. lionagi/lions/coder/coder.py +0 -184
  539. lionagi/lions/coder/util.py +0 -101
  540. lionagi/lions/director/__init__.py +0 -0
  541. lionagi/lions/judge/__init__.py +0 -0
  542. lionagi/lions/judge/config.py +0 -8
  543. lionagi/lions/judge/data/__init__.py +0 -0
  544. lionagi/lions/judge/data/sample_codes.py +0 -526
  545. lionagi/lions/judge/data/sample_rurbic.py +0 -48
  546. lionagi/lions/judge/forms/__init__.py +0 -0
  547. lionagi/lions/judge/forms/code_analysis_form.py +0 -126
  548. lionagi/lions/judge/rubric.py +0 -34
  549. lionagi/lions/judge/services/__init__.py +0 -0
  550. lionagi/lions/judge/services/judge_code.py +0 -49
  551. lionagi/lions/researcher/__init__.py +0 -0
  552. lionagi/lions/researcher/data_source/__init__.py +0 -0
  553. lionagi/lions/researcher/data_source/finhub_.py +0 -192
  554. lionagi/lions/researcher/data_source/google_.py +0 -207
  555. lionagi/lions/researcher/data_source/wiki_.py +0 -98
  556. lionagi/lions/researcher/data_source/yfinance_.py +0 -21
  557. lionagi/operations/brainstorm.py +0 -87
  558. lionagi/operations/config.py +0 -6
  559. lionagi/operations/rank.py +0 -102
  560. lionagi/operations/score.py +0 -144
  561. lionagi/operations/select.py +0 -141
  562. lionagi-0.4.0.dist-info/METADATA +0 -241
  563. lionagi-0.4.0.dist-info/RECORD +0 -249
  564. /lionagi/{core/_setting → integrations/anthropic_/api_endpoints/messages/response}/__init__.py +0 -0
  565. /lionagi/{core/agent → integrations/groq_/api_endpoints}/__init__.py +0 -0
  566. /lionagi/{core/agent/eval → integrations/ollama_/api_endpoints/completion}/__init__.py +0 -0
  567. /lionagi/{core/agent/learn → integrations/ollama_/api_endpoints/embedding}/__init__.py +0 -0
  568. /lionagi/{core/agent/plan → integrations/openai_}/__init__.py +0 -0
  569. /lionagi/{core/director → integrations/openai_/api_endpoints/chat_completions/response}/__init__.py +0 -0
  570. /lionagi/{core/director/operations → integrations/openai_/image_token_calculator}/__init__.py +0 -0
  571. /lionagi/{core/engine → integrations/perplexity_/api_endpoints}/__init__.py +0 -0
  572. /lionagi/{core/executor → integrations/perplexity_/api_endpoints/chat_completions}/__init__.py +0 -0
  573. /lionagi/{core/generic/registry/component_registry → integrations/perplexity_/api_endpoints/chat_completions/request}/__init__.py +0 -0
  574. /lionagi/{core/rule → integrations/perplexity_/api_endpoints/chat_completions/response}/__init__.py +0 -0
  575. /lionagi/{core/unit/template → libs/compress}/__init__.py +0 -0
  576. /lionagi/{core/validator → libs/file}/__init__.py +0 -0
  577. /lionagi/{core/work → libs/func}/__init__.py +0 -0
  578. /lionagi/{experimental → libs/package}/__init__.py +0 -0
  579. /lionagi/{core/agent/plan/plan.py → libs/parse/params.py} +0 -0
  580. /lionagi/{experimental/compressor → protocols}/__init__.py +0 -0
  581. /lionagi/{experimental/directive/parser → protocols/adapters}/__init__.py +0 -0
  582. /lionagi/{experimental/directive/template → protocols/registries}/__init__.py +0 -0
  583. /lionagi/{experimental/evaluator → strategies}/__init__.py +0 -0
  584. {lionagi-0.4.0.dist-info → lionagi-0.5.1.dist-info/licenses}/LICENSE +0 -0
@@ -1,671 +0,0 @@
1
- import json
2
- import re
3
- from collections.abc import Generator, Iterable
4
- from functools import singledispatch
5
- from typing import Any, Type
6
-
7
- import pandas as pd
8
- from pydantic import BaseModel
9
-
10
- number_regex = re.compile(r"-?\d+\.?\d*")
11
-
12
-
13
- # to_list functions with datatype overloads
14
- @singledispatch
15
- def to_list(
16
- input_, /, *, flatten: bool = True, dropna: bool = True
17
- ) -> list[Any]:
18
- """
19
- Converts the input object to a list. This function is capable of handling various input types,
20
- utilizing single dispatch to specialize for different types such as list, tuple, and set.
21
- The default implementation handles general iterables, excluding strings, bytes, bytearrays,
22
- and dictionaries, by attempting to convert them to a list, optionally flattening and dropping
23
- None values based on the provided arguments.
24
-
25
- Specialized implementations may use additional keyword arguments specific to their conversion logic.
26
-
27
- Args:
28
- input_ (Any): The input object to convert to a list.
29
- flatten (bool): If True, and the input is a nested list, the function will attempt to flatten it.
30
- dropna (bool): If True, None values will be removed from the resulting list.
31
-
32
- Returns:
33
- list[Any]: A list representation of the input, with modifications based on `flatten` and `dropna`.
34
-
35
- Raises:
36
- ValueError: If the input type is unsupported or cannot be converted to a list.
37
-
38
- Note:
39
- - This function uses `@singledispatch` to handle different input types via overloading.
40
- - The default behavior for dictionaries is to wrap them in a list without flattening.
41
- - For specific behaviors with lists, tuples, sets, and other types, see the registered implementations.
42
- """
43
- try:
44
- if not isinstance(input_, Iterable) or isinstance(
45
- input_, (str, bytes, bytearray, dict)
46
- ):
47
- return [input_]
48
- iterable_list = list(input_)
49
- return (
50
- _flatten_list(iterable_list, dropna) if flatten else iterable_list
51
- )
52
- except Exception as e:
53
- raise ValueError(
54
- f"Could not convert {type(input_)} object to list: {e}"
55
- ) from e
56
-
57
-
58
- @to_list.register(list)
59
- def _(input_, /, *, flatten: bool = True, dropna: bool = True) -> list[Any]:
60
- return _flatten_list(input_, dropna) if flatten else input_
61
-
62
-
63
- @to_list.register(tuple)
64
- def _(input_, /, *, flatten=True, dropna=True):
65
- """Specialized implementation of `to_list` for handling tuple inputs."""
66
- return _flatten_list(list(input_), dropna) if flatten else list(input_)
67
-
68
-
69
- @to_list.register(set)
70
- def _(input_, /, *, dropna=True):
71
- """Specialized implementation of `to_list` for handling set inputs."""
72
- return list(_dropna_iterator(list(input_))) if dropna else list(input_)
73
-
74
-
75
- # to_dict functions with datatype overloads
76
- @singledispatch
77
- def to_dict(input_, /, *args, **kwargs) -> dict[Any, Any]:
78
- """
79
- Converts the input object to a dictionary. This base function raises a ValueError for unsupported types.
80
- The function is overloaded to handle specific input types such as dict, str, pandas.Series, pandas.DataFrame,
81
- and Pydantic's BaseModel, utilizing the single dispatch mechanism for type-specific conversions.
82
-
83
- Args:
84
- input_ (Any): The input object to convert to a dictionary.
85
- *args: Variable length argument list for additional options in type-specific handlers.
86
- **kwargs: Arbitrary keyword arguments for additional options in type-specific handlers.
87
-
88
- Returns:
89
- dict[Any, Any]: A dictionary representation of the input object.
90
-
91
- Raises:
92
- ValueError: If the input type is not supported or cannot be converted to a dictionary.
93
-
94
- Note:
95
- - For specific behaviors with dict, str, pandas.Series, pandas.DataFrame, and BaseModel,
96
- see the registered implementations.
97
- """
98
- try:
99
- return dict(input_, *args, **kwargs)
100
- except Exception as e:
101
- raise TypeError(
102
- f"Input type not supported: {type(input_).__name__}. {e}"
103
- ) from e
104
-
105
-
106
- @to_dict.register(dict)
107
- def _(input_) -> dict[Any, Any]:
108
- """
109
- Handles dictionary inputs directly, returning the input without modification.
110
-
111
- Args:
112
- input_ (dict[Any, Any]): The dictionary to be returned.
113
-
114
- Returns:
115
- dict[Any, Any]: The input dictionary, unchanged.
116
- """
117
- return input_
118
-
119
-
120
- @to_dict.register(str)
121
- def _(input_, /, *args, **kwargs) -> dict[Any, Any]:
122
- """
123
- Converts a JSON-formatted string to a dictionary.
124
-
125
- Args:
126
- input_ (str): The JSON string to convert.
127
- *args: Variable length argument list for json.loads().
128
- **kwargs: Arbitrary keyword arguments for json.loads().
129
-
130
- Returns:
131
- dict[Any, Any]: The dictionary representation of the JSON string.
132
-
133
- Raises:
134
- ValueError: If the string cannot be decoded into a dictionary.
135
- """
136
- try:
137
- return json.loads(input_, *args, **kwargs)
138
- except json.JSONDecodeError as e:
139
- raise ValueError(f"Could not convert input_ to dict: {e}") from e
140
-
141
-
142
- @to_dict.register(pd.Series)
143
- def _(input_, /, *args, **kwargs) -> dict[Any, Any]:
144
- """
145
- Converts a pandas Series to a dictionary.
146
-
147
- Args:
148
- input_ (pd.Series): The pandas Series to convert.
149
- *args: Variable length argument list for Series.to_dict().
150
- **kwargs: Arbitrary keyword arguments for Series.to_dict().
151
-
152
- Returns:
153
- dict[Any, Any]: The dictionary representation of the pandas Series.
154
- """
155
- return input_.to_dict(*args, **kwargs)
156
-
157
-
158
- @to_dict.register(pd.DataFrame)
159
- def _(
160
- input_, /, *args, orient: str = "list", as_list: bool = False, **kwargs
161
- ) -> dict[Any, Any] | list[dict[Any, Any]]:
162
- """
163
- Converts a pandas DataFrame to a dictionary or a list of dictionaries, based on the `orient` and `as_list` parameters.
164
-
165
- Args:
166
- input_ (pd.DataFrame): The pandas DataFrame to convert.
167
- *args: Variable length argument list for DataFrame.to_dict() or DataFrame.iterrows().
168
- orient (str): The orientation of the data. Default is 'list'.
169
- as_list (bool): If True, returns a list of dictionaries, one for each row. Default is False.
170
- **kwargs: Arbitrary keyword arguments for DataFrame.to_dict().
171
-
172
- Returns:
173
- dict[Any, Any] | list[dict[Any, Any]]: Depending on `as_list`, either a dictionary representation
174
- of the DataFrame or a list of dictionaries, one for each row.
175
- """
176
- if as_list:
177
- return [row.to_dict(*args, **kwargs) for _, row in input_.iterrows()]
178
- return input_.to_dict(*args, orient=orient, **kwargs)
179
-
180
-
181
- @to_dict.register(BaseModel)
182
- def _(input_, /, *args, **kwargs) -> dict[Any, Any]:
183
- """
184
- Converts a Pydantic BaseModel instance to a dictionary.
185
-
186
- Args:
187
- input_ (BaseModel): The Pydantic BaseModel instance to convert.
188
- *args: Variable length argument list for the model's dict() method.
189
- **kwargs: Arbitrary keyword arguments for the model's dict() method.
190
-
191
- Returns:
192
- dict[Any, Any]: The dictionary representation of the BaseModel instance.
193
- """
194
- return input_.model_dump(*args, **kwargs)
195
-
196
-
197
- # to_str functions with datatype overloads
198
- @singledispatch
199
- def to_str(input_) -> str:
200
- """
201
- Converts the input object to a string. This function utilizes single dispatch to handle
202
- specific input types such as dict, str, list, pandas.Series, and pandas.DataFrame,
203
- providing type-specific conversions to string format.
204
-
205
- Args:
206
- input_ (Any): The input object to convert to a string.
207
- *args: Variable length argument list for additional options in type-specific handlers.
208
- **kwargs: Arbitrary keyword arguments for additional options in type-specific handlers.
209
-
210
- Returns:
211
- str: A string representation of the input object.
212
-
213
- Note:
214
- - The base implementation simply uses the str() function for conversion.
215
- - For detailed behaviors with dict, str, list, pandas.Series, and pandas.DataFrame,
216
- refer to the registered implementations.
217
- """
218
- return str(input_)
219
-
220
-
221
- @to_str.register(dict)
222
- def _(input_, /, *args, **kwargs) -> str:
223
- """
224
- Converts a dictionary to a JSON-formatted string.
225
-
226
- Args:
227
- input_ (dict): The dictionary to convert.
228
- *args: Variable length argument list for json.dumps().
229
- **kwargs: Arbitrary keyword arguments for json.dumps().
230
-
231
- Returns:
232
- str: The JSON string representation of the dictionary.
233
- """
234
- return json.dumps(input_, *args, **kwargs)
235
-
236
-
237
- @to_str.register(str)
238
- def _(input_) -> str:
239
- """
240
- Returns the input string unchanged.
241
-
242
- Args:
243
- input_ (str): The input string.
244
- *args: Ignored.
245
- **kwargs: Ignored.
246
-
247
- Returns:
248
- str: The input string, unchanged.
249
- """
250
- return input_
251
-
252
-
253
- @to_str.register(list)
254
- def _(input_, /, *args, as_list: bool = False, **kwargs) -> str | list[str]:
255
- """
256
- Converts a list to a string. Optionally, the function can return a string representation
257
- of the list itself or join the string representations of its elements.
258
-
259
- Args:
260
- input_ (list): The list to convert.
261
- *args: Variable length argument list for additional options in element conversion.
262
- as_list (bool): If True, returns the string representation of the list. If False,
263
- returns the elements joined by a comma. Default is False.
264
- **kwargs: Arbitrary keyword arguments for additional options in element conversion.
265
-
266
- Returns:
267
- str: Depending on `as_list`, either the string representation of the list or a string
268
- of the elements joined by a comma.
269
- """
270
- lst_ = [to_str(item, *args, **kwargs) for item in input_]
271
- return lst_ if as_list else ", ".join(lst_)
272
-
273
-
274
- @to_str.register(pd.Series)
275
- def _(input_, /, *args, **kwargs) -> str:
276
- """
277
- Converts a pandas Series to a JSON-formatted string.
278
-
279
- Args:
280
- input_ (pd.Series): The pandas Series to convert.
281
- *args: Variable length argument list for Series.to_json().
282
- **kwargs: Arbitrary keyword arguments for Series.to_json().
283
-
284
- Returns:
285
- str: The JSON string representation of the pandas Series.
286
- """
287
- return input_.to_json(*args, **kwargs)
288
-
289
-
290
- @to_str.register(pd.DataFrame)
291
- def _(input_, /, *args, as_list: bool = False, **kwargs) -> str | list[str]:
292
- """
293
- Converts a pandas DataFrame to a JSON-formatted string. Optionally, can convert to a list of dictionaries
294
- first if `as_list` is True, then to a string representation of that list.
295
-
296
- Args:
297
- input_ (pd.DataFrame): The pandas DataFrame to convert.
298
- *args: Variable length argument list for additional options in conversion.
299
- as_list (bool): If True, converts the DataFrame to a list of dictionaries before converting
300
- to a string. Default is False.
301
- **kwargs: Arbitrary keyword arguments for DataFrame.to_json() or to_dict().
302
-
303
- Returns:
304
- str: Depending on `as_list`, either a JSON string representation of the DataFrame or a string
305
- representation of a list of dictionaries derived from the DataFrame.
306
- """
307
- if as_list:
308
- return to_dict(input_, as_list=True, *args, **kwargs)
309
- return input_.to_json(*args, **kwargs)
310
-
311
-
312
- # to_df functions with datatype overloads
313
-
314
-
315
- @singledispatch
316
- def to_df(
317
- input_: Any,
318
- /,
319
- *,
320
- how: str = "all",
321
- drop_kwargs: dict[str, Any] | None = None,
322
- reset_index: bool = True,
323
- **kwargs: Any,
324
- ) -> pd.DataFrame:
325
- """
326
- Converts various input types to a pandas DataFrame, with options for handling missing data and resetting the index.
327
- This function is overloaded to handle specific data structures such as lists of dictionaries, lists of pandas objects (DataFrames or Series), and more.
328
-
329
- The base implementation attempts to directly convert the input to a DataFrame, applying dropna and reset_index as specified.
330
-
331
- Args:
332
- input_ (Any): The input data to convert into a DataFrame. Accepts a wide range of types thanks to overloads.
333
- how (str): Specifies how missing values are dropped. Passed directly to DataFrame.dropna().
334
- drop_kwargs (dict[str, Any] | None): Additional keyword arguments for DataFrame.dropna().
335
- reset_index (bool): If True, the DataFrame index will be reset, removing the index labels.
336
- **kwargs: Additional keyword arguments passed to the pandas DataFrame constructor.
337
-
338
- Returns:
339
- pd.DataFrame: A pandas DataFrame constructed from the input data.
340
-
341
- Raises:
342
- ValueError: If there is an error during the conversion process.
343
-
344
- Note:
345
- - This function is overloaded to provide specialized behavior for different input types, enhancing its flexibility.
346
- """
347
-
348
- if drop_kwargs is None:
349
- drop_kwargs = {}
350
-
351
- try:
352
- dfs = pd.DataFrame(input_, **kwargs)
353
- dfs = dfs.dropna(**(drop_kwargs | {"how": how}))
354
- return dfs.reset_index(drop=True) if reset_index else dfs
355
-
356
- except Exception as e:
357
- raise ValueError(f"Error converting input_ to DataFrame: {e}") from e
358
-
359
-
360
- @to_df.register(list)
361
- def _(
362
- input_,
363
- /,
364
- *,
365
- how: str = "all",
366
- drop_kwargs: dict | None = None,
367
- reset_index: bool = True,
368
- **kwargs,
369
- ) -> pd.DataFrame:
370
- if not input_:
371
- return pd.DataFrame()
372
- if not isinstance(
373
- input_[0], (pd.DataFrame, pd.Series, pd.core.generic.NDFrame)
374
- ):
375
- if drop_kwargs is None:
376
- drop_kwargs = {}
377
- try:
378
- dfs = pd.DataFrame(input_, **kwargs)
379
- dfs = dfs.dropna(**(drop_kwargs | {"how": how}))
380
- return dfs.reset_index(drop=True) if reset_index else dfs
381
- except Exception as e:
382
- raise ValueError(
383
- f"Error converting input_ to DataFrame: {e}"
384
- ) from e
385
-
386
- dfs = ""
387
- if drop_kwargs is None:
388
- drop_kwargs = {}
389
- try:
390
- dfs = pd.concat(input_, **kwargs)
391
-
392
- except Exception as e1:
393
- try:
394
- input_ = to_list(input_)
395
- dfs = input_[0]
396
- if len(input_) > 1:
397
- for i in input_[1:]:
398
- dfs = pd.concat([dfs, i], **kwargs)
399
-
400
- except Exception as e2:
401
- raise ValueError(
402
- f"Error converting input_ to DataFrame: {e1}, {e2}"
403
- ) from e2
404
-
405
- dfs.dropna(**(drop_kwargs | {"how": how}), inplace=True)
406
- return dfs.reset_index(drop=True) if reset_index else dfs
407
-
408
-
409
- def to_num(
410
- input_: Any,
411
- /,
412
- *,
413
- upper_bound: int | float | None = None,
414
- lower_bound: int | float | None = None,
415
- num_type: type[int | float] = float,
416
- precision: int | None = None,
417
- ) -> int | float:
418
- """
419
- Converts the input to a numeric value of specified type, with optional bounds and precision.
420
-
421
- Args:
422
- input_ (Any): The input value to convert. Can be of any type that `to_str` can handle.
423
- upper_bound (float | None): The upper bound for the numeric value. If specified, values above this bound will raise an error.
424
- lower_bound (float | None): The lower bound for the numeric value. If specified, values below this bound will raise an error.
425
- num_type (Type[int | float]): The numeric type to convert to. Can be `int` or `float`.
426
- precision (int | None): The number of decimal places for the result. Applies only to `float` type.
427
-
428
- Returns:
429
- int | float: The converted numeric value, adhering to specified type and precision.
430
-
431
- Raises:
432
- ValueError: If the input cannot be converted to a number, or if it violates the specified bounds.
433
- """
434
- str_ = to_str(input_)
435
- return _str_to_num(str_, upper_bound, lower_bound, num_type, precision)
436
-
437
-
438
- def to_readable_dict(input_: Any) -> str:
439
- """
440
- Converts a given input to a readable dictionary format
441
- """
442
-
443
- try:
444
- dict_ = to_dict(input_)
445
- return (
446
- json.dumps(dict_, indent=4) if isinstance(input_, dict) else input_
447
- )
448
- except Exception as e:
449
- raise ValueError(
450
- f"Could not convert given input to readable dict: {e}"
451
- ) from e
452
-
453
-
454
- def is_same_dtype(
455
- input_: list | dict, dtype: type | None = None, return_dtype=False
456
- ) -> bool:
457
- """
458
- Checks if all elements in a list or dictionary values are of the same data type.
459
-
460
- Args:
461
- input_ (list | dict): The input list or dictionary to check.
462
- dtype (Type | None): The data type to check against. If None, uses the type of the first element.
463
-
464
- Returns:
465
- bool: True if all elements are of the same type (or if the input is empty), False otherwise.
466
- """
467
- if not input_:
468
- return True
469
-
470
- iterable = input_.values() if isinstance(input_, dict) else input_
471
- first_element_type = type(next(iter(iterable), None))
472
-
473
- dtype = dtype or first_element_type
474
-
475
- a = all(isinstance(element, dtype) for element in iterable)
476
- return a, dtype if return_dtype else a
477
-
478
-
479
- def xml_to_dict(root) -> dict[str, Any]:
480
- import xml.etree.ElementTree as ET
481
- from collections import defaultdict
482
-
483
- def parse_xml(element: ET.Element, parent: dict[str, Any]):
484
- children = list(element)
485
- if children:
486
- d = defaultdict(list)
487
- for child in children:
488
- parse_xml(child, d)
489
- parent[element.tag].append(d if len(d) > 1 else d[next(iter(d))])
490
- else:
491
- parent[element.tag].append(element.text)
492
-
493
- result = defaultdict(list)
494
- parse_xml(root, result)
495
- return {k: v[0] if len(v) == 1 else v for k, v in result.items()}
496
-
497
-
498
- def strip_lower(input_: Any) -> str:
499
- """
500
- Converts the input to a lowercase string with leading and trailing whitespace removed.
501
-
502
- Args:
503
- input_ (Any): The input value to convert and process.
504
-
505
- Returns:
506
- str: The processed string.
507
-
508
- Raises:
509
- ValueError: If the input cannot be converted to a string.
510
- """
511
- try:
512
- return str(input_).strip().lower()
513
- except Exception as e:
514
- raise ValueError(
515
- f"Could not convert input_ to string: {input_}, Error: {e}"
516
- )
517
-
518
-
519
- def is_structure_homogeneous(
520
- structure: Any, return_structure_type: bool = False
521
- ) -> bool | tuple[bool, type | None]:
522
- """
523
- checks if a nested structure is homogeneous, meaning it doesn't contain a mix
524
- of lists and dictionaries.
525
-
526
- Args: structure: The nested structure to check. return_structure_type: Flag to
527
- indicate whether to return the type of homogeneous structure.
528
-
529
- Returns: If return_structure_type is False, returns a boolean indicating
530
- whether the structure is homogeneous. if return_structure_type is True,
531
- returns a tuple containing a boolean indicating whether the structure is
532
- homogeneous, and the type of the homogeneous structure if it is homogeneous (
533
- either list | dict, or None).
534
-
535
- examples:
536
- >>> _is_structure_homogeneous({'a': {'b': 1}, 'c': {'d': 2}})
537
- True
538
-
539
- >>> _is_structure_homogeneous({'a': {'b': 1}, 'c': [1, 2]})
540
- False
541
- """
542
-
543
- # noinspection PyShadowingNames
544
- def _check_structure(substructure):
545
- structure_type = None
546
- if isinstance(substructure, list):
547
- structure_type = list
548
- for item in substructure:
549
- if not isinstance(item, structure_type) and isinstance(
550
- item, (list | dict)
551
- ):
552
- return False, None
553
- result, _ = _check_structure(item)
554
- if not result:
555
- return False, None
556
- elif isinstance(substructure, dict):
557
- structure_type = dict
558
- for item in substructure.values():
559
- if not isinstance(item, structure_type) and isinstance(
560
- item, (list | dict)
561
- ):
562
- return False, None
563
- result, _ = _check_structure(item)
564
- if not result:
565
- return False, None
566
- return True, structure_type
567
-
568
- is_, structure_type = _check_structure(structure)
569
- return (is_, structure_type) if return_structure_type else is_
570
-
571
-
572
- def is_homogeneous(
573
- iterables: list[Any] | dict[Any, Any], type_check: type
574
- ) -> bool:
575
- if isinstance(iterables, list):
576
- return all(isinstance(it, type_check) for it in iterables)
577
- return isinstance(iterables, type_check)
578
-
579
-
580
- def _str_to_num(
581
- input_: str,
582
- upper_bound: float | None = None,
583
- lower_bound: float | None = None,
584
- num_type: type[int | float] = int,
585
- precision: int | None = None,
586
- ) -> int | float:
587
- number_str = _extract_first_number(input_)
588
- if number_str is None:
589
- raise ValueError(f"No numeric values found in the string: {input_}")
590
-
591
- number = _convert_to_num(number_str, num_type, precision)
592
-
593
- if upper_bound is not None and number > upper_bound:
594
- raise ValueError(
595
- f"Number {number} is greater than the upper bound of {upper_bound}."
596
- )
597
-
598
- if lower_bound is not None and number < lower_bound:
599
- raise ValueError(
600
- f"Number {number} is less than the lower bound of {lower_bound}."
601
- )
602
-
603
- return number
604
-
605
-
606
- def _extract_first_number(input_: str) -> str | None:
607
- match = number_regex.search(input_)
608
- return match.group(0) if match else None
609
-
610
-
611
- def _convert_to_num(
612
- number_str: str,
613
- num_type: type[int | float] = int,
614
- precision: int | None = None,
615
- ) -> int | float:
616
- if num_type is int:
617
- return int(float(number_str))
618
- elif num_type is float:
619
- number = float(number_str)
620
- return round(number, precision) if precision is not None else number
621
- else:
622
- raise ValueError(f"Invalid number type: {num_type}")
623
-
624
-
625
- def _dropna_iterator(lst_: list[Any]) -> iter:
626
- return (item for item in lst_ if item is not None)
627
-
628
-
629
- def _flatten_list(lst_: list[Any], dropna: bool = True) -> list[Any]:
630
- """
631
- flatten a nested list, optionally removing None values.
632
-
633
- Args:
634
- lst_ (list[Any]): A nested list to flatten.
635
- dropna (bool): If True, None values are removed. default is True.
636
-
637
- Returns:
638
- list[Any]: A flattened list.
639
-
640
- examples:
641
- >>> flatten_list([[1, 2], [3, None]], dropna=True)
642
- [1, 2, 3]
643
- >>> flatten_list([[1, [2, None]], 3], dropna=False)
644
- [1, 2, None, 3]
645
- """
646
- flattened_list = list(_flatten_list_generator(lst_, dropna))
647
- return list(_dropna_iterator(flattened_list)) if dropna else flattened_list
648
-
649
-
650
- def _flatten_list_generator(
651
- lst_: list[Any], dropna: bool = True
652
- ) -> Generator[Any, None, None]:
653
- """
654
- Generator for flattening a nested list.
655
-
656
- Args:
657
- lst_ (list[Any]): A nested list to flatten.
658
- dropna (bool): If True, None values are omitted. Default is True.
659
-
660
- Yields:
661
- Generator[Any, None, None]: A generator yielding flattened elements.
662
-
663
- Examples:
664
- >>> list(_flatten_list_generator([[1, [2, None]], 3], dropna=False))
665
- [1, 2, None, 3]
666
- """
667
- for i in lst_:
668
- if isinstance(i, list):
669
- yield from _flatten_list_generator(i, dropna)
670
- else:
671
- yield i