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.
- flock/__init__.py +12 -217
- flock/agent.py +678 -0
- flock/api/themes.py +71 -0
- flock/artifacts.py +79 -0
- flock/cli.py +75 -0
- flock/components.py +173 -0
- flock/dashboard/__init__.py +28 -0
- flock/dashboard/collector.py +283 -0
- flock/dashboard/events.py +182 -0
- flock/dashboard/launcher.py +230 -0
- flock/dashboard/service.py +537 -0
- flock/dashboard/websocket.py +235 -0
- flock/engines/__init__.py +6 -0
- flock/engines/dspy_engine.py +856 -0
- flock/examples.py +128 -0
- flock/{core/util → helper}/cli_helper.py +4 -3
- flock/{core/logging → logging}/__init__.py +2 -3
- flock/{core/logging → logging}/formatters/enum_builder.py +3 -4
- flock/{core/logging → logging}/formatters/theme_builder.py +19 -44
- flock/{core/logging → logging}/formatters/themed_formatter.py +69 -115
- flock/{core/logging → logging}/logging.py +77 -61
- flock/{core/logging → logging}/telemetry.py +20 -26
- flock/{core/logging → logging}/telemetry_exporter/base_exporter.py +2 -2
- flock/{core/logging → logging}/telemetry_exporter/file_exporter.py +6 -9
- flock/{core/logging → logging}/telemetry_exporter/sqlite_exporter.py +2 -3
- flock/{core/logging → logging}/trace_and_logged.py +20 -24
- flock/mcp/__init__.py +91 -0
- flock/{core/mcp/mcp_client.py → mcp/client.py} +103 -154
- flock/{core/mcp/mcp_config.py → mcp/config.py} +62 -117
- flock/mcp/manager.py +255 -0
- flock/mcp/servers/sse/__init__.py +1 -1
- flock/mcp/servers/sse/flock_sse_server.py +11 -53
- flock/mcp/servers/stdio/__init__.py +1 -1
- flock/mcp/servers/stdio/flock_stdio_server.py +8 -48
- flock/mcp/servers/streamable_http/flock_streamable_http_server.py +17 -62
- flock/mcp/servers/websockets/flock_websocket_server.py +7 -40
- flock/{core/mcp/flock_mcp_tool.py → mcp/tool.py} +16 -26
- flock/mcp/types/__init__.py +42 -0
- flock/{core/mcp → mcp}/types/callbacks.py +9 -15
- flock/{core/mcp → mcp}/types/factories.py +7 -6
- flock/{core/mcp → mcp}/types/handlers.py +13 -18
- flock/{core/mcp → mcp}/types/types.py +70 -74
- flock/{core/mcp → mcp}/util/helpers.py +1 -1
- flock/orchestrator.py +645 -0
- flock/registry.py +148 -0
- flock/runtime.py +262 -0
- flock/service.py +140 -0
- flock/store.py +69 -0
- flock/subscription.py +111 -0
- flock/themes/andromeda.toml +1 -1
- flock/themes/apple-system-colors.toml +1 -1
- flock/themes/arcoiris.toml +1 -1
- flock/themes/atomonelight.toml +1 -1
- flock/themes/ayu copy.toml +1 -1
- flock/themes/ayu-light.toml +1 -1
- flock/themes/belafonte-day.toml +1 -1
- flock/themes/belafonte-night.toml +1 -1
- flock/themes/blulocodark.toml +1 -1
- flock/themes/breeze.toml +1 -1
- flock/themes/broadcast.toml +1 -1
- flock/themes/brogrammer.toml +1 -1
- flock/themes/builtin-dark.toml +1 -1
- flock/themes/builtin-pastel-dark.toml +1 -1
- flock/themes/catppuccin-latte.toml +1 -1
- flock/themes/catppuccin-macchiato.toml +1 -1
- flock/themes/catppuccin-mocha.toml +1 -1
- flock/themes/cga.toml +1 -1
- flock/themes/chalk.toml +1 -1
- flock/themes/ciapre.toml +1 -1
- flock/themes/coffee-theme.toml +1 -1
- flock/themes/cyberpunkscarletprotocol.toml +1 -1
- flock/themes/dark+.toml +1 -1
- flock/themes/darkermatrix.toml +1 -1
- flock/themes/darkside.toml +1 -1
- flock/themes/desert.toml +1 -1
- flock/themes/django.toml +1 -1
- flock/themes/djangosmooth.toml +1 -1
- flock/themes/doomone.toml +1 -1
- flock/themes/dotgov.toml +1 -1
- flock/themes/dracula+.toml +1 -1
- flock/themes/duckbones.toml +1 -1
- flock/themes/encom.toml +1 -1
- flock/themes/espresso.toml +1 -1
- flock/themes/everblush.toml +1 -1
- flock/themes/fairyfloss.toml +1 -1
- flock/themes/fideloper.toml +1 -1
- flock/themes/fishtank.toml +1 -1
- flock/themes/flexoki-light.toml +1 -1
- flock/themes/floraverse.toml +1 -1
- flock/themes/framer.toml +1 -1
- flock/themes/galizur.toml +1 -1
- flock/themes/github.toml +1 -1
- flock/themes/grass.toml +1 -1
- flock/themes/grey-green.toml +1 -1
- flock/themes/gruvboxlight.toml +1 -1
- flock/themes/guezwhoz.toml +1 -1
- flock/themes/harper.toml +1 -1
- flock/themes/hax0r-blue.toml +1 -1
- flock/themes/hopscotch.256.toml +1 -1
- flock/themes/ic-green-ppl.toml +1 -1
- flock/themes/iceberg-dark.toml +1 -1
- flock/themes/japanesque.toml +1 -1
- flock/themes/jubi.toml +1 -1
- flock/themes/kibble.toml +1 -1
- flock/themes/kolorit.toml +1 -1
- flock/themes/kurokula.toml +1 -1
- flock/themes/materialdesigncolors.toml +1 -1
- flock/themes/matrix.toml +1 -1
- flock/themes/mellifluous.toml +1 -1
- flock/themes/midnight-in-mojave.toml +1 -1
- flock/themes/monokai-remastered.toml +1 -1
- flock/themes/monokai-soda.toml +1 -1
- flock/themes/neon.toml +1 -1
- flock/themes/neopolitan.toml +1 -1
- flock/themes/nord-light.toml +1 -1
- flock/themes/ocean.toml +1 -1
- flock/themes/onehalfdark.toml +1 -1
- flock/themes/onehalflight.toml +1 -1
- flock/themes/palenighthc.toml +1 -1
- flock/themes/paulmillr.toml +1 -1
- flock/themes/pencildark.toml +1 -1
- flock/themes/pnevma.toml +1 -1
- flock/themes/purple-rain.toml +1 -1
- flock/themes/purplepeter.toml +1 -1
- flock/themes/raycast-dark.toml +1 -1
- flock/themes/red-sands.toml +1 -1
- flock/themes/relaxed.toml +1 -1
- flock/themes/retro.toml +1 -1
- flock/themes/rose-pine.toml +1 -1
- flock/themes/royal.toml +1 -1
- flock/themes/ryuuko.toml +1 -1
- flock/themes/sakura.toml +1 -1
- flock/themes/scarlet-protocol.toml +1 -1
- flock/themes/seoulbones-dark.toml +1 -1
- flock/themes/shades-of-purple.toml +1 -1
- flock/themes/smyck.toml +1 -1
- flock/themes/softserver.toml +1 -1
- flock/themes/solarized-darcula.toml +1 -1
- flock/themes/square.toml +1 -1
- flock/themes/sugarplum.toml +1 -1
- flock/themes/thayer-bright.toml +1 -1
- flock/themes/tokyonight.toml +1 -1
- flock/themes/tomorrow.toml +1 -1
- flock/themes/ubuntu.toml +1 -1
- flock/themes/ultradark.toml +1 -1
- flock/themes/ultraviolent.toml +1 -1
- flock/themes/unikitty.toml +1 -1
- flock/themes/urple.toml +1 -1
- flock/themes/vesper.toml +1 -1
- flock/themes/vimbones.toml +1 -1
- flock/themes/wildcherry.toml +1 -1
- flock/themes/wilmersdorf.toml +1 -1
- flock/themes/wryan.toml +1 -1
- flock/themes/xcodedarkhc.toml +1 -1
- flock/themes/xcodelight.toml +1 -1
- flock/themes/zenbones-light.toml +1 -1
- flock/themes/zenwritten-dark.toml +1 -1
- flock/utilities.py +301 -0
- flock/{components/utility → utility}/output_utility_component.py +68 -53
- flock/visibility.py +107 -0
- flock_core-0.5.56b0.dist-info/METADATA +747 -0
- flock_core-0.5.56b0.dist-info/RECORD +398 -0
- flock_core-0.5.56b0.dist-info/entry_points.txt +2 -0
- {flock_core-0.5.0b28.dist-info → flock_core-0.5.56b0.dist-info}/licenses/LICENSE +1 -1
- flock/adapter/__init__.py +0 -14
- flock/adapter/azure_adapter.py +0 -68
- flock/adapter/chroma_adapter.py +0 -73
- flock/adapter/faiss_adapter.py +0 -97
- flock/adapter/pinecone_adapter.py +0 -51
- flock/adapter/vector_base.py +0 -47
- flock/cli/assets/release_notes.md +0 -140
- flock/cli/config.py +0 -8
- flock/cli/constants.py +0 -36
- flock/cli/create_agent.py +0 -1
- flock/cli/create_flock.py +0 -280
- flock/cli/execute_flock.py +0 -620
- flock/cli/load_agent.py +0 -1
- flock/cli/load_examples.py +0 -1
- flock/cli/load_flock.py +0 -192
- flock/cli/load_release_notes.py +0 -20
- flock/cli/loaded_flock_cli.py +0 -254
- flock/cli/manage_agents.py +0 -459
- flock/cli/registry_management.py +0 -889
- flock/cli/runner.py +0 -41
- flock/cli/settings.py +0 -857
- flock/cli/utils.py +0 -135
- flock/cli/view_results.py +0 -29
- flock/cli/yaml_editor.py +0 -396
- flock/components/__init__.py +0 -30
- flock/components/evaluation/__init__.py +0 -9
- flock/components/evaluation/declarative_evaluation_component.py +0 -606
- flock/components/routing/__init__.py +0 -15
- flock/components/routing/conditional_routing_component.py +0 -494
- flock/components/routing/default_routing_component.py +0 -103
- flock/components/routing/llm_routing_component.py +0 -206
- flock/components/utility/__init__.py +0 -22
- flock/components/utility/example_utility_component.py +0 -250
- flock/components/utility/feedback_utility_component.py +0 -206
- flock/components/utility/memory_utility_component.py +0 -550
- flock/components/utility/metrics_utility_component.py +0 -700
- flock/config.py +0 -61
- flock/core/__init__.py +0 -110
- flock/core/agent/__init__.py +0 -16
- flock/core/agent/default_agent.py +0 -216
- flock/core/agent/flock_agent_components.py +0 -104
- flock/core/agent/flock_agent_execution.py +0 -101
- flock/core/agent/flock_agent_integration.py +0 -260
- flock/core/agent/flock_agent_lifecycle.py +0 -186
- flock/core/agent/flock_agent_serialization.py +0 -381
- flock/core/api/__init__.py +0 -10
- flock/core/api/custom_endpoint.py +0 -45
- flock/core/api/endpoints.py +0 -254
- flock/core/api/main.py +0 -162
- flock/core/api/models.py +0 -97
- flock/core/api/run_store.py +0 -224
- flock/core/api/runner.py +0 -44
- flock/core/api/service.py +0 -214
- flock/core/component/__init__.py +0 -15
- flock/core/component/agent_component_base.py +0 -309
- flock/core/component/evaluation_component.py +0 -62
- flock/core/component/routing_component.py +0 -74
- flock/core/component/utility_component.py +0 -69
- flock/core/config/flock_agent_config.py +0 -58
- flock/core/config/scheduled_agent_config.py +0 -40
- flock/core/context/context.py +0 -213
- flock/core/context/context_manager.py +0 -37
- flock/core/context/context_vars.py +0 -10
- flock/core/evaluation/utils.py +0 -396
- flock/core/execution/batch_executor.py +0 -369
- flock/core/execution/evaluation_executor.py +0 -438
- flock/core/execution/local_executor.py +0 -31
- flock/core/execution/opik_executor.py +0 -103
- flock/core/execution/temporal_executor.py +0 -164
- flock/core/flock.py +0 -634
- flock/core/flock_agent.py +0 -336
- flock/core/flock_factory.py +0 -613
- flock/core/flock_scheduler.py +0 -166
- flock/core/flock_server_manager.py +0 -136
- flock/core/interpreter/python_interpreter.py +0 -689
- flock/core/mcp/__init__.py +0 -1
- flock/core/mcp/flock_mcp_server.py +0 -680
- flock/core/mcp/mcp_client_manager.py +0 -201
- flock/core/mcp/types/__init__.py +0 -1
- flock/core/mixin/dspy_integration.py +0 -403
- flock/core/mixin/prompt_parser.py +0 -125
- flock/core/orchestration/__init__.py +0 -15
- flock/core/orchestration/flock_batch_processor.py +0 -94
- flock/core/orchestration/flock_evaluator.py +0 -113
- flock/core/orchestration/flock_execution.py +0 -295
- flock/core/orchestration/flock_initialization.py +0 -149
- flock/core/orchestration/flock_server_manager.py +0 -67
- flock/core/orchestration/flock_web_server.py +0 -117
- flock/core/registry/__init__.py +0 -45
- flock/core/registry/agent_registry.py +0 -69
- flock/core/registry/callable_registry.py +0 -139
- flock/core/registry/component_discovery.py +0 -142
- flock/core/registry/component_registry.py +0 -64
- flock/core/registry/config_mapping.py +0 -64
- flock/core/registry/decorators.py +0 -137
- flock/core/registry/registry_hub.py +0 -205
- flock/core/registry/server_registry.py +0 -57
- flock/core/registry/type_registry.py +0 -86
- flock/core/serialization/__init__.py +0 -13
- flock/core/serialization/callable_registry.py +0 -52
- flock/core/serialization/flock_serializer.py +0 -832
- flock/core/serialization/json_encoder.py +0 -41
- flock/core/serialization/secure_serializer.py +0 -175
- flock/core/serialization/serializable.py +0 -342
- flock/core/serialization/serialization_utils.py +0 -412
- flock/core/util/file_path_utils.py +0 -223
- flock/core/util/hydrator.py +0 -309
- flock/core/util/input_resolver.py +0 -164
- flock/core/util/loader.py +0 -59
- flock/core/util/splitter.py +0 -219
- flock/di.py +0 -27
- flock/platform/docker_tools.py +0 -49
- flock/platform/jaeger_install.py +0 -86
- flock/webapp/__init__.py +0 -1
- flock/webapp/app/__init__.py +0 -0
- flock/webapp/app/api/__init__.py +0 -0
- flock/webapp/app/api/agent_management.py +0 -241
- flock/webapp/app/api/execution.py +0 -709
- flock/webapp/app/api/flock_management.py +0 -129
- flock/webapp/app/api/registry_viewer.py +0 -30
- flock/webapp/app/chat.py +0 -665
- flock/webapp/app/config.py +0 -104
- flock/webapp/app/dependencies.py +0 -117
- flock/webapp/app/main.py +0 -1070
- flock/webapp/app/middleware.py +0 -113
- flock/webapp/app/models_ui.py +0 -7
- flock/webapp/app/services/__init__.py +0 -0
- flock/webapp/app/services/feedback_file_service.py +0 -363
- flock/webapp/app/services/flock_service.py +0 -337
- flock/webapp/app/services/sharing_models.py +0 -81
- flock/webapp/app/services/sharing_store.py +0 -762
- flock/webapp/app/templates/theme_mapper.html +0 -326
- flock/webapp/app/theme_mapper.py +0 -812
- flock/webapp/app/utils.py +0 -85
- flock/webapp/run.py +0 -215
- flock/webapp/static/css/chat.css +0 -301
- flock/webapp/static/css/components.css +0 -167
- flock/webapp/static/css/header.css +0 -39
- flock/webapp/static/css/layout.css +0 -46
- flock/webapp/static/css/sidebar.css +0 -127
- flock/webapp/static/css/two-pane.css +0 -48
- flock/webapp/templates/base.html +0 -200
- flock/webapp/templates/chat.html +0 -152
- flock/webapp/templates/chat_settings.html +0 -19
- flock/webapp/templates/flock_editor.html +0 -16
- flock/webapp/templates/index.html +0 -12
- flock/webapp/templates/partials/_agent_detail_form.html +0 -93
- flock/webapp/templates/partials/_agent_list.html +0 -18
- flock/webapp/templates/partials/_agent_manager_view.html +0 -51
- flock/webapp/templates/partials/_agent_tools_checklist.html +0 -14
- flock/webapp/templates/partials/_chat_container.html +0 -15
- flock/webapp/templates/partials/_chat_messages.html +0 -57
- flock/webapp/templates/partials/_chat_settings_form.html +0 -85
- flock/webapp/templates/partials/_create_flock_form.html +0 -50
- flock/webapp/templates/partials/_dashboard_flock_detail.html +0 -17
- flock/webapp/templates/partials/_dashboard_flock_file_list.html +0 -16
- flock/webapp/templates/partials/_dashboard_flock_properties_preview.html +0 -28
- flock/webapp/templates/partials/_dashboard_upload_flock_form.html +0 -16
- flock/webapp/templates/partials/_dynamic_input_form_content.html +0 -22
- flock/webapp/templates/partials/_env_vars_table.html +0 -23
- flock/webapp/templates/partials/_execution_form.html +0 -118
- flock/webapp/templates/partials/_execution_view_container.html +0 -28
- flock/webapp/templates/partials/_flock_file_list.html +0 -23
- flock/webapp/templates/partials/_flock_properties_form.html +0 -52
- flock/webapp/templates/partials/_flock_upload_form.html +0 -16
- flock/webapp/templates/partials/_header_flock_status.html +0 -5
- flock/webapp/templates/partials/_load_manager_view.html +0 -49
- flock/webapp/templates/partials/_registry_table.html +0 -25
- flock/webapp/templates/partials/_registry_viewer_content.html +0 -70
- flock/webapp/templates/partials/_results_display.html +0 -78
- flock/webapp/templates/partials/_settings_env_content.html +0 -9
- flock/webapp/templates/partials/_settings_theme_content.html +0 -14
- flock/webapp/templates/partials/_settings_view.html +0 -36
- flock/webapp/templates/partials/_share_chat_link_snippet.html +0 -11
- flock/webapp/templates/partials/_share_link_snippet.html +0 -35
- flock/webapp/templates/partials/_sidebar.html +0 -74
- flock/webapp/templates/partials/_streaming_results_container.html +0 -195
- flock/webapp/templates/partials/_structured_data_view.html +0 -40
- flock/webapp/templates/partials/_theme_preview.html +0 -36
- flock/webapp/templates/registry_viewer.html +0 -84
- flock/webapp/templates/shared_run_page.html +0 -140
- flock/workflow/__init__.py +0 -0
- flock/workflow/activities.py +0 -196
- flock/workflow/agent_activities.py +0 -24
- flock/workflow/agent_execution_activity.py +0 -202
- flock/workflow/flock_workflow.py +0 -214
- flock/workflow/temporal_config.py +0 -96
- flock/workflow/temporal_setup.py +0 -68
- flock_core-0.5.0b28.dist-info/METADATA +0 -274
- flock_core-0.5.0b28.dist-info/RECORD +0 -561
- flock_core-0.5.0b28.dist-info/entry_points.txt +0 -2
- /flock/{core/logging → logging}/formatters/themes.py +0 -0
- /flock/{core/logging → logging}/span_middleware/baggage_span_processor.py +0 -0
- /flock/{core/mcp → mcp}/util/__init__.py +0 -0
- {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
|
flock/core/mcp/__init__.py
DELETED
|
@@ -1 +0,0 @@
|
|
|
1
|
-
"""Flock MCP package."""
|