flock-core 0.5.0b28__py3-none-any.whl → 0.5.56b0__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.

Potentially problematic release.


This version of flock-core might be problematic. Click here for more details.

Files changed (359) hide show
  1. flock/__init__.py +12 -217
  2. flock/agent.py +678 -0
  3. flock/api/themes.py +71 -0
  4. flock/artifacts.py +79 -0
  5. flock/cli.py +75 -0
  6. flock/components.py +173 -0
  7. flock/dashboard/__init__.py +28 -0
  8. flock/dashboard/collector.py +283 -0
  9. flock/dashboard/events.py +182 -0
  10. flock/dashboard/launcher.py +230 -0
  11. flock/dashboard/service.py +537 -0
  12. flock/dashboard/websocket.py +235 -0
  13. flock/engines/__init__.py +6 -0
  14. flock/engines/dspy_engine.py +856 -0
  15. flock/examples.py +128 -0
  16. flock/{core/util → helper}/cli_helper.py +4 -3
  17. flock/{core/logging → logging}/__init__.py +2 -3
  18. flock/{core/logging → logging}/formatters/enum_builder.py +3 -4
  19. flock/{core/logging → logging}/formatters/theme_builder.py +19 -44
  20. flock/{core/logging → logging}/formatters/themed_formatter.py +69 -115
  21. flock/{core/logging → logging}/logging.py +77 -61
  22. flock/{core/logging → logging}/telemetry.py +20 -26
  23. flock/{core/logging → logging}/telemetry_exporter/base_exporter.py +2 -2
  24. flock/{core/logging → logging}/telemetry_exporter/file_exporter.py +6 -9
  25. flock/{core/logging → logging}/telemetry_exporter/sqlite_exporter.py +2 -3
  26. flock/{core/logging → logging}/trace_and_logged.py +20 -24
  27. flock/mcp/__init__.py +91 -0
  28. flock/{core/mcp/mcp_client.py → mcp/client.py} +103 -154
  29. flock/{core/mcp/mcp_config.py → mcp/config.py} +62 -117
  30. flock/mcp/manager.py +255 -0
  31. flock/mcp/servers/sse/__init__.py +1 -1
  32. flock/mcp/servers/sse/flock_sse_server.py +11 -53
  33. flock/mcp/servers/stdio/__init__.py +1 -1
  34. flock/mcp/servers/stdio/flock_stdio_server.py +8 -48
  35. flock/mcp/servers/streamable_http/flock_streamable_http_server.py +17 -62
  36. flock/mcp/servers/websockets/flock_websocket_server.py +7 -40
  37. flock/{core/mcp/flock_mcp_tool.py → mcp/tool.py} +16 -26
  38. flock/mcp/types/__init__.py +42 -0
  39. flock/{core/mcp → mcp}/types/callbacks.py +9 -15
  40. flock/{core/mcp → mcp}/types/factories.py +7 -6
  41. flock/{core/mcp → mcp}/types/handlers.py +13 -18
  42. flock/{core/mcp → mcp}/types/types.py +70 -74
  43. flock/{core/mcp → mcp}/util/helpers.py +1 -1
  44. flock/orchestrator.py +645 -0
  45. flock/registry.py +148 -0
  46. flock/runtime.py +262 -0
  47. flock/service.py +140 -0
  48. flock/store.py +69 -0
  49. flock/subscription.py +111 -0
  50. flock/themes/andromeda.toml +1 -1
  51. flock/themes/apple-system-colors.toml +1 -1
  52. flock/themes/arcoiris.toml +1 -1
  53. flock/themes/atomonelight.toml +1 -1
  54. flock/themes/ayu copy.toml +1 -1
  55. flock/themes/ayu-light.toml +1 -1
  56. flock/themes/belafonte-day.toml +1 -1
  57. flock/themes/belafonte-night.toml +1 -1
  58. flock/themes/blulocodark.toml +1 -1
  59. flock/themes/breeze.toml +1 -1
  60. flock/themes/broadcast.toml +1 -1
  61. flock/themes/brogrammer.toml +1 -1
  62. flock/themes/builtin-dark.toml +1 -1
  63. flock/themes/builtin-pastel-dark.toml +1 -1
  64. flock/themes/catppuccin-latte.toml +1 -1
  65. flock/themes/catppuccin-macchiato.toml +1 -1
  66. flock/themes/catppuccin-mocha.toml +1 -1
  67. flock/themes/cga.toml +1 -1
  68. flock/themes/chalk.toml +1 -1
  69. flock/themes/ciapre.toml +1 -1
  70. flock/themes/coffee-theme.toml +1 -1
  71. flock/themes/cyberpunkscarletprotocol.toml +1 -1
  72. flock/themes/dark+.toml +1 -1
  73. flock/themes/darkermatrix.toml +1 -1
  74. flock/themes/darkside.toml +1 -1
  75. flock/themes/desert.toml +1 -1
  76. flock/themes/django.toml +1 -1
  77. flock/themes/djangosmooth.toml +1 -1
  78. flock/themes/doomone.toml +1 -1
  79. flock/themes/dotgov.toml +1 -1
  80. flock/themes/dracula+.toml +1 -1
  81. flock/themes/duckbones.toml +1 -1
  82. flock/themes/encom.toml +1 -1
  83. flock/themes/espresso.toml +1 -1
  84. flock/themes/everblush.toml +1 -1
  85. flock/themes/fairyfloss.toml +1 -1
  86. flock/themes/fideloper.toml +1 -1
  87. flock/themes/fishtank.toml +1 -1
  88. flock/themes/flexoki-light.toml +1 -1
  89. flock/themes/floraverse.toml +1 -1
  90. flock/themes/framer.toml +1 -1
  91. flock/themes/galizur.toml +1 -1
  92. flock/themes/github.toml +1 -1
  93. flock/themes/grass.toml +1 -1
  94. flock/themes/grey-green.toml +1 -1
  95. flock/themes/gruvboxlight.toml +1 -1
  96. flock/themes/guezwhoz.toml +1 -1
  97. flock/themes/harper.toml +1 -1
  98. flock/themes/hax0r-blue.toml +1 -1
  99. flock/themes/hopscotch.256.toml +1 -1
  100. flock/themes/ic-green-ppl.toml +1 -1
  101. flock/themes/iceberg-dark.toml +1 -1
  102. flock/themes/japanesque.toml +1 -1
  103. flock/themes/jubi.toml +1 -1
  104. flock/themes/kibble.toml +1 -1
  105. flock/themes/kolorit.toml +1 -1
  106. flock/themes/kurokula.toml +1 -1
  107. flock/themes/materialdesigncolors.toml +1 -1
  108. flock/themes/matrix.toml +1 -1
  109. flock/themes/mellifluous.toml +1 -1
  110. flock/themes/midnight-in-mojave.toml +1 -1
  111. flock/themes/monokai-remastered.toml +1 -1
  112. flock/themes/monokai-soda.toml +1 -1
  113. flock/themes/neon.toml +1 -1
  114. flock/themes/neopolitan.toml +1 -1
  115. flock/themes/nord-light.toml +1 -1
  116. flock/themes/ocean.toml +1 -1
  117. flock/themes/onehalfdark.toml +1 -1
  118. flock/themes/onehalflight.toml +1 -1
  119. flock/themes/palenighthc.toml +1 -1
  120. flock/themes/paulmillr.toml +1 -1
  121. flock/themes/pencildark.toml +1 -1
  122. flock/themes/pnevma.toml +1 -1
  123. flock/themes/purple-rain.toml +1 -1
  124. flock/themes/purplepeter.toml +1 -1
  125. flock/themes/raycast-dark.toml +1 -1
  126. flock/themes/red-sands.toml +1 -1
  127. flock/themes/relaxed.toml +1 -1
  128. flock/themes/retro.toml +1 -1
  129. flock/themes/rose-pine.toml +1 -1
  130. flock/themes/royal.toml +1 -1
  131. flock/themes/ryuuko.toml +1 -1
  132. flock/themes/sakura.toml +1 -1
  133. flock/themes/scarlet-protocol.toml +1 -1
  134. flock/themes/seoulbones-dark.toml +1 -1
  135. flock/themes/shades-of-purple.toml +1 -1
  136. flock/themes/smyck.toml +1 -1
  137. flock/themes/softserver.toml +1 -1
  138. flock/themes/solarized-darcula.toml +1 -1
  139. flock/themes/square.toml +1 -1
  140. flock/themes/sugarplum.toml +1 -1
  141. flock/themes/thayer-bright.toml +1 -1
  142. flock/themes/tokyonight.toml +1 -1
  143. flock/themes/tomorrow.toml +1 -1
  144. flock/themes/ubuntu.toml +1 -1
  145. flock/themes/ultradark.toml +1 -1
  146. flock/themes/ultraviolent.toml +1 -1
  147. flock/themes/unikitty.toml +1 -1
  148. flock/themes/urple.toml +1 -1
  149. flock/themes/vesper.toml +1 -1
  150. flock/themes/vimbones.toml +1 -1
  151. flock/themes/wildcherry.toml +1 -1
  152. flock/themes/wilmersdorf.toml +1 -1
  153. flock/themes/wryan.toml +1 -1
  154. flock/themes/xcodedarkhc.toml +1 -1
  155. flock/themes/xcodelight.toml +1 -1
  156. flock/themes/zenbones-light.toml +1 -1
  157. flock/themes/zenwritten-dark.toml +1 -1
  158. flock/utilities.py +301 -0
  159. flock/{components/utility → utility}/output_utility_component.py +68 -53
  160. flock/visibility.py +107 -0
  161. flock_core-0.5.56b0.dist-info/METADATA +747 -0
  162. flock_core-0.5.56b0.dist-info/RECORD +398 -0
  163. flock_core-0.5.56b0.dist-info/entry_points.txt +2 -0
  164. {flock_core-0.5.0b28.dist-info → flock_core-0.5.56b0.dist-info}/licenses/LICENSE +1 -1
  165. flock/adapter/__init__.py +0 -14
  166. flock/adapter/azure_adapter.py +0 -68
  167. flock/adapter/chroma_adapter.py +0 -73
  168. flock/adapter/faiss_adapter.py +0 -97
  169. flock/adapter/pinecone_adapter.py +0 -51
  170. flock/adapter/vector_base.py +0 -47
  171. flock/cli/assets/release_notes.md +0 -140
  172. flock/cli/config.py +0 -8
  173. flock/cli/constants.py +0 -36
  174. flock/cli/create_agent.py +0 -1
  175. flock/cli/create_flock.py +0 -280
  176. flock/cli/execute_flock.py +0 -620
  177. flock/cli/load_agent.py +0 -1
  178. flock/cli/load_examples.py +0 -1
  179. flock/cli/load_flock.py +0 -192
  180. flock/cli/load_release_notes.py +0 -20
  181. flock/cli/loaded_flock_cli.py +0 -254
  182. flock/cli/manage_agents.py +0 -459
  183. flock/cli/registry_management.py +0 -889
  184. flock/cli/runner.py +0 -41
  185. flock/cli/settings.py +0 -857
  186. flock/cli/utils.py +0 -135
  187. flock/cli/view_results.py +0 -29
  188. flock/cli/yaml_editor.py +0 -396
  189. flock/components/__init__.py +0 -30
  190. flock/components/evaluation/__init__.py +0 -9
  191. flock/components/evaluation/declarative_evaluation_component.py +0 -606
  192. flock/components/routing/__init__.py +0 -15
  193. flock/components/routing/conditional_routing_component.py +0 -494
  194. flock/components/routing/default_routing_component.py +0 -103
  195. flock/components/routing/llm_routing_component.py +0 -206
  196. flock/components/utility/__init__.py +0 -22
  197. flock/components/utility/example_utility_component.py +0 -250
  198. flock/components/utility/feedback_utility_component.py +0 -206
  199. flock/components/utility/memory_utility_component.py +0 -550
  200. flock/components/utility/metrics_utility_component.py +0 -700
  201. flock/config.py +0 -61
  202. flock/core/__init__.py +0 -110
  203. flock/core/agent/__init__.py +0 -16
  204. flock/core/agent/default_agent.py +0 -216
  205. flock/core/agent/flock_agent_components.py +0 -104
  206. flock/core/agent/flock_agent_execution.py +0 -101
  207. flock/core/agent/flock_agent_integration.py +0 -260
  208. flock/core/agent/flock_agent_lifecycle.py +0 -186
  209. flock/core/agent/flock_agent_serialization.py +0 -381
  210. flock/core/api/__init__.py +0 -10
  211. flock/core/api/custom_endpoint.py +0 -45
  212. flock/core/api/endpoints.py +0 -254
  213. flock/core/api/main.py +0 -162
  214. flock/core/api/models.py +0 -97
  215. flock/core/api/run_store.py +0 -224
  216. flock/core/api/runner.py +0 -44
  217. flock/core/api/service.py +0 -214
  218. flock/core/component/__init__.py +0 -15
  219. flock/core/component/agent_component_base.py +0 -309
  220. flock/core/component/evaluation_component.py +0 -62
  221. flock/core/component/routing_component.py +0 -74
  222. flock/core/component/utility_component.py +0 -69
  223. flock/core/config/flock_agent_config.py +0 -58
  224. flock/core/config/scheduled_agent_config.py +0 -40
  225. flock/core/context/context.py +0 -213
  226. flock/core/context/context_manager.py +0 -37
  227. flock/core/context/context_vars.py +0 -10
  228. flock/core/evaluation/utils.py +0 -396
  229. flock/core/execution/batch_executor.py +0 -369
  230. flock/core/execution/evaluation_executor.py +0 -438
  231. flock/core/execution/local_executor.py +0 -31
  232. flock/core/execution/opik_executor.py +0 -103
  233. flock/core/execution/temporal_executor.py +0 -164
  234. flock/core/flock.py +0 -634
  235. flock/core/flock_agent.py +0 -336
  236. flock/core/flock_factory.py +0 -613
  237. flock/core/flock_scheduler.py +0 -166
  238. flock/core/flock_server_manager.py +0 -136
  239. flock/core/interpreter/python_interpreter.py +0 -689
  240. flock/core/mcp/__init__.py +0 -1
  241. flock/core/mcp/flock_mcp_server.py +0 -680
  242. flock/core/mcp/mcp_client_manager.py +0 -201
  243. flock/core/mcp/types/__init__.py +0 -1
  244. flock/core/mixin/dspy_integration.py +0 -403
  245. flock/core/mixin/prompt_parser.py +0 -125
  246. flock/core/orchestration/__init__.py +0 -15
  247. flock/core/orchestration/flock_batch_processor.py +0 -94
  248. flock/core/orchestration/flock_evaluator.py +0 -113
  249. flock/core/orchestration/flock_execution.py +0 -295
  250. flock/core/orchestration/flock_initialization.py +0 -149
  251. flock/core/orchestration/flock_server_manager.py +0 -67
  252. flock/core/orchestration/flock_web_server.py +0 -117
  253. flock/core/registry/__init__.py +0 -45
  254. flock/core/registry/agent_registry.py +0 -69
  255. flock/core/registry/callable_registry.py +0 -139
  256. flock/core/registry/component_discovery.py +0 -142
  257. flock/core/registry/component_registry.py +0 -64
  258. flock/core/registry/config_mapping.py +0 -64
  259. flock/core/registry/decorators.py +0 -137
  260. flock/core/registry/registry_hub.py +0 -205
  261. flock/core/registry/server_registry.py +0 -57
  262. flock/core/registry/type_registry.py +0 -86
  263. flock/core/serialization/__init__.py +0 -13
  264. flock/core/serialization/callable_registry.py +0 -52
  265. flock/core/serialization/flock_serializer.py +0 -832
  266. flock/core/serialization/json_encoder.py +0 -41
  267. flock/core/serialization/secure_serializer.py +0 -175
  268. flock/core/serialization/serializable.py +0 -342
  269. flock/core/serialization/serialization_utils.py +0 -412
  270. flock/core/util/file_path_utils.py +0 -223
  271. flock/core/util/hydrator.py +0 -309
  272. flock/core/util/input_resolver.py +0 -164
  273. flock/core/util/loader.py +0 -59
  274. flock/core/util/splitter.py +0 -219
  275. flock/di.py +0 -27
  276. flock/platform/docker_tools.py +0 -49
  277. flock/platform/jaeger_install.py +0 -86
  278. flock/webapp/__init__.py +0 -1
  279. flock/webapp/app/__init__.py +0 -0
  280. flock/webapp/app/api/__init__.py +0 -0
  281. flock/webapp/app/api/agent_management.py +0 -241
  282. flock/webapp/app/api/execution.py +0 -709
  283. flock/webapp/app/api/flock_management.py +0 -129
  284. flock/webapp/app/api/registry_viewer.py +0 -30
  285. flock/webapp/app/chat.py +0 -665
  286. flock/webapp/app/config.py +0 -104
  287. flock/webapp/app/dependencies.py +0 -117
  288. flock/webapp/app/main.py +0 -1070
  289. flock/webapp/app/middleware.py +0 -113
  290. flock/webapp/app/models_ui.py +0 -7
  291. flock/webapp/app/services/__init__.py +0 -0
  292. flock/webapp/app/services/feedback_file_service.py +0 -363
  293. flock/webapp/app/services/flock_service.py +0 -337
  294. flock/webapp/app/services/sharing_models.py +0 -81
  295. flock/webapp/app/services/sharing_store.py +0 -762
  296. flock/webapp/app/templates/theme_mapper.html +0 -326
  297. flock/webapp/app/theme_mapper.py +0 -812
  298. flock/webapp/app/utils.py +0 -85
  299. flock/webapp/run.py +0 -215
  300. flock/webapp/static/css/chat.css +0 -301
  301. flock/webapp/static/css/components.css +0 -167
  302. flock/webapp/static/css/header.css +0 -39
  303. flock/webapp/static/css/layout.css +0 -46
  304. flock/webapp/static/css/sidebar.css +0 -127
  305. flock/webapp/static/css/two-pane.css +0 -48
  306. flock/webapp/templates/base.html +0 -200
  307. flock/webapp/templates/chat.html +0 -152
  308. flock/webapp/templates/chat_settings.html +0 -19
  309. flock/webapp/templates/flock_editor.html +0 -16
  310. flock/webapp/templates/index.html +0 -12
  311. flock/webapp/templates/partials/_agent_detail_form.html +0 -93
  312. flock/webapp/templates/partials/_agent_list.html +0 -18
  313. flock/webapp/templates/partials/_agent_manager_view.html +0 -51
  314. flock/webapp/templates/partials/_agent_tools_checklist.html +0 -14
  315. flock/webapp/templates/partials/_chat_container.html +0 -15
  316. flock/webapp/templates/partials/_chat_messages.html +0 -57
  317. flock/webapp/templates/partials/_chat_settings_form.html +0 -85
  318. flock/webapp/templates/partials/_create_flock_form.html +0 -50
  319. flock/webapp/templates/partials/_dashboard_flock_detail.html +0 -17
  320. flock/webapp/templates/partials/_dashboard_flock_file_list.html +0 -16
  321. flock/webapp/templates/partials/_dashboard_flock_properties_preview.html +0 -28
  322. flock/webapp/templates/partials/_dashboard_upload_flock_form.html +0 -16
  323. flock/webapp/templates/partials/_dynamic_input_form_content.html +0 -22
  324. flock/webapp/templates/partials/_env_vars_table.html +0 -23
  325. flock/webapp/templates/partials/_execution_form.html +0 -118
  326. flock/webapp/templates/partials/_execution_view_container.html +0 -28
  327. flock/webapp/templates/partials/_flock_file_list.html +0 -23
  328. flock/webapp/templates/partials/_flock_properties_form.html +0 -52
  329. flock/webapp/templates/partials/_flock_upload_form.html +0 -16
  330. flock/webapp/templates/partials/_header_flock_status.html +0 -5
  331. flock/webapp/templates/partials/_load_manager_view.html +0 -49
  332. flock/webapp/templates/partials/_registry_table.html +0 -25
  333. flock/webapp/templates/partials/_registry_viewer_content.html +0 -70
  334. flock/webapp/templates/partials/_results_display.html +0 -78
  335. flock/webapp/templates/partials/_settings_env_content.html +0 -9
  336. flock/webapp/templates/partials/_settings_theme_content.html +0 -14
  337. flock/webapp/templates/partials/_settings_view.html +0 -36
  338. flock/webapp/templates/partials/_share_chat_link_snippet.html +0 -11
  339. flock/webapp/templates/partials/_share_link_snippet.html +0 -35
  340. flock/webapp/templates/partials/_sidebar.html +0 -74
  341. flock/webapp/templates/partials/_streaming_results_container.html +0 -195
  342. flock/webapp/templates/partials/_structured_data_view.html +0 -40
  343. flock/webapp/templates/partials/_theme_preview.html +0 -36
  344. flock/webapp/templates/registry_viewer.html +0 -84
  345. flock/webapp/templates/shared_run_page.html +0 -140
  346. flock/workflow/__init__.py +0 -0
  347. flock/workflow/activities.py +0 -196
  348. flock/workflow/agent_activities.py +0 -24
  349. flock/workflow/agent_execution_activity.py +0 -202
  350. flock/workflow/flock_workflow.py +0 -214
  351. flock/workflow/temporal_config.py +0 -96
  352. flock/workflow/temporal_setup.py +0 -68
  353. flock_core-0.5.0b28.dist-info/METADATA +0 -274
  354. flock_core-0.5.0b28.dist-info/RECORD +0 -561
  355. flock_core-0.5.0b28.dist-info/entry_points.txt +0 -2
  356. /flock/{core/logging → logging}/formatters/themes.py +0 -0
  357. /flock/{core/logging → logging}/span_middleware/baggage_span_processor.py +0 -0
  358. /flock/{core/mcp → mcp}/util/__init__.py +0 -0
  359. {flock_core-0.5.0b28.dist-info → flock_core-0.5.56b0.dist-info}/WHEEL +0 -0
@@ -1,689 +0,0 @@
1
- # =========== Copyright 2023 @ CAMEL-AI.org. All Rights Reserved. ===========
2
- # Licensed under the Apache License, Version 2.0 (the “License”);
3
- # you may not use this file except in compliance with the License.
4
- # You may obtain a copy of the License at
5
- #
6
- # http://www.apache.org/licenses/LICENSE-2.0
7
- #
8
- # Unless required by applicable law or agreed to in writing, software
9
- # distributed under the License is distributed on an “AS IS” BASIS,
10
- # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
11
- # See the License for the specific language governing permissions and
12
- # limitations under the License.
13
- # =========== Copyright 2023 @ CAMEL-AI.org. All Rights Reserved. ===========
14
- import ast
15
- import builtins
16
- import difflib
17
- import importlib
18
- import re
19
- import typing
20
- from collections.abc import Mapping
21
- from typing import (
22
- Any,
23
- )
24
-
25
- from opentelemetry import trace
26
-
27
- from flock.core.logging.logging import get_logger
28
-
29
- tracer = trace.get_tracer(__name__)
30
- logger = get_logger("interpreter")
31
-
32
-
33
- class InterpreterError(ValueError):
34
- r"""An error raised when the interpreter cannot evaluate a Python
35
- expression, due to syntax error or unsupported operations.
36
- """
37
-
38
- pass
39
-
40
- class BreakException(Exception):
41
- """Signal a 'break' from the simulated loop."""
42
- pass
43
-
44
- class ContinueException(Exception):
45
- """Signal a 'continue' in the simulated loop."""
46
- pass
47
-
48
-
49
- class PythonInterpreter:
50
- r"""A customized python interpreter to control the execution of
51
- LLM-generated codes. The interpreter makes sure the code can only execute
52
- functions given in action space and import white list. It also supports
53
- fuzzy variable matching to receive uncertain input variable name.
54
-
55
- Args:
56
- action_space (Dict[str, Any]): A dictionary mapping action names to
57
- their corresponding functions or objects.
58
- import_white_list (Optional[List[str]], optional): A list of allowed modules.
59
- verbose (bool, optional): If True, the interpreter prints log messages
60
- as it executes the code. (default: False)
61
- """
62
-
63
- def __init__(
64
- self,
65
- action_space: dict[str, Any],
66
- import_white_list: list[str] | None = None,
67
- verbose: bool = False,
68
- ) -> None:
69
- self.action_space = action_space
70
- self.state = self.action_space.copy()
71
- self.fuzz_state: dict[str, Any] = {}
72
- self.import_white_list = import_white_list or [
73
- "math",
74
- "random",
75
- "datetime",
76
- "time",
77
- "string",
78
- "collections",
79
- "itertools",
80
- "functools",
81
- "typing",
82
- "enum",
83
- "json",
84
- "ast",
85
- "numpy",
86
- "sympy",
87
- "pandas",
88
- ] # default imports
89
- self.verbose = verbose
90
-
91
- def log(self, message: str) -> None:
92
- """Print a log message immediately."""
93
- # print(message, flush=True)
94
- logger.info(message, flush=True)
95
-
96
- def execute(
97
- self,
98
- code: str,
99
- state: dict[str, Any] | None = None,
100
- fuzz_state: dict[str, Any] | None = None,
101
- keep_state: bool = True,
102
- ) -> Any:
103
- r"""Execute the input python codes in a secure environment.
104
-
105
- [Documentation omitted for brevity]
106
- """
107
- try:
108
- if state is not None:
109
- self.state.update(state)
110
- if fuzz_state is not None:
111
- self.fuzz_state.update(fuzz_state)
112
-
113
- try:
114
- expression = ast.parse(code)
115
- except SyntaxError as e:
116
- error_line = code.splitlines()[e.lineno - 1]
117
- self.log(
118
- f"[Interpreter] Syntax error in code at line {e.lineno}: {error_line}\nError: {e}")
119
- return(
120
- f"Syntax error in code at line {e.lineno}: {error_line}\nError: {e}"
121
- )
122
-
123
- result = None
124
- if self.verbose:
125
- self.log("[Interpreter] Starting code execution...")
126
-
127
- for idx, node in enumerate(expression.body):
128
- # Log the AST node being executed (using unparse if available)
129
- if self.verbose:
130
- try:
131
- node_repr = ast.unparse(node)
132
- except Exception:
133
- node_repr = ast.dump(node)
134
- self.log(f"[Interpreter] Executing node {idx}: {node_repr}")
135
-
136
- try:
137
- line_result = self._execute_ast(node)
138
- except InterpreterError as e:
139
- if not keep_state:
140
- self.clear_state()
141
- msg = f"Evaluation of the code stopped at node {idx}. See:\n{e}"
142
- return msg
143
- if line_result is not None:
144
- result = line_result
145
- if self.verbose:
146
- self.log(f"[Interpreter] Node {idx} result: {result}")
147
-
148
- if self.verbose:
149
- self.log("[Interpreter] Finished code execution.")
150
- if not keep_state:
151
- self.clear_state()
152
-
153
- return result
154
- except Exception as e:
155
- self.log(
156
- f"[Interpreter] Error during code execution: {e}")
157
- return f"[Interpreter] Error during code execution: {e}"
158
-
159
- def clear_state(self) -> None:
160
- r"""Initialize :obj:`state` and :obj:`fuzz_state`"""
161
- self.state = self.action_space.copy()
162
- self.fuzz_state = {}
163
-
164
- # ast.Index is deprecated after python 3.9, which cannot pass type check,
165
- # but is still necessary for older versions.
166
- @typing.no_type_check
167
- def _execute_ast(self, expression: ast.AST) -> Any:
168
- if isinstance(expression, ast.Assign):
169
- return self._execute_assign(expression)
170
- elif isinstance(expression, ast.Attribute):
171
- value = self._execute_ast(expression.value)
172
- return getattr(value, expression.attr)
173
- elif isinstance(expression, ast.AugAssign):
174
- return self._execute_augassign(expression)
175
- elif isinstance(expression, ast.BinOp):
176
- return self._execute_binop(expression)
177
- elif isinstance(expression, ast.BoolOp):
178
- return self._execute_condition(expression)
179
- elif isinstance(expression, ast.Call):
180
- return self._execute_call(expression)
181
- elif isinstance(expression, ast.Compare):
182
- return self._execute_condition(expression)
183
- elif isinstance(expression, ast.Constant):
184
- return expression.value
185
- elif isinstance(expression, ast.Dict):
186
- result: dict = {}
187
- for k, v in zip(expression.keys, expression.values):
188
- if k is not None:
189
- result[self._execute_ast(k)] = self._execute_ast(v)
190
- else:
191
- result.update(self._execute_ast(v))
192
- return result
193
- elif isinstance(expression, ast.Expr):
194
- return self._execute_ast(expression.value)
195
- elif isinstance(expression, ast.For):
196
- return self._execute_for(expression)
197
- elif isinstance(expression, ast.FormattedValue):
198
- return self._execute_ast(expression.value)
199
- elif isinstance(expression, ast.FunctionDef):
200
- self.state[expression.name] = expression
201
- return None
202
- elif isinstance(expression, ast.GeneratorExp):
203
- return self._execute_generatorexp(expression)
204
- elif isinstance(expression, ast.If):
205
- return self._execute_if(expression)
206
- elif isinstance(expression, ast.IfExp):
207
- return self._execute_ifexp(expression)
208
- elif isinstance(expression, ast.Import):
209
- self._execute_import(expression)
210
- return None
211
- elif isinstance(expression, ast.ImportFrom):
212
- self._execute_import_from(expression)
213
- return None
214
- elif hasattr(ast, "Index") and isinstance(expression, ast.Index):
215
- return self._execute_ast(expression.value)
216
- elif isinstance(expression, ast.JoinedStr):
217
- return "".join(
218
- [str(self._execute_ast(v)) for v in expression.values]
219
- )
220
- elif isinstance(expression, ast.Lambda):
221
- return self._execute_lambda(expression)
222
- elif isinstance(expression, ast.List):
223
- return [self._execute_ast(elt) for elt in expression.elts]
224
- elif isinstance(expression, ast.Name):
225
- return self._execute_name(expression)
226
- elif isinstance(expression, ast.Return):
227
- return self._execute_ast(expression.value)
228
- elif isinstance(expression, ast.Subscript):
229
- return self._execute_subscript(expression)
230
- elif isinstance(expression, ast.Tuple):
231
- return tuple([self._execute_ast(elt) for elt in expression.elts])
232
- elif isinstance(expression, ast.UnaryOp):
233
- return self._execute_unaryop(expression)
234
- elif isinstance(expression, ast.While):
235
- return self._execute_while(expression)
236
- elif isinstance(expression, ast.ListComp):
237
- return self._execute_listcomp(expression)
238
- elif isinstance(expression, ast.DictComp):
239
- return self._execute_dictcomp(expression)
240
- elif isinstance(expression, ast.SetComp):
241
- return self._execute_setcomp(expression)
242
- elif isinstance(expression, ast.Break):
243
- raise BreakException()
244
- elif isinstance(expression, ast.Continue):
245
- raise ContinueException()
246
- elif isinstance(expression, ast.Try):
247
- return self._execute_try(expression)
248
- elif isinstance(expression, ast.Raise):
249
- return self._execute_raise(expression)
250
- elif isinstance(expression, ast.Pass):
251
- return None
252
- elif isinstance(expression, ast.Assert):
253
- return self._execute_assert(expression)
254
- else:
255
- return(
256
- f"{expression.__class__.__name__} is not supported."
257
- )
258
-
259
- def _execute_assign(self, assign: ast.Assign) -> Any:
260
- targets = assign.targets
261
- result = self._execute_ast(assign.value)
262
-
263
- for target in targets:
264
- self._assign(target, result)
265
- return result
266
-
267
- def _assign(self, target: ast.expr, value: Any):
268
- if isinstance(target, ast.Name):
269
- self.state[target.id] = value
270
- elif isinstance(target, ast.Tuple):
271
- if not isinstance(value, tuple):
272
- return(
273
- f"Expected type tuple, but got {value.__class__.__name__} instead."
274
- )
275
- if len(target.elts) != len(value):
276
- return(
277
- f"Expected {len(target.elts)} values but got {len(value)}."
278
- )
279
- for t, v in zip(target.elts, value):
280
- self.state[self._execute_ast(t)] = v
281
- else:
282
- return(
283
- f"Unsupported variable type. Expected ast.Name or ast.Tuple, got {target.__class__.__name__} instead."
284
- )
285
-
286
- def _execute_call(self, call: ast.Call) -> Any:
287
- callable_func = self._execute_ast(call.func)
288
-
289
- args = [self._execute_ast(arg) for arg in call.args]
290
- kwargs = {
291
- keyword.arg: self._execute_ast(keyword.value)
292
- for keyword in call.keywords
293
- }
294
- if isinstance(callable_func, ast.FunctionDef):
295
- old_state = self.state.copy()
296
- for param_name, arg_value in zip(
297
- [param.arg for param in callable_func.args.args], args
298
- ):
299
- self.state[param_name] = arg_value
300
- result = None
301
- for stmt in callable_func.body:
302
- result = self._execute_ast(stmt)
303
- if isinstance(stmt, ast.Return):
304
- break
305
- self.state = old_state
306
- return result
307
- return callable_func(*args, **kwargs)
308
-
309
- def _execute_augassign(self, augassign: ast.AugAssign):
310
- current_value = self.state[augassign.target.id]
311
- increment_value = self._execute_ast(augassign.value)
312
- if not (
313
- isinstance(current_value, (int, float))
314
- and isinstance(increment_value, (int, float))
315
- ):
316
- return(
317
- f"Invalid types for augmented assignment: {type(current_value)}, {type(increment_value)}"
318
- )
319
- if isinstance(augassign.op, ast.Add):
320
- new_value = current_value + increment_value
321
- elif isinstance(augassign.op, ast.Sub):
322
- new_value = current_value - increment_value
323
- elif isinstance(augassign.op, ast.Mult):
324
- new_value = current_value * increment_value
325
- elif isinstance(augassign.op, ast.Div):
326
- new_value = current_value / increment_value
327
- else:
328
- return(
329
- f"Augmented assignment operator {augassign.op} is not supported"
330
- )
331
- self._assign(augassign.target, new_value)
332
- return new_value
333
-
334
- def _execute_subscript(self, subscript: ast.Subscript):
335
- index = self._execute_ast(subscript.slice)
336
- value = self._execute_ast(subscript.value)
337
- if not isinstance(subscript.ctx, ast.Load):
338
- return(
339
- f"{subscript.ctx.__class__.__name__} is not supported for subscript."
340
- )
341
- if isinstance(value, (list, tuple)):
342
- return value[int(index)]
343
- if index in value:
344
- return value[index]
345
- if isinstance(index, str) and isinstance(value, Mapping):
346
- close_matches = difflib.get_close_matches(index, list(value.keys()))
347
- if len(close_matches) > 0:
348
- return value[close_matches[0]]
349
- return(f"Could not index {value} with '{index}'.")
350
-
351
- def _execute_name(self, name: ast.Name):
352
- if name.id in dir(builtins):
353
- return getattr(builtins, name.id)
354
- if isinstance(name.ctx, ast.Store):
355
- return name.id
356
- elif isinstance(name.ctx, ast.Load):
357
- return self._get_value_from_state(name.id)
358
- else:
359
- return(f"{name.ctx} is not supported.")
360
-
361
- def _execute_condition(self, condition):
362
- if isinstance(condition, ast.BoolOp):
363
- if isinstance(condition.op, ast.And):
364
- results = [
365
- self._execute_ast(value) for value in condition.values
366
- ]
367
- return all(results)
368
- elif isinstance(condition.op, ast.Or):
369
- results = [
370
- self._execute_ast(value) for value in condition.values
371
- ]
372
- return any(results)
373
- else:
374
- return(
375
- f"Boolean operator {condition.op} is not supported"
376
- )
377
- elif isinstance(condition, ast.Compare):
378
- if len(condition.ops) > 1:
379
- return(
380
- "Cannot evaluate conditions with multiple operators"
381
- )
382
- left = self._execute_ast(condition.left)
383
- comparator = condition.ops[0]
384
- right = self._execute_ast(condition.comparators[0])
385
- if isinstance(comparator, ast.Eq):
386
- return left == right
387
- elif isinstance(comparator, ast.NotEq):
388
- return left != right
389
- elif isinstance(comparator, ast.Lt):
390
- return left < right
391
- elif isinstance(comparator, ast.LtE):
392
- return left <= right
393
- elif isinstance(comparator, ast.Gt):
394
- return left > right
395
- elif isinstance(comparator, ast.GtE):
396
- return left >= right
397
- elif isinstance(comparator, ast.Is):
398
- return left is right
399
- elif isinstance(comparator, ast.IsNot):
400
- return left is not right
401
- elif isinstance(comparator, ast.In):
402
- return left in right
403
- elif isinstance(comparator, ast.NotIn):
404
- return left not in right
405
- else:
406
- return("Unsupported comparison operator")
407
- elif isinstance(condition, ast.UnaryOp):
408
- return self._execute_unaryop(condition)
409
- elif isinstance(condition, ast.Name) or isinstance(condition, ast.Call):
410
- return bool(self._execute_ast(condition))
411
- elif isinstance(condition, ast.Constant):
412
- return bool(condition.value)
413
- else:
414
- return(
415
- f"Unsupported condition type: {type(condition).__name__}"
416
- )
417
-
418
- def _execute_if(self, if_statement: ast.If):
419
- result = None
420
- if self._execute_condition(if_statement.test):
421
- for line in if_statement.body:
422
- line_result = self._execute_ast(line)
423
- if line_result is not None:
424
- result = line_result
425
- else:
426
- for line in if_statement.orelse:
427
- line_result = self._execute_ast(line)
428
- if line_result is not None:
429
- result = line_result
430
- return result
431
-
432
- def _execute_ifexp(self, ifexp: ast.IfExp) -> Any:
433
- test_result = self._execute_condition(ifexp.test)
434
- if test_result:
435
- return self._execute_ast(ifexp.body)
436
- else:
437
- return self._execute_ast(ifexp.orelse)
438
-
439
- def _execute_import(self, import_module: ast.Import) -> None:
440
- for module in import_module.names:
441
- self._validate_import(module.name)
442
- alias = module.asname or module.name
443
- self.state[alias] = importlib.import_module(module.name)
444
-
445
- def _execute_import_from(self, import_from: ast.ImportFrom):
446
- if import_from.module is None:
447
- return('"from . import" is not supported.')
448
- for import_name in import_from.names:
449
- full_name = import_from.module + f".{import_name.name}"
450
- self._validate_import(full_name)
451
- imported_module = importlib.import_module(import_from.module)
452
- alias = import_name.asname or import_name.name
453
- self.state[alias] = getattr(imported_module, import_name.name)
454
-
455
- # Note: Two versions of _execute_for and _execute_while appear in this file.
456
- # We keep both as provided, but you may wish to consolidate these in your code.
457
-
458
- def _execute_for(self, for_statement: ast.For):
459
- result = None
460
- try:
461
- for value in self._execute_ast(for_statement.iter):
462
- self._assign(for_statement.target, value)
463
- try:
464
- for line in for_statement.body:
465
- line_result = self._execute_ast(line)
466
- if line_result is not None:
467
- result = line_result
468
- except ContinueException:
469
- continue
470
- except BreakException:
471
- pass
472
- return result
473
-
474
- def _execute_while(self, while_statement: ast.While):
475
- result = None
476
- try:
477
- while self._execute_condition(while_statement.test):
478
- try:
479
- for line in while_statement.body:
480
- line_result = self._execute_ast(line)
481
- if line_result is not None:
482
- result = line_result
483
- except ContinueException:
484
- continue
485
- except BreakException:
486
- pass
487
- return result
488
-
489
- def _execute_try(self, try_statement: ast.Try):
490
- try:
491
- for line in try_statement.body:
492
- self._execute_ast(line)
493
- except Exception as e:
494
- handled = False
495
- for handler in try_statement.handlers:
496
- if handler.type is None or isinstance(
497
- e, self._execute_ast(handler.type)
498
- ):
499
- if handler.name:
500
- self.state[handler.name.id] = e
501
- for line in handler.body:
502
- self._execute_ast(line)
503
- handled = True
504
- break
505
- if not handled:
506
- raise
507
- finally:
508
- for line in try_statement.finalbody:
509
- self._execute_ast(line)
510
-
511
- def _execute_raise(self, raise_statement: ast.Raise):
512
- if raise_statement.exc:
513
- exception = self._execute_ast(raise_statement.exc)
514
- raise exception
515
- else:
516
- raise
517
-
518
- def _execute_assert(self, assert_statement: ast.Assert):
519
- test_result = self._execute_condition(assert_statement.test)
520
- if not test_result:
521
- if assert_statement.msg:
522
- msg = self._execute_ast(assert_statement.msg)
523
- raise AssertionError(msg)
524
- else:
525
- raise AssertionError
526
-
527
- def _execute_lambda(self, lambda_node: ast.Lambda) -> Any:
528
- def lambda_function(*args):
529
- old_state = self.state.copy()
530
- for param, arg in zip(lambda_node.args.args, args):
531
- self.state[param.arg] = arg
532
- result = self._execute_ast(lambda_node.body)
533
- self.state = old_state # Restore the state
534
- return result
535
-
536
- return lambda_function
537
-
538
- def _validate_import(self, full_name: str):
539
- tmp_name = ""
540
- found_name = False
541
- for name in full_name.split("."):
542
- tmp_name += name if tmp_name == "" else f".{name}"
543
- if tmp_name in self.import_white_list:
544
- found_name = True
545
- return
546
- if not found_name:
547
- return(
548
- f"It is not permitted to import modules "
549
- f"than module white list (try to import {full_name})."
550
- )
551
-
552
- def _execute_binop(self, binop: ast.BinOp):
553
- left = self._execute_ast(binop.left)
554
- operator = binop.op
555
- right = self._execute_ast(binop.right)
556
-
557
- if isinstance(operator, ast.Add):
558
- return left + right
559
- elif isinstance(operator, ast.Sub):
560
- return left - right
561
- elif isinstance(operator, ast.Mult):
562
- return left * right
563
- elif isinstance(operator, ast.Div):
564
- return left / right
565
- elif isinstance(operator, ast.FloorDiv):
566
- return left // right
567
- elif isinstance(operator, ast.Mod):
568
- return left % right
569
- elif isinstance(operator, ast.Pow):
570
- return left**right
571
- elif isinstance(operator, ast.LShift):
572
- return left << right
573
- elif isinstance(operator, ast.RShift):
574
- return left >> right
575
- elif isinstance(operator, ast.BitAnd):
576
- return left & right
577
- elif isinstance(operator, ast.BitOr):
578
- return left | right
579
- elif isinstance(operator, ast.BitXor):
580
- return left ^ right
581
- elif isinstance(operator, ast.MatMult):
582
- return left @ right
583
- else:
584
- return(f"Operator not supported: {operator}")
585
-
586
- def _execute_unaryop(self, unaryop: ast.UnaryOp):
587
- operand = self._execute_ast(unaryop.operand)
588
- operator = unaryop.op
589
-
590
- if isinstance(operator, ast.UAdd):
591
- return +operand
592
- elif isinstance(operator, ast.USub):
593
- return -operand
594
- elif isinstance(operator, ast.Not):
595
- return not operand
596
- elif isinstance(operator, ast.Invert):
597
- return ~operand
598
- else:
599
- return(f"Operator not supported: {operator}")
600
-
601
- def _execute_listcomp(self, comp: ast.ListComp):
602
- return self._execute_comp(comp.elt, comp.generators)
603
-
604
- def _execute_setcomp(self, comp: ast.SetComp):
605
- return set(self._execute_comp(comp.elt, comp.generators))
606
-
607
- def _execute_dictcomp(self, comp: ast.DictComp):
608
- keys = self._execute_comp(comp.key, comp.generators)
609
- values = self._execute_comp(comp.value, comp.generators)
610
- return dict(zip(keys, values))
611
-
612
- def _execute_comp(self, elt, generators):
613
- # Base-case: wrap the single element in a list so that
614
- # callers can safely .extend() it.
615
- if not generators:
616
- return [self._execute_ast(elt)]
617
-
618
- gen = generators[0]
619
- acc: list[Any] = []
620
- for value in self._execute_ast(gen.iter):
621
- self._assign(gen.target, value)
622
- if all(self._execute_condition(if_cond) for if_cond in gen.ifs):
623
- acc.extend(self._execute_comp(elt, generators[1:]))
624
- return acc
625
-
626
-
627
- def _execute_generatorexp(self, genexp: ast.GeneratorExp):
628
- def generator():
629
- for value in self._execute_comp(genexp.elt, genexp.generators):
630
- yield value
631
-
632
- return generator()
633
-
634
- def _get_value_from_state(self, key: str) -> Any:
635
- if key in self.state:
636
- return self.state[key]
637
- elif key in self.fuzz_state:
638
- return self.fuzz_state[key]
639
- else:
640
- return(f"The variable `{key}` is not defined.")
641
-
642
-
643
- class TextPrompt(str):
644
- r"""A class that represents a text prompt. The :obj:`TextPrompt` class
645
- extends the built-in :obj:`str` class to provide a property for retrieving
646
- the set of keywords in the prompt.
647
- """
648
-
649
- @property
650
- def key_words(self) -> set[str]:
651
- pattern = re.compile(r"\{([^{}]+)\}")
652
- found = pattern.findall(self)
653
- return set(found)
654
-
655
- def format(self, *args: Any, **kwargs: Any) -> "TextPrompt":
656
- default_kwargs = {key: "{" + f"{key}" + "}" for key in self.key_words}
657
- default_kwargs.update(kwargs)
658
- return TextPrompt(super().format(*args, **default_kwargs))
659
-
660
-
661
- class CodePrompt(TextPrompt):
662
- r"""A class that represents a code prompt. It extends the :obj:`TextPrompt`
663
- class with a :obj:`code_type` property.
664
- """
665
-
666
- def __new__(cls, *args: Any, **kwargs: Any) -> "CodePrompt":
667
- code_type = kwargs.pop("code_type", None)
668
- instance = super().__new__(cls, *args, **kwargs)
669
- instance._code_type = code_type
670
- return instance
671
-
672
- @property
673
- def code_type(self) -> str | None:
674
- return self._code_type
675
-
676
- def set_code_type(self, code_type: str) -> None:
677
- self._code_type = code_type
678
-
679
- def execute(
680
- self,
681
- interpreter: PythonInterpreter | None = None,
682
- user_variable: dict[str, Any] | None = None,
683
- ) -> tuple[Any, PythonInterpreter]:
684
- if not interpreter:
685
- interpreter = PythonInterpreter(action_space=globals())
686
- execution_res = interpreter.execute(
687
- self, fuzz_state=user_variable, keep_state=True
688
- )
689
- return execution_res, interpreter
@@ -1 +0,0 @@
1
- """Flock MCP package."""